(.+?)<\/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 `
`
}
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 `
`
case 'image':
return `
\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 `
\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 ``
}
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)
}
})
}