| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052 |
- <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%;
- }
- #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;
- }
- 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", // "chat" | "completion"
- char: "Llama",
- user: "User",
- image_selected: ''
- })
- const params = signal({
- n_predict: 400,
- temperature: 0.7,
- repeat_last_n: 256, // 0 = disable penalty, -1 = context size
- repeat_penalty: 1.18, // 1.0 = disabled
- top_k: 40, // <= 0 to use vocab size
- top_p: 0.95, // 1.0 = disabled
- min_p: 0.05, // 0 = disabled
- typical_p: 1.0, // 1.0 = disabled
- presence_penalty: 0.0, // 0.0 = disabled
- frequency_penalty: 0.0, // 0.0 = disabled
- mirostat: 0, // 0/1/2
- mirostat_tau: 5, // target entropy
- mirostat_eta: 0.1, // learning rate
- grammar: '',
- n_probs: 0, // no completion_probabilities,
- min_keep: 0, // min probs from each sampler,
- image_data: [],
- cache_prompt: true,
- api_key: ''
- })
- /* START: Support for storing prompt templates and parameters in browsers LocalStorage */
- 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;
- }
- }
- // create a container for user templates and settings
- const savedUserTemplates = signal({})
- const selectedUserTemplate = signal({ name: '', template: { session: {}, params: {} } })
- // let's import locally saved templates and settings if there are any
- // user templates and settings are stored in one object
- // in form of { "templatename": "templatedata" } and { "settingstemplatename":"settingsdata" }
- console.log('Importing saved templates')
- let importedTemplates = local_storage_getDataAsObject('user_templates')
- if (importedTemplates) {
- // saved templates were successfully imported.
- console.log('Processing saved templates and updating default template')
- params.value = { ...params.value, image_data: [] };
- //console.log(importedTemplates);
- savedUserTemplates.value = importedTemplates;
- //override default template
- savedUserTemplates.value.default = { session: session.value, params: params.value }
- local_storage_setDataFromObject('user_templates', savedUserTemplates.value)
- } else {
- // no saved templates detected.
- 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() {
- // get autosaved last used template
- 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')
- // no autosaved last used template was found, so load from default.
- userTemplateResetToDefault()
- }
- console.log('Applying template')
- // and update internal data from templates
- userTemplateApply(selectedUserTemplate.value)
- }
- //console.log(savedUserTemplates.value)
- //console.log(selectedUserTemplate.value)
- function userTemplateAutosave() {
- console.log('Template Autosave...')
- if (selectedUserTemplate.value.name == 'default') {
- // we don't want to save over default template, so let's create a new one
- let newTemplateName = 'UserTemplate-' + Date.now().toString()
- let newTemplate = { 'name': newTemplateName, 'data': { 'session': session.value, 'params': params.value } }
- console.log('Saving as ' + newTemplateName)
- // save in the autosave slot
- local_storage_setDataFromObject('user_templates_last', newTemplate)
- // and load it back and apply
- 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()
- /* END: Support for storing prompt templates and parameters in browsers LocalStorage */
- const llamaStats = signal(null)
- const controller = signal(null)
- // currently generating a completion?
- const generating = computed(() => controller.value != null)
- // has the user started a chat?
- const chatStarted = computed(() => session.value.transcript.length > 0)
- const transcriptUpdate = (transcript) => {
- session.value = {
- ...session.value,
- transcript
- }
- }
- // simple template replace
- 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;
- }
- // send message to server
- 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(() => {
- // scroll to bottom (if needed)
- 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(() => {
- // autosave template on every change
- 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 })}
- ${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) {
- // Not for byte pair
- 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}
- </>
- `
- });
- }
- // poor mans markdown replacement
- 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>
- `
- }
- // simple popover impl
- 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}>
- `}
- `;
- };
- // Source: preact-portal (https://github.com/developit/preact-portal/blob/master/src/preact-portal.js)
- /** Redirect rendering of descendants into the given CSS selector */
- 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;
- // clean up old node if moving bases:
- 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;
- }
- }
- // high-order component that renders its first child if it exists.
- // used as a conditional rendering proxy.
- 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>
- <h1>llama.cpp</h1>
- </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>
|