requests.ts 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118
  1. import type { ChatRequest, ChatReponse } from "./api/chat/typing";
  2. import { Message } from "./store";
  3. const makeRequestParam = (
  4. messages: Message[],
  5. options?: {
  6. filterBot?: boolean;
  7. stream?: boolean;
  8. }
  9. ): ChatRequest => {
  10. let sendMessages = messages.map((v) => ({
  11. role: v.role,
  12. content: v.content,
  13. }));
  14. if (options?.filterBot) {
  15. sendMessages = sendMessages.filter((m) => m.role !== "assistant");
  16. }
  17. return {
  18. model: "gpt-3.5-turbo",
  19. messages: sendMessages,
  20. stream: options?.stream,
  21. };
  22. };
  23. export async function requestChat(messages: Message[]) {
  24. const req: ChatRequest = makeRequestParam(messages, { filterBot: true });
  25. const res = await fetch("/api/chat", {
  26. method: "POST",
  27. headers: {
  28. "Content-Type": "application/json",
  29. },
  30. body: JSON.stringify(req),
  31. });
  32. return (await res.json()) as ChatReponse;
  33. }
  34. export async function requestChatStream(
  35. messages: Message[],
  36. options?: {
  37. filterBot?: boolean;
  38. onMessage: (message: string, done: boolean) => void;
  39. onError: (error: Error) => void;
  40. }
  41. ) {
  42. const req = makeRequestParam(messages, {
  43. stream: true,
  44. filterBot: options?.filterBot,
  45. });
  46. const controller = new AbortController();
  47. const reqTimeoutId = setTimeout(() => controller.abort(), 10000);
  48. try {
  49. const res = await fetch("/api/chat-stream", {
  50. method: "POST",
  51. headers: {
  52. "Content-Type": "application/json",
  53. },
  54. body: JSON.stringify(req),
  55. signal: controller.signal,
  56. });
  57. clearTimeout(reqTimeoutId);
  58. let responseText = "";
  59. const finish = () => {
  60. options?.onMessage(responseText, true);
  61. controller.abort();
  62. };
  63. if (res.ok) {
  64. const reader = res.body?.getReader();
  65. const decoder = new TextDecoder();
  66. while (true) {
  67. // handle time out, will stop if no response in 10 secs
  68. const resTimeoutId = setTimeout(() => finish(), 10000);
  69. const content = await reader?.read();
  70. clearTimeout(resTimeoutId);
  71. const text = decoder.decode(content?.value);
  72. responseText += text;
  73. const done = !content || content.done;
  74. options?.onMessage(responseText, false);
  75. if (done) {
  76. break;
  77. }
  78. }
  79. finish();
  80. } else {
  81. console.error("Stream Error");
  82. options?.onError(new Error("Stream Error"));
  83. }
  84. } catch (err) {
  85. console.error("NetWork Error");
  86. options?.onError(new Error("NetWork Error"));
  87. }
  88. }
  89. export async function requestWithPrompt(messages: Message[], prompt: string) {
  90. messages = messages.concat([
  91. {
  92. role: "user",
  93. content: prompt,
  94. date: new Date().toLocaleString(),
  95. },
  96. ]);
  97. const res = await requestChat(messages);
  98. return res.choices.at(0)?.message?.content ?? "";
  99. }