import DOMPurify from 'isomorphic-dompurify'; import markdownit from 'markdown-it' import { getCurrentCharacter, type Database, type Message, type character, type customscript, type groupChat, type triggerscript } from './storage/database.svelte'; import { DBState } from './stores.svelte'; import { getFileSrc } from './globalApi.svelte'; import { processScriptFull } from './process/scripts'; import { get } from 'svelte/store'; import css, { type CssAtRuleAST } from '@adobe/css-tools' import { SizeStore, selectedCharID } from './stores.svelte'; import { calcString } from './process/infunctions'; import { findCharacterbyId, getPersonaPrompt, getUserIcon, getUserName, parseKeyValue, sfc32, sleep, uuidtoNumber } from './util'; import { getInlayAsset } from './process/files/inlays'; import { getModuleAssets, getModuleLorebooks } from './process/modules'; import type { OpenAIChat } from './process/index.svelte'; import hljs from 'highlight.js/lib/core' import 'highlight.js/styles/atom-one-dark.min.css' const markdownItOptions = { html: true, breaks: true, linkify: false, typographer: true, quotes: '\u{E9b0}\u{E9b1}\u{E9b2}\u{E9b3}', //placeholder characters to convert to real quotes } const md = markdownit(markdownItOptions) const mdHighlight = markdownit({ highlight: function (str, lang) { if(lang){ return ``+ str +''; } return '' }, ...markdownItOptions }) md.disable(['code']) mdHighlight.disable(['code']) DOMPurify.addHook("uponSanitizeElement", (node: HTMLElement, data) => { if (data.tagName === "iframe") { const src = node.getAttribute("src") || ""; if (!src.startsWith("https://www.youtube.com/embed/")) { return node.parentNode.removeChild(node); } } }); DOMPurify.addHook("uponSanitizeAttribute", (node, data) => { switch(data.attrName){ case 'style':{ data.attrValue = data.attrValue.replace(/(absolute)|(z-index)|(fixed)/g, '') break } case 'class':{ if(data.attrValue){ data.attrValue = data.attrValue.split(' ').map((v) => { if(v.startsWith('hljs')){ return v } return "x-risu-" + v }).join(' ') } break } case 'href':{ if(data.attrValue.startsWith('http://') || data.attrValue.startsWith('https://')){ node.setAttribute('target', '_blank') break } data.attrValue = '' break } } }) DOMPurify.addHook('uponSanitizeAttribute', (node, data) => { if (['IMG', 'SOURCE', 'STYLE'].includes(node.nodeName) && data.attrName === 'src' && data.attrValue.startsWith('asset://localhost/')) { data.forceKeepAttr = true; } }); function renderMarkdown(md:markdownit, data:string){ let quotes = ['“', '”', '‘', '’'] if(DBState.db?.customQuotes){ quotes = DBState.db.customQuotesData ?? quotes } let text = md.render(data.replace(/“|”/g, '"').replace(/‘|’/g, "'")) if(DBState.db?.unformatQuotes){ text = text.replace(/\uE9b0/gu, quotes[0]).replace(/\uE9b1/gu, quotes[1]) text = text.replace(/\uE9b2/gu, quotes[2]).replace(/\uE9b3/gu, quotes[3]) } else{ text = text.replace(/\uE9b0/gu, '' + quotes[0]).replace(/\uE9b1/gu, quotes[1] + '') text = text.replace(/\uE9b2/gu, '' + quotes[2]).replace(/\uE9b3/gu, quotes[3] + '') } return text } async function renderHighlightableMarkdown(data:string) { let rendered = renderMarkdown(mdHighlight, data) const highlightPlaceholders = rendered.match(/(.+?)<\/pre-hljs-placeholder>/gms) if (!highlightPlaceholders){ return rendered } for (const placeholder of highlightPlaceholders){ try { let lang = placeholder.match(/lang="(.+?)"/)?.[1] const code = placeholder.match(/(.+?)<\/pre-hljs-placeholder>/ms)?.[1] if (!lang || !code){ continue } //import language if not already loaded //we do not refactor this to a function because we want to keep vite to only import the languages that are needed let languageModule:any = null switch(lang){ case 'js': case 'javascript':{ lang = 'javascript' if(!hljs.getLanguage('javascript')){ languageModule = await import('highlight.js/lib/languages/javascript') } break } case 'py': case 'python':{ lang = 'python' if(!hljs.getLanguage('python')){ languageModule = await import('highlight.js/lib/languages/python') } break } case 'css':{ lang = 'css' if(!hljs.getLanguage('css')){ languageModule = await import('highlight.js/lib/languages/css') } break } case 'xml': case 'html':{ lang = 'xml' if(!hljs.getLanguage('xml')){ languageModule = await import('highlight.js/lib/languages/xml') } break } case 'lua':{ lang = 'lua' if(!hljs.getLanguage('lua')){ languageModule = await import('highlight.js/lib/languages/lua') } break } case 'dart':{ lang = 'dart' if(!hljs.getLanguage('dart')){ languageModule = await import('highlight.js/lib/languages/dart') } break } case 'java':{ lang = 'java' if(!hljs.getLanguage('java')){ languageModule = await import('highlight.js/lib/languages/java') } break } case 'rust':{ lang = 'rust' if(!hljs.getLanguage('rust')){ languageModule = await import('highlight.js/lib/languages/rust') } break } case 'c': case 'cpp':{ lang = 'cpp' if(!hljs.getLanguage('cpp')){ languageModule = await import('highlight.js/lib/languages/cpp') } break } case 'csharp': case 'cs':{ lang = 'csharp' if(!hljs.getLanguage('csharp')){ languageModule = await import('highlight.js/lib/languages/csharp') } break } case 'ts': case 'typescript':{ lang = 'typescript' if(!hljs.getLanguage('typescript')){ languageModule = await import('highlight.js/lib/languages/typescript') } break } case 'json':{ lang = 'json' if(!hljs.getLanguage('json')){ languageModule = await import('highlight.js/lib/languages/json') } break } case 'yaml':{ lang = 'yaml' if(!hljs.getLanguage('yaml')){ languageModule = await import('highlight.js/lib/languages/yaml') } break } case 'shell':{ lang = 'shell' if(!hljs.getLanguage('shell')){ languageModule = await import('highlight.js/lib/languages/shell') } break } case 'bash':{ lang = 'bash' if(!hljs.getLanguage('bash')){ languageModule = await import('highlight.js/lib/languages/bash') } break } default:{ lang = 'none' } } if(languageModule){ hljs.registerLanguage(lang, languageModule.default) } if(lang === 'none'){ rendered = rendered.replace(placeholder, `
${md.utils.escapeHtml(code)}
`) } else{ const highlighted = hljs.highlight(code, { language: lang, ignoreIllegals: true }).value rendered = rendered.replace(placeholder, `
${highlighted}
`) } } catch (error) { } } return rendered } export const assetRegex = /{{(raw|path|img|image|video|audio|bg|emotion|asset|video-img|source)::(.+?)}}/g async function parseAdditionalAssets(data:string, char:simpleCharacterArgument|character, mode:'normal'|'back', mode2:'unset'|'pre'|'post' = 'unset'){ const assetWidthString = (DBState.db.assetWidth && DBState.db.assetWidth !== -1 || DBState.db.assetWidth === 0) ? `max-width:${DBState.db.assetWidth}rem;` : '' let assetPaths:{[key:string]:{ path:string ext?:string }} = {} let emoPaths:{[key:string]:{ path:string }} = {} if(char.additionalAssets){ for(const asset of char.additionalAssets){ const assetPath = await getFileSrc(asset[1]) assetPaths[asset[0].toLocaleLowerCase()] = { path: assetPath, ext: asset[2] } } } if(char.emotionImages){ for(const emo of char.emotionImages){ const emoPath = await getFileSrc(emo[1]) emoPaths[emo[0].toLocaleLowerCase()] = { path: emoPath, } } } const moduleAssets = getModuleAssets() if(moduleAssets.length > 0){ for(const asset of moduleAssets){ const assetPath = await getFileSrc(asset[1]) assetPaths[asset[0].toLocaleLowerCase()] = { path: assetPath, ext: asset[2] } } } const videoExtention = ['mp4', 'webm', 'avi', 'm4p', 'm4v'] let needsSourceAccess = false data = data.replaceAll(assetRegex, (full:string, type:string, name:string) => { name = name.toLocaleLowerCase() if(type === 'emotion'){ const path = emoPaths[name]?.path if(!path){ return '' } return `${path}` } if(type === 'source'){ needsSourceAccess = true switch(name){ case 'char':{ return '\uE9b4CHAR\uE9b4' } case 'user': { return '\uE9b4USER\uE9b4' } } } let path = assetPaths[name] if(!path){ if(DBState.db.legacyMediaFindings){ return '' } path = getClosestMatch(name, assetPaths) if(!path){ return '' } } switch(type){ case 'raw': case 'path': return path.path case 'img': return `${path.path}` case 'image': return `
${path.path}
\n` case 'video': return `\n` case 'video-img': return `\n` case 'audio': return `\n` case 'bg': if(mode === 'back'){ return `
` } break case 'asset':{ if(path.ext && videoExtention.includes(path.ext)){ return `\n` } return `${path.path}\n` } } return '' }) if(needsSourceAccess){ const chara = getCurrentCharacter() if(chara.image){} data = data.replace(/\uE9b4CHAR\uE9b4/g, chara.image ? (await getFileSrc(chara.image)) : '' ) data = data.replace(/\uE9b4USER\uE9b4/g, getUserIcon() ? (await getFileSrc(getUserIcon())) : '' ) } return data } function getClosestMatch(name:string, assetPaths:{[key:string]:{path:string, ext?:string}}){ if(Object.keys(assetPaths).length === 0){ return null } //Levenshtein distance, new with 1d array const dest = (a:string, b:string) => { const h = a.length + 1 const w = b.length + 1 let d = new Int16Array(h * w) for(let i=0;i` : '' let postfix = inlayType === 'inlayed' ? `\n\n` : '' switch(asset?.type){ case 'image': data = data.replace(inlay, `${prefix}${postfix}`) break case 'video': data = data.replace(inlay, `${prefix}${postfix}`) break case 'audio': data = data.replace(inlay, `${prefix}${postfix}`) break } } } return data } export interface simpleCharacterArgument{ type: 'simple' additionalAssets?: [string, string, string][] customscript: customscript[] chaId: string, virtualscript?: string emotionImages?: [string, string][] triggerscript?: triggerscript[] } export async function ParseMarkdown( data:string, charArg:(character|simpleCharacterArgument | groupChat | string) = null, mode:'normal'|'back'|'pretranslate' = 'normal', chatID=-1, cbsConditions:CbsConditions = {} ) { let firstParsed = '' const additionalAssetMode = (mode === 'back') ? 'back' : 'normal' let char = (typeof(charArg) === 'string') ? (findCharacterbyId(charArg)) : (charArg) if(char && char.type !== 'group'){ data = await parseAdditionalAssets(data, char, additionalAssetMode, 'pre') firstParsed = data } if(char){ data = (await processScriptFull(char, data, 'editdisplay', chatID, cbsConditions)).data } if(firstParsed !== data && char && char.type !== 'group'){ data = await parseAdditionalAssets(data, char, additionalAssetMode, 'post') } data = await parseInlayAssets(data ?? '') data = encodeStyle(data) if(mode === 'normal'){ data = await renderHighlightableMarkdown(data) } return decodeStyle(DOMPurify.sanitize(data, { ADD_TAGS: ["iframe", "style", "risu-style", "x-em"], ADD_ATTR: ["allow", "allowfullscreen", "frameborder", "scrolling", "risu-btn", 'risu-trigger', 'risu-mark'], })) } export async function postTranslationParse(data:string){ let lines = data.split('\n') for(let i=0;i(.+?)\<\/style\>/gms function encodeStyle(txt:string){ return txt.replaceAll(styleRegex, (f, c1) => { return "" + Buffer.from(c1).toString('hex') + "" }) } const styleDecodeRegex = /\(.+?)\<\/risu-style\>/gms function decodeStyleRule(rule:CssAtRuleAST){ if(rule.type === 'rule'){ if(rule.selectors){ for(let i=0;i { if(v.startsWith('.')){ return ".x-risu-" + v.substring(1) } return v }).join(' ') rule.selectors[i] = ".chattext " + selectors } } } } if(rule.type === 'media' || rule.type === 'supports' || rule.type === 'document' || rule.type === 'host' || rule.type === 'container' ){ for(let i=0;i { try { let text = Buffer.from(txt, 'hex').toString('utf-8') text = risuChatParser(text) const ast = css.parse(text) const rules = ast?.stylesheet?.rules if(rules){ for(let i=0;i${css.stringify(ast)}` } catch (error) { return `CSS ERROR: ${error}`; } }) } export async function hasher(data:Uint8Array){ return Buffer.from(await crypto.subtle.digest("SHA-256", data)).toString('hex'); } export async function convertImage(data:Uint8Array) { if(!DBState.db.imageCompression){ return data } const type = checkImageType(data) if(type !== 'Unknown' && type !== 'WEBP' && type !== 'AVIF'){ return await resizeAndConvert(data) } return data } async function resizeAndConvert(imageData: Uint8Array): Promise { return new Promise((resolve, reject) => { const base64Image = 'data:image/png;base64,' + Buffer.from(imageData).toString('base64'); const image = new Image(); image.onload = () => { URL.revokeObjectURL(base64Image); // Create a canvas const canvas = document.createElement('canvas'); const context = canvas.getContext('2d'); if (!context) { throw new Error('Unable to get 2D context'); } // Compute the new dimensions while maintaining aspect ratio let { width, height } = image; if (width > 3000 || height > 3000) { const aspectRatio = width / height; if (width > height) { width = 3000; height = Math.round(width / aspectRatio); } else { height = 3000; width = Math.round(height * aspectRatio); } } // Resize and draw the image to the canvas canvas.width = width; canvas.height = height; context.drawImage(image, 0, 0, width, height); // Try to convert to WebP let base64 = canvas.toDataURL('image/webp', 75); // If WebP is not supported, convert to JPEG if (base64.indexOf('data:image/webp') != 0) { base64 = canvas.toDataURL('image/jpeg', 75); } // Convert it to Uint8Array const array = Buffer.from(base64.split(',')[1], 'base64'); resolve(array); }; image.src = base64Image; }); } type ImageType = 'JPEG' | 'PNG' | 'GIF' | 'BMP' | 'AVIF' | 'WEBP' | 'Unknown'; export function checkImageType(arr:Uint8Array):ImageType { const isJPEG = arr[0] === 0xFF && arr[1] === 0xD8 && arr[arr.length-2] === 0xFF && arr[arr.length-1] === 0xD9; const isPNG = arr[0] === 0x89 && arr[1] === 0x50 && arr[2] === 0x4E && arr[3] === 0x47 && arr[4] === 0x0D && arr[5] === 0x0A && arr[6] === 0x1A && arr[7] === 0x0A; const isGIF = arr[0] === 0x47 && arr[1] === 0x49 && arr[2] === 0x46 && arr[3] === 0x38 && (arr[4] === 0x37 || arr[4] === 0x39) && arr[5] === 0x61; const isBMP = arr[0] === 0x42 && arr[1] === 0x4D; const isAVIF = arr[4] === 0x66 && arr[5] === 0x74 && arr[6] === 0x79 && arr[7] === 0x70 && arr[8] === 0x61 && arr[9] === 0x76 && arr[10] === 0x69 && arr[11] === 0x66; const isWEBP = arr[0] === 0x52 && arr[1] === 0x49 && arr[2] === 0x46 && arr[3] === 0x46 && arr[8] === 0x57 && arr[9] === 0x45 && arr[10] === 0x42 && arr[11] === 0x50; if (isJPEG) return "JPEG"; if (isPNG) return "PNG"; if (isGIF) return "GIF"; if (isBMP) return "BMP"; if (isAVIF) return "AVIF"; if (isWEBP) return "WEBP"; return "Unknown"; } function wppParser(data:string){ const lines = data.split('\n'); let characterDetails:{[key:string]:string[]} = {}; lines.forEach(line => { // Check for "{" and "}" indicator of object start and end if(line.includes('{')) return; if(line.includes('}')) return; // Extract key and value within brackets let keyBracketStartIndex = line.indexOf('('); let keyBracketEndIndex = line.indexOf(')'); if(keyBracketStartIndex === -1 || keyBracketEndIndex === -1) throw new Error(`Invalid syntax ${line}`); let key = line.substring(0, keyBracketStartIndex).trim(); // Validate Key if(!key) throw new Error(`Missing Key in ${line}`); const valueArray=line.substring(keyBracketStartIndex + 1, keyBracketEndIndex) .split(',') .map(str => str.trim()); // Validate Values for(let i=0;i)/gm type matcherArg = { chatID:number, db:Database, chara:character|string, rmVar:boolean, var?:{[key:string]:string} tokenizeAccurate?:boolean consistantChar?:boolean displaying?:boolean role?:string runVar?:boolean funcName?:string text?:string, recursiveCount?:number lowLevelAccess?:boolean cbsConditions:CbsConditions } export type CbsConditions = { firstmsg?:boolean chatRole?:string } function basicMatcher (p1:string,matcherArg:matcherArg,vars:{[key:string]:string}|null = null ):{ text:string, var:{[key:string]:string} }|string|null { try { if(p1.length > 100000){ return '' } const lowerCased = p1.toLocaleLowerCase() const chatID = matcherArg.chatID const db = matcherArg.db const chara = matcherArg.chara switch(lowerCased){ case 'previous_char_chat': case 'lastcharmessage':{ const selchar = db.characters[get(selectedCharID)] const chat = selchar.chats[selchar.chatPage] let pointer = chatID !== -1 ? chatID - 1 : chat.message.length - 1 while(pointer >= 0){ if(chat.message[pointer].role === 'char'){ return chat.message[pointer].data } pointer-- } return chat.fmIndex === -1 ? selchar.firstMessage : selchar.alternateGreetings[chat.fmIndex] } case 'previous_user_chat': case 'lastusermessage':{ if(chatID !== -1){ const selchar = db.characters[get(selectedCharID)] const chat = selchar.chats[selchar.chatPage] let pointer = chatID - 1 while(pointer >= 0){ if(chat.message[pointer].role === 'user'){ return chat.message[pointer].data } pointer-- } return chat.fmIndex === -1 ? selchar.firstMessage : selchar.alternateGreetings[chat.fmIndex] } return '' } case 'char': case 'bot':{ if(matcherArg.consistantChar){ return 'botname' } let selectedChar = get(selectedCharID) let currentChar = db.characters[selectedChar] if(currentChar && currentChar.type !== 'group'){ return currentChar.nickname || currentChar.name } if(chara){ if(typeof(chara) === 'string'){ return chara } else{ return chara.name } } return currentChar.nickname || currentChar.name } case 'user':{ if(matcherArg.consistantChar){ return 'username' } return getUserName() } case 'personality': case 'char_persona':{ const argChara = chara const achara = (argChara && typeof(argChara) !== 'string') ? argChara : (db.characters[get(selectedCharID)]) if(achara.type === 'group'){ return "" } return risuChatParser(achara.personality, matcherArg) } case 'description': case 'char_desc':{ const argChara = chara const achara = (argChara && typeof(argChara) !== 'string') ? argChara : (db.characters[get(selectedCharID)]) if(achara.type === 'group'){ return "" } return risuChatParser(achara.desc, matcherArg) } case 'scenario':{ const argChara = chara const achara = (argChara && typeof(argChara) !== 'string') ? argChara : (db.characters[get(selectedCharID)]) if(achara.type === 'group'){ return "" } return risuChatParser(achara.scenario, matcherArg) } case 'example_dialogue': case 'example_message':{ const argChara = chara const achara = (argChara && typeof(argChara) !== 'string') ? argChara : (db.characters[get(selectedCharID)]) if(achara.type === 'group'){ return "" } return risuChatParser(achara.exampleMessage, matcherArg) } case 'persona': case 'user_persona':{ return risuChatParser(getPersonaPrompt(), matcherArg) } case 'main_prompt': case 'system_prompt':{ return risuChatParser(db.mainPrompt, matcherArg) } case 'lorebook': case 'world_info':{ const argChara = chara const achara = (argChara && typeof(argChara) !== 'string') ? argChara : (db.characters[get(selectedCharID)]) const selchar = db.characters[get(selectedCharID)] const chat = selchar.chats[selchar.chatPage] const characterLore = (achara.type === 'group') ? [] : (achara.globalLore ?? []) const chatLore = chat.localLore ?? [] const fullLore = characterLore.concat(chatLore.concat(getModuleLorebooks())) return makeArray(fullLore.map((v) => { return JSON.stringify(v) })) } case 'history': case 'messages':{ const selchar = db.characters[get(selectedCharID)] const chat = selchar.chats[selchar.chatPage] return makeArray([{ role: 'char', data: chat.fmIndex === -1 ? selchar.firstMessage : selchar.alternateGreetings[chat.fmIndex] }].concat(chat.message).map((v) => { v = safeStructuredClone(v) v.data = risuChatParser(v.data, matcherArg) return JSON.stringify(v) })) } case 'user_history': case 'user_messages':{ const selchar = db.characters[get(selectedCharID)] const chat = selchar.chats[selchar.chatPage] return makeArray(chat.message.filter((v) => { return v.role === 'user' }).map((v) => { v = safeStructuredClone(v) v.data = risuChatParser(v.data, matcherArg) return JSON.stringify(v) })) } case 'char_history': case 'char_messages':{ const selchar = db.characters[get(selectedCharID)] const chat = selchar.chats[selchar.chatPage] return makeArray(chat.message.filter((v) => { return v.role === 'char' }).map((v) => { v = safeStructuredClone(v) v.data = risuChatParser(v.data, matcherArg) return JSON.stringify(v) })) } case 'jb': case 'jailbreak':{ return risuChatParser(db.jailbreak, matcherArg) } case 'ujb': case 'global_note': case 'system_note':{ return risuChatParser(db.globalNote, matcherArg) } case 'chat_index':{ return chatID.toString() } case 'first_msg_index':{ const selchar = db.characters[get(selectedCharID)] const chat = selchar.chats[selchar.chatPage] return chat.fmIndex.toString() } case 'blank': case 'none':{ return '' } case 'message_time':{ if(matcherArg.tokenizeAccurate){ return `00:00:00` } if(chatID === -1){ return "[Cannot get time]" } const selchar = db.characters[get(selectedCharID)] const chat = selchar.chats[selchar.chatPage] const message = chat.message[chatID] if(!message.time){ return "[Cannot get time, message was sent in older version]" } const date = new Date(message.time) //output time in format like 10:30 AM return date.toLocaleTimeString() } case 'message_date':{ if(matcherArg.tokenizeAccurate){ return `00:00:00` } if(chatID === -1){ return "[Cannot get time]" } const selchar = db.characters[get(selectedCharID)] const chat = selchar.chats[selchar.chatPage] const message = chat.message[chatID] if(!message.time){ return "[Cannot get time, message was sent in older version]" } const date = new Date(message.time) //output date in format like Aug 23, 2021 return date.toLocaleDateString() } case 'message_unixtime_array':{ const selchar = db.characters[get(selectedCharID)] const chat = selchar.chats[selchar.chatPage] return makeArray(chat.message.map((f) => { return `${f.time ?? 0}` })) } case 'unixtime':{ const now = new Date() return (now.getTime() / 1000).toFixed(0) } case 'time':{ const now = new Date() return `${now.getHours()}:${now.getMinutes()}:${now.getSeconds()}` } case 'date':{ const now = new Date() return `${now.getFullYear()}-${now.getMonth() + 1}-${now.getDate()}` } case 'isotime':{ const now = new Date() return `${now.getUTCHours()}:${now.getUTCMinutes()}:${now.getUTCSeconds()}` } case 'isodate':{ const now = new Date() return `${now.getUTCFullYear()}-${now.getUTCMonth() + 1}-${now.getUTCDate()}` } case 'message_idle_duration':{ if(matcherArg.tokenizeAccurate){ return `00:00:00` } if(chatID === -1){ return "[Cannot get time]" } const selchar = db.characters[get(selectedCharID)] const chat = selchar.chats[selchar.chatPage] //get latest user message let pointer = chatID let pointerMode: 'findLast'|'findSecondLast' = 'findLast' let message:Message let previous_message:Message while(pointer >= 0){ if(chat.message[pointer].role === 'user'){ if(pointerMode === 'findLast'){ message = chat.message[pointer] pointerMode = 'findSecondLast' } else{ previous_message = chat.message[pointer] break } } pointer-- } if(!message){ return '[No user message found]' } if(!previous_message){ return '[No previous user message found]' } if(!message.time){ return "[Cannot get time, message was sent in older version]" } if(!previous_message.time){ return "[Cannot get time, previous message was sent in older version]" } let duration = message.time - previous_message.time //output time in format like 10:30:00 let seconds = Math.floor(duration / 1000) let minutes = Math.floor(seconds / 60) let hours = Math.floor(minutes / 60) seconds = seconds % 60 minutes = minutes % 60 //output, like 1:30:00 return hours.toString() + ':' + minutes.toString().padStart(2,'0') + ':' + seconds.toString().padStart(2,'0') } case 'idle_duration':{ if(matcherArg.tokenizeAccurate){ return `00:00:00` } const selchar = db.characters[get(selectedCharID)] const chat = selchar.chats[selchar.chatPage] const messages = chat.message if(messages.length === 0){ return `00:00:00` } const lastMessage = messages[messages.length - 1] if(!lastMessage.time){ return "[Cannot get time, message was sent in older version]" } const now = new Date() let duration = now.getTime() - lastMessage.time let seconds = Math.floor(duration / 1000) let minutes = Math.floor(seconds / 60) let hours = Math.floor(minutes / 60) seconds = seconds % 60 minutes = minutes % 60 return hours.toString() + ':' + minutes.toString().padStart(2,'0') + ':' + seconds.toString().padStart(2,'0') } case 'br': case 'newline':{ return '\n' } case 'model':{ return db.aiModel } case 'axmodel':{ return db.subModel } case 'role': { if(matcherArg.cbsConditions.chatRole){ return matcherArg.cbsConditions.chatRole } if(matcherArg.cbsConditions.firstmsg){ return 'char' } if (chatID !== -1) { const selchar = db.characters[get(selectedCharID)] return selchar.chats[selchar.chatPage].message[chatID].role; } return matcherArg.role ?? 'null' } case 'isfirstmsg': case 'is_first_msg': case 'is_first_message': case 'isfirstmessage':{ if(matcherArg.cbsConditions.firstmsg){ return '1' } return '0' } case 'jbtoggled':{ return db.jailbreakToggle ? '1' : '0' } case 'random':{ return Math.random().toString() } case 'maxcontext':{ return db.maxContext.toString() } case 'lastmessage':{ const selchar = db.characters[get(selectedCharID)] if(!selchar){ return '' } const chat = selchar.chats[selchar.chatPage] return chat.message[chat.message.length - 1].data } case 'lastmessageid': case 'lastmessageindex':{ const selchar = db.characters[get(selectedCharID)] if(!selchar){ return '' } const chat = selchar.chats[selchar.chatPage] return (chat.message.length - 1).toString() } case 'emotionlist':{ const selchar = db.characters[get(selectedCharID)] if(!selchar){ return '' } return makeArray(selchar.emotionImages?.map((f) => { return f[0] })) ?? '' } case 'assetlist':{ const selchar = db.characters[get(selectedCharID)] if(!selchar || selchar.type === 'group'){ return '' } return makeArray(selchar.additionalAssets?.map((f) => { return f[0] })) } case 'prefill_supported':{ return db.aiModel.startsWith('claude') ? '1' : '0' } case 'screen_width':{ return get(SizeStore).w.toString() } case 'screen_height':{ return get(SizeStore).h.toString() } } const arra = p1.split("::") if(arra.length > 1){ const v = arra[1] switch(arra[0]){ case 'tempvar': case 'gettempvar':{ return { text: vars[arra[1]] ?? '', var: vars } } case 'settempvar':{ vars[arra[1]] = arra[2] return { text: '', var: vars } } case 'return':{ vars['__return__'] = arra[1] vars['__force_return__'] = '1' return { text: '', var: vars } } case 'getvar':{ return getChatVar(v) } case 'calc':{ return calcString(v).toString() } case 'addvar':{ if(matcherArg.rmVar){ return '' } if(matcherArg.runVar){ setChatVar(v, (Number(getChatVar(v)) + Number(arra[2])).toString()) return '' } return null } case 'setvar':{ if(matcherArg.rmVar){ return '' } if(matcherArg.runVar){ setChatVar(v, arra[2]) return '' } return null } case 'setdefaultvar':{ if(matcherArg.rmVar){ return '' } if(matcherArg.runVar){ if(!getChatVar(v)){ setChatVar(v, arra[2]) } return '' } return null } case 'getglobalvar':{ return getGlobalChatVar(v) } case 'button':{ return `` } case 'risu':{ return `` } case 'equal':{ return (arra[1] === arra[2]) ? '1' : '0' } case 'not_equal': case 'notequal':{ return (arra[1] !== arra[2]) ? '1' : '0' } case 'greater':{ return (Number(arra[1]) > Number(arra[2])) ? '1' : '0' } case 'less':{ return (Number(arra[1]) < Number(arra[2])) ? '1' : '0' } case 'greater_equal': case 'greaterequal':{ return (Number(arra[1]) >= Number(arra[2])) ? '1' : '0' } case 'less_equal': case 'lessequal':{ return (Number(arra[1]) <= Number(arra[2])) ? '1' : '0' } case 'and':{ return arra[1] === '1' && arra[2] === '1' ? '1' : '0' } case 'or':{ return arra[1] === '1' || arra[2] === '1' ? '1' : '0' } case 'not':{ return arra[1] === '1' ? '0' : '1' } case 'file':{ if(matcherArg.displaying){ return `
${arra[1]}

` } return Buffer.from(arra[2], 'base64').toString('utf-8') } case 'startswith':{ return arra[1].startsWith(arra[2]) ? '1' : '0' } case 'endswith':{ return arra[1].endsWith(arra[2]) ? '1' : '0' } case 'contains':{ return arra[1].includes(arra[2]) ? '1' : '0' } case 'replace':{ return arra[1].replaceAll(arra[2], arra[3]) } case 'split':{ return makeArray(arra[1].split(arra[2])) } case 'join':{ return (parseArray(arra[1])).join(arra[2]) } case 'spread':{ return (parseArray(arra[1])).join('::') } case 'trim':{ return arra[1].trim() } case 'length':{ return arra[1].length.toString() } case 'arraylength': case 'array_length':{ return parseArray(arra[1]).length.toString() } case 'lower':{ return arra[1].toLocaleLowerCase() } case 'upper':{ return arra[1].toLocaleUpperCase() } case 'capitalize':{ return arra[1].charAt(0).toUpperCase() + arra[1].slice(1) } case 'round':{ return Math.round(Number(arra[1])).toString() } case 'floor':{ return Math.floor(Number(arra[1])).toString() } case 'ceil':{ return Math.ceil(Number(arra[1])).toString() } case 'abs':{ return Math.abs(Number(arra[1])).toString() } case 'remaind':{ return (Number(arra[1]) % Number(arra[2])).toString() } case 'previous_chat_log':{ const selchar = db.characters[get(selectedCharID)] const chat = selchar?.chats?.[selchar.chatPage] return chat?.message[chatID - 1]?.data ?? 'Out of range' } case 'tonumber':{ return (arra[1].split('').filter((v) => { return !isNaN(Number(v)) || v === '.' })).join('') } case 'pow':{ return Math.pow(Number(arra[1]), Number(arra[2])).toString() } case 'arrayelement': case 'array_element':{ return parseArray(arra[1]).at(Number(arra[2])) ?? 'null' } case 'dictelement': case 'dict_element': case 'objectelement': case 'object_element':{ return parseDict(arra[1])[arra[2]] ?? 'null' } case 'object_assert': case 'dict_assert': case 'dictassert': case 'objectassert':{ const dict = parseDict(arra[1]) if(!dict[arra[2]]){ dict[arra[2]] = arra[3] } return JSON.stringify(dict) } case 'element': case 'ele':{ try { const agmts = arra.slice(2) let current = arra[1] for(const arg of agmts){ const parsed = JSON.parse(current) if(parsed === null || (typeof(parsed) !== 'object' && !Array.isArray(parsed))){ return 'null' } current = parsed[arg] if(!current){ return 'null' } } return current } catch (error) { return 'null' } } case 'arrayshift': case 'array_shift':{ const arr = parseArray(arra[1]) arr.shift() return makeArray(arr) } case 'arraypop': case 'array_pop':{ const arr = parseArray(arra[1]) arr.pop() return makeArray(arr) } case 'arraypush': case 'array_push':{ const arr = parseArray(arra[1]) arr.push(arra[2]) return makeArray(arr) } case 'arraysplice': case 'array_splice':{ const arr = parseArray(arra[1]) arr.splice(Number(arra[2]), Number(arra[3]), arra[4]) return makeArray(arr) } case 'arrayassert': case 'array_assert':{ const arr = parseArray(arra[1]) const index = Number(arra[2]) if(index >= arr.length){ arr[index] = arra[3] } return makeArray(arr) } case 'makearray': case 'array': case 'a': case 'make_array':{ return makeArray(arra.slice(1)) } case 'makedict': case 'dict': case 'd': case 'make_dict': case 'makeobject': case 'object': case 'o': case 'make_object':{ const sliced = arra.slice(1) let out = {} for(let i=0;i { let data = '' if(arra.includes('role')){ data += f.role + ': ' } data += f.data return data })) } case 'range':{ const arr = parseArray(arra[1]) const start = arr.length > 1 ? Number(arr[0]) : 0 const end = arr.length > 1 ? Number(arr[1]) : Number(arr[0]) const step = arr.length > 2 ? Number(arr[2]) : 1 let out:string[] = [] for(let i=start;i { return f.name === arra[1] }).id return (db.enabledModules.includes(moduleId) || enabledChatModules.includes(moduleId)) ? '1' : '0' } case 'filter':{ const array = parseArray(arra[1]) const filterTypes = [ 'all', 'nonempty', 'unique', ] let filterType = filterTypes.indexOf(arra[2]) if(filterType === -1){ filterType = 0 } return makeArray(array.filter((f, i) => { switch(filterType){ case 0: return f !== '' && i === array.indexOf(f) case 1: return f !== '' case 2: return i === array.indexOf(f) } })) } case 'all':{ const array = arra.length > 2 ? arra.slice(1) : parseArray(arra[1]) const all = array.every((f) => { return f === '1' }) return all ? '1' : '0' } case 'any':{ const array = arra.length > 2 ? arra.slice(1) : parseArray(arra[1]) const any = array.some((f) => { return f === '1' }) return any ? '1' : '0' } case 'min':{ const val = arra.length > 2 ? arra.slice(1) : parseArray(arra[1]) return Math.min(...val.map((f) => { const num = Number(f) if(isNaN(num)){ return 0 } return num })).toString() } case 'max':{ const val = arra.length > 2 ? arra.slice(1) : parseArray(arra[1]) return Math.max(...val.map((f) => { const num = Number(f) if(isNaN(num)){ return 0 } return num })).toString() } case 'sum':{ const val = arra.length > 2 ? arra.slice(1) : parseArray(arra[1]) return val.map((f) => { const num = Number(f) if(isNaN(num)){ return 0 } return num }).reduce((a, b) => a + b, 0).toString() } case 'average':{ const val = arra.length > 2 ? arra.slice(1) : parseArray(arra[1]) const sum = val.map((f) => { const num = Number(f) if(isNaN(num)){ return 0 } return num }).reduce((a, b) => a + b, 0) return (sum / val.length).toString() } case 'fixnum': case 'fix_num': case 'fixnumber': case 'fix_number':{ return Number(arra[1]).toFixed(Number(arra[2])) } case 'unicode_encode': case 'unicodeencode':{ return arra[1].charCodeAt(arra[2] ? Number(arra[2]) : 0).toString() } case 'unicode_decode': case 'unicodedecode':{ return String.fromCharCode(Number(arra[1])) } case 'hash':{ return ((pickHashRand(0, arra[1]) * 10000000) + 1).toFixed(0).padStart(7, '0') } } } if(p1.startsWith('random')){ if(p1.startsWith('random::')){ const randomIndex = Math.floor(Math.random() * (arra.length - 1)) + 1 if(matcherArg.tokenizeAccurate){ return arra[0] } return arra[randomIndex] } else{ const arr = p1.replace(/\\,/g, '§X').split(/\:|\,/g) const randomIndex = Math.floor(Math.random() * (arr.length - 1)) + 1 if(matcherArg.tokenizeAccurate){ return arra[0] } return arr[randomIndex]?.replace(/§X/g, ',') ?? '' } } if(p1.startsWith('pick')){ const selchar = db.characters[get(selectedCharID)] const selChat = selchar.chats[selchar.chatPage] const cid = selChat.message.length if(p1.startsWith('pick::')){ const randomIndex = Math.floor(pickHashRand(cid, selchar.chaId + (selChat.id ?? '')) * (arra.length - 1)) + 1 if(matcherArg.tokenizeAccurate){ return arra[0] } return arra[randomIndex] } else{ const arr = p1.replace(/\\,/g, '§X').split(/\:|\,/g) const randomIndex = Math.floor(pickHashRand(cid, selchar.chaId + (selChat.id ?? '')) * (arr.length - 1)) + 1 if(matcherArg.tokenizeAccurate){ return arra[0] } return arr[randomIndex]?.replace(/§X/g, ',') ?? '' } } if(p1.startsWith('roll:') || p1.startsWith('rollp:')){ const p = p1.startsWith('rollp:') const arr = p1.split(/\:|\ /g) let ina = arr.at(-1) if(ina.startsWith('d')){ ina = ina.substring(1) } const maxRoll = parseInt(ina) if(isNaN(maxRoll)){ return 'NaN' } if(p){ const selchar = db.characters[get(selectedCharID)] const selChat = selchar.chats[selchar.chatPage] const cid = selChat.message.length return (Math.floor(pickHashRand(cid, selchar.chaId + (selChat.id ?? '')) * maxRoll) + 1).toString() } return (Math.floor(Math.random() * maxRoll) + 1).toString() } if(p1.startsWith('datetimeformat')){ let main = p1.substring("datetimeformat".length + 1) return dateTimeFormat(main) } if(p1.startsWith('? ')){ const substring = p1.substring(2) return calcString(substring).toString() } if(p1.startsWith('//')){ return '' } if(p1.startsWith('hidden_key:')){ return '' } if(p1.startsWith('reverse:')){ return p1.substring( p1[8] === ':' ? 9 : 8 ).split('').reverse().join('') } if(p1.startsWith('comment:')){ if(!matcherArg.displaying){ return '' } return `
${p1.substring(p1[8] === ':' ? 9 : 8)}
` } return null } catch (error) { return null } } function pickHashRand(cid:number,word:string) { let hashAddress = 5515 const rand = (word:string) => { for (let counter = 0; counter { const date = time === 0 ? (new Date()) : (new Date(time)) if(!main){ return '' } if(main.startsWith(':')){ main = main.substring(1) } if(main.length > 300){ return '' } return main .replace(/YYYY/g, date.getFullYear().toString()) .replace(/YY/g, date.getFullYear().toString().substring(2)) .replace(/MMMM/g, Intl.DateTimeFormat('en', { month: 'long' }).format(date)) .replace(/MMM/g, Intl.DateTimeFormat('en', { month: 'short' }).format(date)) .replace(/MM/g, (date.getMonth() + 1).toString().padStart(2, '0')) .replace(/DDDD/g, Math.floor((date.getTime() - new Date(date.getFullYear(), 0, 0).getTime()) / (1000 * 60 * 60 * 24)).toString()) .replace(/DD/g, date.getDate().toString().padStart(2, '0')) .replace(/dddd/g, Intl.DateTimeFormat('en', { weekday: 'long' }).format(date)) .replace(/ddd/g, Intl.DateTimeFormat('en', { weekday: 'short' }).format(date)) .replace(/HH/g, date.getHours().toString().padStart(2, '0')) .replace(/hh/g, (date.getHours() % 12 || 12).toString().padStart(2, '0')) .replace(/mm/g, date.getMinutes().toString().padStart(2, '0')) .replace(/ss/g, date.getSeconds().toString().padStart(2, '0')) .replace(/X/g, Math.floor(date.getTime() / 1000).toString()) .replace(/x/g, date.getTime().toString()) .replace(/A/g, date.getHours() >= 12 ? 'PM' : 'AM') } const smMatcher = (p1:string,matcherArg:matcherArg) => { if(!p1){ return null } const lowerCased = p1.toLocaleLowerCase() const db = matcherArg.db const chara = matcherArg.chara switch(lowerCased){ case 'char': case 'bot':{ if(matcherArg.consistantChar){ return 'botname' } let selectedChar = get(selectedCharID) let currentChar = db.characters[selectedChar] if(currentChar && currentChar.type !== 'group'){ return currentChar.nickname || currentChar.name } if(chara){ if(typeof(chara) === 'string'){ return chara } else{ return chara.name } } return currentChar.nickname || currentChar.name } case 'user':{ if(matcherArg.consistantChar){ return 'username' } return getUserName() } } } const legacyBlockMatcher = (p1:string,matcherArg:matcherArg) => { const bn = p1.indexOf('\n') if(bn === -1){ return null } const logic = p1.substring(0, bn) const content = p1.substring(bn + 1) const statement = logic.split(" ", 2) switch(statement[0]){ case 'if':{ if(["","0","-1"].includes(statement[1])){ return '' } return content.trim() } } return null } type blockMatch = 'ignore'|'parse'|'nothing'|'parse-pure'|'pure'|'each'|'function'|'pure-display' function parseArray(p1:string):string[]{ try { const arr = JSON.parse(p1) if(Array.isArray(arr)){ return arr } return p1.split('§') } catch (error) { return p1.split('§') } } function parseDict(p1:string):{[key:string]:string}{ try { return JSON.parse(p1) } catch (error) { return {} } } function makeArray(p1:string[]):string{ return JSON.stringify(p1.map((f) => { if(typeof(f) === 'string'){ return f.replace(/::/g, '\\u003A\\u003A') } return f })) } function blockStartMatcher(p1:string,matcherArg:matcherArg):{type:blockMatch,type2?:string,funcArg?:string[]}{ if(p1.startsWith('#if') || p1.startsWith('#if_pure ')){ const statement = p1.split(' ', 2) const state = statement[1] if(state === 'true' || state === '1'){ return {type:p1.startsWith('#if_pure') ? 'parse-pure' : 'parse'} } return {type:'ignore'} } if(p1 === '#pure'){ return {type:'pure'} } if(p1 === '#pure_display' || p1 === '#puredisplay'){ return {type:'pure-display'} } if(p1.startsWith('#each')){ let t2 = p1.substring(5).trim() if(t2.startsWith('as ')){ t2 = t2.substring(3).trim() } return {type:'each',type2:t2} } if(p1.startsWith('#func')){ const statement = p1.split(' ') if(statement.length > 1){ return {type:'function',funcArg:statement.slice(1)} } } return {type:'nothing'} } function trimLines(p1:string){ return p1.split('\n').map((v) => { return v.trimStart() }).join('\n').trim() } function blockEndMatcher(p1:string,type:{type:blockMatch,type2?:string},matcherArg:matcherArg):string{ const p1Trimed = p1.trim() switch(type.type){ case 'pure': case 'pure-display': case 'function':{ return p1Trimed } case 'parse': case 'each':{ return trimLines(p1Trimed) } case 'parse-pure':{ return p1 } default:{ return '' } } } export function risuChatParser(da:string, arg:{ chatID?:number db?:Database chara?:string|character|groupChat rmVar?:boolean, var?:{[key:string]:string} tokenizeAccurate?:boolean consistantChar?:boolean visualize?:boolean, role?:string runVar?:boolean functions?:Map callStack?:number cbsConditions?:CbsConditions } = {}):string{ const chatID = arg.chatID ?? -1 const db = arg.db ?? DBState.db const aChara = arg.chara const visualize = arg.visualize ?? false let chara:character|string = null if(aChara){ if(typeof(aChara) !== 'string' && aChara.type === 'group'){ if(aChara.chats[aChara.chatPage].message.length > 0){ const gc = findCharacterbyId(aChara.chats[aChara.chatPage].message.at(-1).saying ?? '') if(gc.name !== 'Unknown Character'){ chara = gc } } else{ chara = 'bot' } } else{ chara = aChara } } if(arg.tokenizeAccurate){ const db = arg.db ?? DBState.db const selchar = chara ?? db.characters[get(selectedCharID)] if(!selchar){ chara = 'bot' } } let pointer = 0; let nested:string[] = [""] let stackType = new Uint8Array(512) let pureModeNest:Map = new Map() let pureModeNestType:Map = new Map() let blockNestType:Map = new Map() let commentMode = false let commentLatest:string[] = [""] let commentV = new Uint8Array(512) let thinkingMode = false let tempVar:{[key:string]:string} = {} let functions:Map = arg.functions ?? (new Map()) arg.callStack = (arg.callStack ?? 0) + 1 if(arg.callStack > 20){ return 'ERROR: Call stack limit reached' } const matcherObj = { chatID: chatID, chara: chara, rmVar: arg.rmVar ?? false, db: db, var: arg.var ?? null, tokenizeAccurate: arg.tokenizeAccurate ?? false, displaying: arg.visualize ?? false, role: arg.role, runVar: arg.runVar ?? false, consistantChar: arg.consistantChar ?? false, cbsConditions: arg.cbsConditions ?? {}, callStack: arg.callStack, } da = da.replace(/\<(user|char|bot)\>/gi, '{{$1}}') const isPureMode = () => { return pureModeNest.size > 0 } const pureModeType = () => { if(pureModeNest.size === 0){ return '' } return pureModeNestType.get(nested.length) ?? [...pureModeNestType.values()].at(-1) ?? '' } while(pointer < da.length){ switch(da[pointer]){ case '{':{ if(da[pointer + 1] !== '{' && da[pointer + 1] !== '#'){ nested[0] += da[pointer] break } pointer++ nested.unshift('') stackType[nested.length] = 1 break } case '#':{ //legacy if statement, deprecated if(da[pointer + 1] !== '}' || nested.length === 1 || stackType[nested.length] !== 1){ nested[0] += da[pointer] break } pointer++ const dat = nested.shift() const mc = legacyBlockMatcher(dat, matcherObj) nested[0] += mc ?? `{#${dat}#}` break } case '}':{ if(da[pointer + 1] !== '}' || nested.length === 1 || stackType[nested.length] !== 1){ nested[0] += da[pointer] break } pointer++ const dat = nested.shift() if(dat.startsWith('#')){ if(isPureMode()){ nested[0] += `{{${dat}}}` nested.unshift('') stackType[nested.length] = 6 break } const matchResult = blockStartMatcher(dat, matcherObj) if(matchResult.type === 'nothing'){ nested[0] += `{{${dat}}}` break } else{ nested.unshift('') stackType[nested.length] = 5 blockNestType.set(nested.length, matchResult) if( matchResult.type === 'ignore' || matchResult.type === 'pure' || matchResult.type === 'each' || matchResult.type === 'function' || matchResult.type === 'pure-display' ){ pureModeNest.set(nested.length, true) pureModeNestType.set(nested.length, "block") } break } } if(dat.startsWith('/')){ if(stackType[nested.length] === 5){ const blockType = blockNestType.get(nested.length) if( blockType.type === 'ignore' || blockType.type === 'pure' || blockType.type === 'each' || blockType.type === 'function' || blockType.type === 'pure-display' ){ pureModeNest.delete(nested.length) pureModeNestType.delete(nested.length) } blockNestType.delete(nested.length) const dat2 = nested.shift() const matchResult = blockEndMatcher(dat2, blockType, matcherObj) if(blockType.type === 'each'){ const subind = blockType.type2.lastIndexOf(' ') const sub = blockType.type2.substring(subind + 1) const array = parseArray(blockType.type2.substring(0, subind)) let added = '' for(let i = 0;i < array.length;i++){ const res = matchResult.replaceAll(`{{slot::${sub}}}`, array[i]) added += res } da = da.substring(0, pointer + 1) + added.trim() + da.substring(pointer + 1) break } if(blockType.type === 'function'){ console.log(matchResult) functions.set(blockType.funcArg[0], { data: matchResult, arg: blockType.funcArg.slice(1) }) break } if(blockType.type === 'pure-display'){ nested[0] += matchResult.replaceAll('{{', '\\{\\{').replaceAll('}}', '\\}\\}') break } if(matchResult === ''){ break } nested[0] += matchResult break } if(stackType[nested.length] === 6){ const sft = nested.shift() nested[0] += sft + `{{${dat}}}` break } } if(dat.startsWith('call::')){ if(arg.callStack && arg.callStack > 20){ nested[0] += `ERROR: Call stack limit reached` break } const argData = dat.split('::').slice(1) const funcName = argData[0] const func = functions.get(funcName) console.log(func) if(func){ let data = func.data for(let i = 0;i < argData.length;i++){ data = data.replaceAll(`{{arg::${i}}}`, argData[i]) } arg.functions = functions nested[0] += risuChatParser(data, arg) break } } const mc = isPureMode() ? null :basicMatcher(dat, matcherObj, tempVar) if(!mc && mc !== ''){ nested[0] += `{{${dat}}}` } else if(typeof(mc) === 'string'){ nested[0] += mc } else{ nested[0] += mc.text tempVar = mc.var if(tempVar['__force_return__']){ return tempVar['__return__'] ?? 'null' } } break } default:{ nested[0] += da[pointer] break } } pointer++ } if(commentMode){ nested = commentLatest stackType = commentV if(thinkingMode){ nested[0] += `
Thinking...
` } commentMode = false } if(nested.length === 1){ return nested[0] } let result = '' while(nested.length > 1){ let dat = (stackType[nested.length] === 1) ? '{{' : "<" dat += nested.shift() result = dat + result } return nested[0] + result } export function getChatVar(key:string){ const selectedChar = get(selectedCharID) const char = DBState.db.characters[selectedChar] if(!char){ return 'null' } const chat = char.chats[char.chatPage] chat.scriptstate = chat.scriptstate ?? {} const state = (chat.scriptstate['$' + key]) if(state === undefined || state === null){ const defaultVariables = parseKeyValue(char.defaultVariables).concat(parseKeyValue(DBState.db.templateDefaultVariables)) const findResult = defaultVariables.find((f) => { return f[0] === key }) if(findResult){ return findResult[1] } return 'null' } return state.toString() } export function getGlobalChatVar(key:string){ return DBState.db.globalChatVariables[key] ?? 'null' } export function setChatVar(key:string, value:string){ const selectedChar = get(selectedCharID) if(!DBState.db.characters[selectedChar].chats[DBState.db.characters[selectedChar].chatPage].scriptstate){ DBState.db.characters[selectedChar].chats[DBState.db.characters[selectedChar].chatPage].scriptstate = {} } DBState.db.characters[selectedChar].chats[DBState.db.characters[selectedChar].chatPage].scriptstate['$' + key] = value } async function editDisplay(text){ let rt = "" if(!text.includes("")){ return text } for(let i=0;i{ //XML type try { const parser = new DOMParser() const dom = `${prompt}` const xmlDoc = parser.parseFromString(dom, "text/xml") const root = xmlDoc.documentElement const errorNode = root.getElementsByTagName('parsererror') if(errorNode.length > 0){ throw new Error('XML Parse Error') //fallback to other parser } const parseNode = (node:Element):string|PromptParsed => { if(node.children.length === 0){ return node.textContent } const data:{[key:string]:string|PromptParsed} = {} for(let i=0;i f !== '').map((v) => { let role:'system'|'user'|'assistant' = 'user' //default separators if(v.startsWith('user' + seperator)){ role = 'user' v = v.substring(4 + seperator.length) } else if(v.startsWith('system' + seperator)){ role = 'system' v = v.substring(6 + seperator.length) } else if(v.startsWith('assistant' + seperator)){ role = 'assistant' v = v.substring(9 + seperator.length) } //space/newline separators else if(v.startsWith('user ') || v.startsWith('user\n')){ role = 'user' v = v.substring(5) } else if(v.startsWith('system ') || v.startsWith('system\n')){ role = 'system' v = v.substring(7) } else if(v.startsWith('assistant ') || v.startsWith('assistant\n')){ role = 'assistant' v = v.substring(10) } v = v.trim() if(v.endsWith(ender)){ v = v.substring(0, v.length - ender.length) } return { role: role, content: risuChatParser(v) } }) }