| | <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%; |
| | background-image: url('llamapattern.png'); |
| | } |
| | |
| | #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; |
| | |
| | background-color: rgba(255,255,255,0.9); |
| | } |
| | |
| | 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; |
| | } |
| | |
| | .right { |
| | display: flex; |
| | flex-direction: row; |
| | gap: 0.5em; |
| | justify-content: flex-end; |
| | } |
| | |
| | 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, |
| | top_k: 40, |
| | top_p: 0.95, |
| | min_p: 0.05, |
| | 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 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: location.pathname.replace(/\/+$/, '') })) { |
| | 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); |
| | } |
| | }); |
| | } |
| | |
| | function MessageInput() { |
| | const message = useSignal("") |
| | |
| | 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="right"> |
| | <button type="submit" disabled=${generating.value}>Send</button> |
| | <button onclick=${uploadImage}>Upload Image</button> |
| | <button onclick=${stop} disabled=${!generating.value}>Stop</button> |
| | <button onclick=${reset}>Reset</button> |
| | </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 isCompletionMode = session.value.type === 'completion' |
| | const chatLine = ([user, data], index) => { |
| | let message |
| | const isArrayMessage = Array.isArray(data) |
| | if (params.value.n_probs > 0 && isArrayMessage) { |
| | message = html`<${Probabilities} data=${data} />` |
| | } else { |
| | const text = isArrayMessage ? |
| | data.map(msg => msg.content).join('').replace(/^\s+/, '') : |
| | data; |
| | message = isCompletionMode ? |
| | text : |
| | html`<${Markdownish} text=${template(text)} />` |
| | } |
| | if (user) { |
| | return html`<p key=${index}><strong>${template(user)}:</strong> ${message}</p>` |
| | } else { |
| | return isCompletionMode ? |
| | html`<span key=${index}>${message}</span>` : |
| | html`<p key=${index}>${message}</p>` |
| | } |
| | }; |
| | |
| | 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 })} |
| | </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 md = params.text |
| | .replace(/&/g, '&') |
| | .replace(/</g, '<') |
| | .replace(/>/g, '>') |
| | .replace(/^#{1,6} (.*)$/gim, '<h3>$1</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 />'); |
| | return html`<span dangerouslySetInnerHTML=${{ __html: md }} />`; |
| | }; |
| | |
| | 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> |
| | <img src="llama_cpp.png" style="width:100%"/> |
| | </header> |
| | |
| | <section id="write"> |
| | <${session.value.type === 'chat' ? MessageInput : CompletionControls} /> |
| | </section> |
| | |
| | <main id="content"> |
| | <${chatStarted.value ? ChatLog : ConfigForm} /> |
| | </main> |
| | |
| | |
| | <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> |
| |
|