@copilotkit/react-core 1.51.3-next.3 → 1.51.3-next.5

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 (65) hide show
  1. package/CHANGELOG.md +19 -0
  2. package/dist/{chunk-OBXQSB7S.mjs → chunk-3HJ2O2MN.mjs} +10 -6
  3. package/dist/chunk-3HJ2O2MN.mjs.map +1 -0
  4. package/dist/{chunk-QNUAXSDP.mjs → chunk-3X3I7OJV.mjs} +12 -6
  5. package/dist/chunk-3X3I7OJV.mjs.map +1 -0
  6. package/dist/{chunk-M7U7RZLU.mjs → chunk-HZWTHNXL.mjs} +2 -2
  7. package/dist/{chunk-4MLKBZGB.mjs → chunk-K2GJI3PT.mjs} +2 -2
  8. package/dist/{chunk-W5BKJN46.mjs → chunk-KINHFG63.mjs} +2 -2
  9. package/dist/{chunk-F555TVE4.mjs → chunk-LMHB2D4J.mjs} +3 -3
  10. package/dist/chunk-LMHB2D4J.mjs.map +1 -0
  11. package/dist/{chunk-2VA5EV2P.mjs → chunk-THCOOWDH.mjs} +3 -3
  12. package/dist/components/copilot-provider/copilotkit.js +13 -7
  13. package/dist/components/copilot-provider/copilotkit.js.map +1 -1
  14. package/dist/components/copilot-provider/copilotkit.mjs +3 -3
  15. package/dist/components/copilot-provider/index.js +13 -7
  16. package/dist/components/copilot-provider/index.js.map +1 -1
  17. package/dist/components/copilot-provider/index.mjs +3 -3
  18. package/dist/components/index.js +13 -7
  19. package/dist/components/index.js.map +1 -1
  20. package/dist/components/index.mjs +3 -3
  21. package/dist/context/index.js +2 -2
  22. package/dist/context/index.js.map +1 -1
  23. package/dist/context/index.mjs +1 -1
  24. package/dist/context/threads-context.js +2 -2
  25. package/dist/context/threads-context.js.map +1 -1
  26. package/dist/context/threads-context.mjs +1 -1
  27. package/dist/hooks/index.js +19 -9
  28. package/dist/hooks/index.js.map +1 -1
  29. package/dist/hooks/index.mjs +5 -5
  30. package/dist/hooks/use-coagent-state-render-bridge.js +11 -5
  31. package/dist/hooks/use-coagent-state-render-bridge.js.map +1 -1
  32. package/dist/hooks/use-coagent-state-render-bridge.mjs +2 -2
  33. package/dist/hooks/use-copilot-chat-headless_c.js +19 -9
  34. package/dist/hooks/use-copilot-chat-headless_c.js.map +1 -1
  35. package/dist/hooks/use-copilot-chat-headless_c.mjs +4 -4
  36. package/dist/hooks/use-copilot-chat.js +19 -9
  37. package/dist/hooks/use-copilot-chat.js.map +1 -1
  38. package/dist/hooks/use-copilot-chat.mjs +4 -4
  39. package/dist/hooks/use-copilot-chat_internal.js +19 -9
  40. package/dist/hooks/use-copilot-chat_internal.js.map +1 -1
  41. package/dist/hooks/use-copilot-chat_internal.mjs +3 -3
  42. package/dist/hooks/use-langgraph-interrupt-render.mjs +1 -1
  43. package/dist/index.js +21 -11
  44. package/dist/index.js.map +1 -1
  45. package/dist/index.mjs +7 -7
  46. package/dist/lib/copilot-task.mjs +4 -4
  47. package/dist/lib/index.mjs +4 -4
  48. package/dist/setupTests.js +3 -1
  49. package/dist/setupTests.js.map +1 -1
  50. package/dist/setupTests.mjs +3 -1
  51. package/dist/setupTests.mjs.map +1 -1
  52. package/package.json +5 -5
  53. package/src/context/__tests__/threads-context.test.tsx +28 -0
  54. package/src/context/threads-context.tsx +2 -2
  55. package/src/hooks/__tests__/use-coagent-state-render.e2e.test.tsx +229 -0
  56. package/src/hooks/use-coagent-state-render-bridge.tsx +15 -5
  57. package/src/hooks/use-copilot-chat_internal.ts +17 -9
  58. package/src/setupTests.ts +2 -0
  59. package/dist/chunk-F555TVE4.mjs.map +0 -1
  60. package/dist/chunk-OBXQSB7S.mjs.map +0 -1
  61. package/dist/chunk-QNUAXSDP.mjs.map +0 -1
  62. /package/dist/{chunk-M7U7RZLU.mjs.map → chunk-HZWTHNXL.mjs.map} +0 -0
  63. /package/dist/{chunk-4MLKBZGB.mjs.map → chunk-K2GJI3PT.mjs.map} +0 -0
  64. /package/dist/{chunk-W5BKJN46.mjs.map → chunk-KINHFG63.mjs.map} +0 -0
  65. /package/dist/{chunk-2VA5EV2P.mjs.map → chunk-THCOOWDH.mjs.map} +0 -0
package/dist/index.mjs CHANGED
@@ -5,7 +5,7 @@ import "./chunk-36KQV2NA.mjs";
5
5
  import "./chunk-LUGEI4YQ.mjs";
6
6
  import {
7
7
  CopilotTask
8
- } from "./chunk-4MLKBZGB.mjs";
8
+ } from "./chunk-K2GJI3PT.mjs";
9
9
  import "./chunk-4CEQJ2X6.mjs";
10
10
  import "./chunk-A6NKSGH3.mjs";
11
11
  import {
@@ -40,16 +40,16 @@ import {
40
40
  } from "./chunk-7DTB7S5V.mjs";
41
41
  import {
42
42
  useCopilotChatHeadless_c
43
- } from "./chunk-W5BKJN46.mjs";
43
+ } from "./chunk-KINHFG63.mjs";
44
44
  import {
45
45
  useCopilotChatSuggestions
46
46
  } from "./chunk-XDFVCQD3.mjs";
47
47
  import {
48
48
  useCopilotChat
49
- } from "./chunk-M7U7RZLU.mjs";
49
+ } from "./chunk-HZWTHNXL.mjs";
50
50
  import {
51
51
  useCopilotChatInternal
52
- } from "./chunk-OBXQSB7S.mjs";
52
+ } from "./chunk-3HJ2O2MN.mjs";
53
53
  import {
54
54
  useLangGraphInterruptRender
55
55
  } from "./chunk-VV56AVPB.mjs";
@@ -71,7 +71,7 @@ import "./chunk-SPCZTZCY.mjs";
71
71
  import {
72
72
  CopilotKit,
73
73
  defaultCopilotContextCategories
74
- } from "./chunk-2VA5EV2P.mjs";
74
+ } from "./chunk-THCOOWDH.mjs";
75
75
  import "./chunk-FJEI7IBX.mjs";
76
76
  import "./chunk-CYDWEPFL.mjs";
77
77
  import {
@@ -86,13 +86,13 @@ import "./chunk-PMAFHQ7P.mjs";
86
86
  import "./chunk-YYN33GSG.mjs";
87
87
  import "./chunk-BVBFFI7K.mjs";
88
88
  import "./chunk-O7ARI5CV.mjs";
89
- import "./chunk-QNUAXSDP.mjs";
89
+ import "./chunk-3X3I7OJV.mjs";
90
90
  import "./chunk-NB2FKV2V.mjs";
91
91
  import {
92
92
  ThreadsContext,
93
93
  ThreadsProvider,
94
94
  useThreads
95
- } from "./chunk-F555TVE4.mjs";
95
+ } from "./chunk-LMHB2D4J.mjs";
96
96
  import "./chunk-WF65O6HX.mjs";
97
97
  import "./chunk-EFL5OBKN.mjs";
98
98
  import "./chunk-6ZLPNY7X.mjs";
@@ -1,10 +1,10 @@
1
1
  import {
2
2
  CopilotTask
3
- } from "../chunk-4MLKBZGB.mjs";
3
+ } from "../chunk-K2GJI3PT.mjs";
4
4
  import "../chunk-4CEQJ2X6.mjs";
5
5
  import "../chunk-JD7BAH7U.mjs";
6
6
  import "../chunk-SPCZTZCY.mjs";
7
- import "../chunk-2VA5EV2P.mjs";
7
+ import "../chunk-THCOOWDH.mjs";
8
8
  import "../chunk-FJEI7IBX.mjs";
9
9
  import "../chunk-CYDWEPFL.mjs";
10
10
  import "../chunk-2IDV5OHF.mjs";
@@ -15,9 +15,9 @@ import "../chunk-PMAFHQ7P.mjs";
15
15
  import "../chunk-YYN33GSG.mjs";
16
16
  import "../chunk-BVBFFI7K.mjs";
17
17
  import "../chunk-O7ARI5CV.mjs";
