store.ts 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344
  1. import { create } from "zustand";
  2. import { persist } from "zustand/middleware";
  3. import { type ChatCompletionResponseMessage } from "openai";
  4. import { requestChat, requestChatStream, requestWithPrompt } from "./requests";
  5. import { trimTopic } from "./utils";
  6. export type Message = ChatCompletionResponseMessage & {
  7. date: string;
  8. streaming?: boolean;
  9. };
  10. export enum SubmitKey {
  11. Enter = "Enter",
  12. CtrlEnter = "Ctrl + Enter",
  13. ShiftEnter = "Shift + Enter",
  14. AltEnter = "Alt + Enter",
  15. }
  16. export enum Theme {
  17. Auto = "auto",
  18. Dark = "dark",
  19. Light = "light",
  20. }
  21. export interface ChatConfig {
  22. maxToken?: number;
  23. historyMessageCount: number; // -1 means all
  24. compressMessageLengthThreshold: number;
  25. sendBotMessages: boolean; // send bot's message or not
  26. submitKey: SubmitKey;
  27. avatar: string;
  28. theme: Theme;
  29. tightBorder: boolean;
  30. }
  31. const DEFAULT_CONFIG: ChatConfig = {
  32. historyMessageCount: 4,
  33. compressMessageLengthThreshold: 500,
  34. sendBotMessages: true as boolean,
  35. submitKey: SubmitKey.CtrlEnter as SubmitKey,
  36. avatar: "1f603",
  37. theme: Theme.Auto as Theme,
  38. tightBorder: false,
  39. };
  40. export interface ChatStat {
  41. tokenCount: number;
  42. wordCount: number;
  43. charCount: number;
  44. }
  45. export interface ChatSession {
  46. id: number;
  47. topic: string;
  48. memoryPrompt: string;
  49. messages: Message[];
  50. stat: ChatStat;
  51. lastUpdate: string;
  52. lastSummarizeIndex: number;
  53. }
  54. const DEFAULT_TOPIC = "新的聊天";
  55. function createEmptySession(): ChatSession {
  56. const createDate = new Date().toLocaleString();
  57. return {
  58. id: Date.now(),
  59. topic: DEFAULT_TOPIC,
  60. memoryPrompt: "",
  61. messages: [
  62. {
  63. role: "assistant",
  64. content: "有什么可以帮你的吗",
  65. date: createDate,
  66. },
  67. ],
  68. stat: {
  69. tokenCount: 0,
  70. wordCount: 0,
  71. charCount: 0,
  72. },
  73. lastUpdate: createDate,
  74. lastSummarizeIndex: 0,
  75. };
  76. }
  77. interface ChatStore {
  78. config: ChatConfig;
  79. sessions: ChatSession[];
  80. currentSessionIndex: number;
  81. removeSession: (index: number) => void;
  82. selectSession: (index: number) => void;
  83. newSession: () => void;
  84. currentSession: () => ChatSession;
  85. onNewMessage: (message: Message) => void;
  86. onUserInput: (content: string) => Promise<void>;
  87. summarizeSession: () => void;
  88. updateStat: (message: Message) => void;
  89. updateCurrentSession: (updater: (session: ChatSession) => void) => void;
  90. updateMessage: (
  91. sessionIndex: number,
  92. messageIndex: number,
  93. updater: (message?: Message) => void
  94. ) => void;
  95. getMessagesWithMemory: () => Message[];
  96. getConfig: () => ChatConfig;
  97. resetConfig: () => void;
  98. updateConfig: (updater: (config: ChatConfig) => void) => void;
  99. clearAllData: () => void;
  100. }
  101. const LOCAL_KEY = "chat-next-web-store";
  102. export const useChatStore = create<ChatStore>()(
  103. persist(
  104. (set, get) => ({
  105. sessions: [createEmptySession()],
  106. currentSessionIndex: 0,
  107. config: {
  108. ...DEFAULT_CONFIG,
  109. },
  110. resetConfig() {
  111. set(() => ({ config: { ...DEFAULT_CONFIG } }));
  112. },
  113. getConfig() {
  114. return get().config;
  115. },
  116. updateConfig(updater) {
  117. const config = get().config;
  118. updater(config);
  119. set(() => ({ config }));
  120. },
  121. selectSession(index: number) {
  122. set({
  123. currentSessionIndex: index,
  124. });
  125. },
  126. removeSession(index: number) {
  127. set((state) => {
  128. let nextIndex = state.currentSessionIndex;
  129. const sessions = state.sessions;
  130. if (sessions.length === 1) {
  131. return {
  132. currentSessionIndex: 0,
  133. sessions: [createEmptySession()],
  134. };
  135. }
  136. sessions.splice(index, 1);
  137. if (nextIndex === index) {
  138. nextIndex -= 1;
  139. }
  140. return {
  141. currentSessionIndex: nextIndex,
  142. sessions,
  143. };
  144. });
  145. },
  146. newSession() {
  147. set((state) => ({
  148. currentSessionIndex: 0,
  149. sessions: [createEmptySession()].concat(state.sessions),
  150. }));
  151. },
  152. currentSession() {
  153. let index = get().currentSessionIndex;
  154. const sessions = get().sessions;
  155. if (index < 0 || index >= sessions.length) {
  156. index = Math.min(sessions.length - 1, Math.max(0, index));
  157. set(() => ({ currentSessionIndex: index }));
  158. }
  159. const session = sessions[index];
  160. return session;
  161. },
  162. onNewMessage(message) {
  163. get().updateCurrentSession(session => {
  164. session.lastUpdate = new Date().toLocaleString()
  165. })
  166. get().updateStat(message);
  167. get().summarizeSession();
  168. },
  169. async onUserInput(content) {
  170. const userMessage: Message = {
  171. role: "user",
  172. content,
  173. date: new Date().toLocaleString(),
  174. };
  175. const botMessage: Message = {
  176. content: "",
  177. role: "assistant",
  178. date: new Date().toLocaleString(),
  179. streaming: true,
  180. };
  181. // get recent messages
  182. const recentMessages = get().getMessagesWithMemory()
  183. const sendMessages = recentMessages.concat(userMessage)
  184. // save user's and bot's message
  185. get().updateCurrentSession((session) => {
  186. session.messages.push(userMessage);
  187. session.messages.push(botMessage);
  188. });
  189. console.log('[User Input] ', sendMessages)
  190. requestChatStream(sendMessages, {
  191. onMessage(content, done) {
  192. if (done) {
  193. botMessage.streaming = false;
  194. get().onNewMessage(botMessage)
  195. } else {
  196. botMessage.content = content;
  197. set(() => ({}));
  198. }
  199. },
  200. onError(error) {
  201. botMessage.content += "\n\n出错了,稍后重试吧";
  202. botMessage.streaming = false;
  203. set(() => ({}));
  204. },
  205. filterBot: !get().config.sendBotMessages,
  206. });
  207. },
  208. getMessagesWithMemory() {
  209. const session = get().currentSession()
  210. const config = get().config
  211. const n = session.messages.length
  212. const recentMessages = session.messages.slice(n - config.historyMessageCount);
  213. const memoryPrompt: Message = {
  214. role: 'system',
  215. content: '这是 ai 和用户的历史聊天总结作为前情提要:' + session.memoryPrompt,
  216. date: ''
  217. }
  218. if (session.memoryPrompt) {
  219. recentMessages.unshift(memoryPrompt)
  220. }
  221. return recentMessages
  222. },
  223. updateMessage(
  224. sessionIndex: number,
  225. messageIndex: number,
  226. updater: (message?: Message) => void
  227. ) {
  228. const sessions = get().sessions;
  229. const session = sessions.at(sessionIndex);
  230. const messages = session?.messages;
  231. updater(messages?.at(messageIndex));
  232. set(() => ({ sessions }));
  233. },
  234. summarizeSession() {
  235. const session = get().currentSession();
  236. if (session.topic === DEFAULT_TOPIC) {
  237. // should summarize topic
  238. requestWithPrompt(
  239. session.messages,
  240. "直接返回这句话的简要主题,不要解释,如果没有主题,请直接返回“闲聊”"
  241. ).then((res) => {
  242. get().updateCurrentSession(
  243. (session) => (session.topic = trimTopic(res))
  244. );
  245. });
  246. }
  247. const config = get().config
  248. const messages = get().getMessagesWithMemory()
  249. const toBeSummarizedMsgs = get().getMessagesWithMemory()
  250. const historyMsgLength = toBeSummarizedMsgs.reduce((pre, cur) => pre + cur.content.length, 0)
  251. const lastSummarizeIndex = session.messages.length
  252. console.log('[Chat History] ', messages, historyMsgLength, config.compressMessageLengthThreshold)
  253. if (historyMsgLength > config.compressMessageLengthThreshold) {
  254. requestChatStream(toBeSummarizedMsgs.concat({
  255. role: 'system',
  256. content: '总结一下 ai 和用户的对话,用作后续的上下文提示 prompt,控制在 100 字以内,你在回复时用 ai 自称',
  257. date: ''
  258. }), {
  259. filterBot: false,
  260. onMessage(message, done) {
  261. session.memoryPrompt = message
  262. if (done) {
  263. console.log('[Memory] ', session.memoryPrompt)
  264. session.lastSummarizeIndex = lastSummarizeIndex
  265. }
  266. },
  267. onError(error) {
  268. console.error('[Summarize] ', error)
  269. },
  270. })
  271. }
  272. },
  273. updateStat(message) {
  274. get().updateCurrentSession((session) => {
  275. session.stat.charCount += message.content.length;
  276. // TODO: should update chat count and word count
  277. });
  278. },
  279. updateCurrentSession(updater) {
  280. const sessions = get().sessions;
  281. const index = get().currentSessionIndex;
  282. updater(sessions[index]);
  283. set(() => ({ sessions }));
  284. },
  285. clearAllData() {
  286. if (confirm('确认清除所有聊天、设置数据?')) {
  287. localStorage.clear()
  288. location.reload()
  289. }
  290. },
  291. }),
  292. {
  293. name: LOCAL_KEY,
  294. }
  295. )
  296. );