@copilotkit/react-core 0.0.0-mme-load-agent-state-20250117154700

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (293) hide show
  1. package/CHANGELOG.md +2017 -0
  2. package/README.md +48 -0
  3. package/dist/chunk-4IYWTPBR.mjs +61 -0
  4. package/dist/chunk-4IYWTPBR.mjs.map +1 -0
  5. package/dist/chunk-5FHSUKQL.mjs +70 -0
  6. package/dist/chunk-5FHSUKQL.mjs.map +1 -0
  7. package/dist/chunk-6BQVZFNO.mjs +35 -0
  8. package/dist/chunk-6BQVZFNO.mjs.map +1 -0
  9. package/dist/chunk-6U3UH3KO.mjs +140 -0
  10. package/dist/chunk-6U3UH3KO.mjs.map +1 -0
  11. package/dist/chunk-BHRJWCZT.mjs +423 -0
  12. package/dist/chunk-BHRJWCZT.mjs.map +1 -0
  13. package/dist/chunk-DCTJZ742.mjs +22 -0
  14. package/dist/chunk-DCTJZ742.mjs.map +1 -0
  15. package/dist/chunk-EFZPSZWO.mjs +1 -0
  16. package/dist/chunk-EFZPSZWO.mjs.map +1 -0
  17. package/dist/chunk-IFBUUIU7.mjs +53 -0
  18. package/dist/chunk-IFBUUIU7.mjs.map +1 -0
  19. package/dist/chunk-IJJTXITZ.mjs +59 -0
  20. package/dist/chunk-IJJTXITZ.mjs.map +1 -0
  21. package/dist/chunk-IPTELVUL.mjs +221 -0
  22. package/dist/chunk-IPTELVUL.mjs.map +1 -0
  23. package/dist/chunk-J6UBNKIR.mjs +23 -0
  24. package/dist/chunk-J6UBNKIR.mjs.map +1 -0
  25. package/dist/chunk-JFMOYUAX.mjs +120 -0
  26. package/dist/chunk-JFMOYUAX.mjs.map +1 -0
  27. package/dist/chunk-LNI2Z45S.mjs +242 -0
  28. package/dist/chunk-LNI2Z45S.mjs.map +1 -0
  29. package/dist/chunk-LUGEI4YQ.mjs +1 -0
  30. package/dist/chunk-LUGEI4YQ.mjs.map +1 -0
  31. package/dist/chunk-MLAS4QUR.mjs +18 -0
  32. package/dist/chunk-MLAS4QUR.mjs.map +1 -0
  33. package/dist/chunk-N37EFAUX.mjs +28 -0
  34. package/dist/chunk-N37EFAUX.mjs.map +1 -0
  35. package/dist/chunk-NR2BTA6Z.mjs +1 -0
  36. package/dist/chunk-NR2BTA6Z.mjs.map +1 -0
  37. package/dist/chunk-O7ARI5CV.mjs +31 -0
  38. package/dist/chunk-O7ARI5CV.mjs.map +1 -0
  39. package/dist/chunk-SKC7AJIV.mjs +61 -0
  40. package/dist/chunk-SKC7AJIV.mjs.map +1 -0
  41. package/dist/chunk-U6JOA64X.mjs +107 -0
  42. package/dist/chunk-U6JOA64X.mjs.map +1 -0
  43. package/dist/chunk-VNTDIJSG.mjs +37 -0
  44. package/dist/chunk-VNTDIJSG.mjs.map +1 -0
  45. package/dist/chunk-VRQFYEQF.mjs +181 -0
  46. package/dist/chunk-VRQFYEQF.mjs.map +1 -0
  47. package/dist/chunk-WZZEEFIX.mjs +500 -0
  48. package/dist/chunk-WZZEEFIX.mjs.map +1 -0
  49. package/dist/chunk-XERJQUHA.mjs +31 -0
  50. package/dist/chunk-XERJQUHA.mjs.map +1 -0
  51. package/dist/chunk-YPSGKPDA.mjs +1 -0
  52. package/dist/chunk-YPSGKPDA.mjs.map +1 -0
  53. package/dist/chunk-YVQNSJ4Z.mjs +90 -0
  54. package/dist/chunk-YVQNSJ4Z.mjs.map +1 -0
  55. package/dist/components/copilot-provider/copilot-messages.d.ts +16 -0
  56. package/dist/components/copilot-provider/copilot-messages.js +203 -0
  57. package/dist/components/copilot-provider/copilot-messages.js.map +1 -0
  58. package/dist/components/copilot-provider/copilot-messages.mjs +10 -0
  59. package/dist/components/copilot-provider/copilot-messages.mjs.map +1 -0
  60. package/dist/components/copilot-provider/copilotkit-props.d.ts +97 -0
  61. package/dist/components/copilot-provider/copilotkit-props.js +19 -0
  62. package/dist/components/copilot-provider/copilotkit-props.js.map +1 -0
  63. package/dist/components/copilot-provider/copilotkit-props.mjs +1 -0
  64. package/dist/components/copilot-provider/copilotkit-props.mjs.map +1 -0
  65. package/dist/components/copilot-provider/copilotkit.d.ts +18 -0
  66. package/dist/components/copilot-provider/copilotkit.js +1037 -0
  67. package/dist/components/copilot-provider/copilotkit.js.map +1 -0
  68. package/dist/components/copilot-provider/copilotkit.mjs +22 -0
  69. package/dist/components/copilot-provider/copilotkit.mjs.map +1 -0
  70. package/dist/components/copilot-provider/index.d.ts +13 -0
  71. package/dist/components/copilot-provider/index.js +1037 -0
  72. package/dist/components/copilot-provider/index.js.map +1 -0
  73. package/dist/components/copilot-provider/index.mjs +20 -0
  74. package/dist/components/copilot-provider/index.mjs.map +1 -0
  75. package/dist/components/error-boundary/error-boundary.d.ts +22 -0
  76. package/dist/components/error-boundary/error-boundary.js +183 -0
  77. package/dist/components/error-boundary/error-boundary.js.map +1 -0
  78. package/dist/components/error-boundary/error-boundary.mjs +12 -0
  79. package/dist/components/error-boundary/error-boundary.mjs.map +1 -0
  80. package/dist/components/error-boundary/error-utils.d.ts +11 -0
  81. package/dist/components/error-boundary/error-utils.js +177 -0
  82. package/dist/components/error-boundary/error-utils.js.map +1 -0
  83. package/dist/components/error-boundary/error-utils.mjs +13 -0
  84. package/dist/components/error-boundary/error-utils.mjs.map +1 -0
  85. package/dist/components/index.d.ts +13 -0
  86. package/dist/components/index.js +1037 -0
  87. package/dist/components/index.js.map +1 -0
  88. package/dist/components/index.mjs +20 -0
  89. package/dist/components/index.mjs.map +1 -0
  90. package/dist/components/toast/exclamation-mark-icon.d.ts +9 -0
  91. package/dist/components/toast/exclamation-mark-icon.js +55 -0
  92. package/dist/components/toast/exclamation-mark-icon.js.map +1 -0
  93. package/dist/components/toast/exclamation-mark-icon.mjs +8 -0
  94. package/dist/components/toast/exclamation-mark-icon.mjs.map +1 -0
  95. package/dist/components/toast/toast-provider.d.ts +30 -0
  96. package/dist/components/toast/toast-provider.js +269 -0
  97. package/dist/components/toast/toast-provider.js.map +1 -0
  98. package/dist/components/toast/toast-provider.mjs +11 -0
  99. package/dist/components/toast/toast-provider.mjs.map +1 -0
  100. package/dist/context/copilot-context.d.ts +144 -0
  101. package/dist/context/copilot-context.js +141 -0
  102. package/dist/context/copilot-context.js.map +1 -0
  103. package/dist/context/copilot-context.mjs +10 -0
  104. package/dist/context/copilot-context.mjs.map +1 -0
  105. package/dist/context/copilot-messages-context.d.ts +15 -0
  106. package/dist/context/copilot-messages-context.js +57 -0
  107. package/dist/context/copilot-messages-context.js.map +1 -0
  108. package/dist/context/copilot-messages-context.mjs +10 -0
  109. package/dist/context/copilot-messages-context.mjs.map +1 -0
  110. package/dist/context/index.d.ts +11 -0
  111. package/dist/context/index.js +164 -0
  112. package/dist/context/index.js.map +1 -0
  113. package/dist/context/index.mjs +17 -0
  114. package/dist/context/index.mjs.map +1 -0
  115. package/dist/hooks/index.d.ts +21 -0
  116. package/dist/hooks/index.js +1350 -0
  117. package/dist/hooks/index.js.map +1 -0
  118. package/dist/hooks/index.mjs +56 -0
  119. package/dist/hooks/index.mjs.map +1 -0
  120. package/dist/hooks/use-chat.d.ts +138 -0
  121. package/dist/hooks/use-chat.js +628 -0
  122. package/dist/hooks/use-chat.js.map +1 -0
  123. package/dist/hooks/use-chat.mjs +12 -0
  124. package/dist/hooks/use-chat.mjs.map +1 -0
  125. package/dist/hooks/use-coagent-state-render.d.ts +60 -0
  126. package/dist/hooks/use-coagent-state-render.js +198 -0
  127. package/dist/hooks/use-coagent-state-render.js.map +1 -0
  128. package/dist/hooks/use-coagent-state-render.mjs +11 -0
  129. package/dist/hooks/use-coagent-state-render.mjs.map +1 -0
  130. package/dist/hooks/use-coagent.d.ts +199 -0
  131. package/dist/hooks/use-coagent.js +1111 -0
  132. package/dist/hooks/use-coagent.js.map +1 -0
  133. package/dist/hooks/use-coagent.mjs +29 -0
  134. package/dist/hooks/use-coagent.mjs.map +1 -0
  135. package/dist/hooks/use-copilot-action.d.ts +140 -0
  136. package/dist/hooks/use-copilot-action.js +372 -0
  137. package/dist/hooks/use-copilot-action.js.map +1 -0
  138. package/dist/hooks/use-copilot-action.mjs +11 -0
  139. package/dist/hooks/use-copilot-action.mjs.map +1 -0
  140. package/dist/hooks/use-copilot-authenticated-action.d.ts +8 -0
  141. package/dist/hooks/use-copilot-authenticated-action.js +421 -0
  142. package/dist/hooks/use-copilot-authenticated-action.js.map +1 -0
  143. package/dist/hooks/use-copilot-authenticated-action.mjs +12 -0
  144. package/dist/hooks/use-copilot-authenticated-action.mjs.map +1 -0
  145. package/dist/hooks/use-copilot-chat.d.ts +48 -0
  146. package/dist/hooks/use-copilot-chat.js +957 -0
  147. package/dist/hooks/use-copilot-chat.js.map +1 -0
  148. package/dist/hooks/use-copilot-chat.mjs +24 -0
  149. package/dist/hooks/use-copilot-chat.mjs.map +1 -0
  150. package/dist/hooks/use-copilot-readable.d.ts +33 -0
  151. package/dist/hooks/use-copilot-readable.js +161 -0
  152. package/dist/hooks/use-copilot-readable.js.map +1 -0
  153. package/dist/hooks/use-copilot-readable.mjs +9 -0
  154. package/dist/hooks/use-copilot-readable.mjs.map +1 -0
  155. package/dist/hooks/use-copilot-runtime-client.d.ts +5 -0
  156. package/dist/hooks/use-copilot-runtime-client.js +182 -0
  157. package/dist/hooks/use-copilot-runtime-client.js.map +1 -0
  158. package/dist/hooks/use-copilot-runtime-client.mjs +10 -0
  159. package/dist/hooks/use-copilot-runtime-client.mjs.map +1 -0
  160. package/dist/hooks/use-flat-category-store.d.ts +9 -0
  161. package/dist/hooks/use-flat-category-store.js +90 -0
  162. package/dist/hooks/use-flat-category-store.js.map +1 -0
  163. package/dist/hooks/use-flat-category-store.mjs +8 -0
  164. package/dist/hooks/use-flat-category-store.mjs.map +1 -0
  165. package/dist/hooks/use-make-copilot-document-readable.d.ts +12 -0
  166. package/dist/hooks/use-make-copilot-document-readable.js +156 -0
  167. package/dist/hooks/use-make-copilot-document-readable.js.map +1 -0
  168. package/dist/hooks/use-make-copilot-document-readable.mjs +9 -0
  169. package/dist/hooks/use-make-copilot-document-readable.mjs.map +1 -0
  170. package/dist/hooks/use-tree.d.ts +18 -0
  171. package/dist/hooks/use-tree.js +172 -0
  172. package/dist/hooks/use-tree.js.map +1 -0
  173. package/dist/hooks/use-tree.mjs +8 -0
  174. package/dist/hooks/use-tree.mjs.map +1 -0
  175. package/dist/index.d.ts +27 -0
  176. package/dist/index.js +2308 -0
  177. package/dist/index.js.map +1 -0
  178. package/dist/index.mjs +83 -0
  179. package/dist/index.mjs.map +1 -0
  180. package/dist/lib/copilot-task.d.ts +97 -0
  181. package/dist/lib/copilot-task.js +195 -0
  182. package/dist/lib/copilot-task.js.map +1 -0
  183. package/dist/lib/copilot-task.mjs +20 -0
  184. package/dist/lib/copilot-task.mjs.map +1 -0
  185. package/dist/lib/index.d.ts +11 -0
  186. package/dist/lib/index.js +197 -0
  187. package/dist/lib/index.js.map +1 -0
  188. package/dist/lib/index.mjs +21 -0
  189. package/dist/lib/index.mjs.map +1 -0
  190. package/dist/types/chat-suggestion-configuration.d.ts +22 -0
  191. package/dist/types/chat-suggestion-configuration.js +19 -0
  192. package/dist/types/chat-suggestion-configuration.js.map +1 -0
  193. package/dist/types/chat-suggestion-configuration.mjs +1 -0
  194. package/dist/types/chat-suggestion-configuration.mjs.map +1 -0
  195. package/dist/types/coagent-action.d.ts +29 -0
  196. package/dist/types/coagent-action.js +19 -0
  197. package/dist/types/coagent-action.js.map +1 -0
  198. package/dist/types/coagent-action.mjs +1 -0
  199. package/dist/types/coagent-action.mjs.map +1 -0
  200. package/dist/types/coagent-state.d.ts +11 -0
  201. package/dist/types/coagent-state.js +19 -0
  202. package/dist/types/coagent-state.js.map +1 -0
  203. package/dist/types/coagent-state.mjs +1 -0
  204. package/dist/types/coagent-state.mjs.map +1 -0
  205. package/dist/types/document-pointer.d.ts +9 -0
  206. package/dist/types/document-pointer.js +19 -0
  207. package/dist/types/document-pointer.js.map +1 -0
  208. package/dist/types/document-pointer.mjs +1 -0
  209. package/dist/types/document-pointer.mjs.map +1 -0
  210. package/dist/types/frontend-action.d.ts +126 -0
  211. package/dist/types/frontend-action.js +53 -0
  212. package/dist/types/frontend-action.js.map +1 -0
  213. package/dist/types/frontend-action.mjs +8 -0
  214. package/dist/types/frontend-action.mjs.map +1 -0
  215. package/dist/types/index.d.ts +7 -0
  216. package/dist/types/index.js +19 -0
  217. package/dist/types/index.js.map +1 -0
  218. package/dist/types/index.mjs +2 -0
  219. package/dist/types/index.mjs.map +1 -0
  220. package/dist/types/system-message.d.ts +3 -0
  221. package/dist/types/system-message.js +19 -0
  222. package/dist/types/system-message.js.map +1 -0
  223. package/dist/types/system-message.mjs +1 -0
  224. package/dist/types/system-message.mjs.map +1 -0
  225. package/dist/utils/dev-console.d.ts +3 -0
  226. package/dist/utils/dev-console.js +42 -0
  227. package/dist/utils/dev-console.js.map +1 -0
  228. package/dist/utils/dev-console.mjs +8 -0
  229. package/dist/utils/dev-console.mjs.map +1 -0
  230. package/dist/utils/extract.d.ts +43 -0
  231. package/dist/utils/extract.js +208 -0
  232. package/dist/utils/extract.js.map +1 -0
  233. package/dist/utils/extract.mjs +18 -0
  234. package/dist/utils/extract.mjs.map +1 -0
  235. package/dist/utils/index.d.ts +13 -0
  236. package/dist/utils/index.js +226 -0
  237. package/dist/utils/index.js.map +1 -0
  238. package/dist/utils/index.mjs +21 -0
  239. package/dist/utils/index.mjs.map +1 -0
  240. package/dist/utils/utils.d.ts +2 -0
  241. package/dist/utils/utils.js +19 -0
  242. package/dist/utils/utils.js.map +1 -0
  243. package/dist/utils/utils.mjs +1 -0
  244. package/dist/utils/utils.mjs.map +1 -0
  245. package/dist/utils/utils.test.d.ts +2 -0
  246. package/dist/utils/utils.test.js +9 -0
  247. package/dist/utils/utils.test.js.map +1 -0
  248. package/dist/utils/utils.test.mjs +7 -0
  249. package/dist/utils/utils.test.mjs.map +1 -0
  250. package/jest.config.js +5 -0
  251. package/package.json +69 -0
  252. package/src/components/copilot-provider/copilot-messages.tsx +45 -0
  253. package/src/components/copilot-provider/copilotkit-props.tsx +102 -0
  254. package/src/components/copilot-provider/copilotkit.tsx +427 -0
  255. package/src/components/copilot-provider/index.ts +3 -0
  256. package/src/components/error-boundary/error-boundary.tsx +42 -0
  257. package/src/components/error-boundary/error-utils.tsx +95 -0
  258. package/src/components/index.ts +1 -0
  259. package/src/components/toast/exclamation-mark-icon.tsx +27 -0
  260. package/src/components/toast/toast-provider.tsx +170 -0
  261. package/src/context/copilot-context.tsx +280 -0
  262. package/src/context/copilot-messages-context.tsx +29 -0
  263. package/src/context/index.ts +8 -0
  264. package/src/hooks/index.ts +12 -0
  265. package/src/hooks/use-chat.ts +682 -0
  266. package/src/hooks/use-coagent-state-render.ts +118 -0
  267. package/src/hooks/use-coagent.ts +394 -0
  268. package/src/hooks/use-copilot-action.ts +289 -0
  269. package/src/hooks/use-copilot-authenticated-action.ts +60 -0
  270. package/src/hooks/use-copilot-chat.ts +310 -0
  271. package/src/hooks/use-copilot-readable.ts +123 -0
  272. package/src/hooks/use-copilot-runtime-client.ts +28 -0
  273. package/src/hooks/use-flat-category-store.ts +104 -0
  274. package/src/hooks/use-make-copilot-document-readable.ts +30 -0
  275. package/src/hooks/use-tree.ts +202 -0
  276. package/src/index.tsx +7 -0
  277. package/src/lib/copilot-task.ts +201 -0
  278. package/src/lib/index.ts +1 -0
  279. package/src/types/chat-suggestion-configuration.ts +23 -0
  280. package/src/types/coagent-action.ts +31 -0
  281. package/src/types/coagent-state.ts +9 -0
  282. package/src/types/document-pointer.ts +7 -0
  283. package/src/types/frontend-action.ts +199 -0
  284. package/src/types/index.ts +8 -0
  285. package/src/types/system-message.ts +4 -0
  286. package/src/utils/dev-console.ts +18 -0
  287. package/src/utils/extract.ts +215 -0
  288. package/src/utils/index.ts +2 -0
  289. package/src/utils/utils.test.ts +7 -0
  290. package/src/utils/utils.ts +1 -0
  291. package/tsconfig.json +12 -0
  292. package/tsup.config.ts +16 -0
  293. package/typedoc.json +4 -0
