mirror of
https://github.com/enricoros/big-AGI.git
synced 2026-05-10 21:50:14 -07:00
128 lines
4.4 KiB
TypeScript
128 lines
4.4 KiB
TypeScript
import { NextRequest, NextResponse } from 'next/server';
|
|
import { createParser } from 'eventsource-parser';
|
|
|
|
import { ApiChatInput, chatCompletionPayload, extractOpenaiChatInputs, postOpenAI } from './chat';
|
|
import { OpenAIAPI } from '@/types/api-openai';
|
|
|
|
|
|
// error function: send them down the stream as text
|
|
const sendErrorAndClose = (controller: ReadableStreamDefaultController, encoder: TextEncoder, message: string) => {
|
|
controller.enqueue(encoder.encode(message));
|
|
controller.close();
|
|
};
|
|
|
|
|
|
async function chatStreamRepeater(input: ApiChatInput, signal: AbortSignal): Promise<ReadableStream> {
|
|
|
|
// Handle the abort event when the connection is closed by the client
|
|
signal.addEventListener('abort', () => {
|
|
console.log('Client closed the connection.');
|
|
});
|
|
|
|
// begin event streaming from the OpenAI API
|
|
const encoder = new TextEncoder();
|
|
|
|
let upstreamResponse: Response;
|
|
try {
|
|
upstreamResponse = await postOpenAI(input.api, '/v1/chat/completions', chatCompletionPayload(input, true), signal);
|
|
} catch (error: any) {
|
|
console.log(error);
|
|
const message = '[OpenAI Issue] ' + (error?.message || typeof error === 'string' ? error : JSON.stringify(error)) + (error?.cause ? ' · ' + error.cause : '');
|
|
return new ReadableStream({
|
|
start: controller => sendErrorAndClose(controller, encoder, message),
|
|
});
|
|
}
|
|
|
|
// decoding and re-encoding loop
|
|
|
|
const onReadableStreamStart = async (controller: ReadableStreamDefaultController) => {
|
|
|
|
let hasBegun = false;
|
|
|
|
// stream response (SSE) from OpenAI is split into multiple chunks. this function
|
|
// will parse the event into a text stream, and re-emit it to the client
|
|
const upstreamParser = createParser(event => {
|
|
|
|
// ignore reconnect interval
|
|
if (event.type !== 'event')
|
|
return;
|
|
|
|
// https://beta.openai.com/docs/api-reference/completions/create#completions/create-stream
|
|
if (event.data === '[DONE]') {
|
|
controller.close();
|
|
return;
|
|
}
|
|
|
|
try {
|
|
const json: OpenAIAPI.Chat.CompletionsResponseChunked = JSON.parse(event.data);
|
|
|
|
// ignore any 'role' delta update
|
|
if (json.choices[0].delta?.role)
|
|
return;
|
|
|
|
// stringify and send the first packet as a JSON object
|
|
if (!hasBegun) {
|
|
hasBegun = true;
|
|
const firstPacket: ApiChatFirstOutput = {
|
|
model: json.model,
|
|
};
|
|
controller.enqueue(encoder.encode(JSON.stringify(firstPacket)));
|
|
}
|
|
|
|
// transmit the text stream
|
|
const text = json.choices[0].delta?.content || '';
|
|
controller.enqueue(encoder.encode(text));
|
|
|
|
} catch (error) {
|
|
// maybe parse error
|
|
console.error('Error parsing OpenAI response', error);
|
|
controller.error(error);
|
|
}
|
|
});
|
|
|
|
// https://web.dev/streams/#asynchronous-iteration
|
|
const decoder = new TextDecoder();
|
|
for await (const upstreamChunk of upstreamResponse.body as any)
|
|
upstreamParser.feed(decoder.decode(upstreamChunk));
|
|
|
|
};
|
|
|
|
return new ReadableStream({
|
|
start: onReadableStreamStart,
|
|
cancel: (reason) => console.log('chatStreamRepeater cancelled', reason),
|
|
});
|
|
}
|
|
|
|
|
|
/**
|
|
* The client will be sent a stream of words. As an extra (an totally optional) 'data channel' we send a
|
|
* string JSON object with the few initial variables. We hope in the future to adopt a better
|
|
* solution (e.g. websockets, but that will exclude deployment in Edge Functions).
|
|
*/
|
|
export interface ApiChatFirstOutput {
|
|
model: string;
|
|
}
|
|
|
|
export default async function handler(req: NextRequest): Promise<Response> {
|
|
try {
|
|
const apiChatInput = await extractOpenaiChatInputs(req);
|
|
const stream: ReadableStream = await chatStreamRepeater(apiChatInput, req.signal);
|
|
return new NextResponse(stream);
|
|
} catch (error: any) {
|
|
if (error.name === 'AbortError') {
|
|
console.log('Fetch request aborted in handler');
|
|
return new Response('Request aborted by the user.', { status: 499 }); // Use 499 status code for client closed request
|
|
} else if (error.code === 'ECONNRESET') {
|
|
console.log('Connection reset by the client in handler');
|
|
return new Response('Connection reset by the client.', { status: 499 }); // Use 499 status code for client closed request
|
|
} else {
|
|
console.error('Fetch request failed:', error);
|
|
return new NextResponse(`[Issue] ${error}`, { status: 400 });
|
|
}
|
|
}
|
|
};
|
|
|
|
//noinspection JSUnusedGlobalSymbols
|
|
export const config = {
|
|
runtime: 'edge',
|
|
}; |