18
- import "../chunk-QNUAXSDP.mjs";
18
+ import "../chunk-3X3I7OJV.mjs";
19
19
  import "../chunk-NB2FKV2V.mjs";
20
- import "../chunk-F555TVE4.mjs";
20
+ import "../chunk-LMHB2D4J.mjs";
21
21
  import "../chunk-WF65O6HX.mjs";
22
22
  import "../chunk-EFL5OBKN.mjs";
23
23
  import "../chunk-6ZLPNY7X.mjs";
@@ -1,11 +1,11 @@
1
1
  import "../chunk-LUGEI4YQ.mjs";
2
2
  import {
3
3
  CopilotTask
4
- } from "../chunk-4MLKBZGB.mjs";
4
+ } from "../chunk-K2GJI3PT.mjs";
5
5
  import "../chunk-4CEQJ2X6.mjs";
6
6
  import "../chunk-JD7BAH7U.mjs";
7
7
  import "../chunk-SPCZTZCY.mjs";
8
- import "../chunk-2VA5EV2P.mjs";
8
+ import "../chunk-THCOOWDH.mjs";
9
9
  import "../chunk-FJEI7IBX.mjs";
10
10
  import "../chunk-CYDWEPFL.mjs";
11
11
  import "../chunk-2IDV5OHF.mjs";
@@ -16,9 +16,9 @@ import "../chunk-PMAFHQ7P.mjs";
16
16
  import "../chunk-YYN33GSG.mjs";
17
17
  import "../chunk-BVBFFI7K.mjs";
18
18
  import "../chunk-O7ARI5CV.mjs";
19
- import "../chunk-QNUAXSDP.mjs";
19
+ import "../chunk-3X3I7OJV.mjs";
20
20
  import "../chunk-NB2FKV2V.mjs";
21
- import "../chunk-F555TVE4.mjs";
21
+ import "../chunk-LMHB2D4J.mjs";
22
22
  import "../chunk-WF65O6HX.mjs";
23
23
  import "../chunk-EFL5OBKN.mjs";
24
24
  import "../chunk-6ZLPNY7X.mjs";
@@ -18,7 +18,9 @@ jest.mock("@copilotkit/shared", () => ({
18
18
  return defaultValue;
19
19
  }
20
20
  }),
21
- CopilotKitAgentDiscoveryError: jest.fn()
21
+ randomId: jest.fn(() => "test-random-id"),
22
+ CopilotKitAgentDiscoveryError: jest.fn(),
23
+ randomUUID: jest.fn(() => "mock-thread-id")
22
24
  }));
23
25
  jest.mock("react-dom/test-utils", () => ({
24
26
  act: jest.fn((fn) => fn())
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/setupTests.ts"],"sourcesContent":["// Mock modules that cause ES module issues\njest.mock(\"@segment/analytics-node\", () => ({\n Analytics: jest.fn().mockImplementation(() => ({\n track: jest.fn(),\n identify: jest.fn(),\n page: jest.fn(),\n group: jest.fn(),\n alias: jest.fn(),\n })),\n}));\n\njest.mock(\"@copilotkit/shared\", () => ({\n parseJson: jest.fn((jsonString, defaultValue) => {\n try {\n return JSON.parse(jsonString);\n } catch {\n return defaultValue;\n }\n }),\n CopilotKitAgentDiscoveryError: jest.fn(),\n}));\n\n// Mock react-dom/test-utils to avoid compatibility issues\njest.mock(\"react-dom/test-utils\", () => ({\n act: jest.fn((fn) => fn()),\n}));\n"],"mappings":";;;AACA,KAAK,KAAK,2BAA2B,OAAO;AAAA,EAC1C,WAAW,KAAK,GAAG,EAAE,mBAAmB,OAAO;AAAA,IAC7C,OAAO,KAAK,GAAG;AAAA,IACf,UAAU,KAAK,GAAG;AAAA,IAClB,MAAM,KAAK,GAAG;AAAA,IACd,OAAO,KAAK,GAAG;AAAA,IACf,OAAO,KAAK,GAAG;AAAA,EACjB,EAAE;AACJ,EAAE;AAEF,KAAK,KAAK,sBAAsB,OAAO;AAAA,EACrC,WAAW,KAAK,GAAG,CAAC,YAAY,iBAAiB;AAC/C,QAAI;AACF,aAAO,KAAK,MAAM,UAAU;AAAA,IAC9B,SAAQ,GAAN;AACA,aAAO;AAAA,IACT;AAAA,EACF,CAAC;AAAA,EACD,+BAA+B,KAAK,GAAG;AACzC,EAAE;AAGF,KAAK,KAAK,wBAAwB,OAAO;AAAA,EACvC,KAAK,KAAK,GAAG,CAAC,OAAO,GAAG,CAAC;AAC3B,EAAE;","names":[]}
1
+ {"version":3,"sources":["../src/setupTests.ts"],"sourcesContent":["// Mock modules that cause ES module issues\njest.mock(\"@segment/analytics-node\", () => ({\n Analytics: jest.fn().mockImplementation(() => ({\n track: jest.fn(),\n identify: jest.fn(),\n page: jest.fn(),\n group: jest.fn(),\n alias: jest.fn(),\n })),\n}));\n\njest.mock(\"@copilotkit/shared\", () => ({\n parseJson: jest.fn((jsonString, defaultValue) => {\n try {\n return JSON.parse(jsonString);\n } catch {\n return defaultValue;\n }\n }),\n randomId: jest.fn(() => \"test-random-id\"),\n CopilotKitAgentDiscoveryError: jest.fn(),\n randomUUID: jest.fn(() => \"mock-thread-id\"),\n}));\n\n// Mock react-dom/test-utils to avoid compatibility issues\njest.mock(\"react-dom/test-utils\", () => ({\n act: jest.fn((fn) => fn()),\n}));\n"],"mappings":";;;AACA,KAAK,KAAK,2BAA2B,OAAO;AAAA,EAC1C,WAAW,KAAK,GAAG,EAAE,mBAAmB,OAAO;AAAA,IAC7C,OAAO,KAAK,GAAG;AAAA,IACf,UAAU,KAAK,GAAG;AAAA,IAClB,MAAM,KAAK,GAAG;AAAA,IACd,OAAO,KAAK,GAAG;AAAA,IACf,OAAO,KAAK,GAAG;AAAA,EACjB,EAAE;AACJ,EAAE;AAEF,KAAK,KAAK,sBAAsB,OAAO;AAAA,EACrC,WAAW,KAAK,GAAG,CAAC,YAAY,iBAAiB;AAC/C,QAAI;AACF,aAAO,KAAK,MAAM,UAAU;AAAA,IAC9B,SAAQ,GAAN;AACA,aAAO;AAAA,IACT;AAAA,EACF,CAAC;AAAA,EACD,UAAU,KAAK,GAAG,MAAM,gBAAgB;AAAA,EACxC,+BAA+B,KAAK,GAAG;AAAA,EACvC,YAAY,KAAK,GAAG,MAAM,gBAAgB;AAC5C,EAAE;AAGF,KAAK,KAAK,wBAAwB,OAAO;AAAA,EACvC,KAAK,KAAK,GAAG,CAAC,OAAO,GAAG,CAAC;AAC3B,EAAE;","names":[]}
@@ -16,7 +16,9 @@ jest.mock("@copilotkit/shared", () => ({
16
16
  return defaultValue;
17
17
  }
18
18
  }),
19
- CopilotKitAgentDiscoveryError: jest.fn()
19
+ randomId: jest.fn(() => "test-random-id"),
20
+ CopilotKitAgentDiscoveryError: jest.fn(),
21
+ randomUUID: jest.fn(() => "mock-thread-id")
20
22
  }));
