forked from city-unit/st-extension-example
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.js
616 lines (538 loc) · 21.9 KB
/
index.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
import { extension_settings, getContext, loadExtensionSettings,ModuleWorkerWrapper, } from "../../../extensions.js";
import {
power_user,
renderStoryString,
} from "../../../../scripts/power-user.js";
import { favsToHotswap, getDeviceInfo } from "../../../../scripts/RossAscends-mods.js";
import {
force_output_sequence,
formatInstructModeChat,
formatInstructModePrompt,
formatInstructModeExamples,
getInstructStoppingSequences,
formatInstructModeSystemPrompt,
} from "../../../../scripts/instruct-mode.js";
import { getTokenCount, getTokenizerModel, initTokenizers, saveTokenCache } from "../../../../scripts/tokenizers.js";
import { getCfgPrompt, getGuidanceScale, initCfg } from "../../../../scripts/cfg-scale.js";
import { getExtensionPrompt,replaceBiasMarkup,getBiasStrings,hideSwipeButtons,baseChatReplace,getRequestHeaders,addOneMessage,deleteLastMessage,deactivateSendButtons,chat,setSendButtonState, eventSource, event_types,saveSettingsDebounced, saveChat,callPopup, name1, name2, this_chid, characters, chat_metadata, getStoppingStrings} from "../../../../script.js";
import { PromptManagerModule } from '../../../PromptManager.js'
import { getWorldInfoPrompt } from "../../../../scripts/world-info.js";
import { delay , resetScrollHeight } from '../../../utils.js'
const extensionName = "Some-Extension";
const extensionFolderPath = `scripts/extensions/third-party/${extensionName}`;
const extensionSettings = extension_settings[extensionName];
const defaultSettings = {};
let models = [
{name: 'MythoMax L2 13B',id: 'mythomax-l2-13b-gptq',description:`A verbose, creative, and uncensored model. Great for fantasy and adventure scenarios.`,context: ''},
{name: 'Mythalion 13B GGUF',id: 'mythalion-13b-gguf',description:`This model was created in collaboration with Gryphe, a mixture of Pygmalion-2 13B and Gryphe's Mythomax L2 13B.`,context: ''},
{name: 'Nous Hermes Llama2 13b',id: 'nous-hermes-llama2-13b',description:`Smart, capable, and uncensored, this model is best for role-play and coherent conversation.`,context: ''},
{name: 'Tiefighter 13b',id: 'llama2-13b-tiefighter',description:`creative, If you let it improvise you get better results than if you drown it in details.`,context: ''},
]
// some global variables
const abortController = new AbortController();
let chat2 = [];
let coreChat = chat.filter(x => !x.is_system);
let current_model = 'mythomax-l2-13b-gptq';
let this_max_context = 8000;
for (let i = coreChat.length - 1, j = 0; i >= 0; i--, j++) {
chat2[i] = formatMessageHistoryItem(coreChat[j], false, false);
}
const get_settings = () => {
return {
mas_new_tokens: $('#max_new_tokens').val(),
top_k: $('#top_k').val(),
top_p: $('#top_p').val(),
typical_p: $('#typical_p').val(),
max_context: $('#max_context').val(),
temperature: $('#temp_textgenerationwebui').val(),
rep_penalty: $('#rep_pen_textgenerationwebui').val(),
}
}
const generate = async () => {
try {
awaiting_message()
const { headers, body, url, path } = await get_api_request_object();
const response = await get_response(headers, body, url, path);
handle_response(response);
}catch(err){
handle_error(err)
}
}
/**
* send a placeholder message as the chracter and send it to the frontend chat!
*/
const awaiting_message = () => {
const message = {
"extra": {
display_text: '...'
},
"name": name2,
"is_user": false,
"send_date": getdate(),
"mes": '...',
}
chat.push(message)
addOneMessage(message)
}
/**
* send a message as the user to the front end
* @param {string} text
*/
const send_as_user = (text) => {
const message = {
"extra": {
display_text: text
},
"name": name1,
"is_user": true,
"send_date": getdate(),
"mes": text
}
chat.push(message)
eventSource.emit(event_types.MESSAGE_SENT, getContext().chat.length - 1);
addOneMessage(message)
eventSource.emit(event_types.USER_MESSAGE_RENDERED, getContext().chat.length - 1);
}
/**
* This function returns the api request object used for a fetch request!
* @returns {Promise<object>}
*/
const get_api_request_object = async () => {
const { mas_new_tokens, max_context, rep_penalty, temperature, top_k, top_p, typical_p } = get_settings();
if (current_model == 'mythomax-l2-13b-gptq' || current_model == 'mythalion-13b-gguf' || current_model == 'nous-hermes-llama2-13b' || current_model == 'llama2-13b-tiefighter'){
return {
path: 'generated_text',
url: 'https://ai.moemate.io/api/chat',
headers: {
'Content-Type': 'application/json',
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; rv:109.0) Gecko/20100101 Firefox/119.0',
'webahost': current_model == 'mythomax-l2-13b-gptq' ? 'llmmyth.dev.moemate.io' : '' || current_model == 'mythalion-13b-gguf' ? 'mythalion.dev.moemate.io' : '' || current_model == 'nous-hermes-llama2-13b' ? 'llmnh2.dev.moemate.io' : '' || current_model == 'llama2-13b-tiefighter' ? 'llm2.dev.moemate.io' : '',
'WebaAuth': 'Bearer',
},
body: JSON.stringify({
inputs: await get_prompt(chat_metadata,getContext().characters,this_chid),
parameters: {
best_of: 1,
decoder_input_details: true,
details: false,
do_sample: true,
max_new_tokens: 500 || mas_new_tokens,
repetition_penalty: 1.15 || rep_penalty,
return_full_text: false,
seed: null,
stop: getInstructStoppingSequences(),
stream: true,
temperature: 0.9 || temperature,
top_k: 10 || top_k,
top_p: 0.6 || top_p,
truncate: null,
typical_p: 0.99 || typical_p,
watermark: false
}
}),
}
}
}
/**
* This function handles the current connection status of the extension!
* @param {boolean} status
*/
const handle_status = (status) => {
if (status === true){
$('#send_form').removeClass('no-connection')
$('#ai_settings_drawer').removeClass('no_connection')
$('#no_connection_icon').hide()
$('.online_status4').replaceWith(` <div class="online_status4">
<div class="online_status_indicator4" style="background-color: rgb(26, 255, 0);"></div>
<div class="online_status_text4" data-i18n="Not connected...">Connected...</div>
</div>`)
$('#ai_settings_drawer .drawer-icon').removeClass('no_connection')
}else {
$('#send_form').addClass('no-connection')
$('#ai_settings_drawer').addClass('no_connection')
$('#no_connection_icon').show()
$('#ai_settings_drawer .drawer-icon').addClass('no_connection')
$('.online_status4').replaceWith(` <div class="online_status4">
<div class="online_status_indicator4" style="background-color: rgb(255, 0, 0);"></div>
<div class="online_status_text4" data-i18n="Not connected...">No connection...</div>
</div>`)
}
}
const display_settings = () => {
try {
$.get(`${extensionFolderPath}/html/settings/${current_model}.html`, function (data) {
$("#api_settings_menu > #settings").replaceWith(data);
})
}catch(err){
toastr.error('No Settings Found!')
}
}
/**
* This function creates the api settings drawer in the top-bar!
*/
const api_settings_drawer = async () => {
$('#top-settings-holder').prepend(
`<div id="ai_settings_drawer" class="drawer no_connection">
<div id='api-settings-toggle' class="drawer-toggle">
<i id='no_connection_icon' style="position: absolute; padding-top: 15px; padding-left: 18px" class="fa-solid fa-circle-exclamation"></i>
<div class="drawer-icon fa-solid fa-server closedIcon" title="API Settings" data-i18n="[title]API Settings"></div>
</div>
</div`)
const data = await $.ajax(`${extensionFolderPath}/html/api_drawer.html`);
$("#ai_settings_drawer").append(data);
$('#api-settings-toggle').on('click', function () {
var icon = $(this).find('.drawer-icon');
var drawer = $(this).parent().find('.drawer-content');
if (drawer.hasClass('resizing')) { return }
var drawerWasOpenAlready = $(this).parent().find('.drawer-content').hasClass('openDrawer');
let targetDrawerID = $(this).parent().find('.drawer-content').attr('id');
const pinnedDrawerClicked = drawer.hasClass('pinnedOpen');
if (!drawerWasOpenAlready) { //to open the drawer
$('.openDrawer').not('.pinnedOpen').addClass('resizing').slideToggle(200, "swing", async function () {
await delay(50); $(this).closest('.drawer-content').removeClass('resizing');
});
$('.openIcon').toggleClass('closedIcon openIcon');
$('.openDrawer').not('.pinnedOpen').toggleClass('closedDrawer openDrawer');
icon.toggleClass('openIcon closedIcon');
drawer.toggleClass('openDrawer closedDrawer');
//console.log(targetDrawerID);
if (targetDrawerID === 'right-nav-panel') {
$(this).closest('.drawer').find('.drawer-content').addClass('resizing').slideToggle({
duration: 200,
easing: "swing",
start: function () {
jQuery(this).css('display', 'flex'); //flex needed to make charlist scroll
},
complete: async function () {
favsToHotswap();
await delay(50);
$(this).closest('.drawer-content').removeClass('resizing');
$("#rm_print_characters_block").trigger("scroll");
}
})
} else {
$(this).closest('.drawer').find('.drawer-content').addClass('resizing').slideToggle(200, "swing", async function () {
await delay(50); $(this).closest('.drawer-content').removeClass('resizing');
});
}
// Set the height of "autoSetHeight" textareas within the drawer to their scroll height
$(this).closest('.drawer').find('.drawer-content textarea.autoSetHeight').each(function () {
resetScrollHeight($(this));
});
} else if (drawerWasOpenAlready) { //to close manually
icon.toggleClass('closedIcon openIcon');
if (pinnedDrawerClicked) {
$(drawer).addClass('resizing').slideToggle(200, "swing", async function () {
await delay(50); $(this).removeClass('resizing');
});
}
else {
$('.openDrawer').not('.pinnedOpen').addClass('resizing').slideToggle(200, "swing", async function () {
await delay(50); $(this).closest('.drawer-content').removeClass('resizing');
});
}
drawer.toggleClass('closedDrawer openDrawer');
}
});
$('#api_button_connect').on('click' , function(){
handle_status(true)
})
models.forEach(m => {
$('#model').append(`<option value=${m.id}>${m.name}</option>`)
})
const deviceInfo = getDeviceInfo();
if (deviceInfo && deviceInfo.device.type === 'desktop') {
// @ts-ignore
$('#model').select2({
placeholder: 'Select a model',
searchInputPlaceholder: 'Search models...',
searchInputCssClass: 'text_pole',
width: '100%',
templateResult: getModelTemplate,
});
}
jQuery(async () => {
$('#api_settings_menu').append('<div id="settings"></div>')
$("#model").on('change' , function(){
let model = String($(this).val());
current_model = model
display_settings()
})
})
display_settings()
}
/**
*
* @param {object} headers - being the headers used for the fetch request!
* @param {object} body - being the body used for the fetch request!
* @param {string} url - being the url fetched from!
* @param {string} path - being the path to the response message object(string)
*/
const get_response = async (headers,body,url,path) => {
const args = {
headers: headers,
body: body,
}
return await fetch(url, { signal: abortController.signal, method: 'POST',...args }).then(async response => {
const data = await response.json()
if (response.ok) {
return data[path]
}else{
handle_error(response)
}
})
}
/**
* This function takes in an parameter of input that it displays in the frontend chat
* @param {string} input - being the message to display
*/
const handle_response = (input) => {
if (chat[chat.length-1].mes == '...' && chat[chat.length-1].is_user == false){
deleteLastMessage()
}
Array.from(getStoppingStrings().keys()).some(key => input.replaceAll(String(key),''))
const message = {
"extra": {
display_text: String(input),
},
"name": name2,
"is_user": false,
"send_date": getdate(),
"swipe_id:": 0,
"swipes": [
{
"send_date": "November 21, 2023 6:05pm",
"gen_started": "2023-11-21T18:05:52.068Z",
"gen_finished": "2023-11-21T18:05:54.535Z",
"extra": {
"api": "novel",
"model": "kayra-v1"
}
}
],
"mes": String(input),
}
chat.push(message)
eventSource.emit(event_types.MESSAGE_RECEIVED, getContext().chat.length - 1);
addOneMessage(message)
eventSource.emit(event_types.CHARACTER_MESSAGE_RENDERED, getContext().chat.length - 1);
}
const handle_error = (err) => {
if (chat[chat.length-1].mes == '...' && chat[chat.length-1].is_user == false){
deleteLastMessage()
}
console.log(err)
toastr.error(JSON.stringify(err))
}
const handle_abort = () => {
if (chat[chat.length-1].mes == '...' && chat[chat.length-1].is_user == false){
deleteLastMessage()
}
}
/**
*
* @param {object} chat_metadata
* @param {object} characters
* @param {number} this_chid
* @returns - the prompt for the api request
*/
const get_prompt = async (chat_metadata, characters, this_chid) => {
const isInstruct = power_user.instruct.enabled
let current_chat = ''
for (const object of getContext().chat) {
current_chat += formatInstructModeChat(object.name, object.mes, object.isUser, object.isNarrator, object.forceAvatar, name1, name2, force_output_sequence.FIRST)
}
const textareaText = $("#send_text").val();
let { messageBias, promptBias, isUserPromptBias } = getBiasStrings(textareaText, 'normal');
let mesExamples = formatInstructModeExamples(baseChatReplace(characters[this_chid].mes_example.trim(), name1, name2), name1, name2)
let { worldInfoString, worldInfoBefore, worldInfoAfter, worldInfoDepth } = await getWorldInfoPrompt(chat2, this_max_context);
const storyStringParams = {
description: baseChatReplace(characters[this_chid].description.trim(), name1, name2),
personality: baseChatReplace(characters[this_chid].personality.trim(), name1, name2),
persona: baseChatReplace(power_user.persona_description.trim(), name1, name2),
scenario: baseChatReplace(chat_metadata['scenario'] || '', name1, name2),
system: power_user.prefer_character_prompt ? baseChatReplace(characters[this_chid].data?.system_prompt?.trim(), name1, name2) : '',
char: name2,
user: name1,
wiBefore: worldInfoBefore,
wiAfter: worldInfoAfter,
loreBefore: worldInfoBefore,
loreAfter: worldInfoAfter,
};
const storyString = renderStoryString(storyStringParams);
console.log(storyString)
return `${storyString}\n${mesExamples}\n${current_chat}\n${formatInstructModePrompt(name2,false,promptBias,name1,name2)}`
}
const onSend = () => {
if (getContext().characterId == null){
toastr.warning('dumbass select a character!')
}else if ($('#send_form').hasClass('no_connection') || current_model == null) {
toastr.warning('Select a model first!')
}else{
const textarea = $("#send_text");
const text = textarea.val();
send_as_user(String(text));
textarea.val("");
generate();
}
}
const hjack_send = () => {
$("#send_text").on("keydown", function(event) {
if (event.keyCode === 13) {
if (event.shiftKey) {
const textarea = $(this);
const cursorPosition = textarea.prop("selectionStart");
const text = textarea.val;
// @ts-ignore
const newText = text.substring(0, cursorPosition) + "\n" + text.substring(cursorPosition);
textarea.val(newText);
event.preventDefault();
} else {
event.preventDefault();
onSend();
}
}
})
}
jQuery(() => {
$("#send_textarea").replaceWith(`<textarea id="send_text" placeholder="Type a message, Enter + Shift for new line" name="text" style="height: 35px;"></textarea>`)
hjack_send()
api_settings_drawer()
$('#option_regenerate').on('click', () => {
if (chat[chat.length-1].is_user == false){
deleteLastMessage()
}
generate();
})
$(".option_regenerate").replaceWith(`<a class="option_regenerate"><i class="fa-lg fa-solid fa-repeat"></i><span data-i18n="Regenerate">Regenerate</span></a>`)
$("#option_convert_to_group").hide();
$("#rm_button_group_chats").hide();
$("#option_impersonate").hide();
$("#sys-settings-button").hide();
$("#option_continue").hide();
$("#ai-config-button").hide();
});
// function getCombinedPrompt(isNegative) {
// const cfgGuidanceScale = getGuidanceScale();
// // Only return if the guidance scale doesn't exist or the value is 1
// // Also don't return if constructing the neutral prompt
// if (isNegative && (!cfgGuidanceScale || cfgGuidanceScale?.value === 1)) {
// return;
// }
// // Deep clone
// let finalMesSend = structuredClone(mesSend);
// // TODO: Rewrite getExtensionPrompt to not require multiple for loops
// // Set all extension prompts where insertion depth > mesSend length
// if (finalMesSend.length) {
// for (let upperDepth = MAX_INJECTION_DEPTH; upperDepth >= finalMesSend.length; upperDepth--) {
// const upperAnchor = getExtensionPrompt(extension_prompt_types.IN_CHAT, upperDepth);
// if (upperAnchor && upperAnchor.length) {
// finalMesSend[0].extensionPrompts.push(upperAnchor);
// }
// }
// }
// finalMesSend.forEach((mesItem, index) => {
// if (index === 0) {
// return;
// }
// const anchorDepth = Math.abs(index - finalMesSend.length);
// // NOTE: Depth injected here!
// const extensionAnchor = getExtensionPrompt(extension_prompt_types.IN_CHAT, anchorDepth);
// if (anchorDepth >= 0 && extensionAnchor && extensionAnchor.length) {
// mesItem.extensionPrompts.push(extensionAnchor);
// }
// });
// // TODO: Move zero-depth anchor append to work like CFG and bias appends
// if (zeroDepthAnchor?.length && !isContinue) {
// console.debug(/\s/.test(finalMesSend[finalMesSend.length - 1].message.slice(-1)))
// finalMesSend[finalMesSend.length - 1].message +=
// /\s/.test(finalMesSend[finalMesSend.length - 1].message.slice(-1))
// ? zeroDepthAnchor
// : `${zeroDepthAnchor}`;
// }
// let cfgPrompt = {};
// if (cfgGuidanceScale && cfgGuidanceScale?.value !== 1) {
// cfgPrompt = getCfgPrompt(cfgGuidanceScale, isNegative);
// }
// if (cfgPrompt && cfgPrompt?.value) {
// if (cfgPrompt?.depth === 0) {
// finalMesSend[finalMesSend.length - 1].message +=
// /\s/.test(finalMesSend[finalMesSend.length - 1].message.slice(-1))
// ? cfgPrompt.value
// : ` ${cfgPrompt.value}`;
// } else {
// // TODO: Make all extension prompts use an array/splice method
// const lengthDiff = mesSend.length - cfgPrompt.depth;
// const cfgDepth = lengthDiff >= 0 ? lengthDiff : 0;
// finalMesSend[cfgDepth].extensionPrompts.push(`${cfgPrompt.value}\n`);
// }
// }
// // Add prompt bias after everything else
// // Always run with continue
// if (!isInstruct && !isImpersonate) {
// if (promptBias.trim().length !== 0) {
// finalMesSend[finalMesSend.length - 1].message +=
// /\s/.test(finalMesSend[finalMesSend.length - 1].message.slice(-1))
// ? promptBias.trimStart()
// : ` ${promptBias.trimStart()}`;
// }
// }
// // Prune from prompt cache if it exists
// if (generatedPromptCache.length !== 0) {
// generatedPromptCache = cleanupPromptCache(generatedPromptCache);
// }
// // Right now, everything is suffixed with a newline
// mesSendString = finalMesSend.map((e) => `${e.extensionPrompts.join('')}${e.message}`).join('');
// // add chat preamble
// mesSendString = addChatsPreamble(mesSendString);
// // add a custom dingus (if defined)
// mesSendString = addChatsSeparator(mesSendString);
// let combinedPrompt =
// beforeScenarioAnchor +
// storyString +
// afterScenarioAnchor +
// mesExmString +
// mesSendString +
// generatedPromptCache;
// combinedPrompt = combinedPrompt.replace(/\r/gm, '');
// if (power_user.collapse_newlines) {
// combinedPrompt = collapseNewlines(combinedPrompt);
// }
// return combinedPrompt;
// }
//
const getdate = () => {
const currentDateTime = new Date();
return currentDateTime.toLocaleString('en-US', {
year: 'numeric',
month: 'long',
day: 'numeric',
hour: 'numeric',
minute: 'numeric',
hour12: true
});
}
function formatMessageHistoryItem(chatItem, isInstruct, forceOutputSequence) {
const isNarratorType = false;
const characterName = chatItem?.name ? chatItem.name : name2;
const itemName = chatItem.is_user ? chatItem['name'] : characterName;
const shouldPrependName = !isNarratorType;
let textResult = shouldPrependName ? `${itemName}: ${chatItem.mes}\n` : `${chatItem.mes}\n`;
if (isInstruct) {
textResult = formatInstructModeChat(itemName, chatItem.mes, chatItem.is_user, isNarratorType, chatItem.force_avatar, name1, name2, forceOutputSequence);
}
textResult = replaceBiasMarkup(textResult);
return textResult;
}
const getModelTemplate = (option) => {
const model = models.find(x => x.id === option?.element?.value);
if (!option.id || !model) {
return option.text;
}
return $((`
<div class="flex-container flexFlowColumn">
<div><strong>${DOMPurify.sanitize(model.name)}</strong> | <span>${model.context} ctx</span></div>
<small>${DOMPurify.sanitize(model.description)}</small>
</div>
`));
}