@@ -0,0 +1,682 @@
1
+ import { useRef } from "react";
2
+ import {
3
+ FunctionCallHandler,
4
+ COPILOT_CLOUD_PUBLIC_API_KEY_HEADER,
5
+ CoAgentStateRenderHandler,
6
+ randomId,
7
+ } from "@copilotkit/shared";
8
+ import {
9
+ Message,
10
+ TextMessage,
11
+ ResultMessage,
12
+ convertMessagesToGqlInput,
13
+ filterAdjacentAgentStateMessages,
14
+ filterAgentStateMessages,
15
+ convertGqlOutputToMessages,
16
+ MessageStatusCode,
17
+ MessageRole,
18
+ Role,
19
+ CopilotRequestType,
20
+ ForwardedParametersInput,
21
+ loadMessagesFromJsonRepresentation,
22
+ ExtensionsInput,
23
+ CopilotRuntimeClient,
24
+ } from "@copilotkit/runtime-client-gql";
25
+
26
+ import { CopilotApiConfig } from "../context";
27
+ import { FrontendAction, processActionsForRuntimeRequest } from "../types/frontend-action";
28
+ import { CoagentState } from "../types/coagent-state";
29
+ import { AgentSession } from "../context/copilot-context";
30
+ import { useCopilotRuntimeClient } from "./use-copilot-runtime-client";
31
+ import { useAsyncCallback, useErrorToast } from "../components/error-boundary/error-utils";
32
+
33
+ export type UseChatOptions = {
34
+ /**
35
+ * System messages of the chat. Defaults to an empty array.
36
+ */
37
+ initialMessages?: Message[];
38
+ /**
39
+ * Callback function to be called when a function call is received.
40
+ * If the function returns a `ChatRequest` object, the request will be sent
41
+ * automatically to the API and will be used to update the chat.
42
+ */
43
+ onFunctionCall?: FunctionCallHandler;
44
+
45
+ /**
46
+ * Callback function to be called when a coagent action is received.
47
+ */
48
+ onCoAgentStateRender?: CoAgentStateRenderHandler;
49
+
50
+ /**
51
+ * Function definitions to be sent to the API.
52
+ */
53
+ actions: FrontendAction<any>[];
54
+
55
+ /**
56
+ * The CopilotKit API configuration.
57
+ */
58
+ copilotConfig: CopilotApiConfig;
59
+
60
+ /**
61
+ * The current list of messages in the chat.
62
+ */
63
+ messages: Message[];
64
+ /**
65
+ * The setState-powered method to update the chat messages.
66
+ */
67
+ setMessages: React.Dispatch<React.SetStateAction<Message[]>>;
68
+
69
+ /**
70
+ * A callback to get the latest system message.
71
+ */
72
+ makeSystemMessageCallback: () => TextMessage;
73
+
74
+ /**
75
+ * Whether the API request is in progress
76
+ */
77
+ isLoading: boolean;
78
+
79
+ /**
80
+ * setState-powered method to update the isChatLoading value
81
+ */
82
+ setIsLoading: React.Dispatch<React.SetStateAction<boolean>>;
83
+
84
+ /**
85
+ * The current list of coagent states.
86
+ */
87
+ coagentStatesRef: React.RefObject<Record<string, CoagentState>>;
88
+
89
+ /**
90
+ * setState-powered method to update the agent states
91
+ */
92
+ setCoagentStatesWithRef: React.Dispatch<React.SetStateAction<Record<string, CoagentState>>>;
93
+
94
+ /**
95
+ * The current agent session.
96
+ */
97
+ agentSession: AgentSession | null;
98
+
99
+ /**
100
+ * setState-powered method to update the agent session
101
+ */
102
+ setAgentSession: React.Dispatch<React.SetStateAction<AgentSession | null>>;
103
+
104
+ /**
105
+ * The forwarded parameters.
106
+ */
107
+ forwardedParameters?: Pick<ForwardedParametersInput, "temperature">;
108
+
109
+ /**
110
+ * The current thread ID.
111
+ */
112
+ threadId: string;
113
+ /**
114
+ * set the current thread ID
115
+ */
116
+ setThreadId: (threadId: string) => void;
117
+ /**
118
+ * The current run ID.
119
+ */
120
+ runId: string | null;
121
+ /**
122
+ * set the current run ID
123
+ */
124
+ setRunId: (runId: string | null) => void;
125
+ /**
126
+ * The global chat abort controller.
127
+ */
128
+ chatAbortControllerRef: React.MutableRefObject<AbortController | null>;
129
+ /**
130
+ * The agent lock.
131
+ */
132
+ agentLock: string | null;
133
+ /**
134
+ * The extensions.
135
+ */
136
+ extensions: ExtensionsInput;
137
+ /**
138
+ * The setState-powered method to update the extensions.
139
+ */
140
+ setExtensions: React.Dispatch<React.SetStateAction<ExtensionsInput>>;
141
+ };
142
+
143
+ export type UseChatHelpers = {
144
+ /**
145
+ * Append a user message to the chat list. This triggers the API call to fetch
146
+ * the assistant's response.
147
+ * @param message The message to append
148
+ */
149
+ append: (message: Message, options?: AppendMessageOptions) => Promise<void>;
150
+ /**
151
+ * Reload the last AI chat response for the given chat history. If the last
152
+ * message isn't from the assistant, it will request the API to generate a
153
+ * new response.
154
+ */
155
+ reload: () => Promise<void>;
156
+ /**
157
+ * Abort the current request immediately, keep the generated tokens if any.
158
+ */
159
+ stop: () => void;
160
+
161
+ /**
162
+ * Run the chat completion.
163
+ */
164
+ runChatCompletion: () => Promise<Message[]>;
165
+ };
166
+
167
+ export interface AppendMessageOptions {
168
+ /**
169
+ * Whether to run the chat completion after appending the message. Defaults to `true`.
170
+ */
171
+ followUp?: boolean;
172
+ }
173
+
174
+ export function useChat(options: UseChatOptions): UseChatHelpers {
175
+ const {
176
+ messages,
177
+ setMessages,
178
+ makeSystemMessageCallback,
179
+ copilotConfig,
180
+ setIsLoading,
181
+ initialMessages,
182
+ isLoading,
183
+ actions,
184
+ onFunctionCall,
185
+ onCoAgentStateRender,
186
+ setCoagentStatesWithRef,
187
+ coagentStatesRef,
188
+ agentSession,
189
+ setAgentSession,
190
+ threadId,
191
+ setThreadId,
192
+ runId,
193
+ setRunId,
194
+ chatAbortControllerRef,
195
+ agentLock,
196
+ extensions,
197
+ setExtensions,
198
+ } = options;
199
+ const runChatCompletionRef = useRef<(previousMessages: Message[]) => Promise<Message[]>>();
200
+ const addErrorToast = useErrorToast();
201
+ // We need to keep a ref of coagent states and session because of renderAndWait - making sure
202
+ // the latest state is sent to the API
203
+ // This is a workaround and needs to be addressed in the future
204
+ const agentSessionRef = useRef<AgentSession | null>(agentSession);
205
+ agentSessionRef.current = agentSession;
206
+
207
+ const runIdRef = useRef<string | null>(runId);
208
+ runIdRef.current = runId;
209
+ const extensionsRef = useRef<ExtensionsInput>(extensions);
210
+ extensionsRef.current = extensions;
211
+
212
+ const publicApiKey = copilotConfig.publicApiKey;
213
+
214
+ const headers = {
215
+ ...(copilotConfig.headers || {}),
216
+ ...(publicApiKey ? { [COPILOT_CLOUD_PUBLIC_API_KEY_HEADER]: publicApiKey } : {}),
217
+ };
218
+
219
+ const runtimeClient = useCopilotRuntimeClient({
220
+ url: copilotConfig.chatApiEndpoint,
221
+ publicApiKey: copilotConfig.publicApiKey,
222
+ headers,
223
+ credentials: copilotConfig.credentials,
224
+ });
225
+
226
+ const runChatCompletion = useAsyncCallback(
227
+ async (previousMessages: Message[]): Promise<Message[]> => {
228
+ setIsLoading(true);
229
+
230
+ // this message is just a placeholder. It will disappear once the first real message
231
+ // is received
232
+ let newMessages: Message[] = [
233
+ new TextMessage({
234
+ content: "",
235
+ role: Role.Assistant,
236
+ }),
237
+ ];
238
+
239
+ chatAbortControllerRef.current = new AbortController();
240
+
241
+ setMessages([...previousMessages, ...newMessages]);
242
+
243
+ const systemMessage = makeSystemMessageCallback();
244
+
245
+ const messagesWithContext = [systemMessage, ...(initialMessages || []), ...previousMessages];
246
+
247
+ const isAgentRun = agentSessionRef.current !== null;
248
+
249
+ const stream = runtimeClient.asStream(
250
+ runtimeClient.generateCopilotResponse({
251
+ data: {
252
+ frontend: {
253
+ actions: processActionsForRuntimeRequest(actions),
254
+ url: window.location.href,
255
+ },
256
+ threadId: threadId,
257
+ runId: runIdRef.current,
258
+ extensions: extensionsRef.current,
259
+ messages: convertMessagesToGqlInput(filterAgentStateMessages(messagesWithContext)),
260
+ ...(copilotConfig.cloud
261
+ ? {
262
+ cloud: {
263
+ ...(copilotConfig.cloud.guardrails?.input?.restrictToTopic?.enabled
264
+ ? {
265
+ guardrails: {
266
+ inputValidationRules: {
267
+ allowList:
268
+ copilotConfig.cloud.guardrails.input.restrictToTopic.validTopics,
269
+ denyList:
270
+ copilotConfig.cloud.guardrails.input.restrictToTopic.invalidTopics,
271
+ },
272
+ },
273
+ }
274
+ : {}),
275
+ },
276
+ }
277
+ : {}),
278
+ metadata: {
279
+ requestType: CopilotRequestType.Chat,
280
+ },
281
+ ...(agentSessionRef.current
282
+ ? {
283
+ agentSession: agentSessionRef.current,
284
+ }
285
+ : {}),
286
+ agentStates: Object.values(coagentStatesRef.current!).map((state) => ({
287
+ agentName: state.name,
288
+ state: JSON.stringify(state.state),
289
+ })),
290
+ forwardedParameters: options.forwardedParameters || {},
291
+ },
292
+ properties: copilotConfig.properties,
293
+ signal: chatAbortControllerRef.current?.signal,
294
+ }),
295
+ );
296
+
297
+ const guardrailsEnabled =
298
+ copilotConfig.cloud?.guardrails?.input?.restrictToTopic.enabled || false;
299
+
300
+ const reader = stream.getReader();
301
+
302
+ let executedCoAgentStateRenders: string[] = [];
303
+ let followUp: FrontendAction["followUp"] = undefined;
304
+
305
+ let messages: Message[] = [];
306
+ let syncedMessages: Message[] = [];
307
+
308
+ try {
309
+ while (true) {
310
+ let done, value;
311
+
312
+ try {
313
+ const readResult = await reader.read();
314
+ done = readResult.done;
315
+ value = readResult.value;
316
+ } catch (readError) {
317
+ break;
318
+ }
319
+
320
+ if (done) {
321
+ if (chatAbortControllerRef.current.signal.aborted) {
322
+ return [];
323
+ }
324
+ break;
325
+ }
326
+
327
+ if (!value?.generateCopilotResponse) {
328
+ continue;
329
+ }
330
+
331
+ runIdRef.current = value.generateCopilotResponse.runId || null;
332
+
333
+ // in the output, graphql inserts __typename, which leads to an error when sending it along
334
+ // as input to the next request.
335
+ extensionsRef.current = CopilotRuntimeClient.removeGraphQLTypename(
336
+ value.generateCopilotResponse.extensions || {},
337
+ );
338
+
339
+ // setThreadId(threadIdRef.current);
340
+ setRunId(runIdRef.current);
341
+ setExtensions(extensionsRef.current);
342
+
343
+ messages = convertGqlOutputToMessages(
344
+ filterAdjacentAgentStateMessages(value.generateCopilotResponse.messages),
345
+ );
346
+
347
+ if (messages.length === 0) {
348
+ continue;
349
+ }
350
+
351
+ newMessages = [];
352
+
353
+ // request failed, display error message and quit
354
+ if (
355
+ value.generateCopilotResponse.status?.__typename === "FailedResponseStatus" &&
356
+ value.generateCopilotResponse.status.reason === "GUARDRAILS_VALIDATION_FAILED"
357
+ ) {
358
+ newMessages = [
359
+ new TextMessage({
360
+ role: MessageRole.Assistant,
361
+ content: value.generateCopilotResponse.status.details?.guardrailsReason || "",
362
+ }),
363
+ ];
364
+ setMessages([...previousMessages, ...newMessages]);
365
+ break;
366
+ }
367
+
368
+ // add messages to the chat
369
+ else {
370
+ newMessages = [...messages];
371
+
372
+ for (const message of messages) {
373
+ // execute onCoAgentStateRender handler
374
+ if (
375
+ message.isAgentStateMessage() &&
376
+ !message.active &&
377
+ !executedCoAgentStateRenders.includes(message.id) &&
378
+ onCoAgentStateRender
379
+ ) {
380
+ // Do not execute a coagent action if guardrails are enabled but the status is not known
381
+ if (guardrailsEnabled && value.generateCopilotResponse.status === undefined) {
382
+ break;
383
+ }
384
+ // execute coagent action
385
+ await onCoAgentStateRender({
386
+ name: message.agentName,
387
+ nodeName: message.nodeName,
388
+ state: message.state,
389
+ });
390
+ executedCoAgentStateRenders.push(message.id);
391
+ }
392
+ }
393
+
394
+ const lastAgentStateMessage = [...messages]
395
+ .reverse()
396
+ .find((message) => message.isAgentStateMessage());
397
+
398
+ if (lastAgentStateMessage) {
399
+ if (
400
+ lastAgentStateMessage.state.messages &&
401
+ lastAgentStateMessage.state.messages.length > 0
402
+ ) {
403
+ syncedMessages = loadMessagesFromJsonRepresentation(
404
+ lastAgentStateMessage.state.messages,
405
+ );
406
+ }
407
+ setCoagentStatesWithRef((prevAgentStates) => ({
408
+ ...prevAgentStates,
409
+ [lastAgentStateMessage.agentName]: {
410
+ name: lastAgentStateMessage.agentName,
411
+ state: lastAgentStateMessage.state,
412
+ running: lastAgentStateMessage.running,
413
+ active: lastAgentStateMessage.active,
414
+ threadId: lastAgentStateMessage.threadId,
415
+ nodeName: lastAgentStateMessage.nodeName,
416
+ runId: lastAgentStateMessage.runId,
417
+ },
418
+ }));
419
+ if (lastAgentStateMessage.running) {
420
+ setAgentSession({
421
+ threadId: lastAgentStateMessage.threadId,
422
+ agentName: lastAgentStateMessage.agentName,
423
+ nodeName: lastAgentStateMessage.nodeName,
424
+ });
425
+ } else {
426
+ if (agentLock) {
427
+ setAgentSession({
428
+ threadId: randomId(),
429
+ agentName: agentLock,
430
+ nodeName: undefined,
431
+ });
432
+ } else {
433
+ setAgentSession(null);
434
+ }
435
+ }
436
+ }
437
+ }
438
+
439
+ if (newMessages.length > 0) {
440
+ // Update message state
441
+ setMessages([...previousMessages, ...newMessages]);
442
+ }
443
+ }
444
+ const finalMessages = constructFinalMessages(syncedMessages, previousMessages, newMessages);
445
+
446
+ let didExecuteAction = false;
447
+
448
+ // execute regular action executions that are specific to the frontend (last actions)
449
+ if (onFunctionCall) {
450
+ // Find consecutive action execution messages at the end
451
+ const lastMessages = [];
452
+ for (let i = finalMessages.length - 1; i >= 0; i--) {
453
+ const message = finalMessages[i];
454
+ if (
455
+ message.isActionExecutionMessage() &&
456
+ message.status.code !== MessageStatusCode.Pending
457
+ ) {
458
+ lastMessages.unshift(message);
459
+ } else {
460
+ break;
461
+ }
462
+ }
463
+
464
+ for (const message of lastMessages) {
465
+ // We update the message state before calling the handler so that the render
466
+ // function can be called with `executing` state
467
+ setMessages(finalMessages);
468
+
469
+ const action = actions.find((action) => action.name === message.name);
470
+
471
+ if (action) {
472
+ followUp = action.followUp;
473
+ let result: any;
474
+ let error: Error | null = null;
475
+ try {
476
+ result = await Promise.race([
477
+ onFunctionCall({
478
+ messages: previousMessages,
479
+ name: message.name,
480
+ args: message.arguments,
481
+ }),
482
+ new Promise((resolve) =>
483
+ chatAbortControllerRef.current?.signal.addEventListener("abort", () =>
484
+ resolve("Operation was aborted by the user"),
485
+ ),
486
+ ),
487
+ // if the user stopped generation, we also abort consecutive actions
488
+ new Promise((resolve) => {
489
+ if (chatAbortControllerRef.current?.signal.aborted) {
490
+ resolve("Operation was aborted by the user");
491
+ }
492
+ }),
493
+ ]);
494
+ } catch (e) {
495
+ error = e as Error;
496
+ addErrorToast([error]);
497
+ result = `Failed to execute action ${message.name}. ${error.message}`;
498
+ console.error(`Failed to execute action ${message.name}: ${error}`);
499
+ }
500
+ didExecuteAction = true;
501
+ const messageIndex = finalMessages.findIndex((msg) => msg.id === message.id);
502
+ finalMessages.splice(
503
+ messageIndex + 1,
504
+ 0,
505
+ new ResultMessage({
506
+ id: "result-" + message.id,
507
+ result: ResultMessage.encodeResult(
508
+ error
509
+ ? {
510
+ content: result,
511
+ error: JSON.parse(
512
+ JSON.stringify(error, Object.getOwnPropertyNames(error)),
513
+ ),
514
+ }
515
+ : result,
516
+ ),
517
+ actionExecutionId: message.id,
518
+ actionName: message.name,
519
+ }),
520
+ );
521
+ }
522
+ }
523
+
524
+ setMessages(finalMessages);
525
+ }
526
+
527
+ if (
528
+ // if followUp is not explicitly false
529
+ followUp !== false &&
530
+ // and we executed an action
531
+ (didExecuteAction ||
532
+ // the last message is a server side result
533
+ (!isAgentRun &&
534
+ finalMessages.length &&
535
+ finalMessages[finalMessages.length - 1].isResultMessage())) &&
536
+ // the user did not stop generation
537
+ !chatAbortControllerRef.current?.signal.aborted
538
+ ) {
539
+ // run the completion again and return the result
540
+
541
+ // wait for next tick to make sure all the react state updates
542
+ // - tried using react-dom's flushSync, but it did not work
543
+ await new Promise((resolve) => setTimeout(resolve, 10));
544
+
545
+ return await runChatCompletionRef.current!(finalMessages);
546
+ } else if (chatAbortControllerRef.current?.signal.aborted) {
547
+ // filter out all the action execution messages that do not have a consecutive matching result message
548
+ const repairedMessages = finalMessages.filter((message, actionExecutionIndex) => {
549
+ if (message.isActionExecutionMessage()) {
550
+ return finalMessages.find(
551
+ (msg, resultIndex) =>
552
+ msg.isResultMessage() &&
553
+ msg.actionExecutionId === message.id &&
554
+ resultIndex === actionExecutionIndex + 1,
555
+ );
556
+ }
557
+ return true;
558
+ });
559
+ const repairedMessageIds = repairedMessages.map((message) => message.id);
560
+ setMessages(repairedMessages);
561
+
562
+ // LangGraph needs two pieces of information to continue execution:
563
+ // 1. The threadId
564
+ // 2. The nodeName it came from
565
+ // When stopping the agent, we don't know the nodeName the agent would have ended with
566
+ // Therefore, we set the nodeName to the most reasonable thing we can guess, which
567
+ // is "__end__"
568
+ if (agentSessionRef.current?.nodeName) {
569
+ setAgentSession({
570
+ threadId: agentSessionRef.current.threadId,
571
+ agentName: agentSessionRef.current.agentName,
572
+ nodeName: "__end__",
573
+ });
574
+ }
575
+ // only return new messages that were not filtered out
576
+ return newMessages.filter((message) => repairedMessageIds.includes(message.id));
577
+ } else {
578
+ return newMessages.slice();
579
+ }
580
+ } finally {
581
+ setIsLoading(false);
582
+ }
583
+ },
584
+ [
585
+ messages,
586
+ setMessages,
587
+ makeSystemMessageCallback,
588
+ copilotConfig,
589
+ setIsLoading,
590
+ initialMessages,
591
+ isLoading,
592
+ actions,
593
+ onFunctionCall,
594
+ onCoAgentStateRender,
595
+ setCoagentStatesWithRef,
596
+ coagentStatesRef,
597
+ agentSession,
598
+ setAgentSession,
599
+ ],
600
+ );
601
+
602
+ runChatCompletionRef.current = runChatCompletion;
603
+
604
+ const runChatCompletionAndHandleFunctionCall = useAsyncCallback(
605
+ async (messages: Message[]): Promise<void> => {
606
+ await runChatCompletionRef.current!(messages);
607
+ },
608
+ [messages],
609
+ );
610
+
611
+ const append = useAsyncCallback(
612
+ async (message: Message, options?: AppendMessageOptions): Promise<void> => {
613
+ if (isLoading) {
614
+ return;
615
+ }
616
+
617
+ const newMessages = [...messages, message];
618
+ setMessages(newMessages);
619
+ const followUp = options?.followUp ?? true;
620
+ if (followUp) {
621
+ return runChatCompletionAndHandleFunctionCall(newMessages);
622
+ }
623
+ },
624
+ [isLoading, messages, setMessages, runChatCompletionAndHandleFunctionCall],
625
+ );
626
+
627
+ const reload = useAsyncCallback(async (): Promise<void> => {
628
+ if (isLoading || messages.length === 0) {
629
+ return;
630
+ }
631
+ let newMessages = [...messages];
632
+ const lastMessage = messages[messages.length - 1];
633
+
634
+ if (lastMessage.isTextMessage() && lastMessage.role === "assistant") {
635
+ newMessages = newMessages.slice(0, -1);
636
+ }
637
+
638
+ setMessages(newMessages);
639
+
640
+ return runChatCompletionAndHandleFunctionCall(newMessages);
641
+ }, [isLoading, messages, setMessages, runChatCompletionAndHandleFunctionCall]);
642
+
643
+ const stop = (): void => {
644
+ chatAbortControllerRef.current?.abort("Stop was called");
645
+ };
646
+
647
+ return {
648
+ append,
649
+ reload,
650
+ stop,
651
+ runChatCompletion: () => runChatCompletionRef.current!(messages),
652
+ };
653
+ }
654
+
655
+ function constructFinalMessages(
656
+ syncedMessages: Message[],
657
+ previousMessages: Message[],
658
+ newMessages: Message[],
659
+ ): Message[] {
660
+ const finalMessages =
661
+ syncedMessages.length > 0 ? [...syncedMessages] : [...previousMessages, ...newMessages];
662
+
663
+ if (syncedMessages.length > 0) {
664
+ const messagesWithAgentState = [...previousMessages, ...newMessages];
665
+
666
+ let previousMessageId: string | undefined = undefined;
667
+
668
+ for (const message of messagesWithAgentState) {
669
+ if (message.isAgentStateMessage()) {
670
+ // insert this message into finalMessages after the position of previousMessageId
671
+ const index = finalMessages.findIndex((msg) => msg.id === previousMessageId);
672
+ if (index !== -1) {
673
+ finalMessages.splice(index + 1, 0, message);
674
+ }
675
+ }
676
+
677
+ previousMessageId = message.id;
678
+ }
679
+ }
680
+
681
+ return finalMessages;
682
+ }