@copilotkit/react-core 1.6.0-next.1 → 1.6.0-next.11

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 (76) hide show
  1. package/CHANGELOG.md +77 -0
  2. package/dist/{chunk-XERJQUHA.mjs → chunk-4CEQJ2X6.mjs} +2 -2
  3. package/dist/{chunk-XERJQUHA.mjs.map → chunk-4CEQJ2X6.mjs.map} +1 -1
  4. package/dist/{chunk-VMYLY5KI.mjs → chunk-4EC4JK7H.mjs} +58 -48
  5. package/dist/chunk-4EC4JK7H.mjs.map +1 -0
  6. package/dist/{chunk-ZAU4EMBM.mjs → chunk-F7O6L7C3.mjs} +4 -5
  7. package/dist/chunk-F7O6L7C3.mjs.map +1 -0
  8. package/dist/{chunk-2DPTBHGN.mjs → chunk-FUO5LKSJ.mjs} +115 -63
  9. package/dist/chunk-FUO5LKSJ.mjs.map +1 -0
  10. package/dist/{chunk-4ED56DQX.mjs → chunk-HBCP2KNA.mjs} +3 -4
  11. package/dist/chunk-HBCP2KNA.mjs.map +1 -0
  12. package/dist/{chunk-Z3U4UBG2.mjs → chunk-T4CAQC6B.mjs} +2 -2
  13. package/dist/{chunk-JFCNSGNU.mjs → chunk-Y2T56VSE.mjs} +2 -2
  14. package/dist/components/copilot-provider/copilotkit-props.d.ts +1 -2
  15. package/dist/components/copilot-provider/copilotkit.d.ts +1 -2
  16. package/dist/components/copilot-provider/index.d.ts +1 -2
  17. package/dist/components/index.d.ts +1 -2
  18. package/dist/context/copilot-context.d.ts +10 -148
  19. package/dist/context/index.d.ts +1 -2
  20. package/dist/copilot-context-9ae8990b.d.ts +183 -0
  21. package/dist/hooks/index.d.ts +1 -2
  22. package/dist/hooks/index.js +180 -123
  23. package/dist/hooks/index.js.map +1 -1
  24. package/dist/hooks/index.mjs +6 -6
  25. package/dist/hooks/use-chat.d.ts +1 -2
  26. package/dist/hooks/use-chat.js +113 -64
  27. package/dist/hooks/use-chat.js.map +1 -1
  28. package/dist/hooks/use-chat.mjs +2 -2
  29. package/dist/hooks/use-coagent.d.ts +1 -2
  30. package/dist/hooks/use-coagent.js +171 -112
  31. package/dist/hooks/use-coagent.js.map +1 -1
  32. package/dist/hooks/use-coagent.mjs +4 -4
  33. package/dist/hooks/use-copilot-chat.d.ts +1 -2
  34. package/dist/hooks/use-copilot-chat.js +116 -67
  35. package/dist/hooks/use-copilot-chat.js.map +1 -1
  36. package/dist/hooks/use-copilot-chat.mjs +3 -3
  37. package/dist/hooks/use-langgraph-interrupt-render.js +3 -4
  38. package/dist/hooks/use-langgraph-interrupt-render.js.map +1 -1
  39. package/dist/hooks/use-langgraph-interrupt-render.mjs +1 -1
  40. package/dist/hooks/use-langgraph-interrupt.d.ts +9 -2
  41. package/dist/hooks/use-langgraph-interrupt.js +117 -69
  42. package/dist/hooks/use-langgraph-interrupt.js.map +1 -1
  43. package/dist/hooks/use-langgraph-interrupt.mjs +4 -4
  44. package/dist/index.d.ts +1 -2
  45. package/dist/index.js +187 -130
  46. package/dist/index.js.map +1 -1
  47. package/dist/index.mjs +7 -7
  48. package/dist/lib/copilot-task.d.ts +1 -2
  49. package/dist/lib/copilot-task.js +1 -1
  50. package/dist/lib/copilot-task.js.map +1 -1
  51. package/dist/lib/copilot-task.mjs +2 -2
  52. package/dist/lib/index.d.ts +1 -2
  53. package/dist/lib/index.js +1 -1
  54. package/dist/lib/index.js.map +1 -1
  55. package/dist/lib/index.mjs +2 -2
  56. package/dist/types/frontend-action.d.ts +2 -1
  57. package/dist/types/frontend-action.js +1 -1
  58. package/dist/types/frontend-action.js.map +1 -1
  59. package/dist/types/frontend-action.mjs +1 -1
  60. package/dist/types/interrupt-action.d.ts +10 -35
  61. package/dist/types/interrupt-action.js.map +1 -1
  62. package/dist/utils/extract.d.ts +1 -2
  63. package/dist/utils/index.d.ts +1 -2
  64. package/package.json +3 -3
  65. package/src/hooks/use-chat.ts +134 -61
  66. package/src/hooks/use-coagent.ts +70 -56
  67. package/src/hooks/use-langgraph-interrupt-render.ts +7 -3
  68. package/src/hooks/use-langgraph-interrupt.ts +1 -2
  69. package/src/types/frontend-action.ts +5 -2
  70. package/src/types/interrupt-action.ts +16 -12
  71. package/dist/chunk-2DPTBHGN.mjs.map +0 -1
  72. package/dist/chunk-4ED56DQX.mjs.map +0 -1
  73. package/dist/chunk-VMYLY5KI.mjs.map +0 -1
  74. package/dist/chunk-ZAU4EMBM.mjs.map +0 -1
  75. /package/dist/{chunk-Z3U4UBG2.mjs.map → chunk-T4CAQC6B.mjs.map} +0 -0
  76. /package/dist/{chunk-JFCNSGNU.mjs.map → chunk-Y2T56VSE.mjs.map} +0 -0
package/package.json CHANGED
@@ -9,7 +9,7 @@
9
9
  "publishConfig": {
10
10
  "access": "public"
11
11
  },
12
- "version": "1.6.0-next.1",
12
+ "version": "1.6.0-next.11",
13
13
  "sideEffects": false,
14
14
  "main": "./dist/index.js",
15
15
  "module": "./dist/index.mjs",
@@ -43,8 +43,8 @@
43
43
  "@scarf/scarf": "^1.3.0",
44
44
  "react-markdown": "^8.0.7",
45
45
  "untruncate-json": "^0.0.1",
46
- "@copilotkit/runtime-client-gql": "1.6.0-next.1",
47
- "@copilotkit/shared": "1.6.0-next.1"
46
+ "@copilotkit/runtime-client-gql": "1.6.0-next.11",
47
+ "@copilotkit/shared": "1.6.0-next.11"
48
48
  },
