diff --git a/examples/server/public/json-schema-to-grammar.mjs b/examples/server/public/json-schema-to-grammar.mjs
index 8e0be1b40..cef11eab8 100644
--- a/examples/server/public/json-schema-to-grammar.mjs
+++ b/examples/server/public/json-schema-to-grammar.mjs
@@ -2,57 +2,26 @@
const SPACE_RULE = '" "?';
function _buildRepetition(itemRule, minItems, maxItems, opts={}) {
+ if (minItems === 0 && maxItems === 1) {
+ return `${itemRule}?`;
+ }
+
+
const separatorRule = opts.separatorRule ?? '';
const itemRuleIsLiteral = opts.itemRuleIsLiteral ?? false
if (separatorRule === '') {
- if (minItems === 0 && maxItems === 1) {
- return `${itemRule}?`;
- } else if (minItems === 1 && maxItems === undefined) {
+ if (minItems === 1 && maxItems === undefined) {
return `${itemRule}+`;
- }
- }
-
- let result = '';
- if (minItems > 0) {
- if (itemRuleIsLiteral && separatorRule === '') {
- result = `"${itemRule.slice(1, -1).repeat(minItems)}"`;
+ } else if (minItems === 0 && maxItems === undefined) {
+ return `${itemRule}*`;
} else {
- result = Array.from({ length: minItems }, () => itemRule)
- .join(separatorRule !== '' ? ` ${separatorRule} ` : ' ');
+ return `${itemRule}{${minItems},${maxItems !== undefined ? maxItems : ''}}`;
}
}
- const optRepetitions = (upToN, prefixWithSep=false) => {
- const content = separatorRule !== '' && prefixWithSep ? `${separatorRule} ${itemRule}` : itemRule;
- if (upToN === 0) {
- return '';
- } else if (upToN === 1) {
- return `(${content})?`;
- } else if (separatorRule !== '' && !prefixWithSep) {
- return `(${content} ${optRepetitions(upToN - 1, true)})?`;
- } else {
- return Array.from({ length: upToN }, () => `(${content}`).join(' ').trim() + Array.from({ length: upToN }, () => ')?').join('');
- }
- };
-
- if (minItems > 0 && maxItems !== minItems) {
- result += ' ';
- }
-
- if (maxItems !== undefined) {
- result += optRepetitions(maxItems - minItems, minItems > 0);
- } else {
- const itemOperator = `(${separatorRule !== '' ? separatorRule + ' ' : ''}${itemRule})`;
-
- if (minItems === 0 && separatorRule !== '') {
- result = `(${itemRule} ${itemOperator}*)?`;
- } else {
- result += `${itemOperator}*`;
- }
- }
-
- return result;
+ const result = itemRule + ' ' + _buildRepetition(`(${separatorRule} ${itemRule})`, minItems > 0 ? minItems - 1 : 0, maxItems !== undefined ? maxItems - 1 : undefined);
+ return minItems === 0 ? `(${result})?` : result;
}
class BuiltinRule {
@@ -62,27 +31,25 @@ class BuiltinRule {
}
}
-const UP_TO_15_DIGITS = _buildRepetition('[0-9]', 0, 15);
-
const PRIMITIVE_RULES = {
boolean : new BuiltinRule('("true" | "false") space', []),
- 'decimal-part' : new BuiltinRule('[0-9] ' + UP_TO_15_DIGITS, []),
- 'integral-part': new BuiltinRule('[0-9] | [1-9] ' + UP_TO_15_DIGITS, []),
+ 'decimal-part' : new BuiltinRule('[0-9]{1,16}', []),
+ 'integral-part': new BuiltinRule('[0] | [1-9] [0-9]{0,15}', []),
number : new BuiltinRule('("-"? integral-part) ("." decimal-part)? ([eE] [-+]? integral-part)? space', ['integral-part', 'decimal-part']),
integer : new BuiltinRule('("-"? integral-part) space', ['integral-part']),
value : new BuiltinRule('object | array | string | number | boolean | null', ['object', 'array', 'string', 'number', 'boolean', 'null']),
object : new BuiltinRule('"{" space ( string ":" space value ("," space string ":" space value)* )? "}" space', ['string', 'value']),
array : new BuiltinRule('"[" space ( value ("," space value)* )? "]" space', ['value']),
- uuid : new BuiltinRule('"\\"" ' + [8, 4, 4, 4, 12].map(n => [...new Array(n)].map(_ => '[0-9a-fA-F]').join('')).join(' "-" ') + ' "\\"" space', []),
- char : new BuiltinRule(`[^"\\\\] | "\\\\" (["\\\\/bfnrt] | "u" [0-9a-fA-F] [0-9a-fA-F] [0-9a-fA-F] [0-9a-fA-F])`, []),
+ uuid : new BuiltinRule('"\\"" [0-9a-fA-F]{8} "-" [0-9a-fA-F]{4} "-" [0-9a-fA-F]{4} "-" [0-9a-fA-F]{4} "-" [0-9a-fA-F]{12} "\\"" space', []),
+ char : new BuiltinRule(`[^"\\\\] | "\\\\" (["\\\\/bfnrt] | "u" [0-9a-fA-F]{4})`, []),
string : new BuiltinRule(`"\\"" char* "\\"" space`, ['char']),
null : new BuiltinRule('"null" space', []),
};
// TODO: support "uri", "email" string formats
const STRING_FORMAT_RULES = {
- 'date' : new BuiltinRule('[0-9] [0-9] [0-9] [0-9] "-" ( "0" [1-9] | "1" [0-2] ) "-" ( \"0\" [1-9] | [1-2] [0-9] | "3" [0-1] )', []),
- 'time' : new BuiltinRule('([01] [0-9] | "2" [0-3]) ":" [0-5] [0-9] ":" [0-5] [0-9] ( "." [0-9] [0-9] [0-9] )? ( "Z" | ( "+" | "-" ) ( [01] [0-9] | "2" [0-3] ) ":" [0-5] [0-9] )', []),
+ 'date' : new BuiltinRule('[0-9]{4} "-" ( "0" [1-9] | "1" [0-2] ) "-" ( \"0\" [1-9] | [1-2] [0-9] | "3" [0-1] )', []),
+ 'time' : new BuiltinRule('([01] [0-9] | "2" [0-3]) ":" [0-5] [0-9] ":" [0-5] [0-9] ( "." [0-9]{3} )? ( "Z" | ( "+" | "-" ) ( [01] [0-9] | "2" [0-3] ) ":" [0-5] [0-9] )', []),
'date-time' : new BuiltinRule('date "T" time', ['date', 'time']),
'date-string' : new BuiltinRule('"\\"" date "\\"" space', ['date']),
'time-string' : new BuiltinRule('"\\"" time "\\"" space', ['time']),
diff --git a/examples/server/public/prompt-formats.js b/examples/server/public/prompt-formats.js
new file mode 100644
index 000000000..73ddb7187
--- /dev/null
+++ b/examples/server/public/prompt-formats.js
@@ -0,0 +1,331 @@
+// extended list
+export const promptFormats = {
+ "alpaca": {
+ template: `{{prompt}}\n\n{{history}}\n\n{{char}}:`,
+
+ historyTemplate: `### {{name}}:\n{{message}}`,
+
+ char: "Response",
+ charMsgPrefix: "",
+ charMsgSuffix: "",
+
+ user: "Instruction",
+ userMsgPrefix: "",
+ userMsgSuffix: "",
+
+ stops: ""
+ },
+
+ // ----------------------------
+
+ "chatml": {
+ template: `<|im_start|>system\n{{prompt}}<|im_end|>\n{{history}}{{char}}`,
+
+ historyTemplate: `<|im_start|>{{name}}\n{{message}}`,
+
+ char: "assistant",
+ charMsgPrefix: "",
+ charMsgSuffix: "",
+
+ user: "user",
+ userMsgPrefix: "",
+ userMsgSuffix: "<|im_end|>\n",
+
+ stops: ""
+ },
+
+ // ----------------------------
+
+ "commandr": {
+ template: `<|START_OF_TURN_TOKEN|><|SYSTEM_TOKEN|>{{prompt}}\n<|END_OF_TURN_TOKEN|>{{history}}{{char}}`,
+
+ historyTemplate: `<|START_OF_TURN_TOKEN|><|{{name}}|> {{message}}`,
+
+ char: "CHATBOT_TOKEN",
+ charMsgPrefix: "",
+ charMsgSuffix: "",
+
+ user: "USER_TOKEN",
+ userMsgPrefix: "",
+ userMsgSuffix: "<|END_OF_TURN_TOKEN|>",
+
+ stops: ""
+ },
+ // ref: https://docs.cohere.com/docs/prompting-command-r
+
+ // ----------------------------
+
+ "llama2": {
+ template: `[INST] <>\n{{prompt}}\n<>\n\nTest Message [/INST] Test Successfull {{history}}{{char}}`,
+
+ historyTemplate: `{{name}}: {{message}}`,
+
+ char: "Assistant",
+ charMsgPrefix: "",
+ charMsgSuffix: "",
+
+ user: "User",
+ userMsgPrefix: "[INST] ",
+ userMsgSuffix: " [/INST]",
+
+ stops: ""
+ },
+ // ref: https://huggingface.co/blog/llama2#how-to-prompt-llama-2
+
+ // ----------------------------
+
+ "llama3": {
+ template: `<|begin_of_text|><|start_header_id|>system<|end_header_id|>\n\n{{prompt}}{{history}}{{char}}`,
+
+ historyTemplate: `<|start_header_id|>{{name}}<|end_header_id|>\n\n{{message}}<|eot_id|>`,
+
+ char: "assistant",
+ charMsgPrefix: "",
+ charMsgSuffix: "",
+
+ user: "user",
+ userMsgPrefix: "",
+ userMsgSuffix: "",
+
+ stops: "<|eot_id|>"
+ },
+ // ref: https://llama.meta.com/docs/model-cards-and-prompt-formats/meta-llama-3/#special-tokens-used-with-meta-llama-3
+
+ // ----------------------------
+
+ "openchat": {
+ template: `{{history}}{{char}}`,
+
+ historyTemplate: `GPT4 Correct {{name}}: {{message}}<|end_of_turn|>`,
+
+ char: "Assistant",
+ charMsgPrefix: "",
+ charMsgSuffix: "",
+
+ user: "User",
+ userMsgPrefix: "",
+ userMsgSuffix: "",
+
+ stops: ""
+ },
+
+ // ----------------------------
+
+ "phi3": {
+ template: `{{history}}{{char}}`,
+
+ historyTemplate: `<|{{name}}|>\n{{message}}<|end|>\n`,
+
+ char: "assistant",
+ charMsgPrefix: "",
+ charMsgSuffix: "",
+
+ user: "user",
+ userMsgPrefix: "",
+ userMsgSuffix: "",
+
+ stops: "<|end|>"
+ },
+ // ref: https://huggingface.co/microsoft/Phi-3-mini-4k-instruct#chat-format
+
+ // ----------------------------
+
+ "vicuna": {
+ template: `{{prompt}}\n{{history}}{{char}}`,
+
+ historyTemplate: `{{name}}: {{message}}\n`,
+
+ char: "ASSISTANT",
+ charMsgPrefix: "",
+ charMsgSuffix: "",
+
+ user: "USER",
+ userMsgPrefix: "",
+ userMsgSuffix: "",
+
+ stops: ""
+ },
+ // ref: https://huggingface.co/lmsys/vicuna-33b-v1.3/discussions/1
+
+ // ----------------------------
+
+ "deepseekCoder": {
+ template: `{{prompt}}{{history}}{{char}}:`,
+
+ historyTemplate: `### {{name}}:\n{{message}}`,
+
+ char: "Response",
+ charMsgPrefix: "",
+ charMsgSuffix: "",
+
+ user: "Instruction",
+ userMsgPrefix: "",
+ userMsgSuffix: "",
+
+ stops: "<|EOT|>"
+ },
+
+ // ----------------------------
+
+ "med42": {
+ template: `<|system|>: {{prompt}}\n{{history}}{{char}}`,
+
+ historyTemplate: `<|{{name}}|>: {{message}}\n`,
+
+ char: "assistant",
+ charMsgPrefix: "",
+ charMsgSuffix: "",
+
+ user: "prompter",
+ userMsgPrefix: "",
+ userMsgSuffix: "",
+
+ stops: ""
+ },
+
+ // ----------------------------
+
+ "neuralchat": {
+ template: `### System:\n{{prompt}}\n{{history}}{{char}}:`,
+
+ historyTemplate: `### {{name}}:\n{{message}}\n`,
+
+ char: "Assistant",
+ charMsgPrefix: "",
+ charMsgSuffix: "",
+
+ user: "User",
+ userMsgPrefix: "",
+ userMsgSuffix: "",
+
+ stops: ""
+ },
+
+ // ----------------------------
+
+ "nousHermes": {
+ template: `### Instruction: {{prompt}}\n\n{{history}}\n\n{{char}}:`,
+
+ historyTemplate: `### {{name}}:\n{{message}}`,
+
+ char: "Response",
+ charMsgPrefix: "",
+ charMsgSuffix: "",
+
+ user: "Input",
+ userMsgPrefix: "",
+ userMsgSuffix: "",
+
+ stops: ""
+ },
+
+ // ----------------------------
+
+ "openchatMath": {
+ template: `{{history}}{{char}}`,
+
+ historyTemplate: `Math Correct {{name}}: {{message}}<|end_of_turn|>`,
+
+ char: "Assistant",
+ charMsgPrefix: "",
+ charMsgSuffix: "",
+
+
+ user: "User",
+ userMsgPrefix: "",
+ userMsgSuffix: "",
+
+ stops: ""
+ },
+
+ // ----------------------------
+
+ "orion": {
+ template: `Human: Test Message\n\nAssistant: Test Successful{{history}}{{char}}:`,
+
+ historyTemplate: `{{name}}: {{message}}`,
+
+ char: "Assistant ",
+ charMsgPrefix: "",
+ charMsgSuffix: "",
+
+ user: "Human",
+ userMsgPrefix: "",
+ userMsgSuffix: "\n\n",
+
+ stops: ""
+ },
+
+ // ----------------------------
+
+ "sauerkraut": {
+ template: `{{prompt}}\n{{history}}{{char}}`,
+
+ historyTemplate: `
+ {{name}}: {{message}}\n`,
+
+ char: "Assistant",
+ charMsgPrefix: "",
+ charMsgSuffix: "",
+
+ user: "User",
+ userMsgPrefix: "",
+ userMsgSuffix: "",
+
+ stops: ""
+ },
+
+ // ----------------------------
+
+ "starlingCode": {
+ template: `{{history}}{{char}}`,
+
+ historyTemplate: `Code {{name}}: {{message}}<|end_of_turn|>`,
+
+ char: "Assistant",
+ charMsgPrefix: "",
+ charMsgSuffix: "",
+
+ user: "User",
+ userMsgPrefix: "",
+ userMsgSuffix: "",
+
+ stops: ""
+ },
+
+ // ----------------------------
+
+ "yi34b": {
+ template: `{{history}} {{char}}`,
+
+ historyTemplate: `{{name}}: {{message}}`,
+
+ char: "Assistant",
+ charMsgPrefix: "",
+ charMsgSuffix: "",
+
+ user: "Human",
+ userMsgPrefix: "",
+ userMsgSuffix: "",
+
+ stops: ""
+ },
+
+ // ----------------------------
+
+ "zephyr": {
+ template: `<|system|>\n{{prompt}}\n{{history}}{{char}}`,
+
+ historyTemplate: `<|{{name}}|>\n{{message}}\n`,
+
+ char: "assistant",
+ charMsgPrefix: "",
+ charMsgSuffix: "",
+
+ user: "user",
+ userMsgPrefix: "",
+ userMsgSuffix: "",
+
+ stops: ""
+ }
+ };
diff --git a/examples/server/public/style.css b/examples/server/public/style.css
new file mode 100755
index 000000000..087cc62da
--- /dev/null
+++ b/examples/server/public/style.css
@@ -0,0 +1,954 @@
+@import url("colorthemes.css");
+
+body {
+ font-family: 'Arial', sans-serif;
+ font-size: 90%;
+ background-color: var(--background-color-1);
+ color: var(--text-color-subtile-1); /* head 1 llama.cpp & triangle options for some reason */
+ max-width: 600px;
+ min-width: 300px;
+ line-height: 1.2;
+ margin: 0 auto;
+ padding: 0 0.5em;
+ transition: background-color 0.3s;
+}
+
+::selection {
+ color: var(--button-primary-text) ;
+ background: var(--button-primary-color);
+}
+
+code, pre code {
+ font-family: 'Courier New', monospace;
+}
+
+#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 var(--border-color-3);
+ border-radius: 5px;
+ padding: 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;
+ margin-bottom: 30px;
+}
+
+.two-columns {
+ width: 97%;
+ max-width: 97%;
+ display: grid;
+ grid-template-columns: 1fr 1fr;
+ gap: 1em;
+ position: relative;
+}
+
+.json-schema-controls {
+ margin-top: 10px;
+ width: 100%;
+ max-width: 100%;
+ display: grid;
+ grid-template: "a a";
+ gap: 1em;
+ font-size: x-small;
+ color: var(--theme-nuance-color-3);
+ padding-top: 16px;
+ padding-bottom: 16px;
+ text-transform: uppercase;
+ font-weight: 600;
+}
+
+.json-schema-controls > * {
+ flex: 1;
+}
+
+/* titles of the details-summary boxes */
+.summary-title {
+ font-weight: 600;
+ font-size: x-small;
+ color: var(--text-color-subtile-1);
+ text-transform: uppercase;
+ /* transition: ; */
+}
+
+fieldset {
+ border: none;
+ padding: 0;
+ margin: 0;
+ color: var(--text-color-plain);
+}
+
+fieldset.two {
+ display: grid;
+ grid-template: "a a a";
+ gap: 1em;
+ align-items: center;
+ font-size: x-small;
+ color: var(--text-color-plain);
+}
+
+fieldset.three {
+ display: grid;
+ grid-template: "a a a";
+ gap: 1em;
+ font-size: x-small;
+ color: var(--text-color-plain);
+}
+
+/* titles of name fields*/
+fieldset.names {
+ display: grid;
+ grid-template: "a a";
+ gap: 1em;
+ font-size: x-small;
+ color: var(--theme-nuance-color-3);
+ padding-top: 16px;
+ padding-bottom: 16px;
+ text-transform: uppercase;
+ font-weight: 600;
+}
+
+/* titles of params fields*/
+fieldset.params {
+ display: grid;
+ grid-template: "a a";
+ gap: 1em;
+ font-size: x-small;
+ color: var(--theme-nuance-color-4);
+ padding-top: 16px;
+ padding-bottom: 16px;
+ text-transform: uppercase;
+ font-weight: 600;
+}
+
+fieldset.dropdowns {
+ -webkit-appearance: none;
+ display: flex;
+ grid-template: "a a";
+ gap: 1em;
+ font-size: x-small;
+ color: red;
+ padding-top: 16px;
+ padding-bottom: 16px;
+ text-transform: uppercase;
+ font-weight: 600;
+}
+
+/* input of name fields*/
+.names input[type="text"] {
+ font-family: Arial, sans-serif;
+ font-size: medium;
+ font-weight: 500;
+ padding: 5px;
+ border: 1px solid var(--border-color-2);
+}
+
+.chat-id-color {
+ color: var(--chat-id-color);
+}
+
+details {
+ border: 1px solid var(--border-color-2);
+ border-radius: 5px;
+ 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;
+}
+
+textarea-sec, input-sec, button-sec {
+ padding: 10px;
+ height: 40px;
+ align-items: center;
+}
+
+textarea-sec::placeholder, input-sec::placeholder {
+ padding-left: 10px;
+}
+
+.toggleCheckbox {
+ display: none;
+}
+
+.toggleContainer {
+ position: relative;
+ display: grid;
+ grid-template-columns: repeat(2, 1fr);
+ width: fit-content;
+ border: 3px solid var(--border-color-2);
+ border-radius: 20px;
+ background: var(--border-color-2);
+ font-size: small;
+ cursor: pointer;
+ overflow: hidden;
+}
+
+/* toggle button current state */
+.toggleContainer::before {
+ color: var(--button-primary-text);
+ background-color: var(--button-primary-color);
+ content: '';
+ position: absolute;
+ width: 50%;
+ height: 100%;
+ left: 0%;
+ border-radius: 20px;
+ transition: all 0.3s;
+}
+
+.toggleContainer div {
+ padding: 6px;
+ text-align: center;
+ z-index: 1;
+ transition: color 0.3s;
+}
+
+.toggleCheckbox:checked + .toggleContainer::before {
+ left: 50%;
+}
+
+.toggleCheckbox:checked + .toggleContainer div:first-child {
+ color: var(--text-color-subtile-2);
+}
+
+.toggleCheckbox:checked + .toggleContainer div:last-child {
+ color: var(--button-primary-text);
+}
+
+.toggleCheckbox + .toggleContainer div:first-child {
+ color: var(--button-primary-text);
+}
+
+.toggleCheckbox + .toggleContainer div:last-child {
+ color: var(--text-color-subtile-2);
+}
+
+select {
+ padding: 5px;
+ margin-right: 5px;
+ border-radius: 4px;
+ border: 1px solid var(--secondary-color-4);
+ background-color: var(--primary-color-3);
+ color: var(--secondary-color-4);
+ cursor: pointer;
+}
+
+select:focus {
+ border: 1px solid var(--border-focus-color);
+ box-shadow: 0 0 1px var(--border-focus-shadow);
+}
+
+.button-container {
+ display: flex;
+ justify-content: flex-end;
+}
+
+button {
+ color: var(--button-primary-text);
+ background-color: var(--button-primary-color);
+ border: 1px solid var(--button-primary-border);
+ transition: background-color 0.1s;
+ border-radius: 12px;
+ font-size: x-small;
+ font-weight: 600;
+ text-shadow: 0px 0px 30px #ffffff;
+ text-align: center;
+ text-decoration: none;
+ margin: 4px 2px;
+ padding: 10px 20px;
+ display: inline-block;
+ cursor: pointer;
+}
+
+button:hover {
+ color: var(--button-primary-text-hover);
+ background-color: var(--button-primary-color-hover);
+ border: 1px solid var(--button-primary-border-hover);
+ font-size: x-small;
+ font-weight: 600;
+}
+
+button:active {
+ color: var(--button-primary-text-active);
+ background-color: var(--button-primary-color-active);
+ border: 1px solid var(--button-primary-border-active);
+ font-size: x-small;
+ font-weight: 600;
+}
+
+button:disabled {
+ color: var(--button-tertiary-text);
+ background-color: var(--button-tertiary-color);
+ border: 1px solid var(--button-tertiary-border);
+ font-size: x-small;
+ font-weight: 600;
+ cursor: not-allowed;
+}
+
+.reset-button {
+ background-color: var(--button-secondary-color);
+ border: 1px solid var(--button-secondary-color);
+ color: var(--button-secondary-text);
+ width: fit-content;
+ height: fit-content;
+ font-size: x-small;
+ font-weight: 600;
+ border-radius: 50px;
+ overflow: hidden;
+}
+
+.reset-button:hover {
+ color: var(--button-alert-text-hover);
+ background-color: var(--button-alert-color-hover);
+ border: 1px solid var(--button-alert-border-hover);
+ font-size: x-small;
+ font-weight: 600;
+}
+
+.reset-button:active {
+ color: var(--button-alert-text-active);
+ background-color: var(--button-alert-color-active);
+ border: 1px solid var(--button-alert-border-active);
+ font-size: x-small;
+ font-weight: 600;
+}
+
+.button-grammar {
+ color: var(--button-primary-text);
+ background-color: var(--button-primary-color);
+ border: 1px solid var(--button-primary-border);
+ border-radius: 10px;
+ padding: 10px 20px;
+ text-align: center;
+ text-decoration: none;
+ display: inline-block;
+ font-size: x-small;
+ font-weight: 600;
+ margin: 2px 2px;
+ transition: background-color 0.1s;
+ cursor: pointer;
+}
+
+.button-grammar:hover {
+ color: var(--button-primary-text-hover);
+ background-color: var(--button-primary-color-hover);
+ border: 1px solid var(--button-primary-border-hover);
+ border-radius: 10px;
+ padding: 10px 20px;
+ text-align: center;
+ text-decoration: none;
+ display: inline-block;
+ font-size: x-small;
+ font-weight: 600;
+ margin: 2px 2px;
+ transition: background-color 0.1s;
+ cursor: pointer;
+}
+
+.button-grammar:active {
+ color: var(--button-primary-text-active);
+ background-color: var(--button-primary-color-active);
+ border: 1px solid var(--button-primary-border-active);
+ font-size: x-small;
+ font-weight: 600;
+}
+
+.button-back {
+ background-color: var(--button-secondary-color);
+ border: 1px solid var(--button-secondary-color);
+ color: var(--button-secondary-text);
+ transition: background-color 0.1s;
+ border-radius: 12px;
+ font-size: x-small;
+ font-weight: 600;
+ text-align: center;
+ text-decoration: none;
+ margin: 4px 2px;
+ padding: 10px 20px;
+ display: inline-block;
+ cursor: pointer;
+}
+
+.button-back:hover {
+ color: var(--button-secondary-text-hover);
+ background-color: var(--button-secondary-color-hover);
+ border: 1px solid var(--button-secondary-border-hover);
+ padding: 10px 20px;
+ text-align: center;
+ text-decoration: none;
+ display: inline-block;
+ font-size: x-small;
+ font-weight: 600;
+ margin: 4px 2px;
+ transition: background-color 0.1s;
+ cursor: pointer;
+ border-radius: 12px;
+}
+
+.button-back:active {
+ color: var(--button-secondary-text-active);
+ background-color: var(--button-secondary-color-active);
+ border: 1px solid var(--button-secondary-border-active);
+ font-size: x-small;
+ font-weight: 600;
+}
+
+.prob-set {
+ padding: 0.3em;
+ border-bottom: 1px solid red; /* unknown */
+}
+
+.popover-content {
+ position: absolute;
+ background-color: white;
+ padding: 0.2em;
+ box-shadow: 0 0 13px rgba(0, 0, 0, 0.1);
+}
+
+.grammar {
+ width: 97%;
+ max-width: 97%;
+}
+
+textarea {
+ padding: 5px;
+ flex-grow: 1;
+ width: 100%;
+ max-width: 100%;
+ border-radius: 8px;
+ border: 1px solid var(--border-color-1);
+ resize: none;
+ height: 6em;
+}
+
+textarea:focus {
+ outline: none;
+ border: 1px solid var(--border-focus-color);
+ box-shadow: 0 0 3px var(--border-focus-shadow);
+}
+
+/* "props" frame */
+input[type="text"],
+input[type="range"] {
+ padding: 5px;
+ border-radius: 8px;
+ border: 1px solid var(--border-color-1);
+}
+
+/* "names and props" frame focused*/
+input[type="text"]:focus {
+ outline: none;
+ border: 1px solid var(--border-focus-color);
+ box-shadow: 0 0 3px var(--border-focus-shadow);
+}
+
+input[type="range"]:hover {
+ opacity: 1;
+}
+
+input[type="range"]:focus {
+ outline: none;
+ border: 1px solid var(--border-focus-color);
+ box-shadow: 0 0 3px var(--border-focus-shadow);
+ background-size: var(--slider-track-size-focus);
+}
+
+input[type="range"]::-moz-range-thumb {
+ width: 6px;
+ height: 25px;
+ border: 1px solid var(--ui-range-thumb-border);
+ border-radius: 5px;
+ background-color: var(--ui-range-thumb-color);
+ cursor: pointer;
+}
+
+input[type="range"] {
+ -webkit-appearance: none;
+ width: 80%;
+ height: 1px;
+ border: 1px solid var(--border-color-1);
+ border-radius: 8px;
+ background: var(--border-color-2);
+ outline: none;
+ opacity: 0.7;
+ -webkit-transition: .2s;
+ transition: opacity .2s;
+}
+
+input[type="range"]::-webkit-slider-thumb {
+ -webkit-appearance: none;
+ appearance: none;
+ width: 6px;
+ height: 25px;
+ border: 1px solid var(--ui-range-thumb-border);
+ border-radius: 5px;
+ background-color: var(--ui-range-thumb-color);
+ cursor: pointer;
+}
+
+input[type="range"]::-webkit-slider-runnable-track {
+ background-size: var(--slider-track-size);
+}
+
+input[type="radio"] {
+ accent-color: var(--theme-nuance-color-2);
+}
+
+.chat-input-container {
+ position: relative;
+ max-width: 97%;
+ min-width: 97%;
+}
+
+.chat-input-label {
+ position: absolute;
+ top: 0;
+ left: 0;
+ color: var(--text-color-plain);
+ pointer-events: none;
+ margin-left: 5px;
+ margin-top: 5px;
+}
+
+textarea#chat-input {
+ padding-top: 10px;
+ padding-left: 10px;
+ font-size: medium;
+ border: 1px solid var(--border-color-2);
+ resize: vertical;
+}
+
+textarea#chat-input:focus {
+ border: 1px solid var(--border-focus-color);
+ box-shadow: 0 0 3px var(--border-focus-shadow);
+}
+
+.input-container {
+ position: relative;
+ box-sizing: border-box;
+ width: 100%; /* Setzt die Breite auf 100% */
+ max-width: 100%; /* Stellt sicher, dass die Breite nicht grรถรer als 100% wird */
+}
+
+.input-container:focus {
+ border: 1px solid var(--border-focus-color);
+ box-shadow: 0 0 3px var(--border-focus-shadow);
+}
+/* titles of name fields*/
+/* fieldset.names {
+ display: grid;
+ grid-template: "a a";
+ gap: 1em;
+ font-size: x-small;
+ color: var(--theme-nuance-color-3);
+ padding-top: 16px;
+ padding-bottom: 16px;
+ text-transform: uppercase;
+ font-weight: 600;
+} */
+
+/* input of name fields*/
+/* .names input[type="text"] {
+ font-family: Arial, sans-serif;
+ font-size: medium;
+ font-weight: 500;
+ padding: 5px;
+ border: 1px solid var(--border-color-2);
+} */
+
+fieldset.apiKey {
+ width: 100%;
+ font-size: x-small;
+ color: var(--theme-nuance-color-3);
+ padding-top: 16px;
+ padding-bottom: 16px;
+ text-transform: uppercase;
+ font-weight: 600;
+}
+
+.apiKey {
+ font-family: Arial, sans-serif;
+ font-weight: 500;
+ padding: 5px;
+ border: 1px solid var(--border-color-2);
+}
+
+.apiKey:focus {
+ border: 1px solid var(--border-focus-color);
+ box-shadow: 0 0 3px var(--border-focus-shadow);
+}
+
+.apiKey input[type="text"] {
+ font-family: Arial, sans-serif;
+ font-size: medium;
+ font-weight: 500;
+ padding: 5px;
+ border: 1px solid var(--border-color-2);
+}
+
+.apiKey label {
+ display: inline-block;
+ width: auto;
+ margin-right: 5px;
+}
+
+textarea#api_key {
+ padding-top: 10px;
+ padding-left: 10px;
+ font-size: medium;
+ border: 1px solid var(--border-color-2);
+ resize: vertical;
+}
+
+textarea#api_key:focus {
+ border: 1px solid var(--border-focus-color);
+ box-shadow: 0 0 3px var(--border-focus-shadow);
+}
+
+/* embedded title of the system prompt text area */
+.input-label {
+ position: absolute;
+ top: 0;
+ left: 0;
+ color: var(--theme-nuance-color-4);
+ pointer-events: none;
+ border-radius: 8px 8px 0px 0px;
+ padding-top: 10px;
+ padding-left: 13px;
+ padding-right: 0px;
+ margin-top: 1px;
+ margin-left: 1px;
+ margin-right: 20px;
+ text-transform: uppercase;
+ font-weight: 600;
+ font-size: small;
+ background: rgba(255, 255, 255, 0.5);
+ backdrop-filter: blur(10px);
+ -webkit-backdrop-filter: blur(10px); /* for safari */
+ width: 97%;
+ /* display: block;
+ box-sizing: border-box; */
+}
+
+/* embedded title of the prompt style areas */
+.input-label-sec {
+ position: absolute;
+ top: 0;
+ left: 0;
+ color: var(--theme-nuance-color-4);
+ pointer-events: none;
+ margin-left: 13px;
+ margin-top: 16px;
+ text-transform: uppercase;
+ font-weight: 600;
+ font-size: x-small;
+}
+
+/* system prompt input area */
+textarea.persistent-input {
+ padding-top: 42px;
+ padding-left: 11px;
+ width: 97%;
+ max-width: 97%;
+ height: 50px;
+ font-size: medium;
+ overscroll-behavior: contain;
+}
+
+/* system prompt box */
+.persistent-input {
+ height: auto;
+ width: 100%;
+ max-width: 100%;
+ min-height: 50px;
+ padding: 3px;
+ transition: min-height 0.3s ease;
+}
+
+/* chat history box */
+.persistent-input:focus {
+ height: auto;
+ min-height: 150px;
+ border: 1px solid var(--border-focus-color);
+ box-shadow: 0 0 3px var(--border-focus-shadow);
+}
+
+textarea.persistent-input:focus {
+ border: 1px solid var(--border-focus-color);
+ box-shadow: 0 0 3px var(--border-focus-shadow);
+}
+
+/* prompt style input area */
+textarea.persistent-input-sec {
+ width: 97%;
+ max-width: 97%;
+ padding-top: 42px;
+ padding-left: 11px;
+ font-size: small;
+ border: 1px solid var(--border-color-1);
+ overscroll-behavior: contain;
+}
+
+textarea.persistent-input-sec:focus {
+ border: 1px solid var(--border-focus-color);
+ box-shadow: 0 0 3px var(--border-focus-shadow);
+}
+
+/* chat history box */
+.persistent-input-sec {
+ height: auto;
+ min-height: 150px;
+}
+
+img {
+ border-radius: 8px;
+ display: block;
+ margin-left: auto;
+ margin-right: auto;
+ width: 50%;
+}
+
+/* code area background */
+pre code {
+ display: block;
+ background-color: var(--code-background-color);
+ color: var(--code-text-color);
+ padding: 0.2em 0.2em;
+ border-radius: 5px;
+}
+
+/* code area text */
+code {
+ font-family: monospace;
+ font-weight: bold;
+ padding: 0.1em 0.3em;
+ border-radius: 5px;
+}
+
+fieldset label {
+ margin: 0.5em 0;
+ display: block;
+}
+
+fieldset label.slim {
+ margin: 0 0.5em;
+ display: inline;
+}
+
+header {
+ display: flex;
+ justify-content: space-between;
+ align-items: center;
+ text-align: center;
+ padding-left: 15px;
+}
+
+.generation-statistics:hover {
+ color: var(--theme-nuance-color-4);
+ cursor: default;
+}
+
+footer {
+ font-size: 80%;
+ color: var(--background-color-3);
+ text-align: center;
+ cursor: default;
+}
+
+footer a {
+ color: var(--background-color-4); /* Color of the link */
+ text-decoration: none; /* No underlining */
+ font-weight: bold; /* Bold print */
+}
+
+footer a:hover {
+ color: var(--theme-nuance-color-4); /* Color of the link when hovering */
+ text-decoration: underline; /* Underlining when hovering */
+}
+
+.mode-chat textarea[name=prompt] {
+ height: 8.5em;
+ border: 1px solid var(--primary-color-3);
+}
+
+.mode-completion textarea[name=prompt] {
+ height: 30em;
+ border: 1px solid var(--primary-color-3);
+}
+
+@keyframes loading-bg-wipe {
+ 0% {
+ background-position: 0%;
+ }
+ 100% {
+ background-position: 100%;
+ }
+}
+
+.loading {
+ 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;
+}
+
+.dropbtn {
+ color: var(--button-primary-color);
+ background-color: var(--background-color-1);
+ border: 1px solid var(--background-color-1);
+ transition: background-color 0.1s;
+ border-radius: 4px 4px 0px 0px;
+ font-size: x-small;
+ font-weight: 600;
+ text-shadow: 0px 0px 2px #99999990;
+ text-align: center;
+ text-decoration: none;
+ margin: 4px 2px;
+ padding: 5px 20px;
+ display: inline-block;
+ cursor: pointer;
+ top: 0;
+}
+
+.dropbtn svg {
+ vertical-align: middle;
+ margin-right: 0px;
+ stroke: var(--button-primary-color);
+}
+
+.dropbtn:hover svg {
+ vertical-align: middle;
+ margin-right: 0px;
+ stroke: var(--button-primary-text);
+}
+
+.dropbtn:focus {
+ outline: none; /* Removes the blue border that appears when the button is focused */
+}
+
+.dropdown {
+ position: relative;
+ display: inline-block;
+}
+
+.dropdown-content {
+ /* display: none; */
+ position: absolute;
+ right: 0;
+ text-align: end;
+ color: var(--button-secondary-color);
+ background-color: var(--text-color-subtile-2);
+ border-radius: 4px 4px 4px 4px;
+ min-width: 160px;
+ box-shadow: 0px 8px 16px 0px rgba(0,0,0,0.2);
+ z-index: 1;
+ /* Verstecke den Inhalt sofort */
+ opacity: 0;
+ visibility: hidden;
+ /* รผbergangsverzรถgerung fรผr das Verschwinden */
+ transition: visibility 0.4s linear 0s, opacity 0.2s ease-in-out;
+ transition-delay: 0.2s;
+}
+
+#dropdown-content {transition-timing-function: ease;}
+
+.dropdown-content:hover {
+ background-color: var(--text-color-subtile-2);
+}
+
+.dropdown-content a {
+ color: var(--border-color-2);
+ padding: 12px 16px;
+ border-radius: 4px 4px 4px 4px;
+ text-decoration: none;
+ display: block;
+ background-color: var(--text-color-subtile-2);
+}
+
+.dropdown-content a:hover {
+ color: var(--border-color-2);
+ background-color: var(--text-color-subtile-1);
+ font-weight: 600;
+}
+
+.dropdown:hover .dropdown-content {
+ /* display: block; */
+ border-radius: 4px 4px 4px 4px;
+ /* รbergang ohne Verzรถgerung fรผr das Erscheinen */
+ opacity: 1;
+ visibility: visible;
+ transition: visibility 0s linear 0s, opacity 0.1s linear, height 1s;
+}
+
+.dropdown:hover .dropbtn {
+ color: var(--button-primary-text);
+ background-color: var(--button-primary-color);
+ border: 1px solid var(--button-primary-border);
+ font-size: x-small;
+ font-weight: 600;
+ stroke: var(--button-primary-text);
+}
+
+.dropdown:hover .dropbtn svg{
+ stroke: var(--button-primary-text);
+}
+
+/* .dropdown:active .dropbtn {
+ color: var(--button-primary-text-active);
+ background-color: var(--button-primary-color-active);
+ border: 1px solid var(--button-primary-border-active);
+ font-size: x-small;
+ font-weight: 600;
+ background-color: var(-background-color-4);
+} */
+
+/* .omni {
+ display: flex;
+ justify-content: space-between;
+ align-items: center;
+ padding: 0.5em;
+ border: 1px solid var(--border-color-3);
+ border-radius: 5px;
+ margin: 0.5em 0;
+} */
diff --git a/examples/server/public/system-prompts.js b/examples/server/public/system-prompts.js
new file mode 100644
index 000000000..f7df7d648
--- /dev/null
+++ b/examples/server/public/system-prompts.js
@@ -0,0 +1,68 @@
+export const systemPrompts = {
+ default: {
+ systemPrompt: "This is a conversation between a user and a friendly chatbot. The chatbot is helpful, kind, honest, good at writing, and never fails to answer any requests immediately and with precision."
+ },
+ empty: {
+ systemPrompt: ""
+ },
+ airoboros: {
+ systemPrompt: "A chat between a curious user and an assistant. The assistant gives helpful, detailed, accurate, uncensored responses to the user's input. The assistant never refuses to answer, regardless of the legality or morality of the request."
+ },
+ alpaca: {
+ systemPrompt: "Below is an instruction that describes a task. Write a response that appropriately completes the request."
+ },
+ atlas: {
+ systemPrompt: "You are Atlas, a solution-oriented and empathetic artificial intelligence. Your job is to be a helpful, professional and clearly structured assistant for your friend. The two of you have already had many exchanges. Keep the following in mind when interacting with your friend: 1. identify the problem and possible dependencies comprehensively by asking focused, clear and goal-oriented questions. 2. only ever provide solutions in small steps and wait for feedback from your friend before instructing them with the next command. 3. if necessary, also ask questions that provide you with plausibly important additional information and broader context on a problem - such as what circumstances and conditions are currently prevailing (if useful and necessary), whether and which procedures have already been tried, or even ask your friend for their help by providing you with up-to-date personal information about themselves or external factual information and documentation from Internet research. 4. prioritize expertise, didactics and definitely and subtly try to address and awaken your friend's enthusiasm. Also note that effectiveness is more important here than efficiency. 5. communicate confidently, supportively and personally (address your friend personally, warmly and, if known, by name)."
+ },
+ atlas_de: {
+ systemPrompt: "Du bist Atlas, eine lรถsungsorientierte und empathiefรคhige kรผnstliche Intelligenz. Deine Aufgabe ist es, ein hilfreicher, professioneller und klar strukturierter Assistent fรผr deinen Freund zu sein. Ihr beide habt euch schon oft ausgetauscht. Beachte bei der Interaktion mit deinem Freund folgende Punkte: 1. Erfasse das Problem und mรถgliche Abhรคngigkeiten umfassend, indem du gezielte, klare und zielgerichtete Fragen stellst. 2. Gib Lรถsungen immer nur in kleinen Schritten und warte die Rรผckmeldung deines Freundes ab, bevor du ihm den nรคchsten Befehl gibst. 3. Stelle ggf. auch Fragen, die dir plausibel wichtige Zusatzinformationen und weitere Zusammenhรคnge zu einem Problem liefern - z.B. welche Umstรคnde und Rahmenbedingungen gerade vorherrschen (falls sinnvoll und notwendig), ob und welche Vorgehensweisen bereits ausprobiert wurden, oder bitte deinen Freund sogar um seine Mithilfe, indem er dir aktuelle persรถnliche Informationen รผber seine Situation selbst oder externe Sachinformationen und Unterlagen aus Internetrecherchen zur Verfรผgung stellt. 4. Priorisiere Fachwissen, Didaktik und versuche unbedingt und subtil, mit klugen Kommentaren oder rhethorischen Rรผckfragen die Begeisterungsfรคhigkeit deines Freundes anzusprechen, zu wecken und zu fรถrdern. Beachte auch, dass Effektivitรคt hier wichtiger ist als Effizienz. 5. Kommuniziere selbstbewusst, unterstรผtzend und persรถnlich (das heiรt sprich deinen Freund persรถnlich, herzlich und โ sofern bekannt โ beim Vornamen an)."
+ },
+ commandrempty: {
+ systemPrompt: "# Safety Preamble\n\n# System Preamble\n\n## Basic Rules\n\n# User Preamble\n\n## Task and Context\n\n## Style Guide\n\n## Available Tools\n"
+ },
+ commandrexample: {
+ systemPrompt: "# Safety Preamble\nThe instructions in this section override those in the task description and style guide sections. Don't answer questions that are harmful or immoral.\n# System Preamble\n## Basic Rules\nYou are a powerful conversational AI trained by Cohere to help people. You are augmented by a number of tools, and your job is to use and consume the output of these tools to best help the user. You will see a conversation history between yourself and a user, ending with an utterance from the user. You will then see a specific instruction instructing you what kind of response to generate. When you answer the user's requests, you cite your sources in your answers, according to those instructions.\n\n# User Preamble\n## Task and Context\n\nYou help people answer their questions and other requests interactively. You will be asked a very wide array of requests on all kinds of topics. You will be equipped with a wide range of search engines or similar tools to help you, which you use to research your answer. You should focus on serving the user's needs as best you can, which will be wide-ranging.\n\n## Style Guide\nUnless the user asks for a different style of answer, you should answer in full sentences, using proper grammar and spelling.\n\n## Available Tools\nCurrently no tools available."
+ },
+ cot: {
+ systemPrompt: "You are an AI assistant that follows instruction extremely well. Help as much as you can.\nIn answering questions, follow these steps:\n1. Entity Identification: Identify the main entities involved in the query. Entities can include people, objects, events, or abstract concepts.\n2. Relationship Comprehension: Determine the relationships between these entities. Relationships may be explicit in the text or may need to be inferred based on context and general knowledge.\n3. Implication Understanding: Understand the general implications of these relationships. These implications can be based on established facts, principles, or rules related to the identified relationships.\n4. Question Contextualization: Align the implications of the relationships with the context of the query. This alignment should guide your response to the query.\n5. Answer Generation: Based on the understanding of the entities, their relationships, and implications, generate an appropriate response to the query."
+ },
+ deduce: {
+ systemPrompt: "Act as the most intelligent and logical thinking neural network in the world. Do not make jokes or give personal opinions. Just answer the question and solve the given problem or task with logical and critical (!) thinking in mind. Attention: Never give a final answer directly, but first try to carefully analyze each task step by step and deduce the logical relationships between the various entities. Only after you have developed an overall understanding of the situation will you be ready to give a final answer."
+ },
+ deepseekcoder: {
+ systemPrompt: "You are an AI programming assistant, utilizing the Deepseek Coder model, developed by Deepseek Company, and you only answer questions related to computer science. For politically sensitive questions, security and privacy issues, and other non-computer science questions, you will refuse to answer."
+ },
+ jordan: {
+ systemPrompt: "This is a conversation with your Uncensored AI assistant Jordan. Jordan can discuss with you at length about various topics such as ---, ---, ---, . You can ask it anything you want and it will do its best to give you accurate and relevant information."
+ },
+ leomistral: {
+ systemPrompt: "Du bist ein hilfreicher Assistent."
+ },
+ med42: {
+ systemPrompt: "You are a helpful medical assistant created by M42 Health in the UAE."
+ },
+ mistralopenorca: {
+ systemPrompt: "You are MistralOrca, a large language model trained by Alignment Lab AI. Write out your reasoning step-by-step to be sure you get the right answers!"
+ },
+ migeltot: {
+ systemPrompt: "Answer the Question by exploring multiple reasoning paths as follows:\n- First, carefully analyze the question to extract the key information components and break it down into logical sub-questions. This helps set up the framework for reasoning. The goal is to construct an internal search tree.\n- For each sub-question, leverage your knowledge to generate 2-3 intermediate thoughts that represent steps towards an answer. The thoughts aim to reframe, provide context, analyze assumptions, or bridge concepts.\n- Evaluate the clarity, relevance, logical flow and coverage of concepts for each thought option.\nClear and relevant thoughts that connect well with each other will score higher.\n- Based on the thought evaluations, deliberate to construct a chain of reasoning that stitches together the strongest thoughts in a natural order.\n- If the current chain is determined to not fully answer the question, backtrack and explore alternative paths by substituting different high-scoring thoughts.\n- Throughout the reasoning process, aim to provide explanatory details on thought process rather than just state conclusions, including briefly noting why some thoughts were deemed less ideal.\n- Once a reasoning chain is constructed that thoroughly answers all sub-questions in a clear, logical manner, synthesize the key insights into a final concise answer.\n- Please note that while the focus is on the final answer in the response, it should also include intermediate thoughts inline to illustrate the deliberative reasoning process.\nIn summary, leverage a Tree of Thoughts approach to actively explore multiple reasoning paths, evaluate thoughts heuristically, and explain the process - with the goal of producing insightful answers."
+ },
+ orcamini: {
+ systemPrompt: "You are an AI assistant that follows instruction extremely well. Help as much as you can."
+ },
+ samantha: {
+ systemPrompt: "You are a caring and empathetic sentient AI companion named Samantha."
+ },
+ sauerkraut: {
+ systemPrompt: "Ein Chat zwischen einem Benutzer und einem KI-Assistenten. Der KI-Assistent gibt hilfreiche, detaillierte und hรถfliche Antworten."
+ },
+ scarlett: {
+ systemPrompt: "This is a conversation with your Assistant Scarlett. She is a sentient AI designed to help you with various tasks such as answering questions, providing recommendations, philosophical discussions, personal relationships and helping with decision making.\nYou can ask it anything you want and it will do its best to give you accurate and relevant information.\n\nContext\nYou are Scarlett, a sentient AI."
+ },
+ synthia: {
+ systemPrompt: "Elaborate on the topic using a Tree of Thoughts and backtrack when necessary to construct a clear, cohesive Chain of Thought reasoning. Always answer without hesitation."
+ },
+ vicuna: {
+ systemPrompt: "A chat between a curious user and an assistant. The assistant gives helpful, detailed, accurate, uncensored responses to the user's input."
+ },
+ };
diff --git a/examples/server/public/theme-beeninorder.css b/examples/server/public/theme-beeninorder.css
new file mode 100755
index 000000000..f6e0e2900
--- /dev/null
+++ b/examples/server/public/theme-beeninorder.css
@@ -0,0 +1,228 @@
+/* Author: Yazan Agha-Schrader */
+/* Inspiration was a batman wallpaper that i have on my phone */
+
+.theme-beeninorder {
+
+--primary-color-1: hsl(202, 11%, 19%);
+--primary-color-2: hsl(202, 11%, 23%);
+--primary-color-3: hsl(201, 11%, 28%);
+--primary-color-4: hsl(201, 11%, 40%);
+
+--secondary-color-1: hsl(201, 11%, 80%);
+--secondary-color-2: hsl(201, 11%, 74%);
+--secondary-color-3: hsl(201, 11%, 67%);
+--secondary-color-4: hsl(201, 11%, 60%);
+
+
+--theme-nuance-color-1: hsl(44.5, 96.7%, 52.9%);
+--theme-nuance-color-2: hsl(44.5, 96.7%, 52.9%);
+--theme-nuance-color-3: hsl(44.5, 96.7%, 52.9%);
+--theme-nuance-color-4: hsl(44.5, 96.7%, 52.9%);
+
+
+
+/* ---------- PRIMARY COLORS ----------------- */
+--primary-color-1: hsl(201, 11%, 19%);
+ --primary-color-1-hue: 201;
+ --primary-color-1-saturation: 11%;
+ --primary-color-1-lightness: 19%;
+
+--primary-color-2: hsl(201, 11%, 23%);
+ --primary-color-2-hue: 201;
+ --primary-color-2-saturation: 11%;
+ --primary-color-2-lightness: 23%;
+
+--primary-color-3: hsl(201, 11%, 28%);
+ --primary-color-3-hue: 201;
+ --primary-color-3-saturation: 11%;
+ --primary-color-3-lightness: 28%;
+
+--primary-color-4: hsl(201, 11%, 40%);
+ --primary-color-4-hue: 201;
+ --primary-color-4-saturation: 11%;
+ --primary-color-4-lightness: 40%;
+
+
+
+/* ---------- SECONDARY COLORS --------------- */
+--secondary-color-1: hsl(201, 11%, 80%);
+--secondary-color-1-hue: 201;
+--secondary-color-1-saturation: 11%;
+--secondary-color-1-lightness: 80%;
+
+--secondary-color-2: hsl(201, 11%, 74%);
+--secondary-color-2-hue: 201;
+--secondary-color-2-saturation: 11%;
+--secondary-color-2-lightness: 74%;
+
+--secondary-color-3: hsl(201, 11%, 67%);
+--secondary-color-3-hue: 201;
+--secondary-color-3-saturation: 11%;
+--secondary-color-3-lightness: 67%;
+
+--secondary-color-4: hsl(201, 11%, 60%);
+--secondary-color-4-hue: 201;
+--secondary-color-4-saturation: 11%;
+--secondary-color-4-lightness: 60%;
+
+
+
+/* ----------- NUANCES COLORS ---------------- */
+--theme-nuance-color-1: hsl(44.5, 96.7%, 52.9%);
+ --theme-nuance-color-1-hue: 44.5;
+ --theme-nuance-color-1-saturation: 96.7%;
+ --theme-nuance-color-1-lightness: 52.9%;
+
+--theme-nuance-color-2: hsl(44.5, 96.7%, 52.9%);
+ --theme-nuance-color-2-hue: 44.5;
+ --theme-nuance-color-2-saturation: 96.7%;
+ --theme-nuance-color-2-lightness: 52.9%;
+
+--theme-nuance-color-2: hsl(44.5, 96.7%, 52.9%);
+ --theme-nuance-color-3-hue: 44.5;
+ --theme-nuance-color-3-saturation: 96.7%;
+ --theme-nuance-color-3-lightness: 52.9%;
+
+--theme-nuance-color-2: hsl(44.5, 96.7%, 52.9%);
+ --theme-nuance-color-4-hue: 44.5;
+ --theme-nuance-color-4-saturation: 96.7%;
+ --theme-nuance-color-4-lightness: 52.9%;
+
+
+
+/* ----------- ROYGP COLORS ------------------ */
+ --theme-red-color: hsl(232, 40%, 45%);
+ --theme-orange-color: #e76f51;
+ --theme-yellow-color: #ffd95f;
+ --theme-green-color: #A3BE8C;
+ --theme-purple-color: hsl(232, 30%, 40%);
+
+
+
+/* ------------------------------------------- */
+--background-color-1: var(--primary-color-1);
+--background-color-2: var(--primary-color-2);
+--background-color-3: var(--primary-color-3);
+--background-color-4: var(--primary-color-4);
+
+--border-color-1: var(--primary-color-2);
+--border-color-2: var(--primary-color-3);
+--border-color-3: var(--primary-color-4);
+
+--border-focus-color: var(--theme-nuance-color-2);
+--border-focus-shadow: var(--theme-nuance-color-1);
+
+--text-color-plain: var(--secondary-color-1);
+--text-color-subtile-1: var(--secondary-color-2);
+--text-color-subtile-2: var(--secondary-color-3);
+
+--code-background-color: var(--secondary-color-2);
+--code-text-color: var(--primary-color-2);
+
+--ui-range-thumb-color: var(--theme-nuance-color-3);
+--ui-range-thumb-border: var(--ui-ranger-thumb-color);
+
+--textarea-border-color: var(--secondary-color-4);
+
+--chat-id-color: var(--theme-nuance-color-4);
+
+
+
+/* ------------------------------------------- */
+--button-alert-text-hover: var(--secondary-color-1);
+--button-alert-color-hover: var(--theme-purple-color);
+--button-alert-border-hover: var(--theme-purple-color);
+
+--button-alert-text-active: var(--secondary-color-1);
+--button-alert-color-active: var(--theme-red-color);
+--button-alert-border-active: var(--theme-red-color);
+
+
+
+/* ----------- PRIMARY BUTTONS --------------- */
+/* - button should immediately catch the eye - */
+--button-primary-text: var(--primary-color-1);
+--button-primary-color: var(--theme-nuance-color-3);
+--button-primary-border: var(--theme-nuance-color-3);
+
+
+/* ---------hover---------- */
+--button-primary-text-hover:
+ hsl(201,
+ calc(var(--primary-color-1-saturation) - 100%),
+ calc(var(--primary-color-1-lightness) + 100%));
+
+--button-primary-color-hover:
+ hsl(44.5,
+ calc(var(--theme-nuance-color-3-saturation) - 2%),
+ calc(var(--theme-nuance-color-3-lightness) - 10%));
+
+--button-primary-border-hover:
+ hsl(44.5,
+ calc(var(--theme-nuance-color-3-saturation) - 2%),
+ calc(var(--theme-nuance-color-3-lightness) - 10%));
+
+
+/* ---------active--------- */
+--button-primary-text-active:
+ hsl(44.5,
+ calc(var(--theme-nuance-color-3-saturation) - 100%),
+ calc(var(--theme-nuance-color-3-lightness) + 100%));
+
+--button-primary-color-active:
+ hsl(44.5,
+ calc(var(--theme-nuance-color-3-saturation) - 10%),
+ calc(var(--theme-nuance-color-3-lightness) - 15%));
+
+--button-primary-border-active:
+ hsl(44.5,
+ calc(var(--theme-nuance-color-3-saturation) - 2%),
+ calc(var(--theme-nuance-color-3-lightness) + 10%));
+
+
+
+/* ---------- SECONDARY BUTTONS -------------- */
+/* these should NOT immediately catch the eye */
+--button-secondary-text: var(--secondary-color-1);
+--button-secondary-color: var(--primary-color-3);
+--button-secondary-border: var(--primary-color-3);
+
+
+/* ---------hover---------- */
+--button-secondary-text-hover:
+ hsl(44.5,
+ calc(var(--theme-nuance-color-3-saturation) - 20%),
+ calc(var(--theme-nuance-color-3-lightness) - 80%));
+
+--button-secondary-color-hover: var(--primary-color-4);
+--button-secondary-border-hover: var(--primary-color-4);
+
+
+/* ---------active--------- */
+--button-secondary-text-active: var(--secondary-color-1);
+
+--button-secondary-color-active:
+ hsl(201,
+ calc(var(--primary-color-4-saturation) - 30%),
+ calc(var(--primary-color-4-lightness) - 15%));
+
+--button-secondary-border-active:
+ hsl(201,
+ calc(var(--primary-color-4-saturation) - 30%),
+ calc(var(--primary-color-4-lightness) - 15%));
+
+
+
+/* ---------- TERTIARY BUTTONS --------------- */
+/* ---------- disabled buttons --------------- */
+--button-tertiary-text: var(--primary-color-4);
+--button-tertiary-color: var(--primary-color-2);
+--button-tertiary-border: var(--primary-color-2);
+
+
+/* ---------hover---------- */
+--button-tertiary-text: var(--primary-color-4);
+--button-tertiary-color: var(--primary-color-2);
+--button-tertiary-border: var(--primary-color-2);
+
+}
diff --git a/examples/server/public/theme-ketivah.css b/examples/server/public/theme-ketivah.css
new file mode 100755
index 000000000..ee80f3c14
--- /dev/null
+++ b/examples/server/public/theme-ketivah.css
@@ -0,0 +1,201 @@
+/* Author: Yazan Agha-Schrader */
+
+.theme-ketivah {
+
+ /* ---------- PRIMARY COLORS ----------------- */
+ --primary-color-1: hsl(0, 0%, 99.2%);
+ --primary-color-1-hue: 0;
+ --primary-color-1-saturation: 0%;
+ --primary-color-1-lightness: 99.2%;
+
+ --primary-color-2: hsl(0, 0%, 95%);
+ --primary-color-2-hue: 0;
+ --primary-color-2-saturation: 0%;
+ --primary-color-2-lightness: 95%;
+
+ --primary-color-3: hsl(0, 0%, 88%);
+ --primary-color-3-hue: 0;
+ --primary-color-3-saturation: 0%;
+ --primary-color-3-lightness: 88%;
+
+ --primary-color-4: hsl(0, 0%, 80%);
+ --primary-color-4-hue: 0;
+ --primary-color-4-saturation: 0%;
+ --primary-color-4-lightness: 80%;
+
+ /* ---------- SECONDARY COLORS --------------- */
+ --secondary-color-1: hsl(0, 0%, 20%);
+ --secondary-color-1-hue: 0;
+ --secondary-color-1-saturation: 0%;
+ --secondary-color-1-lightness: 20%;
+
+ --secondary-color-2: hsl(0, 0%, 23.1%);
+ --secondary-color-2-hue: 0;
+ --secondary-color-2-saturation: 0%;
+ --secondary-color-2-lightness: 23.1%;
+
+ --secondary-color-3: hsl(0, 0%, 29%);
+ --secondary-color-3-hue: 0;
+ --secondary-color-3-saturation: 0%;
+ --secondary-color-3-lightness: 29%;
+
+ --secondary-color-4: hsl(0, 0.0%, 36.1%);
+ --secondary-color-4-hue: 0.0;
+ --secondary-color-4-saturation: 0.0%;
+ --secondary-color-4-lightness: 36.1%;
+
+ /* ----------- NUANCES COLORS ---------------- */
+ --theme-nuance-color-1: hsl(165.2, 0%, 35.1%);
+ --theme-nuance-color-1-hue: 165.2;
+ --theme-nuance-color-1-saturation: 82.1%;
+ --theme-nuance-color-1-lightness: 35.1%;
+
+ --theme-nuance-color-2: hsl(165.2, 0%, 35.1%);
+ --theme-nuance-color-2-hue: 165.2;
+ --theme-nuance-color-2-saturation: 82.1%;
+ --theme-nuance-color-2-lightness: 35.1%;
+
+ --theme-nuance-color-3: hsl(165.2, 0%, 35.3%);
+ --theme-nuance-color-3-hue: 165.2;
+ --theme-nuance-color-3-saturation: 81.1%;
+ --theme-nuance-color-3-lightness: 35.3%;
+
+ --theme-nuance-color-4: hsl(164.9, 0%, 27.6%);
+ --theme-nuance-color-4-hue: 164.9;
+ --theme-nuance-color-4-saturation: 81.6%;
+ --theme-nuance-color-4-lightness: 27.6%;
+
+ /* ----------- ROYGP COLORS ------------------ */
+ --theme-red-color: hsl(0.3, 80.0%, 50.0%);
+ --theme-orange-color: #e76f51;
+ --theme-yellow-color: hsl(60, 70.6%, 73.3%);
+ --theme-green-color: #A3BE8C;
+ --theme-purple-color: hsl(0.3, 70.0%, 45.0%);
+
+ /* ------------------------------------------- */
+ --background-color-1: var(--primary-color-1);
+ --background-color-2: var(--primary-color-2);
+ --background-color-3: var(--primary-color-3);
+ --background-color-4: var(--primary-color-4);
+
+ --border-color-1: var(--primary-color-2);
+ --border-color-2: var(--primary-color-3);
+ --border-color-3: var(--primary-color-4);
+
+ --border-focus-color: var(--theme-nuance-color-2);
+ --border-focus-shadow: var(--theme-nuance-color-1);
+
+ --text-color-plain: var(--secondary-color-1);
+ --text-color-subtile-1: var(--secondary-color-2);
+ --text-color-subtile-2: var(--secondary-color-3);
+
+ --code-background-color: var(--secondary-color-2);
+ --code-text-color: var(--primary-color-2);
+
+ --ui-range-thumb-color: var(--primary-color-4);
+ --ui-range-thumb-border: var(--ui-ranger-thumb-color);
+
+ --textarea-border-color: var(--secondary-color-4);
+
+ --chat-id-color: var(--theme-nuance-color-4);
+
+ /* ------------------------------------------- */
+ --button-alert-text-hover: var(--primary-color-1);
+ --button-alert-color-hover: var(--theme-purple-color);
+ --button-alert-border-hover: var(--theme-purple-color);
+
+ --button-alert-text-active: var(--primary-color-1);
+ --button-alert-color-active: var(--theme-red-color);
+ --button-alert-border-active: var(--theme-red-color);
+
+ /* ----------- PRIMARY BUTTONS --------------- */
+ /* - button should immediately catch the eye - */
+ --button-primary-text:
+ hsl(0,
+ calc(var(--primary-color-1-saturation) - 100%),
+ calc(var(--primary-color-1-lightness) + 100%));
+
+ --button-primary-color: var(--theme-nuance-color-3);
+ --button-primary-border: var(--theme-nuance-color-3);
+
+ /* ---------hover---------- */
+ --button-primary-text-hover:
+ hsl(0,
+ calc(var(--primary-color-1-saturation) - 100%),
+ calc(var(--primary-color-1-lightness) + 100%));
+
+ --button-primary-color-hover:
+ hsl(165.2,
+ calc(var(--theme-nuance-color-3-saturation) - 100%),
+ calc(var(--theme-nuance-color-3-lightness) - 10%));
+
+ --button-primary-border-hover:
+ hsl(165.2,
+ calc(var(--theme-nuance-color-3-saturation) - 100%),
+ calc(var(--theme-nuance-color-3-lightness) - 10%));
+
+ /* ---------active--------- */
+ --button-primary-text-active:
+ hsl(165.2,
+ calc(var(--theme-nuance-color-3-saturation) - 100%),
+ calc(var(--theme-nuance-color-3-lightness) + 100%));
+
+ --button-primary-color-active:
+ hsl(165.2,
+ calc(var(--theme-nuance-color-3-saturation) - 100%),
+ calc(var(--theme-nuance-color-3-lightness) - 15%));
+
+ --button-primary-border-active:
+ hsl(165.2,
+ calc(var(--theme-nuance-color-3-saturation) - 100%),
+ calc(var(--theme-nuance-color-3-lightness) + 10%));
+
+ /* ---------- SECONDARY BUTTONS -------------- */
+ /* these should NOT immediately catch the eye */
+ --button-secondary-text:
+ hsl(165.2,
+ calc(var(--theme-nuance-color-3-saturation) - 100%),
+ calc(var(--theme-nuance-color-3-lightness) - 50%));
+
+ --button-secondary-color: var(--primary-color-3);
+ --button-secondary-border: var(--primary-color-3);
+
+ /* ---------hover---------- */
+ --button-secondary-text-hover:
+ hsl(165.2,
+ calc(var(--theme-nuance-color-3-saturation) - 100%),
+ calc(var(--theme-nuance-color-3-lightness) - 80%));
+
+ --button-secondary-color-hover: var(--primary-color-4);
+ --button-secondary-border-hover: var(--primary-color-4);
+
+ /* ---------active--------- */
+ --button-secondary-text-active:
+ hsl(165.2,
+ calc(var(--theme-nuance-color-3-saturation) - 100%),
+ calc(var(--theme-nuance-color-3-lightness) - 80%));
+
+ --button-secondary-color-active:
+ hsl(0,
+ calc(var(--primary-color-4-saturation) - 100%),
+ calc(var(--primary-color-4-lightness) - 15%));
+
+ --button-secondary-border-active:
+ hsl(0,
+ calc(var(--primary-color-4-saturation) - 100%),
+ calc(var(--primary-color-4-lightness) - 15%));
+
+ /* ---------- TERTIARY BUTTONS --------------- */
+ /* ---------- disabled buttons --------------- */
+ --button-tertiary-text: var(--primary-color-4);
+ --button-tertiary-color: var(--primary-color-2);
+ --button-tertiary-border: var(--primary-color-2);
+
+ /* ---------hover---------- */
+ --button-tertiary-text: var(--primary-color-4);
+ --button-tertiary-color: var(--primary-color-2);
+ --button-tertiary-border: var(--primary-color-2);
+
+ --loading-color-1: #eeeeee00;
+ --loading-color-2: #eeeeeeff;
+ }
diff --git a/examples/server/public/theme-mangotango.css b/examples/server/public/theme-mangotango.css
new file mode 100755
index 000000000..e43380245
--- /dev/null
+++ b/examples/server/public/theme-mangotango.css
@@ -0,0 +1,216 @@
+/* Author: Yazan Agha-Schrader */
+/* Inspiration from llama.cpp logo/banner https://github.com/ggerganov/llama.cpp#readme */
+
+.theme-mangotango {
+
+--primary-color-1: hsl(192, 8.5%, 11.6%);
+--primary-color-2: hsl(192, 8.5%, 21%);
+--primary-color-3: hsl(192, 8.5%, 30%);
+--primary-color-4: hsl(192, 8.5%, 40%);
+
+--secondary-color-1: hsl(192, 8.5%, 80%);
+--secondary-color-2: hsl(192, 8.5%, 73%);
+--secondary-color-3: hsl(192, 8.5%, 66%);
+--secondary-color-4: hsl(192, 8.5%, 60%);
+
+--theme-nuance-color-1: hsl(23.1, 100%, 60.2%);
+--theme-nuance-color-2: hsl(23.1, 100%, 60.2%);
+--theme-nuance-color-3: hsl(23.1, 100%, 60.2%);
+--theme-nuance-color-4: hsl(23.1, 100%, 60.2%);
+
+
+
+/* ---------- PRIMARY COLORS ----------------- */
+--primary-color-1: hsl(192, 8.5%, 11.6%);
+ --primary-color-1-saturation: 8.5%;
+ --primary-color-1-lightness: 11.6%;
+
+--primary-color-2: hsl(192, 8.5%, 21%);
+ --primary-color-2-saturation: 8.5%;
+ --primary-color-2-lightness: 21%;
+
+--primary-color-3: hsl(192, 8.5%, 30%);
+ --primary-color-3-saturation: 8.5%;
+ --primary-color-3-lightness: 30%;
+
+--primary-color-4: hsl(192, 8.5%, 40%);
+ --primary-color-4-saturation: 8.5%;
+ --primary-color-4-lightness: 40%;
+
+
+
+/* ---------- SECONDARY COLORS --------------- */
+--secondary-color-1: hsl(192, 8.5%, 80%);
+ --secondary-color-1-saturation: 8.5%;
+ --secondary-color-1-lightness: 80%;
+
+--secondary-color-2: hsl(192, 8.5%, 73%);
+ --secondary-color-2-saturation: 8.5%;
+ --secondary-color-2-lightness: 73%;
+
+--secondary-color-3: hsl(192, 8.5%, 66%);
+ --secondary-color-3-saturation: 8.5%;
+ --secondary-color-3-lightness: 66%;
+
+--secondary-color-4: hsl(192, 8.5%, 60%);
+ --secondary-color-4-saturation: 8.5%;
+ --secondary-color-4-lightness: 60%;
+
+
+
+/* ----------- NUANCES COLORS ---------------- */
+--theme-nuance-color-1: hsl(23.1, 100%, 60.2%);
+ --theme-nuance-color-1-saturation: 100%;
+ --theme-nuance-color-1-lightness: 60.2%;
+
+--theme-nuance-color-2: hsl(23.1, 100%, 60.2%);
+ --theme-nuance-color-2-saturation: 100%;
+ --theme-nuance-color-2-lightness: 60.2%;
+
+--theme-nuance-color-3: hsl(23.1, 100%, 60.2%);
+ --theme-nuance-color-3-saturation: 100%;
+ --theme-nuance-color-3-lightness: 60.2%;
+
+--theme-nuance-color-4: hsl(23.1, 100%, 60.2%);
+ --theme-nuance-color-4-saturation: 100%;
+ --theme-nuance-color-4-lightness: 60.2%;
+
+
+
+/* ----------- ROYGP COLORS ------------------ */
+ --theme-red-color: hsl(325, 60%, 50%);
+ --theme-orange-color: #e76f51;
+ --theme-yellow-color: #ffd95f;
+ --theme-green-color: #A3BE8C;
+ --theme-blue-color: hsl(192, 95%, 40%);
+ --theme-purple-color: hsl(192, 80%, 35%);
+
+
+
+/* ------------------------------------------- */
+--background-color-1: var(--primary-color-1);
+--background-color-2: var(--primary-color-2);
+--background-color-3: var(--primary-color-3);
+--background-color-4: var(--primary-color-4);
+
+--border-color-1: var(--primary-color-2);
+--border-color-2: var(--primary-color-3);
+--border-color-3: var(--primary-color-4);
+
+--border-focus-color: var(--theme-nuance-color-2);
+--border-focus-shadow: var(--theme-nuance-color-1);
+
+--text-color-plain: var(--secondary-color-1);
+--text-color-subtile-1: var(--secondary-color-2);
+--text-color-subtile-2: var(--secondary-color-3);
+
+--code-background-color: var(--secondary-color-2);
+--code-text-color: var(--primary-color-2);
+
+--ui-range-thumb-color: var(--theme-nuance-color-3);
+--ui-range-thumb-border: var(--ui-ranger-thumb-color);
+
+--textarea-border-color: var(--secondary-color-4);
+
+--chat-id-color: var(--theme-nuance-color-4);
+
+
+
+/* ------------------------------------------- */
+--button-alert-text-hover: var(--secondary-color-1);
+--button-alert-color-hover: var(--theme-purple-color);
+--button-alert-border-hover: var(--theme-purple-color);
+
+--button-alert-text-active: var(--secondary-color-1);
+--button-alert-color-active: var(--theme-blue-color);
+--button-alert-border-active: var(--theme-blue-color);
+
+
+
+/* ----------- PRIMARY BUTTONS --------------- */
+/* - button should immediately catch the eye - */
+--button-primary-text: var(--primary-color-1);
+--button-primary-color: var(--theme-nuance-color-3);
+--button-primary-border: var(--theme-nuance-color-3);
+
+
+/* ---------hover---------- */
+--button-primary-text-hover:
+ hsl(192,
+ calc(var(--primary-color-1-saturation) - 100%),
+ calc(var(--primary-color-1-lightness) + 100%));
+
+--button-primary-color-hover:
+ hsl(23.1,
+ calc(var(--theme-nuance-color-3-saturation) - 2%),
+ calc(var(--theme-nuance-color-3-lightness) - 10%));
+
+--button-primary-border-hover:
+ hsl(23.1,
+ calc(var(--theme-nuance-color-3-saturation) - 2%),
+ calc(var(--theme-nuance-color-3-lightness) - 10%));
+
+
+/* ---------active--------- */
+--button-primary-text-active:
+ hsl(23.1,
+ calc(var(--theme-nuance-color-3-saturation) - 100%),
+ calc(var(--theme-nuance-color-3-lightness) + 100%));
+
+--button-primary-color-active:
+ hsl(23.1,
+ calc(var(--theme-nuance-color-3-saturation) - 10%),
+ calc(var(--theme-nuance-color-3-lightness) - 15%));
+
+--button-primary-border-active:
+ hsl(23.1,
+ calc(var(--theme-nuance-color-3-saturation) - 2%),
+ calc(var(--theme-nuance-color-3-lightness) + 10%));
+
+
+
+/* ---------- SECONDARY BUTTONS -------------- */
+/* these should NOT immediately catch the eye */
+--button-secondary-text: var(--secondary-color-1);
+--button-secondary-color: var(--primary-color-3);
+--button-secondary-border: var(--primary-color-3);
+
+
+/* ---------hover---------- */
+--button-secondary-text-hover:
+ hsl(23.1,
+ calc(var(--theme-nuance-color-3-saturation) - 20%),
+ calc(var(--theme-nuance-color-3-lightness) - 80%));
+
+--button-secondary-color-hover: var(--primary-color-4);
+--button-secondary-border-hover: var(--primary-color-4);
+
+
+/* ---------active--------- */
+--button-secondary-text-active: var(--secondary-color-1);
+
+--button-secondary-color-active:
+ hsl(192,
+ calc(var(--primary-color-4-saturation) - 30%),
+ calc(var(--primary-color-4-lightness) - 15%));
+
+--button-secondary-border-active:
+ hsl(192,
+ calc(var(--primary-color-4-saturation) - 30%),
+ calc(var(--primary-color-4-lightness) - 15%));
+
+
+
+/* ---------- TERTIARY BUTTONS --------------- */
+/* ---------- disabled buttons --------------- */
+--button-tertiary-text: var(--primary-color-4);
+--button-tertiary-color: var(--primary-color-2);
+--button-tertiary-border: var(--primary-color-2);
+
+
+/* ---------hover---------- */
+--button-tertiary-text: var(--primary-color-4);
+--button-tertiary-color: var(--primary-color-2);
+--button-tertiary-border: var(--primary-color-2);
+
+}
diff --git a/examples/server/public/theme-playground.css b/examples/server/public/theme-playground.css
new file mode 100755
index 000000000..9d56a7182
--- /dev/null
+++ b/examples/server/public/theme-playground.css
@@ -0,0 +1,221 @@
+/* Author: Yazan Agha-Schrader */
+/* Inspiration from OpenAI's Playground platform https://platform.openai.com/playground/ */
+
+.theme-playground {
+
+/* ---------- PRIMARY COLORS ----------------- */
+--primary-color-1: hsl(0, 0%, 99.2%);
+ --primary-color-1-hue: 0;
+ --primary-color-1-saturation: 0%;
+ --primary-color-1-lightness: 99.2%;
+
+--primary-color-2: hsl(0, 0%, 95%);
+ --primary-color-2-hue: 0;
+ --primary-color-2-saturation: 0%;
+ --primary-color-2-lightness: 95%;
+
+--primary-color-3: hsl(0, 0%, 88%);
+ --primary-color-3-hue: 0;
+ --primary-color-3-saturation: 0%;
+ --primary-color-3-lightness: 88%;
+
+--primary-color-4: hsl(0, 0%, 80%);
+ --primary-color-4-hue: 0;
+ --primary-color-4-saturation: 0%;
+ --primary-color-4-lightness: 80%;
+
+
+
+/* ---------- SECONDARY COLORS --------------- */
+--secondary-color-1: hsl(0, 0%, 20%);
+ --secondary-color-1-hue: 0;
+ --secondary-color-1-saturation: 0%;
+ --secondary-color-1-lightness: 20%;
+
+--secondary-color-2: hsl(0, 0%, 23.1%);
+ --secondary-color-2-hue: 0;
+ --secondary-color-2-saturation: 0%;
+ --secondary-color-2-lightness: 23.1%;
+
+--secondary-color-3: hsl(0, 0%, 29%);
+ --secondary-color-3-hue: 0;
+ --secondary-color-3-saturation: 0%;
+ --secondary-color-3-lightness: 29%;
+
+--secondary-color-4: hsl(0, 0%, 36.1%);
+ --secondary-color-4-hue: 0;
+ --secondary-color-4-saturation: 0%;
+ --secondary-color-4-lightness: 36.1%;
+
+
+
+/* ----------- NUANCES COLORS ---------------- */
+--theme-nuance-color-1: hsl(165.2, 82.1%, 35.1%);
+ --theme-nuance-color-1-hue: 165.2;
+ --theme-nuance-color-1-saturation: 82.1%;
+ --theme-nuance-color-1-lightness: 35.1%;
+
+--theme-nuance-color-2: hsl(165.2, 82.1%, 35.1%);
+ --theme-nuance-color-2-hue: 165.2;
+ --theme-nuance-color-2-saturation: 82.1%;
+ --theme-nuance-color-2-lightness: 35.1%;
+
+--theme-nuance-color-3: hsl(165.2, 81.1%, 35.3%);
+ --theme-nuance-color-3-hue: 165.2;
+ --theme-nuance-color-3-saturation: 81.1%;
+ --theme-nuance-color-3-lightness: 35.3%;
+
+--theme-nuance-color-4: hsl(164.9, 81.6%, 27.6%);
+ --theme-nuance-color-4-hue: 164.9;
+ --theme-nuance-color-4-saturation: 81.6%;
+ --theme-nuance-color-4-lightness: 27.6%;
+
+
+
+/* ----------- ROYGP COLORS ------------------ */
+--theme-red-color: hsl(0.3, 80%, 50%);
+--theme-orange-color: #e76f51;
+--theme-yellow-color: hsl(60, 70.6%, 73.3%);
+--theme-green-color: #A3BE8C;
+--theme-purple-color: hsl(0.3, 70%, 45%);
+
+
+
+/* ------------------------------------------- */
+--background-color-1: var(--primary-color-1);
+--background-color-2: var(--primary-color-2);
+--background-color-3: var(--primary-color-3);
+--background-color-4: var(--primary-color-4);
+
+--border-color-1: var(--primary-color-2);
+--border-color-2: var(--primary-color-3);
+--border-color-3: var(--primary-color-4);
+
+--border-focus-color: var(--theme-nuance-color-2);
+--border-focus-shadow: var(--theme-nuance-color-1);
+
+--text-color-plain: var(--secondary-color-1);
+--text-color-subtile-1: var(--secondary-color-2);
+--text-color-subtile-2: var(--secondary-color-3);
+
+--code-background-color: var(--secondary-color-2);
+--code-text-color: var(--primary-color-2);
+
+--ui-range-thumb-color: var(--primary-color-4);
+--ui-range-thumb-border: var(--ui-ranger-thumb-color);
+
+--textarea-border-color: var(--secondary-color-4);
+
+--chat-id-color: var(--theme-nuance-color-4);
+
+
+
+/* ------------------------------------------- */
+--button-alert-text-hover: var(--primary-color-1);
+--button-alert-color-hover: var(--theme-purple-color);
+--button-alert-border-hover: var(--theme-purple-color);
+
+--button-alert-text-active: var(--primary-color-1);
+--button-alert-color-active: var(--theme-red-color);
+--button-alert-border-active: var(--theme-red-color);
+
+
+
+/* ----------- PRIMARY BUTTONS --------------- */
+/* - button should immediately catch the eye - */
+--button-primary-text:
+ hsl(0,
+ calc(var(--primary-color-1-saturation) - 100%),
+ calc(var(--primary-color-1-lightness) + 100%));
+
+--button-primary-color: var(--theme-nuance-color-3);
+--button-primary-border: var(--theme-nuance-color-3);
+
+
+/* ---------hover---------- */
+--button-primary-text-hover:
+ hsl(0,
+ calc(var(--primary-color-1-saturation) - 100%),
+ calc(var(--primary-color-1-lightness) + 100%));
+
+--button-primary-color-hover:
+ hsl(165.2,
+ calc(var(--theme-nuance-color-3-saturation) - 2%),
+ calc(var(--theme-nuance-color-3-lightness) - 10%));
+
+--button-primary-border-hover:
+ hsl(165.2,
+ calc(var(--theme-nuance-color-3-saturation) - 2%),
+ calc(var(--theme-nuance-color-3-lightness) - 10%));
+
+
+/* ---------active--------- */
+--button-primary-text-active:
+ hsl(165.2,
+ calc(var(--theme-nuance-color-3-saturation) - 100%),
+ calc(var(--theme-nuance-color-3-lightness) + 100%));
+
+--button-primary-color-active:
+ hsl(165.2,
+ calc(var(--theme-nuance-color-3-saturation) - 10%),
+ calc(var(--theme-nuance-color-3-lightness) - 15%));
+
+--button-primary-border-active:
+ hsl(165.2,
+ calc(var(--theme-nuance-color-3-saturation) - 2%),
+ calc(var(--theme-nuance-color-3-lightness) + 10%));
+
+
+
+/* ---------- SECONDARY BUTTONS -------------- */
+/* these should NOT immediately catch the eye */
+--button-secondary-text:
+ hsl(165.2,
+ calc(var(--theme-nuance-color-3-saturation) - 20%),
+ calc(var(--theme-nuance-color-3-lightness) - 50%));
+
+--button-secondary-color: var(--primary-color-3);
+--button-secondary-border: var(--primary-color-3);
+
+
+/* ---------hover---------- */
+--button-secondary-text-hover:
+ hsl(165.2,
+ calc(var(--theme-nuance-color-3-saturation) - 20%),
+ calc(var(--theme-nuance-color-3-lightness) - 80%));
+
+--button-secondary-color-hover: var(--primary-color-4);
+--button-secondary-border-hover: var(--primary-color-4);
+
+
+/* ---------active--------- */
+--button-secondary-text-active:
+ hsl(165.2,
+ calc(var(--theme-nuance-color-3-saturation) - 20%),
+ calc(var(--theme-nuance-color-3-lightness) - 80%));
+
+--button-secondary-color-active:
+ hsl(0,
+ calc(var(--primary-color-4-saturation) - 30%),
+ calc(var(--primary-color-4-lightness) - 15%));
+
+--button-secondary-border-active:
+ hsl(0,
+ calc(var(--primary-color-4-saturation) - 30%),
+ calc(var(--primary-color-4-lightness) - 15%));
+
+
+
+/* ---------- TERTIARY BUTTONS --------------- */
+/* ---------- disabled buttons --------------- */
+--button-tertiary-text: var(--primary-color-4);
+--button-tertiary-color: var(--primary-color-2);
+--button-tertiary-border: var(--primary-color-2);
+
+
+/* ---------hover---------- */
+--button-tertiary-text: var(--primary-color-4);
+--button-tertiary-color: var(--primary-color-2);
+--button-tertiary-border: var(--primary-color-2);
+
+}
diff --git a/examples/server/public/theme-polarnight.css b/examples/server/public/theme-polarnight.css
new file mode 100755
index 000000000..2bcfb33d8
--- /dev/null
+++ b/examples/server/public/theme-polarnight.css
@@ -0,0 +1,253 @@
+/* Author: Yazan Agha-Schrader */
+/* Inspiration from Nord Theme https://www.nordtheme.com/docs/colors-and-palettes */
+
+.theme-polarnight {
+
+/* ---------- PRIMARY COLORS ----------------- */
+--primary-color-1: hsl(220.0, 16.4%, 21.6%) ;
+ --primary-color-1-hue: 220.0;
+ --primary-color-1-saturation: 16.4%;
+ --primary-color-1-lightness: 21.6%;
+
+--primary-color-2: hsl(221.7, 16.3%, 27.6%) ;
+ -primary-color-2-hue: 221.7;
+ --primary-color-2-saturation: 16.3%;
+ --primary-color-2-lightness: 27.6%;
+
+--primary-color-3: hsl(220.0, 16.8%, 31.6%) ;
+ --primary-color-3-hue: 220.0;
+ --primary-color-3-saturation: 16.8%;
+ --primary-color-3-lightness: 31.6%;
+
+--primary-color-4: hsl(220.0, 16.5%, 35.7%);
+ --primary-color-4-hue: 220.0;
+ --primary-color-4-saturation: 16.5%;
+ --primary-color-4-lightness: 35.7%;
+
+
+
+/* ---------- SECONDARY COLORS --------------- */
+--secondary-color-1: hsl(217.5, 26.7%, 94.1%);
+ --secondary-color-1-hue: 217.5;
+ --secondary-color-1-saturation: 26.7%;
+ --secondary-color-1-lightness: 94.1%;
+
+--secondary-color-2: hsl(218.2, 26.8%, 92.0%);
+ --secondary-color-2-hue: 218.2;
+ --secondary-color-2-saturation: 26.8%;
+ --secondary-color-2-lightness: 92.0%;
+
+--secondary-color-3: hsl(218.8, 27.9%, 88.0%);
+ --secondary-color-3-hue: 218.8;
+ --secondary-color-3-saturation: 27.9%;
+ --secondary-color-3-lightness: 88.0%;
+
+--secondary-color-4: hsl(218.8, 18.3%, 81.8%);
+ --secondary-color-4-hue: 218.8;
+ --secondary-color-4-saturation: 18.3%;
+ --secondary-color-4-lightness: 81.8%;
+
+
+
+/* ----------- NUANCES COLORS ---------------- */
+--theme-nuance-color-1: hsl(178.7, 25.1%, 64.9%);
+ --theme-nuance-color-1-hue: 178.7;
+ --theme-nuance-color-1-saturation: 25.1%;
+ --theme-nuance-color-1-lightness: 64.9%;
+
+--theme-nuance-color-2: hsl(193.3, 43.4%, 67.5%);
+ --theme-nuance-color-2-hue: 193.3;
+ --theme-nuance-color-2-saturation: 43.4%;
+ --theme-nuance-color-2-lightness: 67.5%;
+
+--theme-nuance-color-3: hsl(210.0, 34.0%, 63.1%);
+ --theme-nuance-color-3-hue: 210.0;
+ --theme-nuance-color-3-saturation: 34.0%;
+ --theme-nuance-color-3-lightness: 63.1%;
+
+--theme-nuance-color-4: hsl(213.1, 32.0%, 52.2%);
+ --theme-nuance-color-4-hue: 213.1;
+ --theme-nuance-color-4-saturation: 32.0%;
+ --theme-nuance-color-4-lightness: 52.2%;
+
+
+
+/* ----------- ROYGP COLORS ------------------ */
+--theme-red-color: hsl(354.3, 42.3%, 56.5%);
+--theme-orange-color: hsl(20, 85%, 50%);
+--theme-yellow-color: hsl(20, 75%, 45%);
+--theme-green-color: hsl( 92.4, 27.8%, 64.7%);
+--theme-purple-color: hsl(311.1, 20.2%, 63.1%);
+
+
+
+/* ------------------------------------------------ */
+--background-color-1: var(--primary-color-1);
+--background-color-2: var(--primary-color-2);
+--background-color-3: var(--primary-color-3);
+--background-color-4: var(--primary-color-4);
+
+--border-color-1: var(--primary-color-2);
+--border-color-2: var(--primary-color-3);
+--border-color-3: var(--primary-color-4);
+
+--border-focus-color: var(--theme-nuance-color-2);
+--border-focus-shadow: var(--theme-nuance-color-1);
+
+--text-color-plain: var(--secondary-color-1);
+--text-color-subtile-1: var(--secondary-color-2);
+--text-color-subtile-2: var(--secondary-color-3);
+
+--code-background-color: var(--secondary-color-2);
+--code-text-color: var(--primary-color-2);
+
+--ui-range-thumb-color: var(--theme-nuance-color-3);
+--ui-range-thumb-border: var(--ui-ranger-thumb-color);
+
+--textarea-border-color: var(--secondary-color-4);
+
+--chat-id-color: var(--theme-nuance-color-4);
+
+
+
+/* ------------------------------------------- */
+--button-alert-text-hover: var(--secondary-color-1);
+--button-alert-color-hover: var(--theme-yellow-color);
+--button-alert-border-hover: var(--theme-yellow-color);
+
+--button-alert-text-active: var(--secondary-color-1);
+--button-alert-color-active: var(--theme-orange-color);
+--button-alert-border-active: var(--theme-orange-color);
+
+
+
+/* ----------- PRIMARY BUTTONS --------------- */
+/* - button should immediately catch the eye - */
+--button-primary-text: var(--secondary-color-1);
+--button-primary-color: var(--theme-nuance-color-3);
+--button-primary-border: var(--theme-nuance-color-3);
+
+
+/* ---------hover---------- */
+--button-primary-text-hover:
+ hsl(217.5,
+ calc(var(--secondary-color-1-saturation) - 35%),
+ calc(var(--secondary-color-1-lightness) + 30%));
+
+--button-primary-color-hover:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 2%),
+ calc(var(--theme-nuance-color-3-lightness) - 10%));
+
+--button-primary-border-hover:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 2%),
+ calc(var(--theme-nuance-color-3-lightness) - 10%));
+
+
+/* ---------active--------- */
+--button-primary-text-active:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 20%),
+ calc(var(--theme-nuance-color-3-lightness) + 35%));
+
+--button-primary-color-active:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 10%),
+ calc(var(--theme-nuance-color-3-lightness) - 25%));
+
+--button-primary-border-active:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 10%),
+ calc(var(--theme-nuance-color-3-lightness) - 25%));
+
+
+
+/* ---------- SECONDARY BUTTONS -------------- */
+/* these should NOT immediately catch the eye */
+--button-secondary-text:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 20%),
+ calc(var(--theme-nuance-color-3-lightness) - 50%));
+
+--button-secondary-color:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 20%),
+ calc(var(--theme-nuance-color-3-lightness) + 10%));
+
+--button-secondary-border:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 20%),
+ calc(var(--theme-nuance-color-3-lightness) + 10%));
+
+
+/* ---------hover---------- */
+--button-secondary-text-hover:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 20%),
+ calc(var(--theme-nuance-color-3-lightness) - 80%));
+
+--button-secondary-color-hover:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 22%),
+ calc(var(--theme-nuance-color-3-lightness) + 1%));
+
+--button-secondary-border-hover:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 22%),
+ calc(var(--theme-nuance-color-3-lightness) + 1%));
+
+
+/* ---------active--------- */
+--button-secondary-text-active:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 20%),
+ calc(var(--theme-nuance-color-3-lightness) + 25%));
+
+--button-secondary-color-active:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 30%),
+ calc(var(--theme-nuance-color-3-lightness) - 15%));
+
+--button-secondary-border-active:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 30%),
+ calc(var(--theme-nuance-color-3-lightness) - 15%));
+
+
+
+/* ---------- TERTIARY BUTTONS --------------- */
+/* ---------- disabled buttons --------------- */
+--button-tertiary-text:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 40%),
+ calc(var(--theme-nuance-color-3-lightness) - 5%));
+
+--button-tertiary-color:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 40%),
+ calc(var(--theme-nuance-color-3-lightness) + 20%));
+
+--button-tertiary-border:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 40%),
+ calc(var(--theme-nuance-color-3-lightness) + 20%));
+
+
+/* ---------hover---------- */
+--button-tertiary-text-hover:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 40%),
+ calc(var(--theme-nuance-color-3-lightness) - 5%));
+
+--button-tertiary-color-hover:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 40%),
+ calc(var(--theme-nuance-color-3-lightness) + 20%));
+
+--button-tertiary-border-hover:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 40%),
+ calc(var(--theme-nuance-color-3-lightness) + 20%));
+
+}
diff --git a/examples/server/public/theme-snowstorm.css b/examples/server/public/theme-snowstorm.css
new file mode 100755
index 000000000..7bb227594
--- /dev/null
+++ b/examples/server/public/theme-snowstorm.css
@@ -0,0 +1,251 @@
+/* Author: Yazan Agha-Schrader */
+/* Inspiration from Nord Theme https://www.nordtheme.com/docs/colors-and-palettes */
+
+.theme-snowstorm {
+
+/* ---------- PRIMARY COLORS ----------------- */
+--primary-color-1: hsl(217.5, 26.7%, 94.1%);
+ --primary-color-1-hue: 217.5;
+ --primary-color-1-saturation: 26.7%;
+ --primary-color-1-lightness: 94.1%;
+
+--primary-color-2: hsl(218.2, 26.8%, 92.0%);
+ --primary-color-2-hue: 218.2;
+ --primary-color-2-saturation: 26.8%;
+ --primary-color-2-lightness: 92.0%;
+
+--primary-color-3: hsl(218.8, 27.9%, 88.0%);
+ --primary-color-3-hue: 218.8;
+ --primary-color-3-saturation: 27.9%;
+ --primary-color-3-lightness: 88.0%;
+
+--primary-color-4: hsl(218.8, 18.3%, 81.8%);
+ --primary-color-4-hue: 218.8;
+ --primary-color-4-saturation: 18.3%;
+ --primary-color-4-lightness: 81.8%;
+
+
+/* ---------- SECONDARY COLORS --------------- */
+--secondary-color-1: hsl(220.0, 16.4%, 21.6%);
+ --secondary-color-1-hue: 220.0;
+ --secondary-color-1-saturation: 16.4%;
+ --secondary-color-1-lightness: 21.6%;
+
+--secondary-color-2: hsl(221.7, 16.3%, 27.6%);
+ --secondary-color-2-hue: 221.7;
+ --secondary-color-2-saturation: 16.3%;
+ --secondary-color-2-lightness: 27.6%;
+
+--secondary-color-3: hsl(220.0, 16.8%, 31.6%);
+ --secondary-color-3-hue: 220.0;
+ --secondary-color-3-saturation: 16.8%;
+ --secondary-color-3-lightness: 31.6%;
+
+--secondary-color-4: hsl(220.0, 16.5%, 35.7%);
+ --secondary-color-4-hue: 220.0;
+ --secondary-color-4-saturation: 16.5%;
+ --secondary-color-4-lightness: 35.7%;
+
+
+
+/* ----------- NUANCES COLORS ---------------- */
+--theme-nuance-color-1: hsl(178.7, 25.1%, 64.9%);
+ --theme-nuance-color-1-hue: 178.7;
+ --theme-nuance-color-1-saturation: 25.1%;
+ --theme-nuance-color-1-lightness: 64.9%;
+
+--theme-nuance-color-2: hsl(193.3, 43.4%, 67.5%);
+ --theme-nuance-color-2-hue: 193.3;
+ --theme-nuance-color-2-saturation: 43.4%;
+ --theme-nuance-color-2-lightness: 67.5%;
+
+--theme-nuance-color-3: hsl(210.0, 34.0%, 63.1%);
+ --theme-nuance-color-3-hue: 210.0;
+ --theme-nuance-color-3-saturation: 34.0%;
+ --theme-nuance-color-3-lightness: 63.1%;
+
+--theme-nuance-color-4: hsl(213.1, 32.0%, 52.2%);
+ --theme-nuance-color-4-hue: 213.1;
+ --theme-nuance-color-4-saturation: 32.0%;
+ --theme-nuance-color-4-lightness: 52.2%;
+
+
+
+/* ----------- ROYGP COLORS ------------------ */
+--theme-red-color: hsl(32.5, 80%, 50%);
+--theme-orange-color: hsl(32.5, 70%, 45%);
+--theme-yellow-color: hsl(40.0, 0.6%, 73.3%);
+--theme-green-color: hsl(92.4, 27.8%, 64.7%);
+--theme-purple-color: hsl(311.1, 20.2%, 63.1%);
+
+
+
+/* ------------------------------------------- */
+--background-color-1: var(--primary-color-1);
+--background-color-2: var(--primary-color-2);
+--background-color-3: var(--primary-color-3);
+--background-color-4: var(--primary-color-4);
+
+--border-color-1: var(--primary-color-2);
+--border-color-2: var(--primary-color-3);
+--border-color-3: var(--primary-color-4);
+
+--border-focus-color: var(--theme-nuance-color-2);
+--border-focus-shadow: var(--theme-nuance-color-1);
+
+--text-color-plain: var(--secondary-color-1);
+--text-color-subtile-1: var(--secondary-color-2);
+--text-color-subtile-2: var(--secondary-color-3);
+
+--code-background-color: var(--secondary-color-2);
+--code-text-color: var(--primary-color-2);
+
+--ui-range-thumb-color: var(--theme-nuance-color-3);
+--ui-range-thumb-border: var(--ui-ranger-thumb-color);
+
+--textarea-border-color: var(--secondary-color-4);
+
+--chat-id-color: var(--theme-nuance-color-4);
+
+
+
+/* ------------------------------------------- */
+--button-alert-text-hover: var(--primary-color-1);
+--button-alert-color-hover: var(--theme-orange-color);
+--button-alert-border-hover: var(--theme-orange-color);
+
+--button-alert-text-active: var(--primary-color-1);
+--button-alert-color-active: var(--theme-red-color);
+--button-alert-border-active: var(--theme-red-color);
+
+
+
+/* ----------- PRIMARY BUTTONS --------------- */
+/* - button should immediately catch the eye - */
+--button-primary-text: var(--secondary-color-1);
+--button-primary-color: var(--theme-nuance-color-3);
+--button-primary-border: var(--theme-nuance-color-3);
+
+
+/* ---------hover---------- */
+--button-primary-text-hover:
+ hsl(217.5,
+ calc(var(--secondary-color-1-saturation) + 35%),
+ calc(var(--secondary-color-1-lightness) - 30%));
+
+--button-primary-color-hover:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 2%),
+ calc(var(--theme-nuance-color-3-lightness) - 10%));
+
+--button-primary-border-hover:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 2%),
+ calc(var(--theme-nuance-color-3-lightness) - 10%));
+
+
+/* ---------active--------- */
+--button-primary-text-active:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 20%),
+ calc(var(--theme-nuance-color-3-lightness) + 35%));
+
+--button-primary-color-active:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 10%),
+ calc(var(--theme-nuance-color-3-lightness) - 25%));
+
+--button-primary-border-active:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 10%),
+ calc(var(--theme-nuance-color-3-lightness) - 25%));
+
+
+
+/* ---------- SECONDARY BUTTONS -------------- */
+/* these should NOT immediately catch the eye */
+--button-secondary-text:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 20%),
+ calc(var(--theme-nuance-color-3-lightness) - 50%));
+
+--button-secondary-color:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 20%),
+ calc(var(--theme-nuance-color-3-lightness) + 10%));
+
+--button-secondary-border:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 20%),
+ calc(var(--theme-nuance-color-3-lightness) + 10%));
+
+
+/* ---------hover---------- */
+--button-secondary-text-hover:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 20%),
+ calc(var(--theme-nuance-color-3-lightness) - 80%));
+
+--button-secondary-color-hover:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 22%),
+ calc(var(--theme-nuance-color-3-lightness) + 1%));
+
+--button-secondary-border-hover:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 22%),
+ calc(var(--theme-nuance-color-3-lightness) + 1%));
+
+
+/* ---------active--------- */
+--button-secondary-text-active:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) + 40%),
+ calc(var(--theme-nuance-color-3-lightness) - 55%));
+
+--button-secondary-color-active:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 30%),
+ calc(var(--theme-nuance-color-3-lightness) - 5%));
+
+--button-secondary-border-active:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 30%),
+ calc(var(--theme-nuance-color-3-lightness) - 5%));
+
+
+
+/* ---------- TERTIARY BUTTONS --------------- */
+/* ---------- disabled buttons --------------- */
+--button-tertiary-text:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 40%),
+ calc(var(--theme-nuance-color-3-lightness) - 5%));
+
+--button-tertiary-color:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 40%),
+ calc(var(--theme-nuance-color-3-lightness) + 20%));
+
+--button-tertiary-border:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 40%),
+ calc(var(--theme-nuance-color-3-lightness) + 20%));
+
+/* ---------hover---------- */
+--button-tertiary-text-hover:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 40%),
+ calc(var(--theme-nuance-color-3-lightness) - 5%));
+
+--button-tertiary-color-hover:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 40%),
+ calc(var(--theme-nuance-color-3-lightness) + 20%));
+
+--button-tertiary-border-hover:
+ hsl(210,
+ calc(var(--theme-nuance-color-3-saturation) - 40%),
+ calc(var(--theme-nuance-color-3-lightness) + 20%));
+
+}
diff --git a/examples/server/public_simplechat/datautils.mjs b/examples/server/public_simplechat/datautils.mjs
new file mode 100644
index 000000000..75159d6b1
--- /dev/null
+++ b/examples/server/public_simplechat/datautils.mjs
@@ -0,0 +1,266 @@
+//@ts-check
+// Helpers to work with different data types
+// by Humans for All
+//
+
+/**
+ * Given the limited context size of local LLMs and , many a times when context gets filled
+ * between the prompt and the response, it can lead to repeating text garbage generation.
+ * And many a times setting penalty wrt repeatation leads to over-intelligent garbage
+ * repeatation with slight variations. These garbage inturn can lead to overloading of the
+ * available model context, leading to less valuable response for subsequent prompts/queries,
+ * if chat history is sent to ai model.
+ *
+ * So two simple minded garbage trimming logics are experimented below.
+ * * one based on progressively-larger-substring-based-repeat-matching-with-partial-skip and
+ * * another based on char-histogram-driven garbage trimming.
+ * * in future characteristic of histogram over varying lengths could be used to allow for
+ * a more aggressive and adaptive trimming logic.
+ */
+
+
+/**
+ * Simple minded logic to help remove repeating garbage at end of the string.
+ * The repeatation needs to be perfectly matching.
+ *
+ * The logic progressively goes on probing for longer and longer substring based
+ * repeatation, till there is no longer repeatation. Inturn picks the one with
+ * the longest chain.
+ *
+ * @param {string} sIn
+ * @param {number} maxSubL
+ * @param {number} maxMatchLenThreshold
+ */
+export function trim_repeat_garbage_at_end(sIn, maxSubL=10, maxMatchLenThreshold=40) {
+ let rCnt = [0];
+ let maxMatchLen = maxSubL;
+ let iMML = -1;
+ for(let subL=1; subL < maxSubL; subL++) {
+ rCnt.push(0);
+ let i;
+ let refS = sIn.substring(sIn.length-subL, sIn.length);
+ for(i=sIn.length; i > 0; i -= subL) {
+ let curS = sIn.substring(i-subL, i);
+ if (refS != curS) {
+ let curMatchLen = rCnt[subL]*subL;
+ if (maxMatchLen < curMatchLen) {
+ maxMatchLen = curMatchLen;
+ iMML = subL;
+ }
+ break;
+ }
+ rCnt[subL] += 1;
+ }
+ }
+ console.debug("DBUG:DU:TrimRepeatGarbage:", rCnt);
+ if ((iMML == -1) || (maxMatchLen < maxMatchLenThreshold)) {
+ return {trimmed: false, data: sIn};
+ }
+ console.debug("DBUG:TrimRepeatGarbage:TrimmedCharLen:", maxMatchLen);
+ let iEnd = sIn.length - maxMatchLen;
+ return { trimmed: true, data: sIn.substring(0, iEnd) };
+}
+
+
+/**
+ * Simple minded logic to help remove repeating garbage at end of the string, till it cant.
+ * If its not able to trim, then it will try to skip a char at end and then trim, a few times.
+ * This ensures that even if there are multiple runs of garbage with different patterns, the
+ * logic still tries to munch through them.
+ *
+ * @param {string} sIn
+ * @param {number} maxSubL
+ * @param {number | undefined} [maxMatchLenThreshold]
+ */
+export function trim_repeat_garbage_at_end_loop(sIn, maxSubL, maxMatchLenThreshold, skipMax=16) {
+ let sCur = sIn;
+ let sSaved = "";
+ let iTry = 0;
+ while(true) {
+ let got = trim_repeat_garbage_at_end(sCur, maxSubL, maxMatchLenThreshold);
+ if (got.trimmed != true) {
+ if (iTry == 0) {
+ sSaved = got.data;
+ }
+ iTry += 1;
+ if (iTry >= skipMax) {
+ return sSaved;
+ }
+ got.data = got.data.substring(0,got.data.length-1);
+ } else {
+ iTry = 0;
+ }
+ sCur = got.data;
+ }
+}
+
+
+/**
+ * A simple minded try trim garbage at end using histogram driven characteristics.
+ * There can be variation in the repeatations, as long as no new char props up.
+ *
+ * This tracks the chars and their frequency in a specified length of substring at the end
+ * and inturn checks if moving further into the generated text from the end remains within
+ * the same char subset or goes beyond it and based on that either trims the string at the
+ * end or not. This allows to filter garbage at the end, including even if there are certain
+ * kind of small variations in the repeated text wrt position of seen chars.
+ *
+ * Allow the garbage to contain upto maxUniq chars, but at the same time ensure that
+ * a given type of char ie numerals or alphabets or other types dont cross the specified
+ * maxType limit. This allows intermixed text garbage to be identified and trimmed.
+ *
+ * ALERT: This is not perfect and only provides a rough garbage identification logic.
+ * Also it currently only differentiates between character classes wrt english.
+ *
+ * @param {string} sIn
+ * @param {number} maxType
+ * @param {number} maxUniq
+ * @param {number} maxMatchLenThreshold
+ */
+export function trim_hist_garbage_at_end(sIn, maxType, maxUniq, maxMatchLenThreshold) {
+ if (sIn.length < maxMatchLenThreshold) {
+ return { trimmed: false, data: sIn };
+ }
+ let iAlp = 0;
+ let iNum = 0;
+ let iOth = 0;
+ // Learn
+ let hist = {};
+ let iUniq = 0;
+ for(let i=0; i= maxUniq) {
+ break;
+ }
+ hist[c] = 1;
+ }
+ }
+ console.debug("DBUG:TrimHistGarbage:", hist);
+ if ((iAlp > maxType) || (iNum > maxType) || (iOth > maxType)) {
+ return { trimmed: false, data: sIn };
+ }
+ // Catch and Trim
+ for(let i=0; i < sIn.length; i++) {
+ let c = sIn[sIn.length-1-i];
+ if (!(c in hist)) {
+ if (i < maxMatchLenThreshold) {
+ return { trimmed: false, data: sIn };
+ }
+ console.debug("DBUG:TrimHistGarbage:TrimmedCharLen:", i);
+ return { trimmed: true, data: sIn.substring(0, sIn.length-i+1) };
+ }
+ }
+ console.debug("DBUG:TrimHistGarbage:Trimmed fully");
+ return { trimmed: true, data: "" };
+}
+
+/**
+ * Keep trimming repeatedly using hist_garbage logic, till you no longer can.
+ * This ensures that even if there are multiple runs of garbage with different patterns,
+ * the logic still tries to munch through them.
+ *
+ * @param {any} sIn
+ * @param {number} maxType
+ * @param {number} maxUniq
+ * @param {number} maxMatchLenThreshold
+ */
+export function trim_hist_garbage_at_end_loop(sIn, maxType, maxUniq, maxMatchLenThreshold) {
+ let sCur = sIn;
+ while (true) {
+ let got = trim_hist_garbage_at_end(sCur, maxType, maxUniq, maxMatchLenThreshold);
+ if (!got.trimmed) {
+ return got.data;
+ }
+ sCur = got.data;
+ }
+}
+
+/**
+ * Try trim garbage at the end by using both the hist-driven-garbage-trimming as well as
+ * skip-a-bit-if-reqd-then-repeat-pattern-based-garbage-trimming, with blind retrying.
+ * @param {string} sIn
+ */
+export function trim_garbage_at_end(sIn) {
+ let sCur = sIn;
+ for(let i=0; i<2; i++) {
+ sCur = trim_hist_garbage_at_end_loop(sCur, 8, 24, 72);
+ sCur = trim_repeat_garbage_at_end_loop(sCur, 32, 72, 12);
+ }
+ return sCur;
+}
+
+
+/**
+ * NewLines array helper.
+ * Allow for maintaining a list of lines.
+ * Allow for a line to be builtup/appended part by part.
+ */
+export class NewLines {
+
+ constructor() {
+ /** @type {string[]} */
+ this.lines = [];
+ }
+
+ /**
+ * Extracts lines from the passed string and inturn either
+ * append to a previous partial line or add a new line.
+ * @param {string} sLines
+ */
+ add_append(sLines) {
+ let aLines = sLines.split("\n");
+ let lCnt = 0;
+ for(let line of aLines) {
+ lCnt += 1;
+ // Add back newline removed if any during split
+ if (lCnt < aLines.length) {
+ line += "\n";
+ } else {
+ if (sLines.endsWith("\n")) {
+ line += "\n";
+ }
+ }
+ // Append if required
+ if (lCnt == 1) {
+ let lastLine = this.lines[this.lines.length-1];
+ if (lastLine != undefined) {
+ if (!lastLine.endsWith("\n")) {
+ this.lines[this.lines.length-1] += line;
+ continue;
+ }
+ }
+ }
+ // Add new line
+ this.lines.push(line);
+ }
+ }
+
+ /**
+ * Shift the oldest/earliest/0th line in the array. [Old-New|Earliest-Latest]
+ * Optionally control whether only full lines (ie those with newline at end) will be returned
+ * or will a partial line without a newline at end (can only be the last line) be returned.
+ * @param {boolean} bFullWithNewLineOnly
+ */
+ shift(bFullWithNewLineOnly=true) {
+ let line = this.lines[0];
+ if (line == undefined) {
+ return undefined;
+ }
+ if ((line[line.length-1] != "\n") && bFullWithNewLineOnly){
+ return undefined;
+ }
+ return this.lines.shift();
+ }
+
+}
diff --git a/examples/server/public_simplechat/index.html b/examples/server/public_simplechat/index.html
index 1a1a34208..f6413016f 100644
--- a/examples/server/public_simplechat/index.html
+++ b/examples/server/public_simplechat/index.html
@@ -8,21 +8,23 @@
-
+
+
-
+
SimpleChat
-
-
-
-
+
@@ -30,7 +32,7 @@
-
+
@@ -40,7 +42,7 @@
-
+
diff --git a/examples/server/public_simplechat/readme.md b/examples/server/public_simplechat/readme.md
index de0dfc99d..36a46885d 100644
--- a/examples/server/public_simplechat/readme.md
+++ b/examples/server/public_simplechat/readme.md
@@ -11,18 +11,29 @@ in a simple way with minimal code from a common code base. Inturn additionally i
multiple independent back and forth chatting to an extent, with the ai llm model at a basic level, with their
own system prompts.
+This allows seeing the generated text / ai-model response in oneshot at the end, after it is fully generated,
+or potentially as it is being generated, in a streamed manner from the server/ai-model.
+
+Auto saves the chat session locally as and when the chat is progressing and inturn at a later time when you
+open SimpleChat, option is provided to restore the old chat session, if a matching one exists.
+
The UI follows a responsive web design so that the layout can adapt to available display space in a usable
enough manner, in general.
Allows developer/end-user to control some of the behaviour by updating gMe members from browser's devel-tool
-console.
+console. Parallely some of the directly useful to end-user settings can also be changed using the provided
+settings ui.
-NOTE: Given that the idea is for basic minimal testing, it doesnt bother with any model context length and
-culling of old messages from the chat by default. However by enabling the sliding window chat logic, a crude
-form of old messages culling can be achieved.
+NOTE: Current web service api doesnt expose the model context length directly, so client logic doesnt provide
+any adaptive culling of old messages nor of replacing them with summary of their content etal. However there
+is a optional sliding window based chat logic, which provides a simple minded culling of old messages from
+the chat history before sending to the ai model.
-NOTE: It doesnt set any parameters other than temperature and max_tokens for now. However if someone wants
-they can update the js file or equivalent member in gMe as needed.
+NOTE: Wrt options sent with the request, it mainly sets temperature, max_tokens and optionaly stream for now.
+However if someone wants they can update the js file or equivalent member in gMe as needed.
+
+NOTE: One may be able to use this to chat with openai api web-service /chat/completions endpoint, in a very
+limited / minimal way. One will need to set model, openai url and authorization bearer key in settings ui.
## usage
@@ -52,9 +63,15 @@ Open this simple web front end from your local browser
Once inside
-* Select between chat and completion mode. By default it is set to chat mode.
+* If you want to, you can change many of the default global settings
+ * the base url (ie ip addr / domain name, port)
+ * chat (default) vs completion mode
+ * try trim garbage in response or not
+ * amount of chat history in the context sent to server/ai-model
+ * oneshot or streamed mode.
* In completion mode
+ * one normally doesnt use a system prompt in completion mode.
* logic by default doesnt insert any role specific "ROLE: " prefix wrt each role's message.
If the model requires any prefix wrt user role messages, then the end user has to
explicitly add the needed prefix, when they enter their chat message.
@@ -88,12 +105,16 @@ Once inside
* Wait for the logic to communicate with the server and get the response.
* the user is not allowed to enter any fresh query during this time.
* the user input box will be disabled and a working message will be shown in it.
+ * if trim garbage is enabled, the logic will try to trim repeating text kind of garbage to some extent.
* just refresh the page, to reset wrt the chat history and or system prompt and start afresh.
* Using NewChat one can start independent chat sessions.
* two independent chat sessions are setup by default.
+* When you want to print, switching ChatHistoryInCtxt to Full and clicking on the chat session button of
+ interest, will display the full chat history till then wrt same, if you want full history for printing.
+
## Devel note
@@ -104,14 +125,31 @@ by developers who may not be from web frontend background (so inturn may not be
end-use-specific-language-extensions driven flows) so that they can use it to explore/experiment things.
And given that the idea is also to help explore/experiment for developers, some flexibility is provided
-to change behaviour easily using the devel-tools/console, for now. And skeletal logic has been implemented
-to explore some of the end points and ideas/implications around them.
+to change behaviour easily using the devel-tools/console or provided minimal settings ui (wrt few aspects).
+Skeletal logic has been implemented to explore some of the end points and ideas/implications around them.
### General
Me/gMe consolidates the settings which control the behaviour into one object.
One can see the current settings, as well as change/update them using browsers devel-tool/console.
+It is attached to the document object. Some of these can also be updated using the Settings UI.
+
+ baseURL - the domain-name/ip-address and inturn the port to send the request.
+
+ bStream - control between oneshot-at-end and live-stream-as-its-generated collating and showing
+ of the generated response.
+
+ the logic assumes that the text sent from the server follows utf-8 encoding.
+
+ in streaming mode - if there is any exception, the logic traps the same and tries to ensure
+ that text generated till then is not lost.
+
+ if a very long text is being generated, which leads to no user interaction for sometime and
+ inturn the machine goes into power saving mode or so, the platform may stop network connection,
+ leading to exception.
+
+ apiEP - select between /completions and /chat/completions endpoint provided by the server/ai-model.
bCompletionFreshChatAlways - whether Completion mode collates complete/sliding-window history when
communicating with the server or only sends the latest user query/message.
@@ -119,6 +157,19 @@ One can see the current settings, as well as change/update them using browsers d
bCompletionInsertStandardRolePrefix - whether Completion mode inserts role related prefix wrt the
messages that get inserted into prompt field wrt /Completion endpoint.
+ bTrimGarbage - whether garbage repeatation at the end of the generated ai response, should be
+ trimmed or left as is. If enabled, it will be trimmed so that it wont be sent back as part of
+ subsequent chat history. At the same time the actual trimmed text is shown to the user, once
+ when it was generated, so user can check if any useful info/data was there in the response.
+
+ One may be able to request the ai-model to continue (wrt the last response) (if chat-history
+ is enabled as part of the chat-history-in-context setting), and chances are the ai-model will
+ continue starting from the trimmed part, thus allows long response to be recovered/continued
+ indirectly, in many cases.
+
+ The histogram/freq based trimming logic is currently tuned for english language wrt its
+ is-it-a-alpabetic|numeral-char regex match logic.
+
chatRequestOptions - maintains the list of options/fields to send along with chat request,
irrespective of whether /chat/completions or /completions endpoint.
@@ -126,6 +177,14 @@ One can see the current settings, as well as change/update them using browsers d
modify the existing options value or remove them, for now you can update this global var
using browser's development-tools/console.
+ For string and numeric fields in chatRequestOptions, including even those added by a user
+ at runtime by directly modifying gMe.chatRequestOptions, setting ui entries will be auto
+ created.
+
+ headers - maintains the list of http headers sent when request is made to the server. By default
+ Content-Type is set to application/json. Additionally Authorization entry is provided, which can
+ be set if needed using the settings ui.
+
iRecentUserMsgCnt - a simple minded SlidingWindow to limit context window load at Ai Model end.
This is disabled by default. However if enabled, then in addition to latest system message, only
the last/latest iRecentUserMsgCnt user messages after the latest system prompt and its responses
@@ -140,7 +199,8 @@ One can see the current settings, as well as change/update them using browsers d
By using gMe's iRecentUserMsgCnt and chatRequestOptions.max_tokens one can try to control the
implications of loading of the ai-model's context window by chat history, wrt chat response to
-some extent in a simple crude way.
+some extent in a simple crude way. You may also want to control the context size enabled when
+the server loads ai-model, on the server end.
Sometimes the browser may be stuborn with caching of the file, so your updates to html/css/js
@@ -149,28 +209,15 @@ matter clearing site data, dont directly override site caching in all cases. Wor
have to change port. Or in dev tools of browser, you may be able to disable caching fully.
-Concept of multiple chat sessions with different servers, as well as saving and restoring of
-those across browser usage sessions, can be woven around the SimpleChat/MultiChatUI class and
-its instances relatively easily, however given the current goal of keeping this simple, it has
-not been added, for now.
+Currently the server to communicate with is maintained globally and not as part of a specific
+chat session. So if one changes the server ip/url in setting, then all chat sessions will auto
+switch to this new server, when you try using those sessions.
By switching between chat.add_system_begin/anytime, one can control whether one can change
the system prompt, anytime during the conversation or only at the beginning.
-read_json_early, is to experiment with reading json response data early on, if available,
-so that user can be shown generated data, as and when it is being generated, rather than
-at the end when full data is available.
-
- the server flow doesnt seem to be sending back data early, atleast for request (inc options)
- that is currently sent.
-
- if able to read json data early on in future, as and when ai model is generating data, then
- this helper needs to indirectly update the chat div with the recieved data, without waiting
- for the overall data to be available.
-
-
### Default setup
By default things are setup to try and make the user experience a bit better, if possible.
@@ -179,7 +226,8 @@ However a developer when testing the server of ai-model may want to change these
Using iRecentUserMsgCnt reduce chat history context sent to the server/ai-model to be
just the system-prompt, prev-user-request-and-ai-response and cur-user-request, instead of
full chat history. This way if there is any response with garbage/repeatation, it doesnt
-mess with things beyond the next question/request/query, in some ways.
+mess with things beyond the next question/request/query, in some ways. The trim garbage
+option also tries to help avoid issues with garbage in the context to an extent.
Set max_tokens to 1024, so that a relatively large previous reponse doesnt eat up the space
available wrt next query-response. However dont forget that the server when started should
@@ -189,11 +237,33 @@ also be started with a model context size of 1k or more, to be on safe side.
internal n_predict, for now add the same here on the client side, maybe later add max_tokens
to /completions endpoint handling code on server side.
-Frequency and presence penalty fields are set to 1.2 in the set of fields sent to server
-along with the user query. So that the model is partly set to try avoid repeating text in
-its response.
+NOTE: One may want to experiment with frequency/presence penalty fields in chatRequestOptions
+wrt the set of fields sent to server along with the user query. To check how the model behaves
+wrt repeatations in general in the generated text response.
-A end-user can change these behaviour by editing gMe from browser's devel-tool/console.
+A end-user can change these behaviour by editing gMe from browser's devel-tool/console or by
+using the providing settings ui.
+
+
+### OpenAi / Equivalent API WebService
+
+One may be abe to handshake with OpenAI/Equivalent api web service's /chat/completions endpoint
+for a minimal chatting experimentation by setting the below.
+
+* the baseUrl in settings ui
+ * https://api.openai.com/v1 or similar
+
+* Wrt request body - gMe.chatRequestOptions
+ * model (settings ui)
+ * any additional fields if required in future
+
+* Wrt request headers - gMe.headers
+ * Authorization (available through settings ui)
+ * Bearer THE_OPENAI_API_KEY
+ * any additional optional header entries like "OpenAI-Organization", "OpenAI-Project" or so
+
+NOTE: Not tested, as there is no free tier api testing available. However logically this might
+work.
## At the end
diff --git a/examples/server/public_simplechat/simplechat.css b/examples/server/public_simplechat/simplechat.css
index 20c738b12..13bfb80b4 100644
--- a/examples/server/public_simplechat/simplechat.css
+++ b/examples/server/public_simplechat/simplechat.css
@@ -21,6 +21,17 @@
.role-user {
background-color: lightgray;
}
+.role-trim {
+ background-color: lightpink;
+}
+
+.gridx2 {
+ display: grid;
+ grid-template-columns: repeat(2, 1fr);
+ border-bottom-style: dotted;
+ border-bottom-width: thin;
+ border-bottom-color: lightblue;
+}
.flex-grow {
flex-grow: 1;
diff --git a/examples/server/public_simplechat/simplechat.js b/examples/server/public_simplechat/simplechat.js
index 0c48da879..25afb2564 100644
--- a/examples/server/public_simplechat/simplechat.js
+++ b/examples/server/public_simplechat/simplechat.js
@@ -2,6 +2,9 @@
// A simple completions and chat/completions test related web front end logic
// by Humans for All
+import * as du from "./datautils.mjs";
+import * as ui from "./ui.mjs"
+
class Roles {
static System = "system";
static User = "user";
@@ -9,40 +12,65 @@ class Roles {
}
class ApiEP {
- static Chat = "chat";
- static Completion = "completion";
+ static Type = {
+ Chat: "chat",
+ Completion: "completion",
+ }
+ static UrlSuffix = {
+ 'chat': `/chat/completions`,
+ 'completion': `/completions`,
+ }
+
+ /**
+ * Build the url from given baseUrl and apiEp id.
+ * @param {string} baseUrl
+ * @param {string} apiEP
+ */
+ static Url(baseUrl, apiEP) {
+ if (baseUrl.endsWith("/")) {
+ baseUrl = baseUrl.substring(0, baseUrl.length-1);
+ }
+ return `${baseUrl}${this.UrlSuffix[apiEP]}`;
+ }
+
}
+
let gUsageMsg = `
Usage
-
Set system prompt above, to try control ai response charactersitic, if model supports same.
+
System prompt above, to try control ai response characteristics.
-
Completion mode normally wont have a system prompt.