route.ts 1.4 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152
  1. import { createParser } from "eventsource-parser";
  2. import { NextRequest } from "next/server";
  3. import { requestOpenai } from "../common";
  4. async function createStream(req: NextRequest) {
  5. const encoder = new TextEncoder();
  6. const decoder = new TextDecoder();
  7. const res = await requestOpenai(req);
  8. const stream = new ReadableStream({
  9. async start(controller) {
  10. function onParse(event: any) {
  11. if (event.type === "event") {
  12. const data = event.data;
  13. // https://beta.openai.com/docs/api-reference/completions/create#completions/create-stream
  14. if (data === "[DONE]") {
  15. controller.close();
  16. return;
  17. }
  18. try {
  19. const json = JSON.parse(data);
  20. const text = json.choices[0].delta.content;
  21. const queue = encoder.encode(text);
  22. controller.enqueue(queue);
  23. } catch (e) {
  24. controller.error(e);
  25. }
  26. }
  27. }
  28. const parser = createParser(onParse);
  29. for await (const chunk of res.body as any) {
  30. parser.feed(decoder.decode(chunk));
  31. }
  32. },
  33. });
  34. return stream;
  35. }
  36. export async function POST(req: NextRequest) {
  37. try {
  38. const stream = await createStream(req);
  39. return new Response(stream);
  40. } catch (error) {
  41. console.error("[Chat Stream]", error);
  42. }
  43. }
  44. export const config = {
  45. runtime: "edge",
  46. };