21
23
  jest.mock("react-dom/test-utils", () => ({
22
24
  act: jest.fn((fn) => fn())
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/setupTests.ts"],"sourcesContent":["// Mock modules that cause ES module issues\njest.mock(\"@segment/analytics-node\", () => ({\n Analytics: jest.fn().mockImplementation(() => ({\n track: jest.fn(),\n identify: jest.fn(),\n page: jest.fn(),\n group: jest.fn(),\n alias: jest.fn(),\n })),\n}));\n\njest.mock(\"@copilotkit/shared\", () => ({\n parseJson: jest.fn((jsonString, defaultValue) => {\n try {\n return JSON.parse(jsonString);\n } catch {\n return defaultValue;\n }\n }),\n CopilotKitAgentDiscoveryError: jest.fn(),\n}));\n\n// Mock react-dom/test-utils to avoid compatibility issues\njest.mock(\"react-dom/test-utils\", () => ({\n act: jest.fn((fn) => fn()),\n}));\n"],"mappings":";AACA,KAAK,KAAK,2BAA2B,OAAO;AAAA,EAC1C,WAAW,KAAK,GAAG,EAAE,mBAAmB,OAAO;AAAA,IAC7C,OAAO,KAAK,GAAG;AAAA,IACf,UAAU,KAAK,GAAG;AAAA,IAClB,MAAM,KAAK,GAAG;AAAA,IACd,OAAO,KAAK,GAAG;AAAA,IACf,OAAO,KAAK,GAAG;AAAA,EACjB,EAAE;AACJ,EAAE;AAEF,KAAK,KAAK,sBAAsB,OAAO;AAAA,EACrC,WAAW,KAAK,GAAG,CAAC,YAAY,iBAAiB;AAC/C,QAAI;AACF,aAAO,KAAK,MAAM,UAAU;AAAA,IAC9B,SAAQ,GAAN;AACA,aAAO;AAAA,IACT;AAAA,EACF,CAAC;AAAA,EACD,+BAA+B,KAAK,GAAG;AACzC,EAAE;AAGF,KAAK,KAAK,wBAAwB,OAAO;AAAA,EACvC,KAAK,KAAK,GAAG,CAAC,OAAO,GAAG,CAAC;AAC3B,EAAE;","names":[]}
1
+ {"version":3,"sources":["../src/setupTests.ts"],"sourcesContent":["// Mock modules that cause ES module issues\njest.mock(\"@segment/analytics-node\", () => ({\n Analytics: jest.fn().mockImplementation(() => ({\n track: jest.fn(),\n identify: jest.fn(),\n page: jest.fn(),\n group: jest.fn(),\n alias: jest.fn(),\n })),\n}));\n\njest.mock(\"@copilotkit/shared\", () => ({\n parseJson: jest.fn((jsonString, defaultValue) => {\n try {\n return JSON.parse(jsonString);\n } catch {\n return defaultValue;\n }\n }),\n randomId: jest.fn(() => \"test-random-id\"),\n CopilotKitAgentDiscoveryError: jest.fn(),\n randomUUID: jest.fn(() => \"mock-thread-id\"),\n}));\n\n// Mock react-dom/test-utils to avoid compatibility issues\njest.mock(\"react-dom/test-utils\", () => ({\n act: jest.fn((fn) => fn()),\n}));\n"],"mappings":";AACA,KAAK,KAAK,2BAA2B,OAAO;AAAA,EAC1C,WAAW,KAAK,GAAG,EAAE,mBAAmB,OAAO;AAAA,IAC7C,OAAO,KAAK,GAAG;AAAA,IACf,UAAU,KAAK,GAAG;AAAA,IAClB,MAAM,KAAK,GAAG;AAAA,IACd,OAAO,KAAK,GAAG;AAAA,IACf,OAAO,KAAK,GAAG;AAAA,EACjB,EAAE;AACJ,EAAE;AAEF,KAAK,KAAK,sBAAsB,OAAO;AAAA,EACrC,WAAW,KAAK,GAAG,CAAC,YAAY,iBAAiB;AAC/C,QAAI;AACF,aAAO,KAAK,MAAM,UAAU;AAAA,IAC9B,SAAQ,GAAN;AACA,aAAO;AAAA,IACT;AAAA,EACF,CAAC;AAAA,EACD,UAAU,KAAK,GAAG,MAAM,gBAAgB;AAAA,EACxC,+BAA+B,KAAK,GAAG;AAAA,EACvC,YAAY,KAAK,GAAG,MAAM,gBAAgB;AAC5C,EAAE;AAGF,KAAK,KAAK,wBAAwB,OAAO;AAAA,EACvC,KAAK,KAAK,GAAG,CAAC,OAAO,GAAG,CAAC;AAC3B,EAAE;","names":[]}
package/package.json CHANGED
@@ -9,7 +9,7 @@
9
9
  "publishConfig": {
10
10
  "access": "public"
11
11
  },
12
- "version": "1.51.3-next.3",
12
+ "version": "1.51.3-next.5",
13
13
  "sideEffects": [
14
14
  "**/*.css"
15
15
  ],
@@ -59,10 +59,10 @@
59
59
  "@scarf/scarf": "^1.3.0",
60
60
  "streamdown": "^1.6.10",
61
61
  "untruncate-json": "^0.0.1",
62
- "@copilotkit/runtime-client-gql": "1.51.3-next.3",
63
- "@copilotkit/shared": "1.51.3-next.3",
64
- "@copilotkitnext/core": "1.51.3-next.3",
65
- "@copilotkitnext/react": "1.51.3-next.3"
62
+ "@copilotkit/runtime-client-gql": "1.51.3-next.5",
63
+ "@copilotkit/shared": "1.51.3-next.5",
64
+ "@copilotkitnext/core": "1.51.3-next.5",
65
+ "@copilotkitnext/react": "1.51.3-next.5"
66
66
  },
67
67
  "keywords": [
68
68
  "copilotkit",
@@ -0,0 +1,28 @@
1
+ import React from "react";
2
+ import { render, screen } from "@testing-library/react";
3
+ import { ThreadsProvider, useThreads } from "../threads-context";
4
+
5
+ function ThreadIdViewer() {
6
+ const { threadId } = useThreads();
7
+ return <div data-testid="threadId">{threadId}</div>;
8
+ }
9
+
10
+ describe("ThreadsProvider", () => {
11
+ it("updates threadId when explicit prop becomes available", () => {
12
+ const { rerender } = render(
13
+ <ThreadsProvider>
14
+ <ThreadIdViewer />
15
+ </ThreadsProvider>,
16
+ );
17
+
18
+ expect(screen.getByTestId("threadId").textContent).toBe("mock-thread-id");
19
+
20
+ rerender(
21
+ <ThreadsProvider threadId="customer-thread">
22
+ <ThreadIdViewer />
23
+ </ThreadsProvider>,
24
+ );
25
+
26
+ expect(screen.getByTestId("threadId").textContent).toBe("customer-thread");
27
+ });
28
+ });
@@ -14,9 +14,9 @@ export interface ThreadsProviderProps {
14
14
  }
15
15
 
16
16
  export function ThreadsProvider({ children, threadId: explicitThreadId }: ThreadsProviderProps) {
17
- const [internalThreadId, setThreadId] = useState<string>(explicitThreadId ?? randomUUID());
17
+ const [internalThreadId, setThreadId] = useState<string>(() => randomUUID());
18
18
 
19
- const threadId = internalThreadId;
19
+ const threadId = explicitThreadId ?? internalThreadId;
20
20
 
21
21
  return (
22
22
  <ThreadsContext.Provider
@@ -0,0 +1,229 @@
1
+ import React from "react";
2
+ import { act, render, screen, waitFor } from "@testing-library/react";
3
+ import { useCoAgentStateRender } from "../use-coagent-state-render";
4
+ import { CoAgentStateRenderBridge } from "../use-coagent-state-render-bridge";
5
+ import {
6
+ CoAgentStateRendersProvider,
7
+ CopilotContext,
8
+ type CopilotContextParams,
9
+ } from "../../context";
10
+
11
+ const mockAgent = {
12
+ state: {},
13
+ isRunning: true,
14
+ subscribe: jest.fn(),
15
+ };
16
+
17
+ let lastSubscriber: any = null;
18
+
19
+ jest.mock("@copilotkitnext/react", () => ({
20
+ useAgent: jest.fn(() => ({ agent: mockAgent })),
21
+ }));
22
+
23
+ jest.mock("../../components/toast/toast-provider", () => ({
24
+ useToast: () => ({
25
+ setBannerError: jest.fn(),
26
+ addToast: jest.fn(),
27
+ }),
28
+ }));
29
+
30
+ function TestHarness({ snapshot }: { snapshot: string }) {
31
+ useCoAgentStateRender<{ current_step?: string }>({
32
+ name: "test-agent",
33
+ render: ({ state }) => (
34
+ <div data-testid="state">{state.current_step ?? "none"}</div>
35
+ ),
36
+ });
37
+
38
+ return (
39
+ <CoAgentStateRenderBridge
40
+ agentId="test-agent"
41
+ message={{ id: "msg-1", role: "assistant" }}
42
+ position="after"
43
+ runId="run-1"
44
+ messageIndex={0}
45
+ messageIndexInRun={0}
46
+ numberOfMessagesInRun={1}
47
+ stateSnapshot={snapshot}
48
+ />
49
+ );
50
+ }
51
+
52
+ function SnapshotHarness({
53
+ snapshot,
54
+ message,
55
+ }: {
56
+ snapshot: string;
57
+ message: { id: string; role: string; content?: string };
58
+ }) {
59
+ useCoAgentStateRender<{ current_step?: string }>({
60
+ name: "test-agent",
61
+ render: ({ state }) => (
62
+ <div data-testid="state">{state.current_step ?? "none"}</div>
63
+ ),
64
+ });
65
+
66
+ return (
67
+ <CoAgentStateRenderBridge
68
+ agentId="test-agent"
69
+ message={message}
70
+ position="after"
71
+ runId="run-early"
72
+ messageIndex={0}
73
+ messageIndexInRun={0}
74
+ numberOfMessagesInRun={1}
75
+ stateSnapshot={snapshot}
76
+ />
77
+ );
78
+ }
79
+
80
+ function LiveStateHarness({ message }: { message: { id: string; role: string } }) {
81
+ useCoAgentStateRender<{ current_step?: string }>({
82
+ name: "test-agent",
83
+ render: ({ state }) => (
84
+ <div data-testid="state">{state.current_step ?? "none"}</div>
85
+ ),
86
+ });
87
+
88
+ return (
89
+ <CoAgentStateRenderBridge
90
+ agentId="test-agent"
91
+ message={message}
92
+ position="after"
93
+ runId="run-live"
94
+ messageIndex={0}
95
+ messageIndexInRun={0}
96
+ numberOfMessagesInRun={1}
97
+ stateSnapshot={undefined}
98
+ />
99
+ );
100
+ }
101
+
102
+ describe("useCoAgentStateRender", () => {
103
+ beforeEach(() => {
104
+ lastSubscriber = null;
105
+ mockAgent.state = {};
106
+ mockAgent.subscribe.mockImplementation((subscriber: any) => {
107
+ lastSubscriber = subscriber;
108
+ return { unsubscribe: jest.fn() };
109
+ });
110
+ });
111
+
112
+ it("re-renders when state snapshots change", async () => {
113
+ const copilotContextValue = {
114
+ chatComponentsCache: { current: { actions: {}, coAgentStateRenders: {} } },
115
+ availableAgents: [],
116
+ } as CopilotContextParams;
117
+
118
+ const { rerender } = render(
119
+ <CopilotContext.Provider value={copilotContextValue}>
120
+ <CoAgentStateRendersProvider>
121
+ <TestHarness snapshot={JSON.stringify({ current_step: "Processing..." })} />
122
+ </CoAgentStateRendersProvider>
123
+ </CopilotContext.Provider>,
124
+ );
125
+
126
+ await waitFor(() => {
127
+ expect(screen.getByTestId("state").textContent).toBe("Processing...");
128
+ });
129
+
130
+ rerender(
131
+ <CopilotContext.Provider value={copilotContextValue}>
132
+ <CoAgentStateRendersProvider>
133
+ <TestHarness snapshot={JSON.stringify({ current_step: "Thinking..." })} />
134
+ </CoAgentStateRendersProvider>
135
+ </CopilotContext.Provider>,
136
+ );
137
+
138
+ await waitFor(() => {
139
+ expect(screen.getByTestId("state").textContent).toBe("Thinking...");
140
+ });
141
+ });
142
+
143
+ it("renders snapshots that arrive before assistant text", async () => {
144
+ const copilotContextValue = {
145
+ chatComponentsCache: { current: { actions: {}, coAgentStateRenders: {} } },
146
+ availableAgents: [],
147
+ } as CopilotContextParams;
148
+
149
+ const baseMessage = { id: "msg-early", role: "assistant", content: "" };
150
+
151
+ const { rerender } = render(
152
+ <CopilotContext.Provider value={copilotContextValue}>
153
+ <CoAgentStateRendersProvider>
154
+ <SnapshotHarness
155
+ message={baseMessage}
156
+ snapshot={JSON.stringify({ current_step: "Processing..." })}
157
+ />
158
+ </CoAgentStateRendersProvider>
159
+ </CopilotContext.Provider>,
160
+ );
161
+
162
+ await waitFor(() => {
163
+ expect(screen.getByTestId("state").textContent).toBe("Processing...");
164
+ });
165
+
166
+ rerender(
167
+ <CopilotContext.Provider value={copilotContextValue}>
168
+ <CoAgentStateRendersProvider>
169
+ <SnapshotHarness
170
+ message={baseMessage}
171
+ snapshot={JSON.stringify({ current_step: "Thinking..." })}
172
+ />
173
+ </CoAgentStateRendersProvider>
174
+ </CopilotContext.Provider>,
175
+ );
176
+
177
+ await waitFor(() => {
178
+ expect(screen.getByTestId("state").textContent).toBe("Thinking...");
179
+ });
180
+
181
+ rerender(
182
+ <CopilotContext.Provider value={copilotContextValue}>
183
+ <CoAgentStateRendersProvider>
184
+ <SnapshotHarness
185
+ message={{ ...baseMessage, content: "Hello" }}
186
+ snapshot={JSON.stringify({ current_step: "Finalizing..." })}
187
+ />
188
+ </CoAgentStateRendersProvider>
189
+ </CopilotContext.Provider>,
190
+ );
191
+
192
+ await waitFor(() => {
193
+ expect(screen.getByTestId("state").textContent).toBe("Finalizing...");
194
+ });
195
+ });
196
+
197
+ it("re-renders from live agent state updates", async () => {
198
+ const copilotContextValue = {
199
+ chatComponentsCache: { current: { actions: {}, coAgentStateRenders: {} } },
200
+ availableAgents: [],
201
+ } as CopilotContextParams;
202
+
203
+ render(
204
+ <CopilotContext.Provider value={copilotContextValue}>
205
+ <CoAgentStateRendersProvider>
206
+ <LiveStateHarness message={{ id: "msg-live", role: "assistant" }} />
207
+ </CoAgentStateRendersProvider>
208
+ </CopilotContext.Provider>,
209
+ );
210
+
211
+ mockAgent.state = { current_step: "Processing..." };
212
+ act(() => {
213
+ lastSubscriber?.onStateChanged?.({ state: mockAgent.state });
214
+ });
215
+
216
+ await waitFor(() => {
217
+ expect(screen.getByTestId("state").textContent).toBe("Processing...");
218
+ });
219
+
220
+ mockAgent.state = { current_step: "Thinking..." };
221
+ act(() => {
222
+ lastSubscriber?.onStateChanged?.({ state: mockAgent.state });
223
+ });
224
+
225
+ await waitFor(() => {
226
+ expect(screen.getByTestId("state").textContent).toBe("Thinking...");
227
+ });
228
+ });
229
+ });
@@ -123,6 +123,7 @@ export function useCoagentStateRenderBridge(agentId: string, props: CoAgentState
123
123
  const { coAgentStateRenders, claimsRef } = useCoAgentStateRenders();
124
124
  const { agent } = useAgent({ agentId });
125
125
  const [nodeName, setNodeName] = useState<string | undefined>(undefined);
126
+ const [, forceUpdate] = useState(0);
126
127
 
127
128
  const runId = props.runId ?? message.runId;
128
129
  const effectiveRunId = runId || "pending";
@@ -130,6 +131,9 @@ export function useCoagentStateRenderBridge(agentId: string, props: CoAgentState
130
131
  useEffect(() => {
131
132
  if (!agent) return;
132
133
  const subscriber: AgentSubscriber = {
134
+ onStateChanged: () => {
135
+ forceUpdate((value) => value + 1);
136
+ },
133
137
  onStepStartedEvent: ({ event }) => {
134
138
  if (event.stepName !== nodeName) {
135
139
  setNodeName(event.stepName);
@@ -248,12 +252,18 @@ export function useCoagentStateRenderBridge(agentId: string, props: CoAgentState
248
252
  }
249
253
 
250
254
  // If we found state, and given that now there's a claim for the current message, let's save it in the claim
251
- if (snapshot && !claimsRef.current[message.id].locked) {
252
- if (stateSnapshot) {
253
- claimsRef.current[message.id].stateSnapshot = snapshot;
254
- claimsRef.current[message.id].locked = true;
255
- } else {
255
+ if (snapshot) {
256
+ const existingSnapshot = claimsRef.current[message.id].stateSnapshot;
257
+ const snapshotChanged =
258
+ stateSnapshot &&
259
+ existingSnapshot !== undefined &&
260
+ !areStatesEquals(existingSnapshot, snapshot);
261
+
262
+ if (!claimsRef.current[message.id].locked || snapshotChanged) {
256
263
  claimsRef.current[message.id].stateSnapshot = snapshot;
264
+ if (stateSnapshot) {
265
+ claimsRef.current[message.id].locked = true;
266
+ }
257
267
  }
258
268
  }
259
269
 
@@ -426,14 +426,9 @@ export function useCopilotChatInternal({
426
426
  if (options?.clearSuggestions) {
427
427
  copilotkit.clearSuggestions(resolvedAgentId);
428
428
  }
429
- agent?.addMessage(message);
430
- if (followUp) {
431
- try {
432
- await copilotkit.runAgent({ agent });
433
- } catch (error) {
434
- console.error("CopilotChat: runAgent failed", error);
435
- }
436
- }
429
+
430
+ // Call onSubmitMessage BEFORE adding message and running agent
431
+ // This allows users to perform actions (e.g., open chat window) before agent starts processing
437
432
  if (onSubmitMessage) {
438
433
  const content =
439
434
  typeof message.content === "string"
@@ -443,7 +438,20 @@ export function useCopilotChatInternal({
443
438
  : message.content && "filename" in message.content
444
439
  ? message.content.filename
445
440
  : "";
446
- onSubmitMessage(content);
441
+ try {
442
+ await onSubmitMessage(content);
443
+ } catch (error) {
444
+ console.error("Error in onSubmitMessage:", error);
445
+ }
446
+ }
447
+
448
+ agent?.addMessage(message);
449
+ if (followUp) {
450
+ try {
451
+ await copilotkit.runAgent({ agent });
452
+ } catch (error) {
453
+ console.error("CopilotChat: runAgent failed", error);
454
+ }
447
455
  }
448
456
  },
449
457
  [agent, copilotkit, resolvedAgentId, onSubmitMessage],
package/src/setupTests.ts CHANGED
@@ -17,7 +17,9 @@ jest.mock("@copilotkit/shared", () => ({
17
17
  return defaultValue;
18
18
  }
19
19
  }),
20
+ randomId: jest.fn(() => "test-random-id"),
20
21
  CopilotKitAgentDiscoveryError: jest.fn(),
22
+ randomUUID: jest.fn(() => "mock-thread-id"),
21
23
  }));
22
24
 
23
25
  // Mock react-dom/test-utils to avoid compatibility issues
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/context/threads-context.tsx"],"sourcesContent":["import React, { createContext, useContext, useState, ReactNode, SetStateAction } from \"react\";\nimport { randomUUID } from \"@copilotkit/shared\";\n\nexport interface ThreadsContextValue {\n threadId: string;\n setThreadId: (value: SetStateAction<string>) => void;\n}\n\nconst ThreadsContext = createContext<ThreadsContextValue | undefined>(undefined);\n\nexport interface ThreadsProviderProps {\n children: ReactNode;\n threadId?: string;\n}\n\nexport function ThreadsProvider({ children, threadId: explicitThreadId }: ThreadsProviderProps) {\n const [internalThreadId, setThreadId] = useState<string>(explicitThreadId ?? randomUUID());\n\n const threadId = internalThreadId;\n\n return (\n <ThreadsContext.Provider\n value={{\n threadId,\n setThreadId,\n }}\n >\n {children}\n </ThreadsContext.Provider>\n );\n}\n\nexport function useThreads() {\n const context = useContext(ThreadsContext);\n if (!context) {\n throw new Error(\"useThreads must be used within ThreadsProvider\");\n }\n return context;\n}\n\nexport { ThreadsContext };\n"],"mappings":";AAAA,SAAgB,eAAe,YAAY,gBAA2C;AACtF,SAAS,kBAAkB;AAoBvB;AAbJ,IAAM,iBAAiB,cAA+C,MAAS;AAOxE,SAAS,gBAAgB,EAAE,UAAU,UAAU,iBAAiB,GAAyB;AAC9F,QAAM,CAAC,kBAAkB,WAAW,IAAI,SAAiB,8CAAoB,WAAW,CAAC;AAEzF,QAAM,WAAW;AAEjB,SACE;AAAA,IAAC,eAAe;AAAA,IAAf;AAAA,MACC,OAAO;AAAA,QACL;AAAA,QACA;AAAA,MACF;AAAA,MAEC;AAAA;AAAA,EACH;AAEJ;AAEO,SAAS,aAAa;AAC3B,QAAM,UAAU,WAAW,cAAc;AACzC,MAAI,CAAC,SAAS;AACZ,UAAM,IAAI,MAAM,gDAAgD;AAAA,EAClE;AACA,SAAO;AACT;","names":[]}
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/hooks/use-copilot-chat_internal.ts"],"sourcesContent":["import { useRef, useEffect, useCallback, useMemo, useState, createElement } from \"react\";\nimport { useCopilotContext } from \"../context/copilot-context\";\nimport { SystemMessageFunction } from \"../types\";\nimport { useAsyncCallback } from \"../components/error-boundary/error-utils\";\nimport { Message } from \"@copilotkit/shared\";\nimport { gqlToAGUI, Message as DeprecatedGqlMessage } from \"@copilotkit/runtime-client-gql\";\nimport { useLangGraphInterruptRender } from \"./use-langgraph-interrupt-render\";\nimport {\n useAgent,\n useCopilotChatConfiguration,\n useCopilotKit,\n useRenderCustomMessages,\n useSuggestions,\n} from \"@copilotkitnext/react\";\nimport { Suggestion } from \"@copilotkitnext/core\";\nimport { useLazyToolRenderer } from \"./use-lazy-tool-renderer\";\nimport { AbstractAgent, AGUIConnectNotImplementedError } from \"@ag-ui/client\";\nimport {\n CoAgentStateRenderBridge,\n type CoAgentStateRenderBridgeProps,\n} from \"./use-coagent-state-render-bridge\";\n\n/**\n * The type of suggestions to use in the chat.\n *\n * `auto` - Suggestions are generated automatically.\n * `manual` - Suggestions are controlled programmatically.\n * `SuggestionItem[]` - Static suggestions array.\n */\nexport type ChatSuggestions = \"auto\" | \"manual\" | Omit<Suggestion, \"isLoading\">[];\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 * Whether to clear the suggestions after appending the message. Defaults to `true`.\n */\n clearSuggestions?: boolean;\n}\n\nexport interface OnStopGenerationArguments {\n /**\n * The name of the currently executing agent.\n */\n currentAgentName: string | undefined;\n\n /**\n * The messages in the chat.\n */\n messages: Message[];\n}\n\nexport type OnReloadMessagesArguments = OnStopGenerationArguments & {\n /**\n * The message on which \"regenerate\" was pressed\n */\n messageId: string;\n};\n\nexport type OnStopGeneration = (args: OnStopGenerationArguments) => void;\n\nexport type OnReloadMessages = (args: OnReloadMessagesArguments) => void;\n\nexport interface UseCopilotChatOptions {\n /**\n * A unique identifier for the chat. If not provided, a random one will be\n * generated. When provided, the `useChat` hook with the same `id` will\n * have shared states across components.\n */\n id?: string;\n\n /**\n * HTTP headers to be sent with the API request.\n */\n headers?: Record<string, string> | Headers;\n\n /**\n * Initial messages to populate the chat with.\n */\n initialMessages?: Message[];\n\n /**\n * A function to generate the system message. Defaults to `defaultSystemMessage`.\n */\n makeSystemMessage?: SystemMessageFunction;\n\n /**\n * Disables inclusion of CopilotKit’s default system message. When true, no system message is sent (this also suppresses any custom message from <code>makeSystemMessage</code>).\n */\n disableSystemMessage?: boolean;\n /**\n * Controls the behavior of suggestions in the chat interface.\n *\n * `auto` (default) - Suggestions are generated automatically:\n * - When the chat is first opened (empty state)\n * - After each message exchange completes\n * - Uses configuration from `useCopilotChatSuggestions` hooks\n *\n * `manual` - Suggestions are controlled programmatically:\n * - Use `setSuggestions()` to set custom suggestions\n * - Use `generateSuggestions()` to trigger AI generation\n * - Access via `useCopilotChat` hook\n *\n * `SuggestionItem[]` - Static suggestions array:\n * - Always shows the same suggestions\n * - No AI generation involved\n */\n suggestions?: ChatSuggestions;\n\n onInProgress?: (isLoading: boolean) => void;\n onSubmitMessage?: (messageContent: string) => Promise<void> | void;\n onStopGeneration?: OnStopGeneration;\n onReloadMessages?: OnReloadMessages;\n}\n\nexport interface MCPServerConfig {\n endpoint: string;\n apiKey?: string;\n}\n\n// Old suggestion item interface, for returning from useCopilotChatInternal\ninterface SuggestionItem {\n title: string;\n message: string;\n partial?: boolean;\n className?: string;\n}\n\nexport interface UseCopilotChatReturn {\n /**\n * @deprecated use `messages` instead, this is an old non ag-ui version of the messages\n * Array of messages currently visible in the chat interface\n *\n * This is the visible messages, not the raw messages from the runtime client.\n */\n visibleMessages: DeprecatedGqlMessage[];\n\n /**\n * The messages that are currently in the chat in AG-UI format.\n */\n messages: Message[];\n\n /** @deprecated use `sendMessage` in `useCopilotChatHeadless_c` instead. This will be removed in a future major version. */\n appendMessage: (message: DeprecatedGqlMessage, options?: AppendMessageOptions) => Promise<void>;\n\n /**\n * Send a new message to the chat\n *\n * ```tsx\n * await sendMessage({\n * id: \"123\",\n * role: \"user\",\n * content: \"Hello, process this request\",\n * });\n * ```\n */\n sendMessage: (message: Message, options?: AppendMessageOptions) => Promise<void>;\n\n /**\n * Replace all messages in the chat\n *\n * ```tsx\n * setMessages([\n * { id: \"123\", role: \"user\", content: \"Hello, process this request\" },\n * { id: \"456\", role: \"assistant\", content: \"Hello, I'm the assistant\" },\n * ]);\n * ```\n *\n * **Deprecated** non-ag-ui version:\n *\n * ```tsx\n * setMessages([\n * new TextMessage({\n * content: \"Hello, process this request\",\n * role: gqlRole.User,\n * }),\n * new TextMessage({\n * content: \"Hello, I'm the assistant\",\n * role: gqlRole.Assistant,\n * ]);\n * ```\n *\n */\n setMessages: (messages: Message[] | DeprecatedGqlMessage[]) => void;\n\n /**\n * Remove a specific message by ID\n *\n * ```tsx\n * deleteMessage(\"123\");\n * ```\n */\n deleteMessage: (messageId: string) => void;\n\n /**\n * Regenerate the response for a specific message\n *\n * ```tsx\n * reloadMessages(\"123\");\n * ```\n */\n reloadMessages: (messageId: string) => Promise<void>;\n\n /**\n * Stop the current message generation\n *\n * ```tsx\n * if (isLoading) {\n * stopGeneration();\n * }\n * ```\n */\n stopGeneration: () => void;\n\n /**\n * Clear all messages and reset chat state\n *\n * ```tsx\n * reset();\n * console.log(messages); // []\n * ```\n */\n reset: () => void;\n\n /**\n * Whether the chat is currently generating a response\n *\n * ```tsx\n * if (isLoading) {\n * console.log(\"Loading...\");\n * } else {\n * console.log(\"Not loading\");\n * }\n */\n isLoading: boolean;\n\n /**\n * Whether the chat agent is available to generate responses\n *\n * ```tsx\n * if (isAvailable) {\n * console.log(\"Loading...\");\n * } else {\n * console.log(\"Not loading\");\n * }\n */\n isAvailable: boolean;\n\n /** Manually trigger chat completion (advanced usage) */\n runChatCompletion: () => Promise<Message[]>;\n\n /** MCP (Model Context Protocol) server configurations */\n mcpServers: MCPServerConfig[];\n\n /** Update MCP server configurations */\n setMcpServers: (mcpServers: MCPServerConfig[]) => void;\n\n /**\n * Current suggestions array\n * Use this to read the current suggestions or in conjunction with setSuggestions for manual control\n */\n suggestions: Suggestion[];\n\n /**\n * Manually set suggestions\n * Useful for manual mode or custom suggestion workflows\n */\n setSuggestions: (suggestions: Omit<Suggestion, \"isLoading\">[]) => void;\n\n /**\n * Trigger AI-powered suggestion generation\n * Uses configurations from useCopilotChatSuggestions hooks\n * Respects global debouncing - only one generation can run at a time\n *\n * ```tsx\n * generateSuggestions();\n * console.log(suggestions); // [suggestion1, suggestion2, suggestion3]\n * ```\n */\n generateSuggestions: () => Promise<void>;\n\n /**\n * Clear all current suggestions\n * Also resets suggestion generation state\n */\n resetSuggestions: () => void;\n\n /** Whether suggestions are currently being generated */\n isLoadingSuggestions: boolean;\n\n /** Interrupt content for human-in-the-loop workflows */\n interrupt: string | React.ReactElement | null;\n\n agent?: ReturnType<typeof useAgent>[\"agent\"];\n\n threadId?: string;\n}\n\nexport function useCopilotChatInternal({\n suggestions,\n onInProgress,\n onSubmitMessage,\n onStopGeneration,\n onReloadMessages,\n}: UseCopilotChatOptions = {}): UseCopilotChatReturn {\n const { copilotkit } = useCopilotKit();\n const { threadId, agentSession } = useCopilotContext();\n const existingConfig = useCopilotChatConfiguration();\n const [agentAvailable, setAgentAvailable] = useState(false);\n\n // Apply priority: props > existing config > defaults\n const resolvedAgentId = existingConfig?.agentId ?? \"default\";\n const { agent } = useAgent({ agentId: resolvedAgentId });\n\n useEffect(() => {\n const connect = async (agent: AbstractAgent) => {\n setAgentAvailable(false);\n try {\n await copilotkit.connectAgent({ agent });\n setAgentAvailable(true);\n } catch (error) {\n if (error instanceof AGUIConnectNotImplementedError) {\n // connect not implemented, ignore\n } else {\n throw error;\n }\n }\n };\n if (agent && existingConfig?.threadId && agent.threadId !== existingConfig.threadId) {\n agent.threadId = existingConfig.threadId;\n connect(agent);\n }\n return () => {};\n }, [existingConfig?.threadId, agent, copilotkit, resolvedAgentId]);\n\n useEffect(() => {\n onInProgress?.(Boolean(agent?.isRunning));\n }, [agent?.isRunning, onInProgress]);\n\n const interrupt = useLangGraphInterruptRender(agent);\n\n const reset = () => {\n agent?.setMessages([]);\n agent?.setState(null);\n };\n\n const deleteMessage = useCallback(\n (messageId: string) => {\n const filteredMessages = (agent?.messages ?? []).filter(\n (message) => message.id !== messageId,\n );\n agent?.setMessages(filteredMessages);\n },\n [agent?.setMessages, agent?.messages],\n );\n\n const latestDelete = useUpdatedRef(deleteMessage);\n const latestDeleteFunc = useCallback(\n (messageId: string) => {\n return latestDelete.current(messageId);\n },\n [latestDelete],\n );\n\n const currentSuggestions = useSuggestions({ agentId: resolvedAgentId });\n\n const reload = useAsyncCallback(\n async (reloadMessageId: string): Promise<void> => {\n const messages = agent?.messages ?? [];\n // TODO: get isLoading\n const isLoading = false;\n if (isLoading || messages.length === 0) {\n return;\n }\n\n const reloadMessageIndex = messages.findIndex((msg) => msg.id === reloadMessageId);\n if (reloadMessageIndex === -1) {\n console.warn(`Message with id ${reloadMessageId} not found`);\n return;\n }\n\n const reloadMessageRole = messages[reloadMessageIndex].role;\n if (reloadMessageRole !== \"assistant\") {\n console.warn(`Regenerate cannot be performed on ${reloadMessageRole} role`);\n return;\n }\n let historyCutoff: Message[] = [messages[0]];\n\n if (messages.length > 2 && reloadMessageIndex !== 0) {\n // message to regenerate from is now first.\n // Work backwards to find the first the closest user message\n const lastUserMessageBeforeRegenerate = messages\n .slice(0, reloadMessageIndex)\n .reverse()\n .find((msg) => msg.role === \"user\");\n\n if (!lastUserMessageBeforeRegenerate) {\n historyCutoff = [messages[0]];\n } else {\n const indexOfLastUserMessageBeforeRegenerate = messages.findIndex(\n (msg) => msg.id === lastUserMessageBeforeRegenerate.id,\n );\n // Include the user message, remove everything after it\n historyCutoff = messages.slice(0, indexOfLastUserMessageBeforeRegenerate + 1);\n }\n } else if (messages.length > 2 && reloadMessageIndex === 0) {\n historyCutoff = [messages[0], messages[1]];\n }\n\n agent?.setMessages(historyCutoff);\n\n if (agent) {\n copilotkit.runAgent({ agent });\n }\n return;\n },\n [agent?.setMessages, copilotkit?.runAgent],\n );\n\n const latestSendMessageFunc = useAsyncCallback(\n async (message: Message, options?: AppendMessageOptions) => {\n if (!agent) return;\n const followUp = options?.followUp ?? true;\n if (options?.clearSuggestions) {\n copilotkit.clearSuggestions(resolvedAgentId);\n }\n agent?.addMessage(message);\n if (followUp) {\n try {\n await copilotkit.runAgent({ agent });\n } catch (error) {\n console.error(\"CopilotChat: runAgent failed\", error);\n }\n }\n if (onSubmitMessage) {\n const content =\n typeof message.content === \"string\"\n ? message.content\n : message.content && \"text\" in message.content\n ? message.content.text\n : message.content && \"filename\" in message.content\n ? message.content.filename\n : \"\";\n onSubmitMessage(content);\n }\n },\n [agent, copilotkit, resolvedAgentId, onSubmitMessage],\n );\n\n const latestAppendFunc = useAsyncCallback(\n async (message: DeprecatedGqlMessage, options?: AppendMessageOptions) => {\n return latestSendMessageFunc(gqlToAGUI([message])[0], options);\n },\n [latestSendMessageFunc],\n );\n\n const latestSetMessagesFunc = useCallback(\n (messages: Message[] | DeprecatedGqlMessage[]) => {\n if (messages.every((message) => message instanceof DeprecatedGqlMessage)) {\n return agent?.setMessages?.(gqlToAGUI(messages));\n }\n return agent?.setMessages?.(messages);\n },\n [agent?.setMessages, agent],\n );\n\n const latestReload = useUpdatedRef(reload);\n const latestReloadFunc = useAsyncCallback(\n async (messageId: string) => {\n onReloadMessages?.({\n messageId,\n currentAgentName: agent?.agentId,\n messages: agent?.messages ?? [],\n });\n return await latestReload.current(messageId);\n },\n [latestReload, agent, onReloadMessages],\n );\n\n const latestStopFunc = useCallback(() => {\n onStopGeneration?.({\n currentAgentName: agent?.agentId,\n messages: agent?.messages ?? [],\n });\n return agent?.abortRun?.();\n }, [onStopGeneration, agent]);\n\n const latestReset = useUpdatedRef(reset);\n const latestResetFunc = useCallback(() => {\n return latestReset.current();\n }, [latestReset]);\n\n const lazyToolRendered = useLazyToolRenderer();\n const renderCustomMessage = useRenderCustomMessages();\n const legacyCustomMessageRenderer = useLegacyCoagentRenderer({\n copilotkit,\n agent,\n agentId: resolvedAgentId,\n threadId: existingConfig?.threadId ?? threadId,\n });\n const allMessages = agent?.messages ?? [];\n const resolvedMessages = useMemo(() => {\n let processedMessages = allMessages.map((message) => {\n if (message.role !== \"assistant\") {\n return message;\n }\n\n const lazyRendered = lazyToolRendered(message, allMessages);\n if (lazyRendered) {\n const renderedGenUi = lazyRendered();\n if (renderedGenUi) {\n return { ...message, generativeUI: () => renderedGenUi };\n }\n }\n\n const bridgeRenderer =\n legacyCustomMessageRenderer || renderCustomMessage\n ? () => {\n const customRender = renderCustomMessage?.({\n message,\n position: \"before\",\n });\n if (customRender) {\n return customRender;\n }\n return legacyCustomMessageRenderer?.({ message, position: \"before\" });\n }\n : null;\n\n if (bridgeRenderer) {\n return { ...message, generativeUI: bridgeRenderer };\n }\n return message;\n });\n\n const hasAssistantMessages = processedMessages.some((msg) => msg.role === \"assistant\");\n\n // TODO: what is this?\n // if (legacyCustomMessageRenderer && !hasAssistantMessages) {\n // const placeholderId = `coagent-state-render-${resolvedAgentId}`;\n // const placeholderMessage: Message = {\n // id: placeholderId,\n // role: \"assistant\",\n // content: \"\",\n // name: \"coagent-state-render\",\n // };\n // processedMessages = [\n // ...processedMessages,\n // {\n // ...placeholderMessage,\n // generativeUI: () =>\n // legacyCustomMessageRenderer({\n // message: placeholderMessage,\n // position: \"before\",\n // }),\n // } as Message,\n // ];\n // }\n\n return processedMessages;\n }, [\n agent?.messages,\n lazyToolRendered,\n allMessages,\n renderCustomMessage,\n // legacyCustomMessageRenderer,\n resolvedAgentId,\n ]);\n\n const renderedSuggestions = useMemo(() => {\n if (Array.isArray(suggestions)) {\n return {\n suggestions: suggestions.map((s) => ({ ...s, isLoading: false })),\n isLoading: false,\n };\n }\n return currentSuggestions;\n }, [suggestions, currentSuggestions]);\n\n // @ts-ignore\n return {\n messages: resolvedMessages,\n sendMessage: latestSendMessageFunc,\n appendMessage: latestAppendFunc,\n setMessages: latestSetMessagesFunc,\n reloadMessages: latestReloadFunc,\n stopGeneration: latestStopFunc,\n reset: latestResetFunc,\n deleteMessage: latestDeleteFunc,\n isAvailable: agentAvailable,\n isLoading: Boolean(agent?.isRunning),\n // mcpServers,\n // setMcpServers,\n suggestions: renderedSuggestions.suggestions,\n setSuggestions: (suggestions: Omit<Suggestion, \"isLoading\">[]) =>\n copilotkit.addSuggestionsConfig({ suggestions }),\n generateSuggestions: async () => copilotkit.reloadSuggestions(resolvedAgentId),\n resetSuggestions: () => copilotkit.clearSuggestions(resolvedAgentId),\n isLoadingSuggestions: renderedSuggestions.isLoading,\n interrupt,\n agent,\n threadId,\n };\n}\n\n// store `value` in a ref and update\n// it whenever it changes.\nfunction useUpdatedRef<T>(value: T) {\n const ref = useRef(value);\n\n useEffect(() => {\n ref.current = value;\n }, [value]);\n\n return ref;\n}\n\ntype LegacyRenderParams = {\n message: Message;\n position: \"before\" | \"after\";\n};\n\ntype LegacyRenderer = ((args: LegacyRenderParams) => any) | null;\n\nfunction useLegacyCoagentRenderer({\n copilotkit,\n agent,\n agentId,\n threadId,\n}: {\n copilotkit: ReturnType<typeof useCopilotKit>[\"copilotkit\"];\n agent?: AbstractAgent;\n agentId: string;\n threadId?: string;\n}): LegacyRenderer {\n return useMemo(() => {\n if (!copilotkit || !agent) {\n return null;\n }\n\n return ({ message, position }: LegacyRenderParams) => {\n const effectiveThreadId = threadId ?? agent.threadId ?? \"default\";\n const existingRunId = copilotkit.getRunIdForMessage(agentId, effectiveThreadId, message.id);\n const runId = existingRunId || `pending:${message.id}`;\n const messageIndex = Math.max(\n agent.messages.findIndex((msg) => msg.id === message.id),\n 0,\n );\n\n const bridgeProps: CoAgentStateRenderBridgeProps = {\n message: message as any,\n position,\n runId,\n messageIndex,\n messageIndexInRun: 0,\n numberOfMessagesInRun: 1,\n agentId,\n stateSnapshot: (message as any).state,\n };\n\n return createElement(CoAgentStateRenderBridge, bridgeProps) as any;\n };\n }, [agent, agentId, copilotkit, threadId]);\n}\n\nexport function defaultSystemMessage(\n contextString: string,\n additionalInstructions?: string,\n): string {\n return (\n `\nPlease act as an efficient, competent, conscientious, and industrious professional assistant.\n\nHelp the user achieve their goals, and you do so in a way that is as efficient as possible, without unnecessary fluff, but also without sacrificing professionalism.\nAlways be polite and respectful, and prefer brevity over verbosity.\n\nThe user has provided you with the following context:\n\\`\\`\\`\n${contextString}\n\\`\\`\\`\n\nThey have also provided you with functions you can call to initiate actions on their behalf, or functions you can call to receive more information.\n\nPlease assist them as best you can.\n\nYou can ask them for clarifying questions if needed, but don't be annoying about it. If you can reasonably 'fill in the blanks' yourself, do so.\n\nIf you would like to call a function, call it without saying anything else.\nIn case of a function error:\n- If this error stems from incorrect function parameters or syntax, you may retry with corrected arguments.\n- If the error's source is unclear or seems unrelated to your input, do not attempt further retries.\n` + (additionalInstructions ? `\\n\\n${additionalInstructions}` : \"\")\n );\n}\n"],"mappings":"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","names":["agent","_a","_b","suggestions"]}
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/hooks/use-coagent-state-render-bridge.tsx"],"sourcesContent":["import { ReactCustomMessageRendererPosition, useAgent } from \"@copilotkitnext/react\";\nimport { useCallback, useEffect, useMemo, useState } from \"react\";\nimport type { AgentSubscriber } from \"@ag-ui/client\";\nimport { useCoAgentStateRenders } from \"../context\";\nimport { dataToUUID, parseJson } from \"@copilotkit/shared\";\n\nfunction getStateWithoutConstantKeys(state: any) {\n if (!state) return {};\n const { messages, tools, copilotkit, ...stateWithoutConstantKeys } = state;\n return stateWithoutConstantKeys;\n}\n\n// Function that compares states, without the constant keys\nfunction areStatesEquals(a: any, b: any) {\n if ((a && !b) || (!a && b)) return false;\n const { messages, tools, copilotkit, ...aWithoutConstantKeys } = a;\n const {\n messages: bMessages,\n tools: bTools,\n copilotkit: bCopilotkit,\n ...bWithoutConstantKeys\n } = b;\n\n return JSON.stringify(aWithoutConstantKeys) === JSON.stringify(bWithoutConstantKeys);\n}\n\n/**\n * Bridge hook that connects agent state renders to chat messages.\n *\n * ## Purpose\n * This hook finds matching state render configurations (registered via useCoAgentStateRender)\n * and returns UI to render in chat.\n * It ensures each state render appears bound to a specific message, preventing duplicates while\n * allowing re-binding when the underlying state changes significantly.\n *\n * ## Message-ID-Based Claiming System\n *\n * ### The Problem\n * Multiple bridge component instances render simultaneously (one per message). Without coordination,\n * they would all try to render the same state render, causing duplicates.\n *\n * ### The Solution: Message-ID Claims with State Comparison\n * Each state render is \"claimed\" by exactly one **message ID** (not runId):\n *\n * **Claim Structure**: `claimsRef.current[messageId] = { stateRenderId, runId, stateSnapshot, locked }`\n *\n * **Primary binding is by messageId because**:\n * - runId is not always available immediately (starts as \"pending\")\n * - messageId is the stable identifier throughout the message lifecycle\n * - Claims persist across component remounts via context ref\n *\n * ### Claiming Logic Flow\n *\n * 1. **Message already has a claim**:\n * - Check if the claim matches the current stateRenderId\n * - If yes → render (this message owns this render)\n * - Update runId if it was \"pending\" and now available\n *\n * 2. **State render claimed by another message**:\n * - Compare state snapshots (ignoring constant keys: messages, tools, copilotkit)\n * - If states are identical → block rendering (duplicate)\n * - **If states are different → allow claiming** (new data, new message)\n * - This handles cases where the same render type shows different states in different messages\n *\n * 3. **Unclaimed state render**:\n * - Only allow claiming if runId is \"pending\" (initial render)\n * - If runId is real but no claim exists → block (edge case protection)\n * - Create new claim: `claimsRef.current[messageId] = { stateRenderId, runId }`\n *\n * ### State Snapshot Locking\n *\n * Once a state snapshot is captured and locked for a message:\n * - The UI always renders with the locked snapshot (not live agent.state)\n * - Prevents UI from appearing \"wiped\" during state transitions\n * - Locked when: stateSnapshot prop is available (from message persistence)\n * - Unlocked state: can still update from live agent.state\n *\n * ### Synchronous Claiming (Ref-based)\n *\n * Claims are stored in a context-level ref (not React state):\n * - Multiple bridges render in the same tick\n * - State updates are async - would allow duplicates before update completes\n * - Ref provides immediate, synchronous claim checking\n * - Survives component remounts (stored in context, not component)\n *\n * ## Flow Example\n *\n * ```\n * Time 1: Message A renders, runId=undefined, state={progress: 50%}\n * → effectiveRunId = \"pending\"\n * → Claims: claimsRef[\"msgA\"] = { stateRenderId: \"tasks\", runId: \"pending\", stateSnapshot: {progress: 50%} }\n * → Renders UI with 50% progress\n *\n * Time 2: Message B renders, runId=undefined, same state\n * → Checks: \"tasks\" already claimed by msgA with same state\n * → Returns null (blocked - duplicate)\n *\n * Time 3: Real runId appears (e.g., \"run-123\")\n * → Updates claim: claimsRef[\"msgA\"].runId = \"run-123\"\n * → Message A continues rendering\n *\n * Time 4: Agent processes more, state={progress: 100%}\n * → Message A: locked to 50% (stateSnapshot locked)\n * → Message C renders with state={progress: 100%}\n * → Checks: \"tasks\" claimed by msgA but state is DIFFERENT (50% vs 100%)\n * → Allows new claim: claimsRef[\"msgC\"] = { stateRenderId: \"tasks\", runId: \"run-123\", stateSnapshot: {progress: 100%} }\n * → Both messages render independently with their own snapshots\n * ```\n */\nexport interface CoAgentStateRenderBridgeProps {\n message: any;\n position: ReactCustomMessageRendererPosition;\n runId: string;\n messageIndex: number;\n messageIndexInRun: number;\n numberOfMessagesInRun: number;\n agentId: string;\n stateSnapshot: any;\n}\n\nexport function useCoagentStateRenderBridge(agentId: string, props: CoAgentStateRenderBridgeProps) {\n const { stateSnapshot, messageIndexInRun, message } = props;\n const { coAgentStateRenders, claimsRef } = useCoAgentStateRenders();\n const { agent } = useAgent({ agentId });\n const [nodeName, setNodeName] = useState<string | undefined>(undefined);\n\n const runId = props.runId ?? message.runId;\n const effectiveRunId = runId || \"pending\";\n\n useEffect(() => {\n if (!agent) return;\n const subscriber: AgentSubscriber = {\n onStepStartedEvent: ({ event }) => {\n if (event.stepName !== nodeName) {\n setNodeName(event.stepName);\n }\n },\n onStepFinishedEvent: ({ event }) => {\n if (event.stepName === nodeName) {\n setNodeName(undefined);\n }\n },\n };\n\n const { unsubscribe } = agent.subscribe(subscriber);\n return () => {\n unsubscribe();\n };\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, [agentId, nodeName]);\n\n const getStateRender = useCallback(\n (messageId: string) => {\n return Object.entries(coAgentStateRenders).find(([stateRenderId, stateRender]) => {\n if (claimsRef.current[messageId]) {\n return stateRenderId === claimsRef.current[messageId].stateRenderId;\n }\n const matchingAgentName = stateRender.name === agentId;\n const matchesNodeContext = stateRender.nodeName ? stateRender.nodeName === nodeName : true;\n return matchingAgentName && matchesNodeContext;\n });\n },\n [coAgentStateRenders, nodeName, agentId],\n );\n\n // Message ID-based claim system - A state render can only be claimed by one message ID\n const handleRenderRequest = ({\n stateRenderId,\n messageId,\n runId,\n stateSnapshot: renderSnapshot,\n }: {\n stateRenderId: string;\n messageId: string;\n runId?: string;\n stateSnapshot?: any;\n }): boolean => {\n // Check if this message has already claimed this state render\n if (claimsRef.current[messageId]) {\n const canRender = claimsRef.current[messageId].stateRenderId === stateRenderId;\n\n // Update runId if it doesn't exist\n if (\n canRender &&\n runId &&\n (!claimsRef.current[messageId].runId || claimsRef.current[messageId].runId === \"pending\")\n ) {\n claimsRef.current[messageId].runId = runId;\n }\n\n return canRender;\n }\n\n // Do not allow render if any other message has claimed this state render\n const renderClaimedByOtherMessage = Object.values(claimsRef.current).find(\n (c) =>\n c.stateRenderId === stateRenderId &&\n dataToUUID(getStateWithoutConstantKeys(c.stateSnapshot)) ===\n dataToUUID(getStateWithoutConstantKeys(renderSnapshot)),\n );\n if (renderClaimedByOtherMessage) {\n // If:\n // - state render already claimed\n // - snapshot exists in the claiming object and is different from current,\n if (\n renderSnapshot &&\n renderClaimedByOtherMessage.stateSnapshot &&\n !areStatesEquals(renderClaimedByOtherMessage.stateSnapshot, renderSnapshot)\n ) {\n claimsRef.current[messageId] = { stateRenderId, runId };\n return true;\n }\n return false;\n }\n\n // No existing claim anywhere yet – allow this message to claim even if we already know the runId.\n if (!runId) {\n return false;\n }\n\n claimsRef.current[messageId] = { stateRenderId, runId };\n return true;\n };\n\n return useMemo(() => {\n if (messageIndexInRun !== 0) {\n return null;\n }\n\n const [stateRenderId, stateRender] = getStateRender(message.id) ?? [];\n\n if (!stateRender || !stateRenderId) {\n return null;\n }\n\n // Is there any state we can use?\n const snapshot = stateSnapshot ? parseJson(stateSnapshot, stateSnapshot) : agent?.state;\n\n // Synchronously check/claim - returns true if this message can render\n const canRender = handleRenderRequest({\n stateRenderId,\n messageId: message.id,\n runId: effectiveRunId,\n stateSnapshot: snapshot,\n });\n if (!canRender) {\n return null;\n }\n\n // If we found state, and given that now there's a claim for the current message, let's save it in the claim\n if (snapshot && !claimsRef.current[message.id].locked) {\n if (stateSnapshot) {\n claimsRef.current[message.id].stateSnapshot = snapshot;\n claimsRef.current[message.id].locked = true;\n } else {\n claimsRef.current[message.id].stateSnapshot = snapshot;\n }\n }\n\n if (stateRender.handler) {\n stateRender.handler({\n state: stateSnapshot ? parseJson(stateSnapshot, stateSnapshot) : (agent?.state ?? {}),\n nodeName: nodeName ?? \"\",\n });\n }\n\n if (stateRender.render) {\n const status = agent?.isRunning ? \"inProgress\" : \"complete\";\n\n if (typeof stateRender.render === \"string\") return stateRender.render;\n\n return stateRender.render({\n status,\n // Always use state from claim, to make sure the state does not seem \"wiped\" for a fraction of a second\n state: claimsRef.current[message.id].stateSnapshot ?? {},\n nodeName: nodeName ?? \"\",\n });\n }\n }, [\n getStateRender,\n stateSnapshot,\n agent?.state,\n agent?.isRunning,\n nodeName,\n effectiveRunId,\n message.id,\n messageIndexInRun,\n ]);\n}\n\nexport function CoAgentStateRenderBridge(props: CoAgentStateRenderBridgeProps) {\n return useCoagentStateRenderBridge(props.agentId, props);\n}\n"],"mappings":"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","names":["runId","_a"]}