#!/bin/bash set -euo pipefail cd "$(dirname "$0")/.." || exit if [[ -z "${PROMPT_CACHE_FILE+x}" || -z "${CHAT_SAVE_DIR+x}" ]]; then echo >&2 "error: PROMPT_CACHE_FILE and CHAT_SAVE_DIR must be provided" exit 1 fi MODEL="${MODEL:-./models/llama-13b/ggml-model-q4_0.gguf}" PROMPT_TEMPLATE="${PROMPT_TEMPLATE:-./prompts/chat.txt}" USER_NAME="${USER_NAME:-User}" AI_NAME="${AI_NAME:-ChatLLaMa}" DATE_TIME="$(date +%H:%M)" DATE_YEAR="$(date +%Y)" LOG="${CHAT_SAVE_DIR}/main.log" LOG_BG="${CHAT_SAVE_DIR}/main-bg.log" CUR_PROMPT_FILE="${CHAT_SAVE_DIR}/current-prompt.txt" CUR_PROMPT_CACHE="${CHAT_SAVE_DIR}/current-cache.bin" NEXT_PROMPT_FILE="${CHAT_SAVE_DIR}/next-prompt.txt" NEXT_PROMPT_CACHE="${CHAT_SAVE_DIR}/next-cache.bin" SESSION_SIZE_MSG_PATTERN='main: session file matches [[:digit:]]+ / [[:digit:]]+' SAMPLE_TIME_MSG_PATTERN='sample time =[[:space:]]+[[:digit:]]+.[[:digit:]]+ ms /[[:space:]]+[[:digit:]]+' SED_DELETE_MESSAGES="/^(${USER_NAME}:|${AI_NAME}:|\\.\\.\\.)/,\$d" KV_SIZE=2048 KV_ROTATE_POINT=$((KV_SIZE * 3 / 5)) # REVIEW OPTS=(--model "$MODEL" --kv_size "$KV_SIZE" --repeat_last_n 256 "$@") # An unbuffered `tail -c+N` skip_bytes() { LANG=C IFS= read -r -n "$1" -d '' c while LANG=C IFS= read -r -n 1 -d '' c; do printf '%s' "$c" done } mkdir -p "$CHAT_SAVE_DIR" echo >"$LOG" trap "tail -n100 ${LOG}" EXIT if [[ ! -e "$CUR_PROMPT_FILE" ]]; then sed -e "s/\[\[USER_NAME\]\]/${USER_NAME}/g" \ -e "s/\[\[AI_NAME\]\]/${AI_NAME}/g" \ -e "s/\[\[DATE_TIME\]\]/${DATE_TIME}/g" \ -e "s/\[\[DATE_YEAR\]\]/${DATE_YEAR}/g" \ "$PROMPT_TEMPLATE" >"$CUR_PROMPT_FILE" fi if [[ ! -e "$NEXT_PROMPT_FILE" ]]; then sed -r "$SED_DELETE_MESSAGES" "$CUR_PROMPT_FILE" >"$NEXT_PROMPT_FILE" fi if [[ "$(tail -c4 "$NEXT_PROMPT_FILE")" != "..." ]]; then echo '...' >>"$NEXT_PROMPT_FILE" fi if [[ ! -e "$PROMPT_CACHE_FILE" ]]; then echo 'Prompt cache does not exist, building...' # Default batch_size to 64 here for better user feedback during initial prompt processing ./main 2>>"$LOG" \ --batch_size 64 \ "${OPTS[@]}" \ --prompt-cache "$PROMPT_CACHE_FILE" \ --file "$CUR_PROMPT_FILE" \ --n_predict 1 echo echo 'Done!' fi if [[ ! -e "$CUR_PROMPT_CACHE" ]]; then cp "$PROMPT_CACHE_FILE" "$CUR_PROMPT_CACHE" fi if [[ ! -e "$NEXT_PROMPT_CACHE" ]]; then cp "$PROMPT_CACHE_FILE" "$NEXT_PROMPT_CACHE" fi printf '%s ' "$(< "$CUR_PROMPT_FILE")" n_tokens=0 while read -e line; do # Limit generation to remaining context, with a buffer and estimating 2 chars/token for input n_predict=$((KV_SIZE - n_tokens - ${#line} / 2 - 32)) # Swap prompts when we're about to run out of context if ((n_predict <= 0)); then wait # for background main (below) to finish with next prompt mv "$NEXT_PROMPT_FILE" "$CUR_PROMPT_FILE" mv "$NEXT_PROMPT_CACHE" "$CUR_PROMPT_CACHE" sed -r "$SED_DELETE_MESSAGES" "$CUR_PROMPT_FILE" >"$NEXT_PROMPT_FILE" echo '...' >>"$NEXT_PROMPT_FILE" cp "$PROMPT_CACHE_FILE" "$NEXT_PROMPT_CACHE" n_tokens=0 n_predict=$((KV_SIZE / 2)) fi echo " ${line}" >>"$CUR_PROMPT_FILE" if ((n_tokens > KV_ROTATE_POINT)); then echo " ${line}" >>"$NEXT_PROMPT_FILE" fi n_prompt_len_pre=$(($(wc -c <"$CUR_PROMPT_FILE"))) printf '%s: ' "$AI_NAME" >>"$CUR_PROMPT_FILE" ./main 2>>"$LOG" "${OPTS[@]}" \ --prompt-cache "$CUR_PROMPT_CACHE" \ --prompt-cache-all \ --file "$CUR_PROMPT_FILE" \ --reverse-prompt "${USER_NAME}:" \ --n_predict "$n_predict" | skip_bytes 1 | # skip BOS token added by ./main tee "$CUR_PROMPT_FILE.tmp" | # save prompt + generation to tmp file skip_bytes "$n_prompt_len_pre" # print generation mv "$CUR_PROMPT_FILE.tmp" "$CUR_PROMPT_FILE" # if we hit n_predict instead of reverse-prompt, we need to add the prompt if [[ "$(tail -n1 "$CUR_PROMPT_FILE")" != "${USER_NAME}:" ]]; then printf '\n%s:' "$USER_NAME" printf '\n%s:' "$USER_NAME" >> "$CUR_PROMPT_FILE" fi printf ' ' # HACK get num tokens from debug message # TODO get both messages in one go if ! session_size_msg="$(tail -n30 "$LOG" | grep -oE "$SESSION_SIZE_MSG_PATTERN")" || ! sample_time_msg="$(tail -n10 "$LOG" | grep -oE "$SAMPLE_TIME_MSG_PATTERN")"; then echo >&2 "Couldn't get number of tokens from ./main output!" exit 1 fi n_tokens=$(($(cut -d/ -f2 <<<"$session_size_msg") + $(cut -d/ -f2 <<<"$sample_time_msg"))) if ((n_tokens > KV_ROTATE_POINT)); then tail -c+$((n_prompt_len_pre + 1)) "$CUR_PROMPT_FILE" >>"$NEXT_PROMPT_FILE" fi # Update cache for next prompt in background, ideally during user input ./main >>"$LOG_BG" 2>&1 "${OPTS[@]}" \ --prompt-cache "$NEXT_PROMPT_CACHE" \ --file "$NEXT_PROMPT_FILE" \ --n_predict 1 & done