|
<html> |
|
<head> |
|
<meta charset="UTF-8"> |
|
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1" /> |
|
<meta name="color-scheme" content="light dark"> |
|
<title>llama.cpp - chat</title> |
|
|
|
<style> |
|
body { |
|
font-family: system-ui; |
|
font-size: 90%; |
|
} |
|
|
|
.grid-container { |
|
display: grid; |
|
grid-template-columns: auto auto auto; |
|
padding: 10px; |
|
} |
|
|
|
.grid-item { |
|
padding: 5px; |
|
|
|
text-align: center; |
|
} |
|
|
|
#container { |
|
margin: 0em auto; |
|
display: flex; |
|
flex-direction: column; |
|
justify-content: space-between; |
|
height: 100%; |
|
} |
|
|
|
main { |
|
margin: 3px; |
|
display: flex; |
|
flex-direction: column; |
|
justify-content: space-between; |
|
gap: 1em; |
|
|
|
flex-grow: 1; |
|
overflow-y: auto; |
|
|
|
border: 1px solid #ccc; |
|
border-radius: 5px; |
|
padding: 0.5em; |
|
} |
|
|
|
h1 { |
|
text-align: center; |
|
} |
|
|
|
.customlink:link { |
|
color: white; |
|
background-color: #007aff; |
|
font-weight: 600; |
|
text-decoration: none; |
|
float: right; |
|
margin-top: 30px; |
|
display: flex; |
|
flex-direction: row; |
|
gap: 0.5em; |
|
justify-content: flex-end; |
|
border-radius: 4px; |
|
padding: 8px; |
|
} |
|
|
|
.customlink:visited { |
|
color: white; |
|
background-color: #007aff; |
|
font-weight: 600; |
|
text-decoration: none; |
|
float: right; |
|
margin-top: 30px; |
|
display: flex; |
|
flex-direction: row; |
|
gap: 0.5em; |
|
justify-content: flex-end; |
|
padding: 8px; |
|
} |
|
|
|
.customlink:hover { |
|
color: white; |
|
background-color: #0070ee; |
|
font-weight: 600; |
|
text-decoration: none; |
|
float: right; |
|
margin-top: 30px; |
|
display: flex; |
|
flex-direction: row; |
|
gap: 0.5em; |
|
justify-content: flex-end; |
|
padding: 8px; |
|
} |
|
|
|
.customlink:active { |
|
color: #0070ee; |
|
background-color: #80b3ef; |
|
font-weight: 600; |
|
text-decoration: none; |
|
float: right; |
|
margin-top: 30px; |
|
display: flex; |
|
flex-direction: row; |
|
gap: 0.5em; |
|
justify-content: flex-end; |
|
padding: 8px; |
|
} |
|
|
|
body { |
|
max-width: 600px; |
|
min-width: 300px; |
|
line-height: 1.2; |
|
margin: 0 auto; |
|
padding: 0 0.5em; |
|
} |
|
|
|
p { |
|
overflow-wrap: break-word; |
|
word-wrap: break-word; |
|
hyphens: auto; |
|
margin-top: 0.5em; |
|
margin-bottom: 0.5em; |
|
} |
|
|
|
#write form { |
|
margin: 1em 0 0 0; |
|
display: flex; |
|
flex-direction: column; |
|
gap: 0.5em; |
|
align-items: stretch; |
|
} |
|
|
|
.message-controls { |
|
display: flex; |
|
justify-content: flex-end; |
|
} |
|
.message-controls > div:nth-child(2) { |
|
display: flex; |
|
flex-direction: column; |
|
gap: 0.5em; |
|
} |
|
.message-controls > div:nth-child(2) > div { |
|
display: flex; |
|
margin-left: auto; |
|
gap: 0.5em; |
|
} |
|
|
|
fieldset { |
|
border: none; |
|
padding: 0; |
|
margin: 0; |
|
} |
|
|
|
fieldset.two { |
|
display: grid; |
|
grid-template: "a a"; |
|
gap: 1em; |
|
} |
|
|
|
fieldset.three { |
|
display: grid; |
|
grid-template: "a a a"; |
|
gap: 1em; |
|
} |
|
|
|
details { |
|
border: 1px solid #aaa; |
|
border-radius: 4px; |
|
padding: 0.5em 0.5em 0; |
|
margin-top: 0.5em; |
|
} |
|
|
|
summary { |
|
font-weight: bold; |
|
margin: -0.5em -0.5em 0; |
|
padding: 0.5em; |
|
cursor: pointer; |
|
} |
|
|
|
details[open] { |
|
padding: 0.5em; |
|
} |
|
|
|
.prob-set { |
|
padding: 0.3em; |
|
border-bottom: 1px solid #ccc; |
|
} |
|
|
|
.popover-content { |
|
position: absolute; |
|
background-color: white; |
|
padding: 0.2em; |
|
box-shadow: 0 0 10px rgba(0, 0, 0, 0.1); |
|
} |
|
|
|
textarea { |
|
padding: 5px; |
|
flex-grow: 1; |
|
width: 100%; |
|
} |
|
|
|
pre code { |
|
display: block; |
|
background-color: #222; |
|
color: #ddd; |
|
} |
|
|
|
code { |
|
font-family: monospace; |
|
padding: 0.1em 0.3em; |
|
border-radius: 3px; |
|
} |
|
|
|
fieldset label { |
|
margin: 0.5em 0; |
|
display: block; |
|
} |
|
|
|
fieldset label.slim { |
|
margin: 0 0.5em; |
|
display: inline; |
|
} |
|
|
|
header, |
|
footer { |
|
text-align: center; |
|
} |
|
|
|
footer { |
|
font-size: 80%; |
|
color: #888; |
|
} |
|
|
|
.mode-chat textarea[name=prompt] { |
|
height: 4.5em; |
|
} |
|
|
|
.mode-completion textarea[name=prompt] { |
|
height: 10em; |
|
} |
|
|
|
[contenteditable] { |
|
display: inline-block; |
|
white-space: pre-wrap; |
|
outline: 0px solid transparent; |
|
} |
|
|
|
@keyframes loading-bg-wipe { |
|
0% { |
|
background-position: 0%; |
|
} |
|
|
|
100% { |
|
background-position: 100%; |
|
} |
|
} |
|
|
|
.loading { |
|
--loading-color-1: #eeeeee00; |
|
--loading-color-2: #eeeeeeff; |
|
background-size: 50% 100%; |
|
background-image: linear-gradient(90deg, var(--loading-color-1), var(--loading-color-2), var(--loading-color-1)); |
|
animation: loading-bg-wipe 2s linear infinite; |
|
} |
|
|
|
@media (prefers-color-scheme: dark) { |
|
.loading { |
|
--loading-color-1: #22222200; |
|
--loading-color-2: #222222ff; |
|
} |
|
|
|
.popover-content { |
|
background-color: black; |
|
} |
|
} |
|
</style> |
|
|
|
<script type="module"> |
|
import { |
|
html, h, signal, effect, computed, render, useSignal, useEffect, useRef, Component |
|
} from './index.js'; |
|
|
|
import { llama } from './completion.js'; |
|
import { SchemaConverter } from './json-schema-to-grammar.mjs'; |
|
|
|
let selected_image = false; |
|
var slot_id = -1; |
|
|
|
const session = signal({ |
|
prompt: "This is a conversation between User and Llama, a friendly chatbot. Llama is helpful, kind, honest, good at writing, and never fails to answer any requests immediately and with precision.", |
|
template: "{{prompt}}\n\n{{history}}\n{{char}}:", |
|
historyTemplate: "{{name}}: {{message}}", |
|
transcript: [], |
|
type: "chat", |
|
char: "Llama", |
|
user: "User", |
|
image_selected: '' |
|
}) |
|
|
|
const params = signal({ |
|
n_predict: 400, |
|
temperature: 0.7, |
|
repeat_last_n: 256, |
|
repeat_penalty: 1.18, |
|
penalize_nl: false, |
|
dry_multiplier: 0.0, |
|
dry_base: 1.75, |
|
dry_allowed_length: 2, |
|
dry_penalty_last_n: -1, |
|
top_k: 40, |
|
top_p: 0.95, |
|
min_p: 0.05, |
|
xtc_probability: 0.0, |
|
xtc_threshold: 0.1, |
|
typical_p: 1.0, |
|
presence_penalty: 0.0, |
|
frequency_penalty: 0.0, |
|
mirostat: 0, |
|
mirostat_tau: 5, |
|
mirostat_eta: 0.1, |
|
grammar: '', |
|
n_probs: 0, |
|
min_keep: 0, |
|
image_data: [], |
|
cache_prompt: true, |
|
api_key: '' |
|
}) |
|
|
|
|
|
|
|
const local_storage_storageKey = "llamacpp_server_local_storage"; |
|
|
|
function local_storage_setDataFromObject(tag, content) { |
|
localStorage.setItem(local_storage_storageKey + '/' + tag, JSON.stringify(content)); |
|
} |
|
|
|
function local_storage_setDataFromRawText(tag, content) { |
|
localStorage.setItem(local_storage_storageKey + '/' + tag, content); |
|
} |
|
|
|
function local_storage_getDataAsObject(tag) { |
|
const item = localStorage.getItem(local_storage_storageKey + '/' + tag); |
|
if (!item) { |
|
return null; |
|
} else { |
|
return JSON.parse(item); |
|
} |
|
} |
|
|
|
function local_storage_getDataAsRawText(tag) { |
|
const item = localStorage.getItem(local_storage_storageKey + '/' + tag); |
|
if (!item) { |
|
return null; |
|
} else { |
|
return item; |
|
} |
|
} |
|
|
|
|
|
|
|
const savedUserTemplates = signal({}) |
|
const selectedUserTemplate = signal({ name: '', template: { session: {}, params: {} } }) |
|
|
|
|
|
|
|
|
|
|
|
console.log('Importing saved templates') |
|
|
|
let importedTemplates = local_storage_getDataAsObject('user_templates') |
|
|
|
if (importedTemplates) { |
|
|
|
|
|
console.log('Processing saved templates and updating default template') |
|
params.value = { ...params.value, image_data: [] }; |
|
|
|
|
|
savedUserTemplates.value = importedTemplates; |
|
|
|
|
|
savedUserTemplates.value.default = { session: session.value, params: params.value } |
|
local_storage_setDataFromObject('user_templates', savedUserTemplates.value) |
|
} else { |
|
|
|
|
|
console.log('Initializing LocalStorage and saving default template') |
|
|
|
savedUserTemplates.value = { "default": { session: session.value, params: params.value } } |
|
local_storage_setDataFromObject('user_templates', savedUserTemplates.value) |
|
} |
|
|
|
function userTemplateResetToDefault() { |
|
console.log('Resetting template to default') |
|
selectedUserTemplate.value.name = 'default'; |
|
selectedUserTemplate.value.data = savedUserTemplates.value['default']; |
|
} |
|
|
|
function userTemplateApply(t) { |
|
session.value = t.data.session; |
|
session.value = { ...session.value, image_selected: '' }; |
|
params.value = t.data.params; |
|
params.value = { ...params.value, image_data: [] }; |
|
} |
|
|
|
function userTemplateResetToDefaultAndApply() { |
|
userTemplateResetToDefault() |
|
userTemplateApply(selectedUserTemplate.value) |
|
} |
|
|
|
function userTemplateLoadAndApplyAutosaved() { |
|
|
|
let lastUsedTemplate = local_storage_getDataAsObject('user_templates_last') |
|
|
|
if (lastUsedTemplate) { |
|
|
|
console.log('Autosaved template found, restoring') |
|
|
|
selectedUserTemplate.value = lastUsedTemplate |
|
} |
|
else { |
|
|
|
console.log('No autosaved template found, using default template') |
|
|
|
|
|
userTemplateResetToDefault() |
|
} |
|
|
|
console.log('Applying template') |
|
|
|
|
|
userTemplateApply(selectedUserTemplate.value) |
|
} |
|
|
|
|
|
|
|
|
|
function userTemplateAutosave() { |
|
console.log('Template Autosave...') |
|
if (selectedUserTemplate.value.name == 'default') { |
|
|
|
let newTemplateName = 'UserTemplate-' + Date.now().toString() |
|
let newTemplate = { 'name': newTemplateName, 'data': { 'session': session.value, 'params': params.value } } |
|
|
|
console.log('Saving as ' + newTemplateName) |
|
|
|
|
|
local_storage_setDataFromObject('user_templates_last', newTemplate) |
|
|
|
|
|
userTemplateLoadAndApplyAutosaved() |
|
} else { |
|
local_storage_setDataFromObject('user_templates_last', { 'name': selectedUserTemplate.value.name, 'data': { 'session': session.value, 'params': params.value } }) |
|
} |
|
} |
|
|
|
console.log('Checking for autosaved last used template') |
|
userTemplateLoadAndApplyAutosaved() |
|
|
|
|
|
|
|
const tts = window.speechSynthesis; |
|
const ttsVoice = signal(null) |
|
|
|
const llamaStats = signal(null) |
|
const controller = signal(null) |
|
|
|
|
|
const generating = computed(() => controller.value != null) |
|
|
|
|
|
const chatStarted = computed(() => session.value.transcript.length > 0) |
|
|
|
const transcriptUpdate = (transcript) => { |
|
session.value = { |
|
...session.value, |
|
transcript |
|
} |
|
} |
|
|
|
|
|
const template = (str, extraSettings) => { |
|
let settings = session.value; |
|
if (extraSettings) { |
|
settings = { ...settings, ...extraSettings }; |
|
} |
|
return String(str).replaceAll(/\{\{(.*?)\}\}/g, (_, key) => template(settings[key])); |
|
} |
|
|
|
async function runLlama(prompt, llamaParams, char) { |
|
const currentMessages = []; |
|
const history = session.value.transcript; |
|
if (controller.value) { |
|
throw new Error("already running"); |
|
} |
|
controller.value = new AbortController(); |
|
for await (const chunk of llama(prompt, llamaParams, { controller: controller.value, api_url: new URL('.', document.baseURI).href })) { |
|
const data = chunk.data; |
|
|
|
if (data.stop) { |
|
while ( |
|
currentMessages.length > 0 && |
|
currentMessages[currentMessages.length - 1].content.match(/\n$/) != null |
|
) { |
|
currentMessages.pop(); |
|
} |
|
transcriptUpdate([...history, [char, currentMessages]]) |
|
console.log("Completion finished: '", currentMessages.map(msg => msg.content).join(''), "', summary: ", data); |
|
} else { |
|
currentMessages.push(data); |
|
slot_id = data.slot_id; |
|
if (selected_image && !data.multimodal) { |
|
alert("The server was not compiled for multimodal or the model projector can't be loaded."); |
|
return; |
|
} |
|
transcriptUpdate([...history, [char, currentMessages]]) |
|
} |
|
|
|
if (data.timings) { |
|
llamaStats.value = data; |
|
} |
|
} |
|
|
|
controller.value = null; |
|
} |
|
|
|
|
|
const chat = async (msg) => { |
|
if (controller.value) { |
|
console.log('already running...'); |
|
return; |
|
} |
|
|
|
transcriptUpdate([...session.value.transcript, ["{{user}}", msg]]) |
|
|
|
let prompt = template(session.value.template, { |
|
message: msg, |
|
history: session.value.transcript.flatMap( |
|
([name, data]) => |
|
template( |
|
session.value.historyTemplate, |
|
{ |
|
name, |
|
message: Array.isArray(data) ? |
|
data.map(msg => msg.content).join('').replace(/^\s/, '') : |
|
data, |
|
} |
|
) |
|
).join("\n"), |
|
}); |
|
if (selected_image) { |
|
prompt = `A chat between a curious human and an artificial intelligence assistant. The assistant gives helpful, detailed, and polite answers to the human's questions.\nUSER:[img-10]${msg}\nASSISTANT:`; |
|
} |
|
await runLlama(prompt, { |
|
...params.value, |
|
slot_id: slot_id, |
|
stop: ["</s>", template("{{char}}:"), template("{{user}}:")], |
|
}, "{{char}}"); |
|
} |
|
|
|
const runCompletion = () => { |
|
if (controller.value) { |
|
console.log('already running...'); |
|
return; |
|
} |
|
const { prompt } = session.value; |
|
transcriptUpdate([...session.value.transcript, ["", prompt]]); |
|
runLlama(prompt, { |
|
...params.value, |
|
slot_id: slot_id, |
|
stop: [], |
|
}, "").finally(() => { |
|
session.value.prompt = session.value.transcript.map(([_, data]) => |
|
Array.isArray(data) ? data.map(msg => msg.content).join('') : data |
|
).join(''); |
|
session.value.transcript = []; |
|
}) |
|
} |
|
|
|
const stop = (e) => { |
|
e.preventDefault(); |
|
if (controller.value) { |
|
controller.value.abort(); |
|
controller.value = null; |
|
} |
|
} |
|
|
|
const reset = (e) => { |
|
stop(e); |
|
transcriptUpdate([]); |
|
} |
|
|
|
const uploadImage = (e) => { |
|
e.preventDefault(); |
|
document.getElementById("fileInput").click(); |
|
document.getElementById("fileInput").addEventListener("change", function (event) { |
|
const selectedFile = event.target.files[0]; |
|
if (selectedFile) { |
|
const reader = new FileReader(); |
|
reader.onload = function () { |
|
const image_data = reader.result; |
|
session.value = { ...session.value, image_selected: image_data }; |
|
params.value = { |
|
...params.value, image_data: [ |
|
{ data: image_data.replace(/data:image\/[^;]+;base64,/, ''), id: 10 }] |
|
} |
|
}; |
|
selected_image = true; |
|
reader.readAsDataURL(selectedFile); |
|
} |
|
}); |
|
} |
|
|
|
const SpeechRecognition = window.SpeechRecognition || window.webkitSpeechRecognition; |
|
const talkRecognition = SpeechRecognition ? new SpeechRecognition() : null; |
|
function MessageInput() { |
|
const message = useSignal(""); |
|
|
|
const talkActive = useSignal(false); |
|
const sendOnTalk = useSignal(false); |
|
const talkStop = (e) => { |
|
if (e) e.preventDefault(); |
|
|
|
talkActive.value = false; |
|
talkRecognition?.stop(); |
|
} |
|
const talk = (e) => { |
|
e.preventDefault(); |
|
|
|
if (talkRecognition) |
|
talkRecognition.start(); |
|
else |
|
alert("Speech recognition is not supported by this browser."); |
|
} |
|
if(talkRecognition) { |
|
talkRecognition.onstart = () => { |
|
talkActive.value = true; |
|
} |
|
talkRecognition.onresult = (e) => { |
|
if (event.results.length > 0) { |
|
message.value = event.results[0][0].transcript; |
|
if (sendOnTalk.value) { |
|
submit(e); |
|
} |
|
} |
|
} |
|
talkRecognition.onspeechend = () => { |
|
talkStop(); |
|
} |
|
} |
|
|
|
const ttsVoices = useSignal(tts?.getVoices() || []); |
|
const ttsVoiceDefault = computed(() => ttsVoices.value.find(v => v.default)); |
|
if (tts) { |
|
tts.onvoiceschanged = () => { |
|
ttsVoices.value = tts.getVoices(); |
|
} |
|
} |
|
|
|
const submit = (e) => { |
|
stop(e); |
|
chat(message.value); |
|
message.value = ""; |
|
} |
|
|
|
const enterSubmits = (event) => { |
|
if (event.which === 13 && !event.shiftKey) { |
|
submit(event); |
|
} |
|
} |
|
|
|
return html` |
|
<form onsubmit=${submit}> |
|
<div> |
|
<textarea |
|
className=${generating.value ? "loading" : null} |
|
oninput=${(e) => message.value = e.target.value} |
|
onkeypress=${enterSubmits} |
|
placeholder="Say something..." |
|
rows=2 |
|
type="text" |
|
value="${message}" |
|
/> |
|
</div> |
|
<div class="message-controls"> |
|
<div> </div> |
|
<div> |
|
<div> |
|
<button type="submit" disabled=${generating.value || talkActive.value}>Send</button> |
|
<button disabled=${generating.value || talkActive.value} onclick=${uploadImage}>Upload Image</button> |
|
<button onclick=${stop} disabled=${!generating.value}>Stop</button> |
|
<button onclick=${reset}>Reset</button> |
|
</div> |
|
<div> |
|
<a href="#" style="cursor: help;" title="Help" onclick=${e => { |
|
e.preventDefault(); |
|
alert(`STT supported by your browser: ${SpeechRecognition ? 'Yes' : 'No'}\n` + |
|
`(TTS and speech recognition are not provided by llama.cpp)\n` + |
|
`Note: STT requires HTTPS to work.`); |
|
}}>[?]</a> |
|
<button disabled=${generating.value} onclick=${talkActive.value ? talkStop : talk}>${talkActive.value ? "Stop Talking" : "Talk"}</button> |
|
<div> |
|
<input type="checkbox" id="send-on-talk" name="send-on-talk" checked="${sendOnTalk}" onchange=${(e) => sendOnTalk.value = e.target.checked} /> |
|
<label for="send-on-talk" style="line-height: initial;">Send after talking</label> |
|
</div> |
|
</div> |
|
<div> |
|
<a href="#" style="cursor: help;" title="Help" onclick=${e => { |
|
e.preventDefault(); |
|
alert(`TTS supported by your browser: ${tts ? 'Yes' : 'No'}\n(TTS and speech recognition are not provided by llama.cpp)`); |
|
}}>[?]</a> |
|
<label for="tts-voices" style="line-height: initial;">Bot Voice:</label> |
|
<select id="tts-voices" name="tts-voices" onchange=${(e) => ttsVoice.value = e.target.value} style="max-width: 100px;"> |
|
<option value="" selected="${!ttsVoice.value}">None</option> |
|
${[ |
|
...(ttsVoiceDefault.value ? [ttsVoiceDefault.value] : []), |
|
...ttsVoices.value.filter(v => !v.default), |
|
].map( |
|
v => html`<option value="${v.name}" selected="${ttsVoice.value === v.name}">${v.name} (${v.lang}) ${v.default ? '(default)' : ''}</option>` |
|
)} |
|
</select> |
|
</div> |
|
</div> |
|
</div> |
|
</form> |
|
` |
|
} |
|
|
|
function CompletionControls() { |
|
const submit = (e) => { |
|
stop(e); |
|
runCompletion(); |
|
} |
|
return html` |
|
<div> |
|
<button onclick=${submit} type="button" disabled=${generating.value}>Start</button> |
|
<button onclick=${stop} disabled=${!generating.value}>Stop</button> |
|
<button onclick=${reset}>Reset</button> |
|
</div>`; |
|
} |
|
|
|
const ChatLog = (props) => { |
|
const messages = session.value.transcript; |
|
const container = useRef(null) |
|
|
|
useEffect(() => { |
|
|
|
const parent = container.current.parentElement; |
|
if (parent && parent.scrollHeight <= parent.scrollTop + parent.offsetHeight + 300) { |
|
parent.scrollTo(0, parent.scrollHeight) |
|
} |
|
}, [messages]) |
|
|
|
const ttsChatLineActiveIx = useSignal(undefined); |
|
const ttsChatLine = (e, ix, msg) => { |
|
if (e) e.preventDefault(); |
|
|
|
if (!tts || !ttsVoice.value || !('SpeechSynthesisUtterance' in window)) return; |
|
|
|
const ttsVoices = tts.getVoices(); |
|
const voice = ttsVoices.find(v => v.name === ttsVoice.value); |
|
if (!voice) return; |
|
|
|
if (ttsChatLineActiveIx.value !== undefined) { |
|
tts.cancel(); |
|
if (ttsChatLineActiveIx.value === ix) { |
|
ttsChatLineActiveIx.value = undefined; |
|
return; |
|
} |
|
} |
|
|
|
ttsChatLineActiveIx.value = ix; |
|
let ttsUtter = new SpeechSynthesisUtterance(msg); |
|
ttsUtter.voice = voice; |
|
ttsUtter.onend = e => { |
|
ttsChatLineActiveIx.value = undefined; |
|
}; |
|
tts.speak(ttsUtter); |
|
} |
|
|
|
const isCompletionMode = session.value.type === 'completion' |
|
|
|
|
|
const lastCharChatLinesIxs = useSignal([]); |
|
const lastCharChatLinesIxsOld = useSignal([]); |
|
useEffect(() => { |
|
if ( |
|
!isCompletionMode |
|
&& lastCharChatLinesIxs.value.length !== lastCharChatLinesIxsOld.value.length |
|
&& !generating.value |
|
) { |
|
const ix = lastCharChatLinesIxs.value[lastCharChatLinesIxs.value.length - 1]; |
|
if (ix !== undefined) { |
|
const msg = messages[ix]; |
|
ttsChatLine(null, ix, Array.isArray(msg) ? msg[1].map(m => m.content).join('') : msg); |
|
} |
|
|
|
lastCharChatLinesIxsOld.value = structuredClone(lastCharChatLinesIxs.value); |
|
} |
|
}, [generating.value]); |
|
|
|
const chatLine = ([user, data], index) => { |
|
let message |
|
const isArrayMessage = Array.isArray(data); |
|
const text = isArrayMessage ? |
|
data.map(msg => msg.content).join('') : |
|
data; |
|
if (params.value.n_probs > 0 && isArrayMessage) { |
|
message = html`<${Probabilities} data=${data} />` |
|
} else { |
|
message = isCompletionMode ? |
|
text : |
|
html`<${Markdownish} text=${template(text)} />` |
|
} |
|
|
|
const fromBot = user && user === '{{char}}'; |
|
if (fromBot && !lastCharChatLinesIxs.value.includes(index)) |
|
lastCharChatLinesIxs.value.push(index); |
|
|
|
if (user) { |
|
return html` |
|
<div> |
|
<p key=${index}><strong>${template(user)}:</strong> ${message}</p> |
|
${ |
|
fromBot && ttsVoice.value |
|
&& html`<button disabled=${generating.value} onclick=${e => ttsChatLine(e, index, text)} aria-label=${ttsChatLineActiveIx.value === index ? 'Pause' : 'Play'}>${ ttsChatLineActiveIx.value === index ? '⏸️' : '▶️' }</div>` |
|
} |
|
</div> |
|
`; |
|
} else { |
|
return isCompletionMode ? |
|
html`<span key=${index}>${message}</span>` : |
|
html`<div><p key=${index}>${message}</p></div>` |
|
} |
|
}; |
|
|
|
const handleCompletionEdit = (e) => { |
|
session.value.prompt = e.target.innerText; |
|
session.value.transcript = []; |
|
} |
|
|
|
return html` |
|
<div id="chat" ref=${container} key=${messages.length}> |
|
<img style="width: 60%;${!session.value.image_selected ? `display: none;` : ``}" src="${session.value.image_selected}"/> |
|
<span contenteditable=${isCompletionMode} ref=${container} oninput=${handleCompletionEdit}> |
|
${messages.flatMap(chatLine)} |
|
</span> |
|
</div>`; |
|
}; |
|
|
|
const ConfigForm = (props) => { |
|
const updateSession = (el) => session.value = { ...session.value, [el.target.name]: el.target.value } |
|
const updateParams = (el) => params.value = { ...params.value, [el.target.name]: el.target.value } |
|
const updateParamsFloat = (el) => params.value = { ...params.value, [el.target.name]: parseFloat(el.target.value) } |
|
const updateParamsInt = (el) => params.value = { ...params.value, [el.target.name]: Math.floor(parseFloat(el.target.value)) } |
|
const updateParamsBool = (el) => params.value = { ...params.value, [el.target.name]: el.target.checked } |
|
|
|
const grammarJsonSchemaPropOrder = signal('') |
|
const updateGrammarJsonSchemaPropOrder = (el) => grammarJsonSchemaPropOrder.value = el.target.value |
|
const convertJSONSchemaGrammar = async () => { |
|
try { |
|
let schema = JSON.parse(params.value.grammar) |
|
const converter = new SchemaConverter({ |
|
prop_order: grammarJsonSchemaPropOrder.value |
|
.split(',') |
|
.reduce((acc, cur, i) => ({ ...acc, [cur.trim()]: i }), {}), |
|
allow_fetch: true, |
|
}) |
|
schema = await converter.resolveRefs(schema, 'input') |
|
converter.visit(schema, '') |
|
params.value = { |
|
...params.value, |
|
grammar: converter.formatGrammar(), |
|
} |
|
} catch (e) { |
|
alert(`Convert failed: ${e.message}`) |
|
} |
|
} |
|
|
|
const FloatField = ({ label, max, min, name, step, value }) => { |
|
return html` |
|
<div> |
|
<label for="${name}">${label}</label> |
|
<input type="range" id="${name}" min="${min}" max="${max}" step="${step}" name="${name}" value="${value}" oninput=${updateParamsFloat} /> |
|
<span>${value}</span> |
|
</div> |
|
` |
|
}; |
|
|
|
const IntField = ({ label, max, min, name, value }) => { |
|
return html` |
|
<div> |
|
<label for="${name}">${label}</label> |
|
<input type="range" id="${name}" min="${min}" max="${max}" name="${name}" value="${value}" oninput=${updateParamsInt} /> |
|
<span>${value}</span> |
|
</div> |
|
` |
|
}; |
|
|
|
const BoolField = ({ label, name, value }) => { |
|
return html` |
|
<div> |
|
<label for="${name}">${label}</label> |
|
<input type="checkbox" id="${name}" name="${name}" checked="${value}" onclick=${updateParamsBool} /> |
|
</div> |
|
` |
|
}; |
|
|
|
const userTemplateReset = (e) => { |
|
e.preventDefault(); |
|
userTemplateResetToDefaultAndApply() |
|
} |
|
|
|
const UserTemplateResetButton = () => { |
|
if (selectedUserTemplate.value.name == 'default') { |
|
return html` |
|
<button disabled>Using default template</button> |
|
` |
|
} |
|
|
|
return html` |
|
<button onclick=${userTemplateReset}>Reset all to default</button> |
|
` |
|
}; |
|
|
|
useEffect(() => { |
|
|
|
userTemplateAutosave() |
|
}, [session.value, params.value]) |
|
|
|
const GrammarControl = () => ( |
|
html` |
|
<div> |
|
<label for="template">Grammar</label> |
|
<textarea id="grammar" name="grammar" placeholder="Use gbnf or JSON Schema+convert" value="${params.value.grammar}" rows=4 oninput=${updateParams}/> |
|
<input type="text" name="prop-order" placeholder="order: prop1,prop2,prop3" oninput=${updateGrammarJsonSchemaPropOrder} /> |
|
<button type="button" onclick=${convertJSONSchemaGrammar}>Convert JSON Schema</button> |
|
</div> |
|
` |
|
); |
|
|
|
const PromptControlFieldSet = () => ( |
|
html` |
|
<fieldset> |
|
<div> |
|
<label htmlFor="prompt">Prompt</label> |
|
<textarea type="text" name="prompt" value="${session.value.prompt}" oninput=${updateSession}/> |
|
</div> |
|
</fieldset> |
|
` |
|
); |
|
|
|
const ChatConfigForm = () => ( |
|
html` |
|
${PromptControlFieldSet()} |
|
|
|
<fieldset class="two"> |
|
<div> |
|
<label for="user">User name</label> |
|
<input type="text" name="user" value="${session.value.user}" oninput=${updateSession} /> |
|
</div> |
|
|
|
<div> |
|
<label for="bot">Bot name</label> |
|
<input type="text" name="char" value="${session.value.char}" oninput=${updateSession} /> |
|
</div> |
|
</fieldset> |
|
|
|
<fieldset> |
|
<div> |
|
<label for="template">Prompt template</label> |
|
<textarea id="template" name="template" value="${session.value.template}" rows=4 oninput=${updateSession}/> |
|
</div> |
|
|
|
<div> |
|
<label for="template">Chat history template</label> |
|
<textarea id="template" name="historyTemplate" value="${session.value.historyTemplate}" rows=1 oninput=${updateSession}/> |
|
</div> |
|
${GrammarControl()} |
|
</fieldset> |
|
` |
|
); |
|
|
|
const CompletionConfigForm = () => ( |
|
html` |
|
${PromptControlFieldSet()} |
|
<fieldset>${GrammarControl()}</fieldset> |
|
` |
|
); |
|
|
|
return html` |
|
<form> |
|
<fieldset class="two"> |
|
<${UserTemplateResetButton}/> |
|
<div> |
|
<label class="slim"><input type="radio" name="type" value="chat" checked=${session.value.type === "chat"} oninput=${updateSession} /> Chat</label> |
|
<label class="slim"><input type="radio" name="type" value="completion" checked=${session.value.type === "completion"} oninput=${updateSession} /> Completion</label> |
|
</div> |
|
</fieldset> |
|
|
|
${session.value.type === 'chat' ? ChatConfigForm() : CompletionConfigForm()} |
|
|
|
<fieldset class="two"> |
|
${IntField({ label: "Predictions", max: 2048, min: -1, name: "n_predict", value: params.value.n_predict })} |
|
${FloatField({ label: "Temperature", max: 2.0, min: 0.0, name: "temperature", step: 0.01, value: params.value.temperature })} |
|
${FloatField({ label: "Penalize repeat sequence", max: 2.0, min: 0.0, name: "repeat_penalty", step: 0.01, value: params.value.repeat_penalty })} |
|
${IntField({ label: "Consider N tokens for penalize", max: 2048, min: 0, name: "repeat_last_n", value: params.value.repeat_last_n })} |
|
${BoolField({ label: "Penalize repetition of newlines", name: "penalize_nl", value: params.value.penalize_nl })} |
|
${IntField({ label: "Top-K sampling", max: 100, min: -1, name: "top_k", value: params.value.top_k })} |
|
${FloatField({ label: "Top-P sampling", max: 1.0, min: 0.0, name: "top_p", step: 0.01, value: params.value.top_p })} |
|
${FloatField({ label: "Min-P sampling", max: 1.0, min: 0.0, name: "min_p", step: 0.01, value: params.value.min_p })} |
|
</fieldset> |
|
<details> |
|
<summary>More options</summary> |
|
<fieldset class="two"> |
|
${FloatField({ label: "Typical P", max: 1.0, min: 0.0, name: "typical_p", step: 0.01, value: params.value.typical_p })} |
|
${FloatField({ label: "Presence penalty", max: 1.0, min: 0.0, name: "presence_penalty", step: 0.01, value: params.value.presence_penalty })} |
|
${FloatField({ label: "Frequency penalty", max: 1.0, min: 0.0, name: "frequency_penalty", step: 0.01, value: params.value.frequency_penalty })} |
|
${FloatField({ label: "DRY Penalty Multiplier", max: 5.0, min: 0.0, name: "dry_multiplier", step: 0.01, value: params.value.dry_multiplier })} |
|
${FloatField({ label: "DRY Base", max: 3.0, min: 1.0, name: "dry_base", step: 0.01, value: params.value.dry_base })} |
|
${IntField({ label: "DRY Allowed Length", max: 10, min: 2, step: 1, name: "dry_allowed_length", value: params.value.dry_allowed_length })} |
|
${IntField({ label: "DRY Penalty Last N", max: 2048, min: -1, step: 16, name: "dry_penalty_last_n", value: params.value.dry_penalty_last_n })} |
|
${FloatField({ label: "XTC probability", max: 1.0, min: 0.0, name: "xtc_probability", step: 0.01, value: params.value.xtc_probability })} |
|
${FloatField({ label: "XTC threshold", max: 0.5, min: 0.0, name: "xtc_threshold", step: 0.01, value: params.value.xtc_threshold })} |
|
</fieldset> |
|
<hr /> |
|
<fieldset class="three"> |
|
<div> |
|
<label><input type="radio" name="mirostat" value="0" checked=${params.value.mirostat == 0} oninput=${updateParamsInt} /> no Mirostat</label> |
|
<label><input type="radio" name="mirostat" value="1" checked=${params.value.mirostat == 1} oninput=${updateParamsInt} /> Mirostat v1</label> |
|
<label><input type="radio" name="mirostat" value="2" checked=${params.value.mirostat == 2} oninput=${updateParamsInt} /> Mirostat v2</label> |
|
</div> |
|
${FloatField({ label: "Mirostat tau", max: 10.0, min: 0.0, name: "mirostat_tau", step: 0.01, value: params.value.mirostat_tau })} |
|
${FloatField({ label: "Mirostat eta", max: 1.0, min: 0.0, name: "mirostat_eta", step: 0.01, value: params.value.mirostat_eta })} |
|
</fieldset> |
|
<fieldset> |
|
${IntField({ label: "Show Probabilities", max: 10, min: 0, name: "n_probs", value: params.value.n_probs })} |
|
</fieldset> |
|
<fieldset> |
|
${IntField({ label: "Min Probabilities from each Sampler", max: 10, min: 0, name: "min_keep", value: params.value.min_keep })} |
|
</fieldset> |
|
<fieldset> |
|
<label for="api_key">API Key</label> |
|
<input type="text" name="api_key" value="${params.value.api_key}" placeholder="Enter API key" oninput=${updateParams} /> |
|
</fieldset> |
|
</details> |
|
</form> |
|
` |
|
} |
|
|
|
const probColor = (p) => { |
|
const r = Math.floor(192 * (1 - p)); |
|
const g = Math.floor(192 * p); |
|
return `rgba(${r},${g},0,0.3)`; |
|
} |
|
|
|
const Probabilities = (params) => { |
|
return params.data.map(msg => { |
|
const { completion_probabilities } = msg; |
|
if ( |
|
!completion_probabilities || |
|
completion_probabilities.length === 0 |
|
) return msg.content |
|
|
|
if (completion_probabilities.length > 1) { |
|
|
|
if (completion_probabilities[0].content.startsWith('byte: \\')) return msg.content |
|
|
|
const splitData = completion_probabilities.map(prob => ({ |
|
content: prob.content, |
|
completion_probabilities: [prob] |
|
})) |
|
return html`<${Probabilities} data=${splitData} />` |
|
} |
|
|
|
const { probs, content } = completion_probabilities[0] |
|
const found = probs.find(p => p.tok_str === msg.content) |
|
const pColor = found ? probColor(found.prob) : 'transparent' |
|
|
|
const popoverChildren = html` |
|
<div class="prob-set"> |
|
${probs.map((p, index) => { |
|
return html` |
|
<div |
|
key=${index} |
|
title=${`prob: ${p.prob}`} |
|
style=${{ |
|
padding: '0.3em', |
|
backgroundColor: p.tok_str === content ? probColor(p.prob) : 'transparent' |
|
}} |
|
> |
|
<span>${p.tok_str}: </span> |
|
<span>${Math.floor(p.prob * 100)}%</span> |
|
</div> |
|
` |
|
})} |
|
</div> |
|
` |
|
|
|
return html` |
|
<${Popover} style=${{ backgroundColor: pColor }} popoverChildren=${popoverChildren}> |
|
${msg.content.match(/\n/gim) ? html`<br />` : msg.content} |
|
</> |
|
` |
|
}); |
|
} |
|
|
|
|
|
const Markdownish = (params) => { |
|
const chunks = params.text.split('```'); |
|
|
|
for (let i = 0; i < chunks.length; i++) { |
|
if (i % 2 === 0) { |
|
chunks[i] = chunks[i] |
|
.replace(/&/g, '&') |
|
.replace(/</g, '<') |
|
.replace(/>/g, '>') |
|
.replace(/(^|\n)#{1,6} ([^\n]*)(?=([^`]*`[^`]*`)*[^`]*$)/g, '$1<h3>$2</h3>') |
|
.replace(/\*\*(.*?)\*\*(?=([^`]*`[^`]*`)*[^`]*$)/g, '<strong>$1</strong>') |
|
.replace(/__(.*?)__(?=([^`]*`[^`]*`)*[^`]*$)/g, '<strong>$1</strong>') |
|
.replace(/\*(.*?)\*(?=([^`]*`[^`]*`)*[^`]*$)/g, '<em>$1</em>') |
|
.replace(/_(.*?)_(?=([^`]*`[^`]*`)*[^`]*$)/g, '<em>$1</em>') |
|
.replace(/```.*?\n([\s\S]*?)```/g, '<pre><code>$1</code></pre>') |
|
.replace(/`(.*?)`/g, '<code>$1</code>') |
|
.replace(/\n/gim, '<br />'); |
|
} else { |
|
chunks[i] = `<pre><code>${chunks[i]}</code></pre>`; |
|
} |
|
} |
|
|
|
const restoredText = chunks.join(''); |
|
|
|
return html`<span dangerouslySetInnerHTML=${{ __html: restoredText }} />`; |
|
}; |
|
|
|
const ModelGenerationInfo = (params) => { |
|
if (!llamaStats.value) { |
|
return html`<span/>` |
|
} |
|
return html` |
|
<span> |
|
${llamaStats.value.tokens_predicted} predicted, ${llamaStats.value.tokens_cached} cached, ${llamaStats.value.timings.predicted_per_token_ms.toFixed()}ms per token, ${llamaStats.value.timings.predicted_per_second.toFixed(2)} tokens per second |
|
</span> |
|
` |
|
} |
|
|
|
|
|
|
|
const Popover = (props) => { |
|
const isOpen = useSignal(false); |
|
const position = useSignal({ top: '0px', left: '0px' }); |
|
const buttonRef = useRef(null); |
|
const popoverRef = useRef(null); |
|
|
|
const togglePopover = () => { |
|
if (buttonRef.current) { |
|
const rect = buttonRef.current.getBoundingClientRect(); |
|
position.value = { |
|
top: `${rect.bottom + window.scrollY}px`, |
|
left: `${rect.left + window.scrollX}px`, |
|
}; |
|
} |
|
isOpen.value = !isOpen.value; |
|
}; |
|
|
|
const handleClickOutside = (event) => { |
|
if (popoverRef.current && !popoverRef.current.contains(event.target) && !buttonRef.current.contains(event.target)) { |
|
isOpen.value = false; |
|
} |
|
}; |
|
|
|
useEffect(() => { |
|
document.addEventListener('mousedown', handleClickOutside); |
|
return () => { |
|
document.removeEventListener('mousedown', handleClickOutside); |
|
}; |
|
}, []); |
|
|
|
return html` |
|
<span style=${props.style} ref=${buttonRef} onClick=${togglePopover}>${props.children}</span> |
|
${isOpen.value && html` |
|
<${Portal} into="#portal"> |
|
<div |
|
ref=${popoverRef} |
|
class="popover-content" |
|
style=${{ |
|
top: position.value.top, |
|
left: position.value.left, |
|
}} |
|
> |
|
${props.popoverChildren} |
|
</div> |
|
</${Portal}> |
|
`} |
|
`; |
|
}; |
|
|
|
|
|
|
|
class Portal extends Component { |
|
componentDidUpdate(props) { |
|
for (let i in props) { |
|
if (props[i] !== this.props[i]) { |
|
return setTimeout(this.renderLayer); |
|
} |
|
} |
|
} |
|
|
|
componentDidMount() { |
|
this.isMounted = true; |
|
this.renderLayer = this.renderLayer.bind(this); |
|
this.renderLayer(); |
|
} |
|
|
|
componentWillUnmount() { |
|
this.renderLayer(false); |
|
this.isMounted = false; |
|
if (this.remote && this.remote.parentNode) this.remote.parentNode.removeChild(this.remote); |
|
} |
|
|
|
findNode(node) { |
|
return typeof node === 'string' ? document.querySelector(node) : node; |
|
} |
|
|
|
renderLayer(show = true) { |
|
if (!this.isMounted) return; |
|
|
|
|
|
if (this.props.into !== this.intoPointer) { |
|
this.intoPointer = this.props.into; |
|
if (this.into && this.remote) { |
|
this.remote = render(html`<${PortalProxy} />`, this.into, this.remote); |
|
} |
|
this.into = this.findNode(this.props.into); |
|
} |
|
|
|
this.remote = render(html` |
|
<${PortalProxy} context=${this.context}> |
|
${show && this.props.children || null} |
|
</${PortalProxy}> |
|
`, this.into, this.remote); |
|
} |
|
|
|
render() { |
|
return null; |
|
} |
|
} |
|
|
|
|
|
class PortalProxy extends Component { |
|
getChildContext() { |
|
return this.props.context; |
|
} |
|
render({ children }) { |
|
return children || null; |
|
} |
|
} |
|
|
|
function App(props) { |
|
useEffect(() => { |
|
const query = new URLSearchParams(location.search).get("q"); |
|
if (query) chat(query); |
|
}, []); |
|
|
|
return html` |
|
<div class="mode-${session.value.type}"> |
|
<header> |
|
<div class="grid-container"> |
|
<div class="grid-item"></div> |
|
<div class="grid-item"><h1>llama.cpp</h1></div> |
|
<div class="grid-item"><a class="customlink" href="index-new.html">New UI</a></div> |
|
</div> |
|
</header> |
|
|
|
<main id="content"> |
|
<${chatStarted.value ? ChatLog : ConfigForm} /> |
|
</main> |
|
|
|
<section id="write"> |
|
<${session.value.type === 'chat' ? MessageInput : CompletionControls} /> |
|
</section> |
|
|
|
<footer> |
|
<p><${ModelGenerationInfo} /></p> |
|
<p>Powered by <a href="https://github.com/ggerganov/llama.cpp">llama.cpp</a> and <a href="https://ggml.ai">ggml.ai</a>.</p> |
|
</footer> |
|
</div> |
|
`; |
|
} |
|
|
|
render(h(App), document.querySelector('#container')); |
|
</script> |
|
</head> |
|
|
|
<body> |
|
<div id="container"> |
|
<input type="file" id="fileInput" accept="image/*" style="display: none;"> |
|
</div> |
|
<div id="portal"></div> |
|
</body> |
|
|
|
</html> |
|
|