mirror of
https://github.com/SillyTavern/SillyTavern.git
synced 2024-12-12 17:36:22 +01:00
473 lines
20 KiB
JavaScript
473 lines
20 KiB
JavaScript
import { chat_metadata, getCurrentChatId, saveSettingsDebounced, sendSystemMessage, system_message_types } from "../script.js";
|
||
import { extension_settings, saveMetadataDebounced } from "./extensions.js";
|
||
import { executeSlashCommands, registerSlashCommand } from "./slash-commands.js";
|
||
|
||
function getLocalVariable(name) {
|
||
if (!chat_metadata.variables) {
|
||
chat_metadata.variables = {};
|
||
}
|
||
|
||
const localVariable = chat_metadata?.variables[name];
|
||
|
||
return (localVariable === '' || isNaN(Number(localVariable))) ? (localVariable || '') : Number(localVariable);
|
||
}
|
||
|
||
function setLocalVariable(name, value) {
|
||
if (!chat_metadata.variables) {
|
||
chat_metadata.variables = {};
|
||
}
|
||
|
||
chat_metadata.variables[name] = value;
|
||
saveMetadataDebounced();
|
||
return value;
|
||
}
|
||
|
||
function getGlobalVariable(name) {
|
||
const globalVariable = extension_settings.variables.global[name];
|
||
|
||
return (globalVariable === '' || isNaN(Number(globalVariable))) ? (globalVariable || '') : Number(globalVariable);
|
||
}
|
||
|
||
function setGlobalVariable(name, value) {
|
||
extension_settings.variables.global[name] = value;
|
||
saveSettingsDebounced();
|
||
}
|
||
|
||
function addLocalVariable(name, value) {
|
||
const currentValue = getLocalVariable(name) || 0;
|
||
const increment = Number(value);
|
||
|
||
if (isNaN(increment) || isNaN(Number(currentValue))) {
|
||
const stringValue = String(currentValue || '') + value;
|
||
setLocalVariable(name, stringValue);
|
||
return stringValue;
|
||
}
|
||
|
||
const newValue = Number(currentValue) + increment;
|
||
|
||
if (isNaN(newValue)) {
|
||
return '';
|
||
}
|
||
|
||
setLocalVariable(name, newValue);
|
||
return newValue;
|
||
}
|
||
|
||
function addGlobalVariable(name, value) {
|
||
const currentValue = getGlobalVariable(name) || 0;
|
||
const increment = Number(value);
|
||
|
||
if (isNaN(increment)|| isNaN(Number(currentValue))) {
|
||
const stringValue = String(currentValue || '') + value;
|
||
setGlobalVariable(name, stringValue);
|
||
return stringValue;
|
||
}
|
||
|
||
const newValue = Number(currentValue) + increment;
|
||
|
||
if (isNaN(newValue)) {
|
||
return '';
|
||
}
|
||
|
||
setGlobalVariable(name, newValue);
|
||
return newValue;
|
||
}
|
||
|
||
export function resolveVariable(name) {
|
||
if (existsLocalVariable(name)) {
|
||
return getLocalVariable(name);
|
||
}
|
||
|
||
if (existsGlobalVariable(name)) {
|
||
return getGlobalVariable(name);
|
||
}
|
||
|
||
return name;
|
||
}
|
||
|
||
export function replaceVariableMacros(input) {
|
||
const lines = input.split('\n');
|
||
|
||
for (let i = 0; i < lines.length; i++) {
|
||
let line = lines[i];
|
||
|
||
// Skip lines without macros
|
||
if (!line || !line.includes('{{')) {
|
||
continue;
|
||
}
|
||
|
||
// Replace {{getvar::name}} with the value of the variable name
|
||
line = line.replace(/{{getvar::([^}]+)}}/gi, (_, name) => {
|
||
name = name.trim();
|
||
return getLocalVariable(name);
|
||
});
|
||
|
||
// Replace {{setvar::name::value}} with empty string and set the variable name to value
|
||
line = line.replace(/{{setvar::([^:]+)::([^}]+)}}/gi, (_, name, value) => {
|
||
name = name.trim();
|
||
setLocalVariable(name, value);
|
||
return '';
|
||
});
|
||
|
||
// Replace {{addvar::name::value}} with empty string and add value to the variable value
|
||
line = line.replace(/{{addvar::([^:]+)::([^}]+)}}/gi, (_, name, value) => {
|
||
name = name.trim();
|
||
addLocalVariable(name, value);;
|
||
return '';
|
||
});
|
||
|
||
// Replace {{getglobalvar::name}} with the value of the global variable name
|
||
line = line.replace(/{{getglobalvar::([^}]+)}}/gi, (_, name) => {
|
||
name = name.trim();
|
||
return getGlobalVariable(name);
|
||
});
|
||
|
||
// Replace {{setglobalvar::name::value}} with empty string and set the global variable name to value
|
||
line = line.replace(/{{setglobalvar::([^:]+)::([^}]+)}}/gi, (_, name, value) => {
|
||
name = name.trim();
|
||
setGlobalVariable(name, value);
|
||
return '';
|
||
});
|
||
|
||
// Replace {{addglobalvar::name::value}} with empty string and add value to the global variable value
|
||
line = line.replace(/{{addglobalvar::([^:]+)::([^}]+)}}/gi, (_, name, value) => {
|
||
name = name.trim();
|
||
addGlobalVariable(name, value);
|
||
return '';
|
||
});
|
||
|
||
lines[i] = line;
|
||
}
|
||
|
||
return lines.join('\n');
|
||
}
|
||
|
||
function listVariablesCallback() {
|
||
if (!chat_metadata.variables) {
|
||
chat_metadata.variables = {};
|
||
}
|
||
|
||
const localVariables = Object.entries(chat_metadata.variables).map(([name, value]) => `${name}: ${value}`);
|
||
const globalVariables = Object.entries(extension_settings.variables.global).map(([name, value]) => `${name}: ${value}`);
|
||
|
||
const localVariablesString = localVariables.length > 0 ? localVariables.join('\n\n') : 'No local variables';
|
||
const globalVariablesString = globalVariables.length > 0 ? globalVariables.join('\n\n') : 'No global variables';
|
||
const chatName = getCurrentChatId();
|
||
|
||
const converter = new showdown.Converter();
|
||
const message = `### Local variables (${chatName}):\n${localVariablesString}\n\n### Global variables:\n${globalVariablesString}`;
|
||
const htmlMessage = DOMPurify.sanitize(converter.makeHtml(message));
|
||
|
||
sendSystemMessage(system_message_types.GENERIC, htmlMessage);
|
||
}
|
||
|
||
async function whileCallback(args, command) {
|
||
const MAX_LOOPS = 100;
|
||
const isGuardOff = ['off', 'false', '0'].includes(args.guard?.toLowerCase());
|
||
const iterations = isGuardOff ? Number.MAX_SAFE_INTEGER : MAX_LOOPS;
|
||
|
||
for (let i = 0; i < iterations; i++) {
|
||
const { a, b, rule } = parseBooleanOperands(args);
|
||
const result = evalBoolean(rule, a, b);
|
||
|
||
if (result && command) {
|
||
await executeSubCommands(command);
|
||
} else {
|
||
break;
|
||
}
|
||
}
|
||
|
||
return '';
|
||
}
|
||
|
||
async function ifCallback(args, command) {
|
||
const { a, b, rule } = parseBooleanOperands(args);
|
||
const result = evalBoolean(rule, a, b);
|
||
|
||
if (result && command) {
|
||
return await executeSubCommands(command);
|
||
} else if (!result && args.else && typeof args.else === 'string' && args.else !== '') {
|
||
return await executeSubCommands(args.else);
|
||
}
|
||
|
||
return '';
|
||
}
|
||
|
||
function existsLocalVariable(name) {
|
||
return chat_metadata.variables && chat_metadata.variables[name] !== undefined;
|
||
}
|
||
|
||
function existsGlobalVariable(name) {
|
||
return extension_settings.variables.global && extension_settings.variables.global[name] !== undefined;
|
||
}
|
||
|
||
function parseBooleanOperands(args) {
|
||
// Resultion order: numeric literal, local variable, global variable, string literal
|
||
function getOperand(operand) {
|
||
if (operand === undefined) {
|
||
return '';
|
||
}
|
||
|
||
const operandNumber = Number(operand);
|
||
|
||
if (!isNaN(operandNumber)) {
|
||
return operandNumber;
|
||
}
|
||
|
||
if (existsLocalVariable(operand)) {
|
||
const operandLocalVariable = getLocalVariable(operand);
|
||
return operandLocalVariable ?? '';
|
||
}
|
||
|
||
if (existsGlobalVariable(operand)) {
|
||
const operandGlobalVariable = getGlobalVariable(operand);
|
||
return operandGlobalVariable ?? '';
|
||
}
|
||
|
||
const stringLiteral = String(operand);
|
||
return stringLiteral || '';
|
||
}
|
||
|
||
const left = getOperand(args.a || args.left || args.first || args.x);
|
||
const right = getOperand(args.b || args.right || args.second || args.y);
|
||
const rule = args.rule;
|
||
|
||
return { a: left, b: right, rule };
|
||
}
|
||
|
||
function evalBoolean(rule, a, b) {
|
||
if (!rule) {
|
||
toastr.warning('The rule must be specified for the boolean comparison.', 'Invalid command');
|
||
throw new Error('Invalid command.');
|
||
}
|
||
|
||
let result = false;
|
||
|
||
if (typeof a === 'string' && typeof b !== 'number') {
|
||
const aString = String(a).toLowerCase();
|
||
const bString = String(b).toLowerCase();
|
||
|
||
switch (rule) {
|
||
case 'in':
|
||
result = aString.includes(bString);
|
||
break;
|
||
case 'nin':
|
||
result = !aString.includes(bString);
|
||
break;
|
||
case 'eq':
|
||
result = aString === bString;
|
||
break;
|
||
case 'neq':
|
||
result = aString !== bString;
|
||
break;
|
||
default:
|
||
toastr.error('Unknown boolean comparison rule for type string.', 'Invalid /if command');
|
||
throw new Error('Invalid command.');
|
||
}
|
||
} else if (typeof a === 'number') {
|
||
const aNumber = Number(a);
|
||
const bNumber = Number(b);
|
||
|
||
switch (rule) {
|
||
case 'not':
|
||
result = !aNumber;
|
||
break;
|
||
case 'gt':
|
||
result = aNumber > bNumber;
|
||
break;
|
||
case 'gte':
|
||
result = aNumber >= bNumber;
|
||
break;
|
||
case 'lt':
|
||
result = aNumber < bNumber;
|
||
break;
|
||
case 'lte':
|
||
result = aNumber <= bNumber;
|
||
break;
|
||
case 'eq':
|
||
result = aNumber === bNumber;
|
||
break;
|
||
case 'neq':
|
||
result = aNumber !== bNumber;
|
||
break;
|
||
default:
|
||
toastr.error('Unknown boolean comparison rule for type number.', 'Invalid command');
|
||
throw new Error('Invalid command.');
|
||
}
|
||
}
|
||
|
||
return result;
|
||
}
|
||
|
||
async function executeSubCommands(command) {
|
||
if (command.startsWith('"')) {
|
||
command = command.slice(1);
|
||
}
|
||
|
||
if (command.endsWith('"')) {
|
||
command = command.slice(0, -1);
|
||
}
|
||
|
||
const unescape = true;
|
||
const result = await executeSlashCommands(command, unescape);
|
||
|
||
if (!result || typeof result !== 'object') {
|
||
return '';
|
||
}
|
||
|
||
return result?.pipe || '';
|
||
}
|
||
|
||
function deleteLocalVariable(name) {
|
||
if (!existsLocalVariable(name)) {
|
||
console.warn(`The local variable "${name}" does not exist.`);
|
||
return '';
|
||
}
|
||
|
||
delete chat_metadata.variables[name];
|
||
saveMetadataDebounced();
|
||
return '';
|
||
}
|
||
|
||
function deleteGlobalVariable(name) {
|
||
if (!existsGlobalVariable(name)) {
|
||
console.warn(`The global variable "${name}" does not exist.`);
|
||
return '';
|
||
}
|
||
|
||
delete extension_settings.variables.global[name];
|
||
saveSettingsDebounced();
|
||
return '';
|
||
}
|
||
|
||
function parseNumericSeries(value) {
|
||
if (typeof value === 'number') {
|
||
return [value];
|
||
}
|
||
|
||
const array = value
|
||
.split(' ')
|
||
.map(i => i.trim())
|
||
.filter(i => i !== '')
|
||
.map(i => isNaN(Number(i)) ? resolveVariable(i) : Number(i))
|
||
.filter(i => !isNaN(i));
|
||
|
||
return array;
|
||
}
|
||
|
||
function performOperation(value, operation, singleOperand = false) {
|
||
if (!value) {
|
||
return 0;
|
||
}
|
||
|
||
const array = parseNumericSeries(value);
|
||
|
||
if (array.length === 0) {
|
||
return 0;
|
||
}
|
||
|
||
const result = singleOperand ? operation(array[0]) : operation(array);
|
||
|
||
if (isNaN(result) || !isFinite(result)) {
|
||
return 0;
|
||
}
|
||
|
||
return result;
|
||
}
|
||
|
||
function addValuesCallback(value) {
|
||
return performOperation(value, (array) => array.reduce((a, b) => a + b, 0));
|
||
}
|
||
|
||
function mulValuesCallback(value) {
|
||
return performOperation(value, (array) => array.reduce((a, b) => a * b, 1));
|
||
}
|
||
|
||
function minValuesCallback(value) {
|
||
return performOperation(value, (array) => Math.min(...array));
|
||
}
|
||
|
||
function maxValuesCallback(value) {
|
||
return performOperation(value, (array) => Math.max(...array));
|
||
}
|
||
|
||
function subValuesCallback(value) {
|
||
return performOperation(value, (array) => array[0] - array[1]);
|
||
}
|
||
|
||
function divValuesCallback(value) {
|
||
return performOperation(value, (array) => {
|
||
if (array[1] === 0) {
|
||
console.warn('Division by zero.');
|
||
return 0;
|
||
}
|
||
return array[0] / array[1];
|
||
});
|
||
}
|
||
|
||
function modValuesCallback(value) {
|
||
return performOperation(value, (array) => {
|
||
if (array[1] === 0) {
|
||
console.warn('Division by zero.');
|
||
return 0;
|
||
}
|
||
return array[0] % array[1];
|
||
});
|
||
}
|
||
|
||
function powValuesCallback(value) {
|
||
return performOperation(value, (array) => Math.pow(array[0], array[1]));
|
||
}
|
||
|
||
function sinValuesCallback(value) {
|
||
return performOperation(value, Math.sin, true);
|
||
}
|
||
|
||
function cosValuesCallback(value) {
|
||
return performOperation(value, Math.cos, true);
|
||
}
|
||
|
||
function logValuesCallback(value) {
|
||
return performOperation(value, Math.log, true);
|
||
}
|
||
|
||
function roundValuesCallback(value) {
|
||
return performOperation(value, Math.round, true);
|
||
}
|
||
|
||
function absValuesCallback(value) {
|
||
return performOperation(value, Math.abs, true);
|
||
}
|
||
|
||
function sqrtValuesCallback(value) {
|
||
return performOperation(value, Math.sqrt, true);
|
||
}
|
||
|
||
export function registerVariableCommands() {
|
||
registerSlashCommand('listvar', listVariablesCallback, [], ' – list registered chat variables', true, true);
|
||
registerSlashCommand('setvar', (args, value) => setLocalVariable(args.key || args.name, value), [], '<span class="monospace">key=varname (value)</span> – set a local variable value and pass it down the pipe, e.g. <tt>/setvar key=color green</tt>', true, true);
|
||
registerSlashCommand('getvar', (_, value) => getLocalVariable(value), [], '<span class="monospace">(key)</span> – get a local variable value and pass it down the pipe, e.g. <tt>/getvar height</tt>', true, true);
|
||
registerSlashCommand('addvar', (args, value) => addLocalVariable(args.key || args.name, value), [], '<span class="monospace">key=varname (increment)</span> – add a value to a local variable and pass the result down the pipe, e.g. <tt>/addvar score 10</tt>', true, true);
|
||
registerSlashCommand('setglobalvar', (args, value) => setGlobalVariable(args.key || args.name, value), [], '<span class="monospace">key=varname (value)</span> – set a global variable value and pass it down the pipe, e.g. <tt>/setglobalvar key=color green</tt>', true, true);
|
||
registerSlashCommand('getglobalvar', (_, value) => getGlobalVariable(value), [], '<span class="monospace">(key)</span> – get a global variable value and pass it down the pipe, e.g. <tt>/getglobalvar height</tt>', true, true);
|
||
registerSlashCommand('addglobalvar', (args, value) => addGlobalVariable(args.key || args.name, value), [], '<span class="monospace">key=varname (increment)</span> – add a value to a global variable and pass the result down the pipe, e.g. <tt>/addglobalvar score 10</tt>', true, true);
|
||
registerSlashCommand('if', ifCallback, [], '<span class="monospace">left=varname1 right=varname2 rule=comparison else="(alt.command)" "(command)"</span> – compare the value of the left operand "a" with the value of the right operand "b", and if the condition yields true, then execute any valid slash command enclosed in quotes and pass the result of the command execution down the pipe. Numeric values and string literals for left and right operands supported. Available rules: gt => a > b, gte => a >= b, lt => a < b, lte => a <= b, eq => a == b, neq => a != b, not => !a, in (strings) => a includes b, nin (strings) => a not includes b, e.g. <tt>/if left=score right=10 rule=gte "/speak You win"</tt> triggers a /speak command if the value of "score" is greater or equals 10.', true, true);
|
||
registerSlashCommand('while', whileCallback, [], '<span class="monospace">left=varname1 right=varname2 rule=comparison "(command)"</span> – compare the value of the left operand "a" with the value of the right operand "b", and if the condition yields true, then execute any valid slash command enclosed in quotes. Numeric values and string literals for left and right operands supported. Available rules: gt => a > b, gte => a >= b, lt => a < b, lte => a <= b, eq => a == b, neq => a != b, not => !a, in (strings) => a includes b, nin (strings) => a not includes b, e.g. <tt>/setvar key=i 0 | /while left=i right=10 rule=let "/addvar key=i 1"</tt> adds 1 to the value of "i" until it reaches 10. Loops are limited to 100 iterations by default, pass guard=off to disable.', true, true);
|
||
registerSlashCommand('flushvar', (_, value) => deleteLocalVariable(value), [], '<span class="monospace">(key)</span> – delete a local variable, e.g. <tt>/flushvar score</tt>', true, true);
|
||
registerSlashCommand('flushglobalvar', (_, value) => deleteGlobalVariable(value), [], '<span class="monospace">(key)</span> – delete a global variable, e.g. <tt>/flushglobalvar score</tt>', true, true);
|
||
registerSlashCommand('add', (_, value) => addValuesCallback(value), [], '<span class="monospace">(a b c d)</span> – performs an addition of the set of values and passes the result down the pipe, can use variable names, e.g. <tt>/add 10 i 30 j</tt>', true, true);
|
||
registerSlashCommand('mul', (_, value) => mulValuesCallback(value), [], '<span class="monospace">(a b c d)</span> – performs a multiplication of the set of values and passes the result down the pipe, can use variable names, e.g. <tt>/mul 10 i 30 j</tt>', true, true);
|
||
registerSlashCommand('max', (_, value) => maxValuesCallback(value), [], '<span class="monospace">(a b c d)</span> – returns the maximum value of the set of values and passes the result down the pipe, can use variable names, e.g. <tt>/max 10 i 30 j</tt>', true, true);
|
||
registerSlashCommand('min', (_, value) => minValuesCallback(value), [], '<span class="monospace">(a b c d)</span> – returns the minimum value of the set of values and passes the result down the pipe, can use variable names, e.g. <tt>/min 10 i 30 j</tt>', true, true);
|
||
registerSlashCommand('sub', (_, value) => subValuesCallback(value), [], '<span class="monospace">(a b)</span> – performs a subtraction of two values and passes the result down the pipe, can use variable names, e.g. <tt>/sub i 5</tt>', true, true);
|
||
registerSlashCommand('div', (_, value) => divValuesCallback(value), [], '<span class="monospace">(a b)</span> – performs a division of two values and passes the result down the pipe, can use variable names, e.g. <tt>/div 10 i</tt>', true, true);
|
||
registerSlashCommand('mod', (_, value) => modValuesCallback(value), [], '<span class="monospace">(a b)</span> – performs a modulo operation of two values and passes the result down the pipe, can use variable names, e.g. <tt>/mod i 2</tt>', true, true);
|
||
registerSlashCommand('pow', (_, value) => powValuesCallback(value), [], '<span class="monospace">(a b)</span> – performs a power operation of two values and passes the result down the pipe, can use variable names, e.g. <tt>/pow i 2</tt>', true, true);
|
||
registerSlashCommand('sin', (_, value) => sinValuesCallback(value), [], '<span class="monospace">(a)</span> – performs a sine operation of a value and passes the result down the pipe, can use variable names, e.g. <tt>/sin i</tt>', true, true);
|
||
registerSlashCommand('cos', (_, value) => cosValuesCallback(value), [], '<span class="monospace">(a)</span> – performs a cosine operation of a value and passes the result down the pipe, can use variable names, e.g. <tt>/cos i</tt>', true, true);
|
||
registerSlashCommand('log', (_, value) => logValuesCallback(value), [], '<span class="monospace">(a)</span> – performs a logarithm operation of a value and passes the result down the pipe, can use variable names, e.g. <tt>/log i</tt>', true, true);
|
||
registerSlashCommand('abs', (_, value) => absValuesCallback(value), [], '<span class="monospace">(a)</span> – performs an absolute value operation of a value and passes the result down the pipe, can use variable names, e.g. <tt>/abs i</tt>', true, true);
|
||
registerSlashCommand('sqrt', (_, value) => sqrtValuesCallback(value), [], '<span class="monospace">(a)</span> – performs a square root operation of a value and passes the result down the pipe, can use variable names, e.g. <tt>/sqrt i</tt>', true, true);
|
||
registerSlashCommand('round', (_, value) => roundValuesCallback(value), [], '<span class="monospace">(a)</span> – rounds a value and passes the result down the pipe, can use variable names, e.g. <tt>/round i</tt>', true, true);
|
||
}
|