|
@@ -14,18 +14,35 @@ function appendUserMessageToCtx(msg) {
|
|
|
|
|
|
let renderCnt = 0;
|
|
|
|
|
|
-function onStreamEvent(elem, data, forceRender=false) {
|
|
|
+function extractDelta(type, data) {
|
|
|
+ if (type === 'openai') {
|
|
|
+ return data.choices?.[0]?.delta?.content;
|
|
|
+ } else if (type === 'claude') {
|
|
|
+ if (data.type !== 'content_block_delta') return '';
|
|
|
+ if (data.delta.type !== 'text_delta') return '';
|
|
|
+ return data.delta.text;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+function isStop(type, rawData) {
|
|
|
+ if (type === "openai") {
|
|
|
+ return rawData === 'data: [DONE]';
|
|
|
+ } else if (type === "claude") {
|
|
|
+ return rawData === 'event: message_stop'
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+function onStreamEvent(type, elem, data, forceRender=false) {
|
|
|
let container = document.getElementById('message-container');
|
|
|
let shouldScroll = false;
|
|
|
if (container.scrollTop + container.clientHeight >= container.scrollHeight - 100) {
|
|
|
shouldScroll = true;
|
|
|
}
|
|
|
-
|
|
|
- let deltaContent = data.choices?.[0]?.delta?.content;
|
|
|
+ let deltaContent = extractDelta(type, data);
|
|
|
if (deltaContent === undefined) deltaContent = '';
|
|
|
currentMessage = currentMessage + deltaContent;
|
|
|
renderCnt++;
|
|
|
- if (renderCnt % 10 !== 0 && !forceRender) return;
|
|
|
+ if (renderCnt % 100 !== 0 && !forceRender) return;
|
|
|
let renderContent = currentMessage.replace('<think>', '<div class="think">\n\n');
|
|
|
renderContent = renderContent.replace('</think>', '\n\n</div>').trim();
|
|
|
renderContent = renderContent
|
|
@@ -33,7 +50,7 @@ function onStreamEvent(elem, data, forceRender=false) {
|
|
|
return `\$\$${p2.trim()}\$\$`;
|
|
|
})
|
|
|
.replace(/(\\\()([\s\S]*?)(\\\))/g, (match, p1, p2, p3) => {
|
|
|
- return `\$${p2.trim()}\$`;
|
|
|
+ return `\\\\\\\(${p2.trim()}\\\\\\\)`;
|
|
|
})
|
|
|
const html = marked.parse(renderContent);
|
|
|
elem.innerHTML = html;
|
|
@@ -42,8 +59,8 @@ function onStreamEvent(elem, data, forceRender=false) {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-function onStreamEnd(elem) {
|
|
|
- onStreamEvent(elem, {}, true)
|
|
|
+function onStreamEnd(type, elem) {
|
|
|
+ onStreamEvent(type, elem, {}, true)
|
|
|
context.push({
|
|
|
role: 'assistant',
|
|
|
content: currentMessage.replace(/<think>[\s\S]*?<\/think>/g, ''),
|
|
@@ -53,17 +70,38 @@ function onStreamEnd(elem) {
|
|
|
MathJax.typesetPromise();
|
|
|
}
|
|
|
|
|
|
+function cleanContext(len) {
|
|
|
+ let getBytes = (str) => {
|
|
|
+ return new Blob([str]).size;
|
|
|
+ };
|
|
|
+ let sum = 0;
|
|
|
+ let splitIndex = context.length;
|
|
|
+ for (let i = context.length - 1; i >= 0; i--) {
|
|
|
+ const bytes = getBytes(context[i].content);
|
|
|
+ if (sum + bytes > len) {
|
|
|
+ splitIndex = i;
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ sum += bytes;
|
|
|
+ splitIndex = 0;
|
|
|
+ }
|
|
|
+ context = context.slice(splitIndex);
|
|
|
+}
|
|
|
+
|
|
|
async function fetchAiStream() {
|
|
|
const headers = {
|
|
|
'Content-Type': 'application/json',
|
|
|
'Authorization': `Bearer ${globalCurrentModel.key}`
|
|
|
};
|
|
|
+ cleanContext(globalCurrentModel.ctxLen);
|
|
|
const requestBody = {
|
|
|
model: globalCurrentModel.model,
|
|
|
messages: context,
|
|
|
- stream: true
|
|
|
+ stream: true,
|
|
|
+ max_tokens: globalCurrentModel.maxTokens,
|
|
|
};
|
|
|
let elem = createMessage('', false);
|
|
|
+ let type = globalCurrentModel.type;
|
|
|
let container = document.getElementById('message-container');
|
|
|
container.appendChild(elem);
|
|
|
container.scrollTop = container.scrollHeight;
|
|
@@ -94,23 +132,26 @@ async function fetchAiStream() {
|
|
|
}
|
|
|
if (getStopGenerating()) {
|
|
|
resetStopGenerating();
|
|
|
- onStreamEnd(elem);
|
|
|
+ onStreamEnd(type, elem);
|
|
|
return;
|
|
|
}
|
|
|
if (line === '') continue;
|
|
|
- if (line.trim() === 'data: [DONE]') break;
|
|
|
+ if (isStop(type, line.trim())) break;
|
|
|
+ if (!line.startsWith('data: ')) {
|
|
|
+ continue;
|
|
|
+ }
|
|
|
try {
|
|
|
const data = JSON.parse(line.slice(6));
|
|
|
- onStreamEvent(elem, data);
|
|
|
+ onStreamEvent(type, elem, data);
|
|
|
} catch (e) {
|
|
|
console.error('Error parsing stream data:', e);
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
- onStreamEnd(elem);
|
|
|
+ onStreamEnd(type, elem);
|
|
|
} catch (error) {
|
|
|
console.error('Error:', error);
|
|
|
- onStreamEnd(elem);
|
|
|
+ onStreamEnd(type, elem);
|
|
|
let p = document.createElement('p');
|
|
|
p.innerHTML = error;
|
|
|
p.style.color = 'red';
|