49
49
  "keywords": [
50
50
  "copilotkit",
@@ -4,6 +4,7 @@ import {
4
4
  COPILOT_CLOUD_PUBLIC_API_KEY_HEADER,
5
5
  CoAgentStateRenderHandler,
6
6
  randomId,
7
+ parseJson,
7
8
  } from "@copilotkit/shared";
8
9
  import {
9
10
  Message,
@@ -22,6 +23,12 @@ import {
22
23
  ExtensionsInput,
23
24
  CopilotRuntimeClient,
24
25
  langGraphInterruptEvent,
26
+ MetaEvent,
27
+ MetaEventName,
28
+ ActionExecutionMessage,
29
+ CopilotKitLangGraphInterruptEvent,
30
+ LangGraphInterruptEvent,
31
+ MetaEventInput,
25
32
  } from "@copilotkit/runtime-client-gql";
26
33
 
27
34
  import { CopilotApiConfig } from "../context";
@@ -34,12 +41,6 @@ import {
34
41
  LangGraphInterruptAction,
35
42
  LangGraphInterruptActionSetter,
36
43
  } from "../types/interrupt-action";
37
- import { MetaEvent, MetaEventName } from "@copilotkit/runtime-client-gql";
38
- import {
39
- CopilotKitLangGraphInterruptEvent,
40
- LangGraphInterruptEvent,
41
- MetaEventInput,
42
- } from "@copilotkit/runtime-client-gql";
43
44
 
44
45
  export type UseChatOptions = {
45
46
  /**
@@ -377,10 +378,7 @@ export function useChat(options: UseChatOptions): UseChatHelpers {
377
378
  (value.generateCopilotResponse?.metaEvents ?? []).forEach((ev) => {
378
379
  if (ev.name === MetaEventName.LangGraphInterruptEvent) {
379
380
  let eventValue = langGraphInterruptEvent(ev as LangGraphInterruptEvent).value;
380
- try {
381
- eventValue = JSON.parse(eventValue);
382
- // In case of unparsable string, we keep the event as is
383
- } catch (e) {}
381
+ eventValue = parseJson(eventValue, eventValue);
384
382
  setLangGraphInterruptAction({
385
383
  event: {
386
384
  ...langGraphInterruptEvent(ev as LangGraphInterruptEvent),
@@ -501,7 +499,7 @@ export function useChat(options: UseChatOptions): UseChatHelpers {
501
499
  setMessages([...previousMessages, ...newMessages]);
502
500
  }
503
501
  }
504
- const finalMessages = constructFinalMessages(
502
+ let finalMessages = constructFinalMessages(
505
503
  [...syncedMessages, ...interruptMessages],
506
504
  previousMessages,
507
505
  newMessages,
@@ -516,7 +514,7 @@ export function useChat(options: UseChatOptions): UseChatHelpers {
516
514
  for (let i = finalMessages.length - 1; i >= 0; i--) {
517
515
  const message = finalMessages[i];
518
516
  if (
519
- message.isActionExecutionMessage() &&
517
+ (message.isActionExecutionMessage() || message.isResultMessage()) &&
520
518
  message.status.code !== MessageStatusCode.Pending
521
519
  ) {
522
520
  lastMessages.unshift(message);
@@ -530,57 +528,63 @@ export function useChat(options: UseChatOptions): UseChatHelpers {
530
528
  // function can be called with `executing` state
531
529
  setMessages(finalMessages);
532
530
 
533
- const action = actions.find((action) => action.name === message.name);
534
-
535
- if (action) {
536
- followUp = action.followUp;
537
- let result: any;
538
- let error: Error | null = null;
539
- try {
540
- result = await Promise.race([
541
- onFunctionCall({
542
- messages: previousMessages,
543
- name: message.name,
544
- args: message.arguments,
545
- }),
546
- new Promise((resolve) =>
547
- chatAbortControllerRef.current?.signal.addEventListener("abort", () =>
548
- resolve("Operation was aborted by the user"),
549
- ),
550
- ),
551
- // if the user stopped generation, we also abort consecutive actions
552
- new Promise((resolve) => {
553
- if (chatAbortControllerRef.current?.signal.aborted) {
554
- resolve("Operation was aborted by the user");
555
- }
556
- }),
557
- ]);
558
- } catch (e) {
559
- error = e as Error;
560
- addErrorToast([error]);
561
- result = `Failed to execute action ${message.name}. ${error.message}`;
562
- console.error(`Failed to execute action ${message.name}: ${error}`);
563
- }
531
+ const action = actions.find(
532
+ (action) => action.name === (message as ActionExecutionMessage).name,
533
+ );
534
+ const currentResultMessagePairedFeAction = message.isResultMessage()
535
+ ? getPairedFeAction(actions, message)
536
+ : null;
537
+
538
+ const executeActionFromMessage = async (
539
+ action: FrontendAction<any>,
540
+ message: ActionExecutionMessage,
541
+ ) => {
542
+ followUp = action?.followUp;
543
+ const resultMessage = await executeAction({
544
+ onFunctionCall,
545
+ previousMessages,
546
+ message,
547
+ chatAbortControllerRef,
548
+ onError: (error: Error) => {
549
+ addErrorToast([error]);
550
+ console.error(`Failed to execute action ${message.name}: ${error}`);
551
+ },
552
+ });
564
553
  didExecuteAction = true;
565
554
  const messageIndex = finalMessages.findIndex((msg) => msg.id === message.id);
566
- finalMessages.splice(
567
- messageIndex + 1,
568
- 0,
569
- new ResultMessage({
570
- id: "result-" + message.id,
571
- result: ResultMessage.encodeResult(
572
- error
573
- ? {
574
- content: result,
575
- error: JSON.parse(
576
- JSON.stringify(error, Object.getOwnPropertyNames(error)),
577
- ),
578
- }
579
- : result,
580
- ),
581
- actionExecutionId: message.id,
582
- actionName: message.name,
583
- }),
555
+ finalMessages.splice(messageIndex + 1, 0, resultMessage);
556
+
557
+ return resultMessage;
558
+ };
559
+
560
+ // execution message which has an action registered with the hook (remote availability):
561
+ // execute that action first, and then the "paired FE action"
562
+ if (action && message.isActionExecutionMessage()) {
563
+ const resultMessage = await executeActionFromMessage(action, message);
564
+ const pairedFeAction = getPairedFeAction(actions, resultMessage);
565
+
566
+ if (pairedFeAction) {
567
+ const newExecutionMessage = new ActionExecutionMessage({
568
+ name: pairedFeAction.name,
569
+ arguments: parseJson(resultMessage.result, resultMessage.result),
570
+ status: message.status,
571
+ createdAt: message.createdAt,
572
+ parentMessageId: message.parentMessageId,
573
+ });
574
+ await executeActionFromMessage(pairedFeAction, newExecutionMessage);
575
+ }
576
+ } else if (message.isResultMessage() && currentResultMessagePairedFeAction) {
577
+ // Actions which are set up in runtime actions array: Grab the result, executed paired FE action with it as args.
578
+ const newExecutionMessage = new ActionExecutionMessage({
579
+ name: currentResultMessagePairedFeAction.name,
580
+ arguments: parseJson(message.result, message.result),
581
+ status: message.status,
582
+ createdAt: message.createdAt,
583
+ });
584
+ finalMessages.push(newExecutionMessage);
585
+ await executeActionFromMessage(
586
+ currentResultMessagePairedFeAction,
587
+ newExecutionMessage,
584
588
  );
585
589
  }
586
590
  }
@@ -777,3 +781,72 @@ function constructFinalMessages(
777
781
 
778
782
  return finalMessages;
779
783
  }
784
+
785
+ async function executeAction({
786
+ onFunctionCall,
787
+ previousMessages,
788
+ message,
789
+ chatAbortControllerRef,
790
+ onError,
791
+ }: {
792
+ onFunctionCall: FunctionCallHandler;
793
+ previousMessages: Message[];
794
+ message: ActionExecutionMessage;
795
+ chatAbortControllerRef: React.MutableRefObject<AbortController | null>;
796
+ onError: (error: Error) => void;
797
+ }) {
798
+ let result: any;
799
+ let error: Error | null = null;
800
+ try {
801
+ result = await Promise.race([
802
+ onFunctionCall({
803
+ messages: previousMessages,
804
+ name: message.name,
805
+ args: message.arguments,
806
+ }),
807
+ new Promise((resolve) =>
808
+ chatAbortControllerRef.current?.signal.addEventListener("abort", () =>
809
+ resolve("Operation was aborted by the user"),
810
+ ),
811
+ ),
812
+ // if the user stopped generation, we also abort consecutive actions
813
+ new Promise((resolve) => {
814
+ if (chatAbortControllerRef.current?.signal.aborted) {
815
+ resolve("Operation was aborted by the user");
816
+ }
817
+ }),
818
+ ]);
819
+ } catch (e) {
820
+ onError(e as Error);
821
+ }
822
+ return new ResultMessage({
823
+ id: "result-" + message.id,
824
+ result: ResultMessage.encodeResult(
825
+ error
826
+ ? {
827
+ content: result,
828
+ error: JSON.parse(JSON.stringify(error, Object.getOwnPropertyNames(error))),
829
+ }
830
+ : result,
831
+ ),
832
+ actionExecutionId: message.id,
833
+ actionName: message.name,
834
+ });
835
+ }
836
+
837
+ function getPairedFeAction(
838
+ actions: FrontendAction<any>[],
839
+ message: ActionExecutionMessage | ResultMessage,
840
+ ) {
841
+ let actionName = null;
842
+ if (message.isActionExecutionMessage()) {
843
+ actionName = message.name;
844
+ } else if (message.isResultMessage()) {
845
+ actionName = message.actionName;
846
+ }
847
+ return actions.find(
848
+ (action) =>
849
+ (action.name === actionName && action.available === "frontend") ||
850
+ action.pairedAction === actionName,
851
+ );
852
+ }
@@ -88,7 +88,7 @@
88
88
  * </PropertyReference>
89
89
  */
90
90
 
91
- import { useEffect, useRef } from "react";
91
+ import { useCallback, useEffect, useMemo, useRef } from "react";
92
92
  import {
93
93
  CopilotContextParams,
94
94
  CopilotMessagesContextParams,
@@ -101,6 +101,7 @@ import { Message } from "@copilotkit/runtime-client-gql";
101
101
  import { useAsyncCallback } from "../components/error-boundary/error-utils";
102
102
  import { useToast } from "../components/toast/toast-provider";
103
103
  import { useCopilotRuntimeClient } from "./use-copilot-runtime-client";
104
+ import { parseJson } from "@copilotkit/shared";
104
105
 
105
106
  interface WithInternalStateManagementAndInitial<T> {
106
107
  /**
@@ -222,12 +223,6 @@ export function useCoAgent<T = any>(options: UseCoagentOptions<T>): UseCoagentRe
222
223
  const lastLoadedThreadId = useRef<string>();
223
224
  const lastLoadedState = useRef<any>();
224
225
 
225
- const isExternalStateManagement = (
226
- options: UseCoagentOptions<T>,
227
- ): options is WithExternalStateManagement<T> => {
228
- return "state" in options && "setState" in options;
229
- };
230
-
231
226
  const { name } = options;
232
227
  useEffect(() => {
233
228
  if (availableAgents?.length && !availableAgents.some((a) => a.name === name)) {
@@ -237,35 +232,12 @@ export function useCoAgent<T = any>(options: UseCoagentOptions<T>): UseCoagentRe
237
232
  }
238
233
  }, [availableAgents]);
239
234
 
240
- const isInternalStateManagementWithInitial = (
241
- options: UseCoagentOptions<T>,
242
- ): options is WithInternalStateManagementAndInitial<T> => {
243
- return "initialState" in options;
244
- };
245
-
246
235
  const messagesContext = useCopilotMessagesContext();
247
236
  const context = { ...generalContext, ...messagesContext };
248
237
  const { coagentStates, coagentStatesRef, setCoagentStatesWithRef, threadId, copilotApiConfig } =
249
238
  context;
250
239
  const { appendMessage, runChatCompletion } = useCopilotChat();
251
240
 
252
- const getCoagentState = (coagentStates: Record<string, CoagentState>, name: string) => {
253
- if (coagentStates[name]) {
254
- return coagentStates[name];
255
- } else {
256
- return {
257
- name,
258
- state: isInternalStateManagementWithInitial(options) ? options.initialState : {},
259
- configurable: options.configurable ?? {},
260
- running: false,
261
- active: false,
262
- threadId: undefined,
263
- nodeName: undefined,
264
- runId: undefined,
265
- };
266
- }
267
- };
268
-
269
241
  const runtimeClient = useCopilotRuntimeClient({
270
242
  url: copilotApiConfig.chatApiEndpoint,
271
243
  publicApiKey: copilotApiConfig.publicApiKey,
@@ -273,19 +245,23 @@ export function useCoAgent<T = any>(options: UseCoagentOptions<T>): UseCoagentRe
273
245
  });
274
246
 
275
247
  // if we manage state internally, we need to provide a function to set the state
276
- const setState = (newState: T | ((prevState: T | undefined) => T)) => {
277
- let coagentState: CoagentState = getCoagentState(coagentStatesRef.current || {}, name);
278
- const updatedState =
279
- typeof newState === "function" ? (newState as Function)(coagentState.state) : newState;
248
+ const setState = useCallback(
249
+ (newState: T | ((prevState: T | undefined) => T)) => {
250
+ // coagentStatesRef.current || {}
251
+ let coagentState: CoagentState = getCoagentState({ coagentStates, name, options });
252
+ const updatedState =
253
+ typeof newState === "function" ? (newState as Function)(coagentState.state) : newState;
280
254
 
281
- setCoagentStatesWithRef({
282
- ...coagentStatesRef.current,
283
- [name]: {
284
- ...coagentState,
285
- state: updatedState,
286
- },
287
- });
288
- };
255
+ setCoagentStatesWithRef({
256
+ ...coagentStatesRef.current,
257
+ [name]: {
258
+ ...coagentState,
259
+ state: updatedState,
260
+ },
261
+ });
262
+ },
263
+ [coagentStates, name],
264
+ );
289
265
 
290
266
  useEffect(() => {
291
267
  const fetchAgentState = async () => {
@@ -302,7 +278,7 @@ export function useCoAgent<T = any>(options: UseCoagentOptions<T>): UseCoagentRe
302
278
  if (result.data?.loadAgentState?.threadExists && newState && newState != "{}") {
303
279
  lastLoadedState.current = newState;
304
280
  lastLoadedThreadId.current = threadId;
305
- const fetchedState = JSON.parse(newState);
281
+ const fetchedState = parseJson(newState, {});
306
282
  isExternalStateManagement(options)
307
283
  ? options.setState(fetchedState)
308
284
  : setState(fetchedState);
@@ -311,8 +287,6 @@ export function useCoAgent<T = any>(options: UseCoagentOptions<T>): UseCoagentRe
311
287
  void fetchAgentState();
312
288
  }, [threadId]);
313
289
 
314
- const coagentState = getCoagentState(coagentStates, name);
315
-
316
290
  // Sync internal state with external state if state management is external
317
291
  useEffect(() => {
318
292
  if (isExternalStateManagement(options)) {
@@ -334,17 +308,20 @@ export function useCoAgent<T = any>(options: UseCoagentOptions<T>): UseCoagentRe
334
308
  );
335
309
 
336
310
  // Return the state and setState function
337
- return {
338
- name,
339
- nodeName: coagentState.nodeName,
340
- threadId: coagentState.threadId,
341
- running: coagentState.running,
342
- state: coagentState.state,
343
- setState: isExternalStateManagement(options) ? options.setState : setState,
344
- start: () => startAgent(name, context),
345
- stop: () => stopAgent(name, context),
346
- run: runAgentCallback,
347
- };
311
+ return useMemo(() => {
312
+ const coagentState = getCoagentState({ coagentStates, name, options });
313
+ return {
314
+ name,
315
+ nodeName: coagentState.nodeName,
316
+ threadId: coagentState.threadId,
317
+ running: coagentState.running,
318
+ state: coagentState.state,
319
+ setState: isExternalStateManagement(options) ? options.setState : setState,
320
+ start: () => startAgent(name, context),
321
+ stop: () => stopAgent(name, context),
322
+ run: runAgentCallback,
323
+ };
324
+ }, [name, coagentStates, options, setState, runAgentCallback]);
348
325
  }
349
326
 
350
327
  export function startAgent(name: string, context: CopilotContextParams) {
@@ -411,3 +388,40 @@ export async function runAgent(
411
388
  await runChatCompletion();
412
389
  }
413
390
  }
391
+
392
+ const isExternalStateManagement = <T>(
393
+ options: UseCoagentOptions<T>,
394
+ ): options is WithExternalStateManagement<T> => {
395
+ return "state" in options && "setState" in options;
396
+ };
397
+
398
+ const isInternalStateManagementWithInitial = <T>(
399
+ options: UseCoagentOptions<T>,
400
+ ): options is WithInternalStateManagementAndInitial<T> => {
401
+ return "initialState" in options;
402
+ };
403
+
404
+ const getCoagentState = <T>({
405
+ coagentStates,
406
+ name,
407
+ options,
408
+ }: {
409
+ coagentStates: Record<string, CoagentState>;
410
+ name: string;
411
+ options: UseCoagentOptions<T>;
412
+ }) => {
413
+ if (coagentStates[name]) {
414
+ return coagentStates[name];
415
+ } else {
416
+ return {
417
+ name,
418
+ state: isInternalStateManagementWithInitial<T>(options) ? options.initialState : {},
419
+ configurable: options.configurable ?? {},
420
+ running: false,
421
+ active: false,
422
+ threadId: undefined,
423
+ nodeName: undefined,
424
+ runId: undefined,
425
+ };
426
+ }
427
+ };
@@ -18,7 +18,8 @@ const InterruptRenderer: React.FC<InterruptProps> = ({ event, result, render, re
18
18
  };
19
19
 
20
20
  export function useLangGraphInterruptRender(): string | React.ReactElement | null {
21
- const { langGraphInterruptAction, setLangGraphInterruptAction } = useCopilotContext();
21
+ const { langGraphInterruptAction, setLangGraphInterruptAction, agentSession } =
22
+ useCopilotContext();
22
23
 
23
24
  const responseRef = React.useRef<string>();
24
25
  const resolveInterrupt = useCallback(
@@ -39,9 +40,12 @@ export function useLangGraphInterruptRender(): string | React.ReactElement | nul
39
40
  )
40
41
  return null;
41
42
 
42
- const { render, handler, event, conditions } = langGraphInterruptAction;
43
+ const { render, handler, event, enabled } = langGraphInterruptAction;
43
44
 
44
- const conditionsMet = executeConditions({ conditions, value: event.value });
45
+ const conditionsMet =
46
+ !agentSession || !enabled
47
+ ? true
48
+ : enabled({ eventValue: event.value, agentMetadata: agentSession });
45
49
  if (!conditionsMet) {
46
50
  return null;
47
51
  }
@@ -35,10 +35,9 @@ export function useLangGraphInterrupt<TEventValue = any>(
35
35
 
36
36
  useEffect(() => {
37
37
  if (!action) return;
38
- // console.log(2, { hasAction, isCurrentAction, conditions: action.conditions })
39
38
  // An action was already set, with no conditions and it's not the action we're using right now.
40
39
  // Show a warning, as this action will not be executed
41
- if (hasAction && !isCurrentAction && !action.conditions?.length) {
40
+ if (hasAction && !isCurrentAction && !action.enabled) {
42
41
  addToast({
43
42
  type: "warning",
44
43
  message: "An action is already registered for the interrupt event",
@@ -127,7 +127,7 @@ export type CatchAllActionRenderProps<T extends Parameter[] | [] = []> =
127
127
  name: string;
128
128
  });
129
129
 
130
- export type FrontendActionAvailability = "disabled" | "enabled" | "remote";
130
+ export type FrontendActionAvailability = "disabled" | "enabled" | "remote" | "frontend";
131
131
 
132
132
  export type FrontendAction<
133
133
  T extends Parameter[] | [] = [],
@@ -139,6 +139,7 @@ export type FrontendAction<
139
139
  */
140
140
  disabled?: boolean;
141
141
  available?: FrontendActionAvailability;
142
+ pairedAction?: string;
142
143
  followUp?: boolean;
143
144
  } & (
144
145
  | {
@@ -177,7 +178,9 @@ export function processActionsForRuntimeRequest(actions: FrontendAction<any>[])
177
178
  (action) =>
178
179
  action.available !== ActionInputAvailability.Disabled &&
179
180
  action.disabled !== true &&
180
- action.name !== "*",
181
+ action.name !== "*" &&
182
+ action.available != "frontend" &&
183
+ !action.pairedAction,
181
184
  )
182
185
  .map((action) => {
183
186
  let available: ActionInputAvailability | undefined = ActionInputAvailability.Enabled;
@@ -1,28 +1,32 @@
1
1
  import { LangGraphInterruptEvent } from "@copilotkit/runtime-client-gql";
2
- import { Condition } from "@copilotkit/shared";
2
+ import { AgentSession } from "../context/copilot-context";
3
+
4
+ export interface LangGraphInterruptRenderHandlerProps<TEventValue = any> {
5
+ event: LangGraphInterruptEvent<TEventValue>;
6
+ resolve: (resolution: string) => void;
7
+ }
8
+
9
+ export interface LangGraphInterruptRenderProps<TEventValue = any> {
10
+ result: unknown;
11
+ event: LangGraphInterruptEvent<TEventValue>;
12
+ resolve: (resolution: string) => void;
13
+ }
3
14
 
4
15
  export interface LangGraphInterruptRender<TEventValue = any> {
5
16
  id: string;
6
17
  /**
7
18
  * The handler function to handle the event.
8
19
  */
9
- handler?: (props: {
10
- event: LangGraphInterruptEvent<TEventValue>;
11
- resolve: (resolution: string) => void;
12
- }) => unknown | Promise<unknown>;
20
+ handler?: (props: LangGraphInterruptRenderHandlerProps<TEventValue>) => any | Promise<any>;
13
21
  /**
14
22
  * The render function to handle the event.
15
23
  */
16
- render?: (props: {
17
- result: unknown;
18
- event: LangGraphInterruptEvent<TEventValue>;
19
- resolve: (resolution: string) => void;
20
- }) => string | React.ReactElement;
24
+ render?: (props: LangGraphInterruptRenderProps<TEventValue>) => string | React.ReactElement;
21
25
  /**
22
- * Conditions to render based on.
26
+ * Method that returns a boolean, indicating if the interrupt action should run
23
27
  * Useful when using multiple interrupts
24
28
  */
25
- conditions?: Condition[];
29
+ enabled?: (args: { eventValue: TEventValue; agentMetadata: AgentSession }) => boolean;
26
30
  }
27
31
 
28
32
  export type LangGraphInterruptAction = LangGraphInterruptRender & {
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/hooks/use-chat.ts"],"sourcesContent":["import React, { useCallback, useRef } from \"react\";\nimport {\n FunctionCallHandler,\n COPILOT_CLOUD_PUBLIC_API_KEY_HEADER,\n CoAgentStateRenderHandler,\n randomId,\n} from \"@copilotkit/shared\";\nimport {\n Message,\n TextMessage,\n ResultMessage,\n convertMessagesToGqlInput,\n filterAdjacentAgentStateMessages,\n filterAgentStateMessages,\n convertGqlOutputToMessages,\n MessageStatusCode,\n MessageRole,\n Role,\n CopilotRequestType,\n ForwardedParametersInput,\n loadMessagesFromJsonRepresentation,\n ExtensionsInput,\n CopilotRuntimeClient,\n langGraphInterruptEvent,\n} from \"@copilotkit/runtime-client-gql\";\n\nimport { CopilotApiConfig } from \"../context\";\nimport { FrontendAction, processActionsForRuntimeRequest } from \"../types/frontend-action\";\nimport { CoagentState } from \"../types/coagent-state\";\nimport { AgentSession } from \"../context/copilot-context\";\nimport { useCopilotRuntimeClient } from \"./use-copilot-runtime-client\";\nimport { useAsyncCallback, useErrorToast } from \"../components/error-boundary/error-utils\";\nimport {\n LangGraphInterruptAction,\n LangGraphInterruptActionSetter,\n} from \"../types/interrupt-action\";\nimport { MetaEvent, MetaEventName } from \"@copilotkit/runtime-client-gql\";\nimport {\n CopilotKitLangGraphInterruptEvent,\n LangGraphInterruptEvent,\n MetaEventInput,\n} from \"@copilotkit/runtime-client-gql\";\n\nexport type UseChatOptions = {\n /**\n * System messages of the chat. Defaults to an empty array.\n */\n initialMessages?: Message[];\n /**\n * Callback function to be called when a function call is received.\n * If the function returns a `ChatRequest` object, the request will be sent\n * automatically to the API and will be used to update the chat.\n */\n onFunctionCall?: FunctionCallHandler;\n\n /**\n * Callback function to be called when a coagent action is received.\n */\n onCoAgentStateRender?: CoAgentStateRenderHandler;\n\n /**\n * Function definitions to be sent to the API.\n */\n actions: FrontendAction<any>[];\n\n /**\n * The CopilotKit API configuration.\n */\n copilotConfig: CopilotApiConfig;\n\n /**\n * The current list of messages in the chat.\n */\n messages: Message[];\n /**\n * The setState-powered method to update the chat messages.\n */\n setMessages: React.Dispatch<React.SetStateAction<Message[]>>;\n\n /**\n * A callback to get the latest system message.\n */\n makeSystemMessageCallback: () => TextMessage;\n\n /**\n * Whether the API request is in progress\n */\n isLoading: boolean;\n\n /**\n * setState-powered method to update the isChatLoading value\n */\n setIsLoading: React.Dispatch<React.SetStateAction<boolean>>;\n\n /**\n * The current list of coagent states.\n */\n coagentStatesRef: React.RefObject<Record<string, CoagentState>>;\n\n /**\n * setState-powered method to update the agent states\n */\n setCoagentStatesWithRef: React.Dispatch<React.SetStateAction<Record<string, CoagentState>>>;\n\n /**\n * The current agent session.\n */\n agentSession: AgentSession | null;\n\n /**\n * setState-powered method to update the agent session\n */\n setAgentSession: React.Dispatch<React.SetStateAction<AgentSession | null>>;\n\n /**\n * The forwarded parameters.\n */\n forwardedParameters?: Pick<ForwardedParametersInput, \"temperature\">;\n\n /**\n * The current thread ID.\n */\n threadId: string;\n /**\n * set the current thread ID\n */\n setThreadId: (threadId: string) => void;\n /**\n * The current run ID.\n */\n runId: string | null;\n /**\n * set the current run ID\n */\n setRunId: (runId: string | null) => void;\n /**\n * The global chat abort controller.\n */\n chatAbortControllerRef: React.MutableRefObject<AbortController | null>;\n /**\n * The agent lock.\n */\n agentLock: string | null;\n /**\n * The extensions.\n */\n extensions: ExtensionsInput;\n /**\n * The setState-powered method to update the extensions.\n */\n setExtensions: React.Dispatch<React.SetStateAction<ExtensionsInput>>;\n\n langGraphInterruptAction: LangGraphInterruptAction | null;\n\n setLangGraphInterruptAction: LangGraphInterruptActionSetter;\n};\n\nexport type UseChatHelpers = {\n /**\n * Append a user message to the chat list. This triggers the API call to fetch\n * the assistant's response.\n * @param message The message to append\n */\n append: (message: Message, options?: AppendMessageOptions) => Promise<void>;\n /**\n * Reload the last AI chat response for the given chat history. If the last\n * message isn't from the assistant, it will request the API to generate a\n * new response.\n */\n reload: () => Promise<void>;\n /**\n * Abort the current request immediately, keep the generated tokens if any.\n */\n stop: () => void;\n\n /**\n * Run the chat completion.\n */\n runChatCompletion: () => Promise<Message[]>;\n};\n\nexport interface AppendMessageOptions {\n /**\n * Whether to run the chat completion after appending the message. Defaults to `true`.\n */\n followUp?: boolean;\n}\n\nexport function useChat(options: UseChatOptions): UseChatHelpers {\n const {\n messages,\n setMessages,\n makeSystemMessageCallback,\n copilotConfig,\n setIsLoading,\n initialMessages,\n isLoading,\n actions,\n onFunctionCall,\n onCoAgentStateRender,\n setCoagentStatesWithRef,\n coagentStatesRef,\n agentSession,\n setAgentSession,\n threadId,\n setThreadId,\n runId,\n setRunId,\n chatAbortControllerRef,\n agentLock,\n extensions,\n setExtensions,\n langGraphInterruptAction,\n setLangGraphInterruptAction,\n } = options;\n const runChatCompletionRef = useRef<(previousMessages: Message[]) => Promise<Message[]>>();\n const addErrorToast = useErrorToast();\n // We need to keep a ref of coagent states and session because of renderAndWait - making sure\n // the latest state is sent to the API\n // This is a workaround and needs to be addressed in the future\n const agentSessionRef = useRef<AgentSession | null>(agentSession);\n agentSessionRef.current = agentSession;\n\n const runIdRef = useRef<string | null>(runId);\n runIdRef.current = runId;\n const extensionsRef = useRef<ExtensionsInput>(extensions);\n extensionsRef.current = extensions;\n\n const publicApiKey = copilotConfig.publicApiKey;\n\n const headers = {\n ...(copilotConfig.headers || {}),\n ...(publicApiKey ? { [COPILOT_CLOUD_PUBLIC_API_KEY_HEADER]: publicApiKey } : {}),\n };\n\n const runtimeClient = useCopilotRuntimeClient({\n url: copilotConfig.chatApiEndpoint,\n publicApiKey: copilotConfig.publicApiKey,\n headers,\n credentials: copilotConfig.credentials,\n });\n\n const runChatCompletion = useAsyncCallback(\n async (previousMessages: Message[]): Promise<Message[]> => {\n setIsLoading(true);\n const interruptEvent = langGraphInterruptAction?.event;\n // In case an interrupt event exist and valid but has no response yet, we cannot process further messages to an agent\n if (\n interruptEvent?.name === MetaEventName.LangGraphInterruptEvent &&\n interruptEvent?.value &&\n !interruptEvent?.response &&\n agentSessionRef.current\n ) {\n addErrorToast([\n new Error(\n \"A message was sent while interrupt is active. This will cause failure on the agent side\",\n ),\n ]);\n }\n\n // this message is just a placeholder. It will disappear once the first real message\n // is received\n let newMessages: Message[] = [\n new TextMessage({\n content: \"\",\n role: Role.Assistant,\n }),\n ];\n\n chatAbortControllerRef.current = new AbortController();\n\n setMessages([...previousMessages, ...newMessages]);\n\n const systemMessage = makeSystemMessageCallback();\n\n const messagesWithContext = [systemMessage, ...(initialMessages || []), ...previousMessages];\n\n const isAgentRun = agentSessionRef.current !== null;\n\n const stream = runtimeClient.asStream(\n runtimeClient.generateCopilotResponse({\n data: {\n frontend: {\n actions: processActionsForRuntimeRequest(actions),\n url: window.location.href,\n },\n threadId: threadId,\n runId: runIdRef.current,\n extensions: extensionsRef.current,\n metaEvents: composeAndFlushMetaEventsInput([langGraphInterruptAction?.event]),\n messages: convertMessagesToGqlInput(filterAgentStateMessages(messagesWithContext)),\n ...(copilotConfig.cloud\n ? {\n cloud: {\n ...(copilotConfig.cloud.guardrails?.input?.restrictToTopic?.enabled\n ? {\n guardrails: {\n inputValidationRules: {\n allowList:\n copilotConfig.cloud.guardrails.input.restrictToTopic.validTopics,\n denyList:\n copilotConfig.cloud.guardrails.input.restrictToTopic.invalidTopics,\n },\n },\n }\n : {}),\n },\n }\n : {}),\n metadata: {\n requestType: CopilotRequestType.Chat,\n },\n ...(agentSessionRef.current\n ? {\n agentSession: agentSessionRef.current,\n }\n : {}),\n agentStates: Object.values(coagentStatesRef.current!).map((state) => ({\n agentName: state.name,\n state: JSON.stringify(state.state),\n configurable: JSON.stringify(state.configurable ?? {}),\n })),\n forwardedParameters: options.forwardedParameters || {},\n },\n properties: copilotConfig.properties,\n signal: chatAbortControllerRef.current?.signal,\n }),\n );\n\n const guardrailsEnabled =\n copilotConfig.cloud?.guardrails?.input?.restrictToTopic.enabled || false;\n\n const reader = stream.getReader();\n\n let executedCoAgentStateRenders: string[] = [];\n let followUp: FrontendAction[\"followUp\"] = undefined;\n\n let messages: Message[] = [];\n let syncedMessages: Message[] = [];\n let interruptMessages: Message[] = [];\n\n try {\n while (true) {\n let done, value;\n\n try {\n const readResult = await reader.read();\n done = readResult.done;\n value = readResult.value;\n } catch (readError) {\n break;\n }\n\n if (done) {\n if (chatAbortControllerRef.current.signal.aborted) {\n return [];\n }\n break;\n }\n\n if (!value?.generateCopilotResponse) {\n continue;\n }\n\n runIdRef.current = value.generateCopilotResponse.runId || null;\n\n // in the output, graphql inserts __typename, which leads to an error when sending it along\n // as input to the next request.\n extensionsRef.current = CopilotRuntimeClient.removeGraphQLTypename(\n value.generateCopilotResponse.extensions || {},\n );\n\n // setThreadId(threadIdRef.current);\n setRunId(runIdRef.current);\n setExtensions(extensionsRef.current);\n let rawMessagesResponse = value.generateCopilotResponse.messages;\n (value.generateCopilotResponse?.metaEvents ?? []).forEach((ev) => {\n if (ev.name === MetaEventName.LangGraphInterruptEvent) {\n let eventValue = langGraphInterruptEvent(ev as LangGraphInterruptEvent).value;\n try {\n eventValue = JSON.parse(eventValue);\n // In case of unparsable string, we keep the event as is\n } catch (e) {}\n setLangGraphInterruptAction({\n event: {\n ...langGraphInterruptEvent(ev as LangGraphInterruptEvent),\n value: eventValue,\n },\n });\n }\n if (ev.name === MetaEventName.CopilotKitLangGraphInterruptEvent) {\n const data = (ev as CopilotKitLangGraphInterruptEvent).data;\n\n // @ts-expect-error -- same type of messages\n rawMessagesResponse = [...rawMessagesResponse, ...data.messages];\n interruptMessages = convertGqlOutputToMessages(\n // @ts-ignore\n filterAdjacentAgentStateMessages(data.messages),\n );\n }\n });\n\n messages = convertGqlOutputToMessages(\n filterAdjacentAgentStateMessages(rawMessagesResponse),\n );\n\n if (messages.length === 0) {\n continue;\n }\n\n newMessages = [];\n\n // request failed, display error message and quit\n if (\n value.generateCopilotResponse.status?.__typename === \"FailedResponseStatus\" &&\n value.generateCopilotResponse.status.reason === \"GUARDRAILS_VALIDATION_FAILED\"\n ) {\n newMessages = [\n new TextMessage({\n role: MessageRole.Assistant,\n content: value.generateCopilotResponse.status.details?.guardrailsReason || \"\",\n }),\n ];\n setMessages([...previousMessages, ...newMessages]);\n break;\n }\n\n // add messages to the chat\n else {\n newMessages = [...messages];\n\n for (const message of messages) {\n // execute onCoAgentStateRender handler\n if (\n message.isAgentStateMessage() &&\n !message.active &&\n !executedCoAgentStateRenders.includes(message.id) &&\n onCoAgentStateRender\n ) {\n // Do not execute a coagent action if guardrails are enabled but the status is not known\n if (guardrailsEnabled && value.generateCopilotResponse.status === undefined) {\n break;\n }\n // execute coagent action\n await onCoAgentStateRender({\n name: message.agentName,\n nodeName: message.nodeName,\n state: message.state,\n });\n executedCoAgentStateRenders.push(message.id);\n }\n }\n\n const lastAgentStateMessage = [...messages]\n .reverse()\n .find((message) => message.isAgentStateMessage());\n\n if (lastAgentStateMessage) {\n if (\n lastAgentStateMessage.state.messages &&\n lastAgentStateMessage.state.messages.length > 0\n ) {\n syncedMessages = loadMessagesFromJsonRepresentation(\n lastAgentStateMessage.state.messages,\n );\n }\n setCoagentStatesWithRef((prevAgentStates) => ({\n ...prevAgentStates,\n [lastAgentStateMessage.agentName]: {\n name: lastAgentStateMessage.agentName,\n state: lastAgentStateMessage.state,\n running: lastAgentStateMessage.running,\n active: lastAgentStateMessage.active,\n threadId: lastAgentStateMessage.threadId,\n nodeName: lastAgentStateMessage.nodeName,\n runId: lastAgentStateMessage.runId,\n },\n }));\n if (lastAgentStateMessage.running) {\n setAgentSession({\n threadId: lastAgentStateMessage.threadId,\n agentName: lastAgentStateMessage.agentName,\n nodeName: lastAgentStateMessage.nodeName,\n });\n } else {\n if (agentLock) {\n setAgentSession({\n threadId: randomId(),\n agentName: agentLock,\n nodeName: undefined,\n });\n } else {\n setAgentSession(null);\n }\n }\n }\n }\n\n if (newMessages.length > 0) {\n // Update message state\n setMessages([...previousMessages, ...newMessages]);\n }\n }\n const finalMessages = constructFinalMessages(\n [...syncedMessages, ...interruptMessages],\n previousMessages,\n newMessages,\n );\n\n let didExecuteAction = false;\n\n // execute regular action executions that are specific to the frontend (last actions)\n if (onFunctionCall) {\n // Find consecutive action execution messages at the end\n const lastMessages = [];\n for (let i = finalMessages.length - 1; i >= 0; i--) {\n const message = finalMessages[i];\n if (\n message.isActionExecutionMessage() &&\n message.status.code !== MessageStatusCode.Pending\n ) {\n lastMessages.unshift(message);\n } else {\n break;\n }\n }\n\n for (const message of lastMessages) {\n // We update the message state before calling the handler so that the render\n // function can be called with `executing` state\n setMessages(finalMessages);\n\n const action = actions.find((action) => action.name === message.name);\n\n if (action) {\n followUp = action.followUp;\n let result: any;\n let error: Error | null = null;\n try {\n result = await Promise.race([\n onFunctionCall({\n messages: previousMessages,\n name: message.name,\n args: message.arguments,\n }),\n new Promise((resolve) =>\n chatAbortControllerRef.current?.signal.addEventListener(\"abort\", () =>\n resolve(\"Operation was aborted by the user\"),\n ),\n ),\n // if the user stopped generation, we also abort consecutive actions\n new Promise((resolve) => {\n if (chatAbortControllerRef.current?.signal.aborted) {\n resolve(\"Operation was aborted by the user\");\n }\n }),\n ]);\n } catch (e) {\n error = e as Error;\n addErrorToast([error]);\n result = `Failed to execute action ${message.name}. ${error.message}`;\n console.error(`Failed to execute action ${message.name}: ${error}`);\n }\n didExecuteAction = true;\n const messageIndex = finalMessages.findIndex((msg) => msg.id === message.id);\n finalMessages.splice(\n messageIndex + 1,\n 0,\n new ResultMessage({\n id: \"result-\" + message.id,\n result: ResultMessage.encodeResult(\n error\n ? {\n content: result,\n error: JSON.parse(\n JSON.stringify(error, Object.getOwnPropertyNames(error)),\n ),\n }\n : result,\n ),\n actionExecutionId: message.id,\n actionName: message.name,\n }),\n );\n }\n }\n\n setMessages(finalMessages);\n }\n\n if (\n // if followUp is not explicitly false\n followUp !== false &&\n // and we executed an action\n (didExecuteAction ||\n // the last message is a server side result\n (!isAgentRun &&\n finalMessages.length &&\n finalMessages[finalMessages.length - 1].isResultMessage())) &&\n // the user did not stop generation\n !chatAbortControllerRef.current?.signal.aborted\n ) {\n // run the completion again and return the result\n\n // wait for next tick to make sure all the react state updates\n // - tried using react-dom's flushSync, but it did not work\n await new Promise((resolve) => setTimeout(resolve, 10));\n\n return await runChatCompletionRef.current!(finalMessages);\n } else if (chatAbortControllerRef.current?.signal.aborted) {\n // filter out all the action execution messages that do not have a consecutive matching result message\n const repairedMessages = finalMessages.filter((message, actionExecutionIndex) => {\n if (message.isActionExecutionMessage()) {\n return finalMessages.find(\n (msg, resultIndex) =>\n msg.isResultMessage() &&\n msg.actionExecutionId === message.id &&\n resultIndex === actionExecutionIndex + 1,\n );\n }\n return true;\n });\n const repairedMessageIds = repairedMessages.map((message) => message.id);\n setMessages(repairedMessages);\n\n // LangGraph needs two pieces of information to continue execution:\n // 1. The threadId\n // 2. The nodeName it came from\n // When stopping the agent, we don't know the nodeName the agent would have ended with\n // Therefore, we set the nodeName to the most reasonable thing we can guess, which\n // is \"__end__\"\n if (agentSessionRef.current?.nodeName) {\n setAgentSession({\n threadId: agentSessionRef.current.threadId,\n agentName: agentSessionRef.current.agentName,\n nodeName: \"__end__\",\n });\n }\n // only return new messages that were not filtered out\n return newMessages.filter((message) => repairedMessageIds.includes(message.id));\n } else {\n return newMessages.slice();\n }\n } finally {\n setIsLoading(false);\n }\n },\n [\n messages,\n setMessages,\n makeSystemMessageCallback,\n copilotConfig,\n setIsLoading,\n initialMessages,\n isLoading,\n actions,\n onFunctionCall,\n onCoAgentStateRender,\n setCoagentStatesWithRef,\n coagentStatesRef,\n agentSession,\n setAgentSession,\n ],\n );\n\n runChatCompletionRef.current = runChatCompletion;\n\n const runChatCompletionAndHandleFunctionCall = useAsyncCallback(\n async (messages: Message[]): Promise<void> => {\n await runChatCompletionRef.current!(messages);\n },\n [messages],\n );\n\n // Go over all events and see that they include data that should be returned to the agent\n const composeAndFlushMetaEventsInput = useCallback(\n (metaEvents: (MetaEvent | undefined | null)[]) => {\n return metaEvents.reduce((acc: MetaEventInput[], event) => {\n if (!event) return acc;\n\n switch (event.name) {\n case MetaEventName.LangGraphInterruptEvent:\n if (event.response) {\n // Flush interrupt event from state\n setLangGraphInterruptAction(null);\n const value = (event as LangGraphInterruptEvent).value;\n return [\n ...acc,\n {\n name: event.name,\n value: typeof value === \"string\" ? value : JSON.stringify(value),\n response:\n typeof event.response === \"string\"\n ? event.response\n : JSON.stringify(event.response),\n },\n ];\n }\n return acc;\n default:\n return acc;\n }\n }, []);\n },\n [setLangGraphInterruptAction],\n );\n\n const append = useAsyncCallback(\n async (message: Message, options?: AppendMessageOptions): Promise<void> => {\n if (isLoading) {\n return;\n }\n\n const newMessages = [...messages, message];\n setMessages(newMessages);\n const followUp = options?.followUp ?? true;\n if (followUp) {\n return runChatCompletionAndHandleFunctionCall(newMessages);\n }\n },\n [isLoading, messages, setMessages, runChatCompletionAndHandleFunctionCall],\n );\n\n const reload = useAsyncCallback(async (): Promise<void> => {\n if (isLoading || messages.length === 0) {\n return;\n }\n let newMessages = [...messages];\n const lastMessage = messages[messages.length - 1];\n\n if (lastMessage.isTextMessage() && lastMessage.role === \"assistant\") {\n newMessages = newMessages.slice(0, -1);\n }\n\n setMessages(newMessages);\n\n return runChatCompletionAndHandleFunctionCall(newMessages);\n }, [isLoading, messages, setMessages, runChatCompletionAndHandleFunctionCall]);\n\n const stop = (): void => {\n chatAbortControllerRef.current?.abort(\"Stop was called\");\n };\n\n return {\n append,\n reload,\n stop,\n runChatCompletion: () => runChatCompletionRef.current!(messages),\n };\n}\n\nfunction constructFinalMessages(\n syncedMessages: Message[],\n previousMessages: Message[],\n newMessages: Message[],\n): Message[] {\n const finalMessages =\n syncedMessages.length > 0 ? [...syncedMessages] : [...previousMessages, ...newMessages];\n\n if (syncedMessages.length > 0) {\n const messagesWithAgentState = [...previousMessages, ...newMessages];\n\n let previousMessageId: string | undefined = undefined;\n\n for (const message of messagesWithAgentState) {\n if (message.isAgentStateMessage()) {\n // insert this message into finalMessages after the position of previousMessageId\n const index = finalMessages.findIndex((msg) => msg.id === previousMessageId);\n if (index !== -1) {\n finalMessages.splice(index + 1, 0, message);\n }\n }\n\n previousMessageId = message.id;\n }\n }\n\n return finalMessages;\n}\n"],"mappings":"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","names":["_a","messages","action","options"]}