From 6337e0bbd6fc0914beca01919d95e0abcd7accca Mon Sep 17 00:00:00 2001 From: LenAnderson Date: Fri, 15 Dec 2023 12:38:26 +0000 Subject: [PATCH 1/7] add /times slash command --- public/scripts/variables.js | 15 +++++++++++++++ 1 file changed, 15 insertions(+) diff --git a/public/scripts/variables.js b/public/scripts/variables.js index 1eecb3b77..48bcb763d 100644 --- a/public/scripts/variables.js +++ b/public/scripts/variables.js @@ -319,6 +319,20 @@ async function whileCallback(args, command) { return ''; } +async function timesCallback(args, value) { + const [repeats, ...commandParts] = value.split(' '); + const command = commandParts.join(' '); + const MAX_LOOPS = 100; + const isGuardOff = ['off', 'false', '0'].includes(args.guard?.toLowerCase()); + const iterations = Math.min(Number(repeats), isGuardOff ? Number.MAX_SAFE_INTEGER : MAX_LOOPS); + + for (let i = 0; i < iterations; i++) { + await executeSubCommands(command.replace(/\{\{timesIndex\}\}/g, i)); + } + + return ''; +} + async function ifCallback(args, command) { const { a, b, rule } = parseBooleanOperands(args); const result = evalBoolean(rule, a, b); @@ -656,6 +670,7 @@ export function registerVariableCommands() { registerSlashCommand('decglobalvar', (_, value) => decrementGlobalVariable(value), [], '(key) – decrement a global variable by 1 and pass the result down the pipe, e.g. /decglobalvar score', true, true); registerSlashCommand('if', ifCallback, [], 'left=varname1 right=varname2 rule=comparison else="(alt.command)" "(command)" – 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. /if left=score right=10 rule=gte "/speak You win" triggers a /speak command if the value of "score" is greater or equals 10.', true, true); registerSlashCommand('while', whileCallback, [], 'left=varname1 right=varname2 rule=comparison "(command)" – 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. /setvar key=i 0 | /while left=i right=10 rule=let "/addvar key=i 1" 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('times', (args, value) => timesCallback(args, value), [], '(repeats) "(command)" – execute any valid slash command enclosed in quotes repeats number of times, e.g. /setvar key=i 1 | /times 5 "/addvar key=i 1" adds 1 to the value of "i" 5 times. {{timesIndex}} is replaced with the iteration number (zero-based), e.g. /times 4 "/echo {{timesIndex}}" echos the numbers 0 through 4. Loops are limited to 100 iterations by default, pass guard=off to disable.', true, true); registerSlashCommand('flushvar', (_, value) => deleteLocalVariable(value), [], '(key) – delete a local variable, e.g. /flushvar score', true, true); registerSlashCommand('flushglobalvar', (_, value) => deleteGlobalVariable(value), [], '(key) – delete a global variable, e.g. /flushglobalvar score', true, true); registerSlashCommand('add', (_, value) => addValuesCallback(value), [], '(a b c d) – performs an addition of the set of values and passes the result down the pipe, can use variable names, e.g. /add 10 i 30 j', true, true); From 74d0ef557290651346e75fad906a6b5bf15e5f69 Mon Sep 17 00:00:00 2001 From: LenAnderson Date: Fri, 15 Dec 2023 19:47:40 +0000 Subject: [PATCH 2/7] make MAX_LOOPS module wide --- public/scripts/variables.js | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/public/scripts/variables.js b/public/scripts/variables.js index 48bcb763d..419a6cfe3 100644 --- a/public/scripts/variables.js +++ b/public/scripts/variables.js @@ -2,6 +2,12 @@ import { chat_metadata, getCurrentChatId, saveSettingsDebounced, sendSystemMessa import { extension_settings, saveMetadataDebounced } from './extensions.js'; import { executeSlashCommands, registerSlashCommand } from './slash-commands.js'; + + +const MAX_LOOPS = 100; + + + function getLocalVariable(name, args = {}) { if (!chat_metadata.variables) { chat_metadata.variables = {}; @@ -301,7 +307,6 @@ function listVariablesCallback() { } 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; @@ -322,7 +327,6 @@ async function whileCallback(args, command) { async function timesCallback(args, value) { const [repeats, ...commandParts] = value.split(' '); const command = commandParts.join(' '); - const MAX_LOOPS = 100; const isGuardOff = ['off', 'false', '0'].includes(args.guard?.toLowerCase()); const iterations = Math.min(Number(repeats), isGuardOff ? Number.MAX_SAFE_INTEGER : MAX_LOOPS); From 279b731f645b0a2cdc7e4b3c8ab2f0bd4a889eb0 Mon Sep 17 00:00:00 2001 From: LenAnderson Date: Fri, 15 Dec 2023 19:48:25 +0000 Subject: [PATCH 3/7] use isFalseBoolean in while and times --- public/scripts/variables.js | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/public/scripts/variables.js b/public/scripts/variables.js index 419a6cfe3..673106637 100644 --- a/public/scripts/variables.js +++ b/public/scripts/variables.js @@ -1,6 +1,7 @@ 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'; +import { isFalseBoolean } from './utils.js'; @@ -307,7 +308,7 @@ function listVariablesCallback() { } async function whileCallback(args, command) { - const isGuardOff = ['off', 'false', '0'].includes(args.guard?.toLowerCase()); + const isGuardOff = isFalseBoolean(args.guard); const iterations = isGuardOff ? Number.MAX_SAFE_INTEGER : MAX_LOOPS; for (let i = 0; i < iterations; i++) { @@ -327,7 +328,7 @@ async function whileCallback(args, command) { async function timesCallback(args, value) { const [repeats, ...commandParts] = value.split(' '); const command = commandParts.join(' '); - const isGuardOff = ['off', 'false', '0'].includes(args.guard?.toLowerCase()); + const isGuardOff = isFalseBoolean(args.guard); const iterations = Math.min(Number(repeats), isGuardOff ? Number.MAX_SAFE_INTEGER : MAX_LOOPS); for (let i = 0; i < iterations; i++) { From 9d1b563d485676bf0e0ef6780fee257d32536c39 Mon Sep 17 00:00:00 2001 From: Cohee <18619528+Cohee1207@users.noreply.github.com> Date: Fri, 15 Dec 2023 22:11:48 +0200 Subject: [PATCH 4/7] Add cache for parsed characters --- src/endpoints/characters.js | 25 ++++++++++++++++++++++++- 1 file changed, 24 insertions(+), 1 deletion(-) diff --git a/src/endpoints/characters.js b/src/endpoints/characters.js index f21c29a5f..367fbdf82 100644 --- a/src/endpoints/characters.js +++ b/src/endpoints/characters.js @@ -22,8 +22,25 @@ const { importRisuSprites } = require('./sprites'); let characters = {}; +// KV-store for parsed character data +const characterDataCache = new Map(); + +/** + * Reads the character card from the specified image file. + * @param {string} img_url - Path to the image file + * @param {string} input_format - 'png' + * @returns {Promise} - Character card data + */ async function charaRead(img_url, input_format) { - return characterCardParser.parse(img_url, input_format); + const stat = fs.statSync(img_url); + const cacheKey = `${img_url}-${stat.mtimeMs}`; + if (characterDataCache.has(cacheKey)) { + return characterDataCache.get(cacheKey); + } + + const result = characterCardParser.parse(img_url, input_format); + characterDataCache.set(cacheKey, result); + return result; } /** @@ -32,6 +49,12 @@ async function charaRead(img_url, input_format) { */ async function charaWrite(img_url, data, target_img, response = undefined, mes = 'ok', crop = undefined) { try { + // Reset the cache + for (const key of characterDataCache.keys()) { + if (key.startsWith(img_url)) { + characterDataCache.delete(key); + } + } // Read the image, resize, and save it as a PNG into the buffer const image = await tryReadImage(img_url, crop); From 7dfa989e4c105ec0e856a54d3babaadd46fb018f Mon Sep 17 00:00:00 2001 From: Cohee <18619528+Cohee1207@users.noreply.github.com> Date: Fri, 15 Dec 2023 22:13:02 +0200 Subject: [PATCH 5/7] Bail early from key iteration --- src/endpoints/characters.js | 1 + 1 file changed, 1 insertion(+) diff --git a/src/endpoints/characters.js b/src/endpoints/characters.js index 367fbdf82..9e0956c04 100644 --- a/src/endpoints/characters.js +++ b/src/endpoints/characters.js @@ -53,6 +53,7 @@ async function charaWrite(img_url, data, target_img, response = undefined, mes = for (const key of characterDataCache.keys()) { if (key.startsWith(img_url)) { characterDataCache.delete(key); + break; } } // Read the image, resize, and save it as a PNG into the buffer From 5bda74b88684dfde475a6da35496000229dd0ce6 Mon Sep 17 00:00:00 2001 From: Cohee <18619528+Cohee1207@users.noreply.github.com> Date: Fri, 15 Dec 2023 23:39:02 +0200 Subject: [PATCH 6/7] Too many lines --- public/scripts/variables.js | 4 ---- 1 file changed, 4 deletions(-) diff --git a/public/scripts/variables.js b/public/scripts/variables.js index 673106637..4a9d07351 100644 --- a/public/scripts/variables.js +++ b/public/scripts/variables.js @@ -3,12 +3,8 @@ import { extension_settings, saveMetadataDebounced } from './extensions.js'; import { executeSlashCommands, registerSlashCommand } from './slash-commands.js'; import { isFalseBoolean } from './utils.js'; - - const MAX_LOOPS = 100; - - function getLocalVariable(name, args = {}) { if (!chat_metadata.variables) { chat_metadata.variables = {}; From 3aa950bb45c4e450552259ba27e02d41c24c5f2e Mon Sep 17 00:00:00 2001 From: AliCat <86847834+alicat22@users.noreply.github.com> Date: Fri, 15 Dec 2023 15:01:05 -0700 Subject: [PATCH 7/7] ZenSlider - Default 10 steps to 20 --- public/scripts/power-user.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/public/scripts/power-user.js b/public/scripts/power-user.js index 21806cb99..787df00bb 100644 --- a/public/scripts/power-user.js +++ b/public/scripts/power-user.js @@ -533,7 +533,7 @@ async function CreateZenSliders(elmnt) { var sliderMax = Number(originalSlider.attr('max')); var sliderValue = originalSlider.val(); var sliderRange = sliderMax - sliderMin; - var numSteps = 10; + var numSteps = 20; var decimals = 2; var offVal, allVal; var stepScale;