| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478 |
- // @ts-check
- // A simple completions and chat/completions test related web front end logic
- // by Humans for All
- class Roles {
- static System = "system";
- static User = "user";
- static Assistant = "assistant";
- }
- class ApiEP {
- static Chat = "chat";
- static Completion = "completion";
- }
- let gUsageMsg = `
- <p> Enter the system prompt above, before entering/submitting any user query.</p>
- <p> Enter your text to the ai assistant below.</p>
- <p> Use shift+enter for inserting enter.</p>
- <p> Refresh the page to start over fresh.</p>
- `;
- class SimpleChat {
- constructor() {
- /**
- * Maintain in a form suitable for common LLM web service chat/completions' messages entry
- * @type {{role: string, content: string}[]}
- */
- this.xchat = [];
- this.iLastSys = -1;
- }
- /**
- * Add an entry into xchat
- * @param {string} role
- * @param {string|undefined|null} content
- */
- add(role, content) {
- if ((content == undefined) || (content == null) || (content == "")) {
- return false;
- }
- this.xchat.push( {role: role, content: content} );
- if (role == Roles.System) {
- this.iLastSys = this.xchat.length - 1;
- }
- return true;
- }
- /**
- * Show the contents in the specified div
- * @param {HTMLDivElement} div
- * @param {boolean} bClear
- */
- show(div, bClear=true) {
- if (bClear) {
- div.replaceChildren();
- }
- let last = undefined;
- for(const x of this.xchat) {
- let entry = document.createElement("p");
- entry.className = `role-${x.role}`;
- entry.innerText = `${x.role}: ${x.content}`;
- div.appendChild(entry);
- last = entry;
- }
- if (last !== undefined) {
- last.scrollIntoView(false);
- } else {
- if (bClear) {
- div.innerHTML = gUsageMsg;
- }
- }
- }
- /**
- * Add needed fields wrt json object to be sent wrt LLM web services completions endpoint
- * Convert the json into string.
- * @param {Object} obj
- */
- request_jsonstr(obj) {
- obj["temperature"] = 0.7;
- return JSON.stringify(obj);
- }
- /**
- * Return a string form of json object suitable for chat/completions
- */
- request_messages_jsonstr() {
- let req = {
- messages: this.xchat,
- }
- return this.request_jsonstr(req);
- }
- /**
- * Return a string form of json object suitable for /completions
- */
- request_prompt_jsonstr() {
- let prompt = "";
- for(const chat of this.xchat) {
- prompt += `${chat.role}: ${chat.content}\n`;
- }
- let req = {
- prompt: prompt,
- }
- return this.request_jsonstr(req);
- }
- /**
- * Allow setting of system prompt, but only at begining.
- * @param {string} sysPrompt
- * @param {string} msgTag
- */
- add_system_begin(sysPrompt, msgTag) {
- if (this.xchat.length == 0) {
- if (sysPrompt.length > 0) {
- return this.add(Roles.System, sysPrompt);
- }
- } else {
- if (sysPrompt.length > 0) {
- if (this.xchat[0].role !== Roles.System) {
- console.error(`ERRR:SimpleChat:SC:${msgTag}:You need to specify system prompt before any user query, ignoring...`);
- } else {
- if (this.xchat[0].content !== sysPrompt) {
- console.error(`ERRR:SimpleChat:SC:${msgTag}:You cant change system prompt, mid way through, ignoring...`);
- }
- }
- }
- }
- return false;
- }
- /**
- * Allow setting of system prompt, at any time.
- * @param {string} sysPrompt
- * @param {string} msgTag
- */
- add_system_anytime(sysPrompt, msgTag) {
- if (sysPrompt.length <= 0) {
- return false;
- }
- if (this.iLastSys < 0) {
- return this.add(Roles.System, sysPrompt);
- }
- let lastSys = this.xchat[this.iLastSys].content;
- if (lastSys !== sysPrompt) {
- return this.add(Roles.System, sysPrompt);
- }
- return false;
- }
- /**
- * Retrieve the latest system prompt.
- */
- get_system_latest() {
- if (this.iLastSys == -1) {
- return "";
- }
- let sysPrompt = this.xchat[this.iLastSys].content;
- return sysPrompt;
- }
- }
- let gBaseURL = "http://127.0.0.1:8080";
- let gChatURL = {
- 'chat': `${gBaseURL}/chat/completions`,
- 'completion': `${gBaseURL}/completions`,
- }
- const gbCompletionFreshChatAlways = true;
- /**
- * Set the class of the children, based on whether it is the idSelected or not.
- * @param {HTMLDivElement} elBase
- * @param {string} idSelected
- * @param {string} classSelected
- * @param {string} classUnSelected
- */
- function el_children_config_class(elBase, idSelected, classSelected, classUnSelected="") {
- for(let child of elBase.children) {
- if (child.id == idSelected) {
- child.className = classSelected;
- } else {
- child.className = classUnSelected;
- }
- }
- }
- /**
- * Create button and set it up.
- * @param {string} id
- * @param {(this: HTMLButtonElement, ev: MouseEvent) => any} callback
- * @param {string | undefined} name
- * @param {string | undefined} innerText
- */
- function el_create_button(id, callback, name=undefined, innerText=undefined) {
- if (!name) {
- name = id;
- }
- if (!innerText) {
- innerText = id;
- }
- let btn = document.createElement("button");
- btn.id = id;
- btn.name = name;
- btn.innerText = innerText;
- btn.addEventListener("click", callback);
- return btn;
- }
- class MultiChatUI {
- constructor() {
- /** @type {Object<string, SimpleChat>} */
- this.simpleChats = {};
- /** @type {string} */
- this.curChatId = "";
- // the ui elements
- this.elInSystem = /** @type{HTMLInputElement} */(document.getElementById("system-in"));
- this.elDivChat = /** @type{HTMLDivElement} */(document.getElementById("chat-div"));
- this.elBtnUser = /** @type{HTMLButtonElement} */(document.getElementById("user-btn"));
- this.elInUser = /** @type{HTMLInputElement} */(document.getElementById("user-in"));
- this.elSelectApiEP = /** @type{HTMLSelectElement} */(document.getElementById("api-ep"));
- this.elDivSessions = /** @type{HTMLDivElement} */(document.getElementById("sessions-div"));
- this.validate_element(this.elInSystem, "system-in");
- this.validate_element(this.elDivChat, "chat-div");
- this.validate_element(this.elInUser, "user-in");
- this.validate_element(this.elSelectApiEP, "api-ep");
- this.validate_element(this.elDivChat, "sessions-div");
- }
- /**
- * Check if the element got
- * @param {HTMLElement | null} el
- * @param {string} msgTag
- */
- validate_element(el, msgTag) {
- if (el == null) {
- throw Error(`ERRR:SimpleChat:MCUI:${msgTag} element missing in html...`);
- } else {
- console.debug(`INFO:SimpleChat:MCUI:${msgTag} Id[${el.id}] Name[${el["name"]}]`);
- }
- }
- /**
- * Reset user input ui.
- * * clear user input
- * * enable user input
- * * set focus to user input
- */
- ui_reset_userinput() {
- this.elInUser.value = "";
- this.elInUser.disabled = false;
- this.elInUser.focus();
- }
- /**
- * Setup the needed callbacks wrt UI, curChatId to defaultChatId and
- * optionally switch to specified defaultChatId.
- * @param {string} defaultChatId
- * @param {boolean} bSwitchSession
- */
- setup_ui(defaultChatId, bSwitchSession=false) {
- this.curChatId = defaultChatId;
- if (bSwitchSession) {
- this.handle_session_switch(this.curChatId);
- }
- this.elBtnUser.addEventListener("click", (ev)=>{
- if (this.elInUser.disabled) {
- return;
- }
- this.handle_user_submit(this.curChatId, this.elSelectApiEP.value).catch((/** @type{Error} */reason)=>{
- let msg = `ERRR:SimpleChat\nMCUI:HandleUserSubmit:${this.curChatId}\n${reason.name}:${reason.message}`;
- console.debug(msg.replace("\n", ":"));
- alert(msg);
- this.ui_reset_userinput();
- });
- });
- this.elInUser.addEventListener("keyup", (ev)=> {
- // allow user to insert enter into their message using shift+enter.
- // while just pressing enter key will lead to submitting.
- if ((ev.key === "Enter") && (!ev.shiftKey)) {
- this.elBtnUser.click();
- ev.preventDefault();
- }
- });
- this.elInSystem.addEventListener("keyup", (ev)=> {
- // allow user to insert enter into the system prompt using shift+enter.
- // while just pressing enter key will lead to setting the system prompt.
- if ((ev.key === "Enter") && (!ev.shiftKey)) {
- let chat = this.simpleChats[this.curChatId];
- chat.add_system_anytime(this.elInSystem.value, this.curChatId);
- chat.show(this.elDivChat);
- ev.preventDefault();
- }
- });
- }
- /**
- * Setup a new chat session and optionally switch to it.
- * @param {string} chatId
- * @param {boolean} bSwitchSession
- */
- new_chat_session(chatId, bSwitchSession=false) {
- this.simpleChats[chatId] = new SimpleChat();
- if (bSwitchSession) {
- this.handle_session_switch(chatId);
- }
- }
- /**
- * Handle user query submit request, wrt specified chat session.
- * @param {string} chatId
- * @param {string} apiEP
- */
- async handle_user_submit(chatId, apiEP) {
- let chat = this.simpleChats[chatId];
- chat.add_system_anytime(this.elInSystem.value, chatId);
- let content = this.elInUser.value;
- if (!chat.add(Roles.User, content)) {
- console.debug(`WARN:SimpleChat:MCUI:${chatId}:HandleUserSubmit:Ignoring empty user input...`);
- return;
- }
- chat.show(this.elDivChat);
- let theBody;
- let theUrl = gChatURL[apiEP]
- if (apiEP == ApiEP.Chat) {
- theBody = chat.request_messages_jsonstr();
- } else {
- theBody = chat.request_prompt_jsonstr();
- }
- this.elInUser.value = "working...";
- this.elInUser.disabled = true;
- console.debug(`DBUG:SimpleChat:MCUI:${chatId}:HandleUserSubmit:${theUrl}:ReqBody:${theBody}`);
- let resp = await fetch(theUrl, {
- method: "POST",
- headers: {
- "Content-Type": "application/json",
- },
- body: theBody,
- });
- let respBody = await resp.json();
- console.debug(`DBUG:SimpleChat:MCUI:${chatId}:HandleUserSubmit:RespBody:${JSON.stringify(respBody)}`);
- let assistantMsg;
- if (apiEP == ApiEP.Chat) {
- assistantMsg = respBody["choices"][0]["message"]["content"];
- } else {
- try {
- assistantMsg = respBody["choices"][0]["text"];
- } catch {
- assistantMsg = respBody["content"];
- }
- }
- chat.add(Roles.Assistant, assistantMsg);
- if (chatId == this.curChatId) {
- chat.show(this.elDivChat);
- } else {
- console.debug(`DBUG:SimpleChat:MCUI:HandleUserSubmit:ChatId has changed:[${chatId}] [${this.curChatId}]`);
- }
- // Purposefully clear at end rather than begin of this function
- // so that one can switch from chat to completion mode and sequece
- // in a completion mode with multiple user-assistant chat data
- // from before to be sent/occur once.
- if ((apiEP == ApiEP.Completion) && (gbCompletionFreshChatAlways)) {
- chat.xchat.length = 0;
- }
- this.ui_reset_userinput();
- }
- /**
- * Show buttons for NewChat and available chat sessions, in the passed elDiv.
- * If elDiv is undefined/null, then use this.elDivSessions.
- * Take care of highlighting the selected chat-session's btn.
- * @param {HTMLDivElement | undefined} elDiv
- */
- show_sessions(elDiv=undefined) {
- if (!elDiv) {
- elDiv = this.elDivSessions;
- }
- elDiv.replaceChildren();
- // Btn for creating new chat session
- let btnNew = el_create_button("New CHAT", (ev)=> {
- if (this.elInUser.disabled) {
- console.error(`ERRR:SimpleChat:MCUI:NewChat:Current session [${this.curChatId}] awaiting response, ignoring request...`);
- alert("ERRR:SimpleChat\nMCUI:NewChat\nWait for response to pending query, before starting new chat session");
- return;
- }
- let chatId = `Chat${Object.keys(this.simpleChats).length}`;
- let chatIdGot = prompt("INFO:SimpleChat\nMCUI:NewChat\nEnter id for new chat session", chatId);
- if (!chatIdGot) {
- console.error("ERRR:SimpleChat:MCUI:NewChat:Skipping based on user request...");
- return;
- }
- this.new_chat_session(chatIdGot, true);
- this.create_session_btn(elDiv, chatIdGot);
- el_children_config_class(elDiv, chatIdGot, "session-selected", "");
- });
- elDiv.appendChild(btnNew);
- // Btns for existing chat sessions
- let chatIds = Object.keys(this.simpleChats);
- for(let cid of chatIds) {
- let btn = this.create_session_btn(elDiv, cid);
- if (cid == this.curChatId) {
- btn.className = "session-selected";
- }
- }
- }
- create_session_btn(elDiv, cid) {
- let btn = el_create_button(cid, (ev)=>{
- let target = /** @type{HTMLButtonElement} */(ev.target);
- console.debug(`DBUG:SimpleChat:MCUI:SessionClick:${target.id}`);
- if (this.elInUser.disabled) {
- console.error(`ERRR:SimpleChat:MCUI:SessionClick:${target.id}:Current session [${this.curChatId}] awaiting response, ignoring switch...`);
- alert("ERRR:SimpleChat\nMCUI:SessionClick\nWait for response to pending query, before switching");
- return;
- }
- this.handle_session_switch(target.id);
- el_children_config_class(elDiv, target.id, "session-selected", "");
- });
- elDiv.appendChild(btn);
- return btn;
- }
- /**
- * Switch ui to the specified chatId and set curChatId to same.
- * @param {string} chatId
- */
- async handle_session_switch(chatId) {
- let chat = this.simpleChats[chatId];
- if (chat == undefined) {
- console.error(`ERRR:SimpleChat:MCUI:HandleSessionSwitch:${chatId} missing...`);
- return;
- }
- this.elInSystem.value = chat.get_system_latest();
- this.elInUser.value = "";
- chat.show(this.elDivChat);
- this.elInUser.focus();
- this.curChatId = chatId;
- console.log(`INFO:SimpleChat:MCUI:HandleSessionSwitch:${chatId} entered...`);
- }
- }
- let gMuitChat;
- const gChatIds = [ "Default", "Other" ];
- function startme() {
- console.log("INFO:SimpleChat:StartMe:Starting...");
- gMuitChat = new MultiChatUI();
- for (let cid of gChatIds) {
- gMuitChat.new_chat_session(cid);
- }
- gMuitChat.setup_ui(gChatIds[0]);
- gMuitChat.show_sessions();
- }
- document.addEventListener("DOMContentLoaded", startme);
|