@copilotkit/runtime-client-gql 1.9.3-next.4 → 1.10.0-next.0

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 (72) hide show
  1. package/CHANGELOG.md +35 -0
  2. package/dist/chunk-2R7M2FWR.mjs +17765 -0
  3. package/dist/chunk-2R7M2FWR.mjs.map +1 -0
  4. package/dist/chunk-CA4VMP2C.mjs +1 -0
  5. package/dist/chunk-CA4VMP2C.mjs.map +1 -0
  6. package/dist/chunk-DELDZXUX.mjs +31 -0
  7. package/dist/chunk-DELDZXUX.mjs.map +1 -0
  8. package/dist/chunk-MTD2RJDJ.mjs +187 -0
  9. package/dist/chunk-MTD2RJDJ.mjs.map +1 -0
  10. package/dist/{chunk-MCVCTWSF.mjs → chunk-UQACSQNW.mjs} +5 -5
  11. package/dist/{chunk-MCVCTWSF.mjs.map → chunk-UQACSQNW.mjs.map} +1 -1
  12. package/dist/chunk-YNQMTL2P.mjs +191 -0
  13. package/dist/chunk-YNQMTL2P.mjs.map +1 -0
  14. package/dist/client/CopilotRuntimeClient.js +1 -1
  15. package/dist/client/CopilotRuntimeClient.js.map +1 -1
  16. package/dist/client/CopilotRuntimeClient.mjs +3 -2
  17. package/dist/client/conversion.mjs +1 -0
  18. package/dist/client/index.d.ts +2 -2
  19. package/dist/client/index.js +1 -1
  20. package/dist/client/index.js.map +1 -1
  21. package/dist/client/index.mjs +5 -4
  22. package/dist/client/types.mjs +1 -0
  23. package/dist/graphql/@generated/fragment-masking.mjs +1 -0
  24. package/dist/graphql/@generated/gql.mjs +1 -0
  25. package/dist/graphql/@generated/graphql.mjs +1 -0
  26. package/dist/graphql/@generated/index.mjs +5 -4
  27. package/dist/graphql/definitions/mutations.mjs +1 -0
  28. package/dist/graphql/definitions/queries.mjs +1 -0
  29. package/dist/index.d.ts +3 -0
  30. package/dist/index.js +370 -1
  31. package/dist/index.js.map +1 -1
  32. package/dist/index.mjs +30 -4
  33. package/dist/magic-string.es-O42ACB6H.mjs +1373 -0
  34. package/dist/magic-string.es-O42ACB6H.mjs.map +1 -0
  35. package/dist/message-conversion/agui-to-gql.d.ts +13 -0
  36. package/dist/message-conversion/agui-to-gql.js +293 -0
  37. package/dist/message-conversion/agui-to-gql.js.map +1 -0
  38. package/dist/message-conversion/agui-to-gql.mjs +26 -0
  39. package/dist/message-conversion/agui-to-gql.mjs.map +1 -0
  40. package/dist/message-conversion/agui-to-gql.test.d.ts +2 -0
  41. package/dist/message-conversion/agui-to-gql.test.js +19958 -0
  42. package/dist/message-conversion/agui-to-gql.test.js.map +1 -0
  43. package/dist/message-conversion/agui-to-gql.test.mjs +565 -0
  44. package/dist/message-conversion/agui-to-gql.test.mjs.map +1 -0
  45. package/dist/message-conversion/gql-to-agui.d.ts +11 -0
  46. package/dist/message-conversion/gql-to-agui.js +227 -0
  47. package/dist/message-conversion/gql-to-agui.js.map +1 -0
  48. package/dist/message-conversion/gql-to-agui.mjs +22 -0
  49. package/dist/message-conversion/gql-to-agui.mjs.map +1 -0
  50. package/dist/message-conversion/gql-to-agui.test.d.ts +2 -0
  51. package/dist/message-conversion/gql-to-agui.test.js +20134 -0
  52. package/dist/message-conversion/gql-to-agui.test.js.map +1 -0
  53. package/dist/message-conversion/gql-to-agui.test.mjs +737 -0
  54. package/dist/message-conversion/gql-to-agui.test.mjs.map +1 -0
  55. package/dist/message-conversion/index.d.ts +6 -0
  56. package/dist/message-conversion/index.js +477 -0
  57. package/dist/message-conversion/index.js.map +1 -0
  58. package/dist/message-conversion/index.mjs +37 -0
  59. package/dist/message-conversion/index.mjs.map +1 -0
  60. package/dist/message-conversion/roundtrip-conversion.test.d.ts +2 -0
  61. package/dist/message-conversion/roundtrip-conversion.test.js +19768 -0
  62. package/dist/message-conversion/roundtrip-conversion.test.js.map +1 -0
  63. package/dist/message-conversion/roundtrip-conversion.test.mjs +219 -0
  64. package/dist/message-conversion/roundtrip-conversion.test.mjs.map +1 -0
  65. package/package.json +7 -5
  66. package/src/index.ts +1 -0
  67. package/src/message-conversion/agui-to-gql.test.ts +640 -0
  68. package/src/message-conversion/agui-to-gql.ts +255 -0
  69. package/src/message-conversion/gql-to-agui.test.ts +844 -0
  70. package/src/message-conversion/gql-to-agui.ts +237 -0
  71. package/src/message-conversion/index.ts +2 -0
  72. package/src/message-conversion/roundtrip-conversion.test.ts +212 -0
@@ -0,0 +1,237 @@
1
+ import * as gql from "../client";
2
+ import agui from "@copilotkit/shared";
3
+ import { MessageStatusCode } from "../graphql/@generated/graphql";
4
+
5
+ // Define valid image formats based on the supported formats in the codebase
6
+ const VALID_IMAGE_FORMATS = ["jpeg", "png", "webp", "gif"] as const;
7
+ type ValidImageFormat = (typeof VALID_IMAGE_FORMATS)[number];
8
+
9
+ // Validation function for image format
10
+ function validateImageFormat(format: string): format is ValidImageFormat {
11
+ return VALID_IMAGE_FORMATS.includes(format as ValidImageFormat);
12
+ }
13
+
14
+ /*
15
+ ----------------------------
16
+ GQL Message -> AGUI Message
17
+ ----------------------------
18
+ */
19
+ export function gqlToAGUI(
20
+ messages: gql.Message[] | gql.Message,
21
+ actions?: Record<string, any>,
22
+ coAgentStateRenders?: Record<string, any>,
23
+ ): agui.Message[] {
24
+ let aguiMessages: agui.Message[] = [];
25
+ messages = Array.isArray(messages) ? messages : [messages];
26
+
27
+ // Create a map of action execution ID to result for completed actions
28
+ const actionResults = new Map<string, string>();
29
+ for (const message of messages) {
30
+ if (message.isResultMessage()) {
31
+ actionResults.set(message.actionExecutionId, message.result);
32
+ }
33
+ }
34
+
35
+ for (const message of messages) {
36
+ if (message.isTextMessage()) {
37
+ aguiMessages.push(gqlTextMessageToAGUIMessage(message));
38
+ } else if (message.isResultMessage()) {
39
+ aguiMessages.push(gqlResultMessageToAGUIMessage(message));
40
+ } else if (message.isActionExecutionMessage()) {
41
+ aguiMessages.push(gqlActionExecutionMessageToAGUIMessage(message, actions, actionResults));
42
+ } else if (message.isAgentStateMessage()) {
43
+ aguiMessages.push(gqlAgentStateMessageToAGUIMessage(message, coAgentStateRenders));
44
+ } else if (message.isImageMessage()) {
45
+ aguiMessages.push(gqlImageMessageToAGUIMessage(message));
46
+ } else {
47
+ throw new Error("Unknown message type");
48
+ }
49
+ }
50
+
51
+ return aguiMessages;
52
+ }
53
+
54
+ function gqlActionExecutionMessageToAGUIMessage(
55
+ message: gql.ActionExecutionMessage,
56
+ actions?: Record<string, any>,
57
+ actionResults?: Map<string, string>,
58
+ ): agui.Message {
59
+ if (actions && Object.values(actions).some((action: any) => action.name === message.name)) {
60
+ const action = Object.values(actions).find((action: any) => action.name === message.name);
61
+
62
+ // Create render function wrapper that provides proper props
63
+ const createRenderWrapper = (originalRender: any) => {
64
+ if (!originalRender) return undefined;
65
+
66
+ return (props?: any) => {
67
+ // Determine the correct status based on the same logic as RenderActionExecutionMessage
68
+ const actionResult = actionResults?.get(message.id);
69
+ let status: "inProgress" | "executing" | "complete" = "inProgress";
70
+
71
+ if (actionResult !== undefined) {
72
+ status = "complete";
73
+ } else if (message.status?.code !== MessageStatusCode.Pending) {
74
+ status = "executing";
75
+ }
76
+
77
+ // Provide the full props structure that the render function expects
78
+ const renderProps = {
79
+ status: props?.status || status,
80
+ args: message.arguments || {},
81
+ result: props?.result || actionResult || undefined,
82
+ respond: props?.respond || (() => {}),
83
+ messageId: message.id,
84
+ ...props,
85
+ };
86
+
87
+ return originalRender(renderProps);
88
+ };
89
+ };
90
+
91
+ return {
92
+ id: message.id,
93
+ role: "assistant",
94
+ content: "",
95
+ toolCalls: [actionExecutionMessageToAGUIMessage(message)],
96
+ generativeUI: createRenderWrapper(action.render),
97
+ name: message.name,
98
+ } as agui.AIMessage;
99
+ }
100
+
101
+ return {
102
+ id: message.id,
103
+ role: "assistant",
104
+ toolCalls: [actionExecutionMessageToAGUIMessage(message)],
105
+ name: message.name,
106
+ };
107
+ }
108
+
109
+ function gqlAgentStateMessageToAGUIMessage(
110
+ message: gql.AgentStateMessage,
111
+ coAgentStateRenders?: Record<string, any>,
112
+ ): agui.Message {
113
+ if (
114
+ coAgentStateRenders &&
115
+ Object.values(coAgentStateRenders).some((render: any) => render.name === message.agentName)
116
+ ) {
117
+ const render = Object.values(coAgentStateRenders).find(
118
+ (render: any) => render.name === message.agentName,
119
+ );
120
+
121
+ // Create render function wrapper that provides proper props
122
+ const createRenderWrapper = (originalRender: any) => {
123
+ if (!originalRender) return undefined;
124
+
125
+ return (props?: any) => {
126
+ // Determine the correct status based on the same logic as RenderActionExecutionMessage
127
+ const state = message.state;
128
+
129
+ // Provide the full props structure that the render function expects
130
+ const renderProps = {
131
+ state: state,
132
+ };
133
+
134
+ return originalRender(renderProps);
135
+ };
136
+ };
137
+
138
+ return {
139
+ id: message.id,
140
+ role: "assistant",
141
+ generativeUI: createRenderWrapper(render.render),
142
+ agentName: message.agentName,
143
+ state: message.state,
144
+ };
145
+ }
146
+
147
+ return {
148
+ id: message.id,
149
+ role: "assistant",
150
+ agentName: message.agentName,
151
+ state: message.state,
152
+ };
153
+ }
154
+
155
+ function actionExecutionMessageToAGUIMessage(
156
+ actionExecutionMessage: gql.ActionExecutionMessage,
157
+ ): agui.ToolCall {
158
+ return {
159
+ id: actionExecutionMessage.id,
160
+ function: {
161
+ name: actionExecutionMessage.name,
162
+ arguments: JSON.stringify(actionExecutionMessage.arguments),
163
+ },
164
+ type: "function",
165
+ };
166
+ }
167
+
168
+ export function gqlTextMessageToAGUIMessage(message: gql.TextMessage): agui.Message {
169
+ switch (message.role) {
170
+ case gql.Role.Developer:
171
+ return {
172
+ id: message.id,
173
+ role: "developer",
174
+ content: message.content,
175
+ };
176
+ case gql.Role.System:
177
+ return {
178
+ id: message.id,
179
+ role: "system",
180
+ content: message.content,
181
+ };
182
+ case gql.Role.Assistant:
183
+ return {
184
+ id: message.id,
185
+ role: "assistant",
186
+ content: message.content,
187
+ };
188
+ case gql.Role.User:
189
+ return {
190
+ id: message.id,
191
+ role: "user",
192
+ content: message.content,
193
+ };
194
+ default:
195
+ throw new Error("Unknown message role");
196
+ }
197
+ }
198
+
199
+ export function gqlResultMessageToAGUIMessage(message: gql.ResultMessage): agui.Message {
200
+ return {
201
+ id: message.id,
202
+ role: "tool",
203
+ content: message.result,
204
+ toolCallId: message.actionExecutionId,
205
+ toolName: message.actionName,
206
+ };
207
+ }
208
+
209
+ export function gqlImageMessageToAGUIMessage(message: gql.ImageMessage): agui.Message {
210
+ // Validate image format
211
+ if (!validateImageFormat(message.format)) {
212
+ throw new Error(
213
+ `Invalid image format: ${message.format}. Supported formats are: ${VALID_IMAGE_FORMATS.join(", ")}`,
214
+ );
215
+ }
216
+
217
+ // Validate that bytes is a non-empty string
218
+ if (!message.bytes || typeof message.bytes !== "string" || message.bytes.trim() === "") {
219
+ throw new Error("Image bytes must be a non-empty string");
220
+ }
221
+
222
+ // Determine the role based on the message role
223
+ const role = message.role === gql.Role.Assistant ? "assistant" : "user";
224
+
225
+ // Create the image message with proper typing
226
+ const imageMessage: agui.Message = {
227
+ id: message.id,
228
+ role,
229
+ content: "",
230
+ image: {
231
+ format: message.format,
232
+ bytes: message.bytes,
233
+ },
234
+ };
235
+
236
+ return imageMessage;
237
+ }
@@ -0,0 +1,2 @@
1
+ export * from "./agui-to-gql";
2
+ export * from "./gql-to-agui";
@@ -0,0 +1,212 @@
1
+ import { describe, test, expect, vi } from "vitest";
2
+ import * as gql from "../client";
3
+ import agui from "@copilotkit/shared";
4
+ import { aguiToGQL } from "./agui-to-gql";
5
+ import { gqlToAGUI } from "./gql-to-agui";
6
+
7
+ // Helper to strip functions for deep equality
8
+ function stripFunctions(obj: any): any {
9
+ if (typeof obj === "function") return undefined;
10
+ if (Array.isArray(obj)) return obj.map(stripFunctions);
11
+ if (obj && typeof obj === "object") {
12
+ const out: any = {};
13
+ for (const k in obj) {
14
+ if (typeof obj[k] !== "function") {
15
+ out[k] = stripFunctions(obj[k]);
16
+ }
17
+ }
18
+ return out;
19
+ }
20
+ return obj;
21
+ }
22
+
23
+ describe("roundtrip message conversion", () => {
24
+ test("text message AGUI -> GQL -> AGUI", () => {
25
+ const aguiMsg: agui.Message = {
26
+ id: "user-1",
27
+ role: "user",
28
+ content: "Hello!",
29
+ };
30
+ const gqlMsgs = aguiToGQL(aguiMsg);
31
+ const aguiMsgs2 = gqlToAGUI(gqlMsgs);
32
+ expect(stripFunctions(aguiMsgs2[0])).toEqual(stripFunctions(aguiMsg));
33
+ });
34
+
35
+ test("text message GQL -> AGUI -> GQL", () => {
36
+ const gqlMsg = new gql.TextMessage({
37
+ id: "assistant-1",
38
+ content: "Hi!",
39
+ role: gql.Role.Assistant,
40
+ });
41
+ const aguiMsgs = gqlToAGUI(gqlMsg);
42
+ const gqlMsgs2 = aguiToGQL(aguiMsgs);
43
+ // Should be equivalent in content, id, and role
44
+ expect(gqlMsgs2[0].id).toBe(gqlMsg.id);
45
+ expect((gqlMsgs2[0] as any).content).toBe(gqlMsg.content);
46
+ expect((gqlMsgs2[0] as any).role).toBe(gqlMsg.role);
47
+ });
48
+
49
+ test("tool message AGUI -> GQL -> AGUI", () => {
50
+ const aguiMsg: agui.Message = {
51
+ id: "tool-1",
52
+ role: "tool",
53
+ content: "Tool result",
54
+ toolCallId: "tool-call-1",
55
+ toolName: "testAction",
56
+ };
57
+ const gqlMsgs = aguiToGQL(aguiMsg);
58
+ const aguiMsgs2 = gqlToAGUI(gqlMsgs);
59
+ expect(stripFunctions(aguiMsgs2[0])).toEqual(stripFunctions(aguiMsg));
60
+ });
61
+
62
+ test("tool message GQL -> AGUI -> GQL", () => {
63
+ const gqlMsg = new gql.ResultMessage({
64
+ id: "tool-1",
65
+ result: "Tool result",
66
+ actionExecutionId: "tool-call-1",
67
+ actionName: "testAction",
68
+ });
69
+ const aguiMsgs = gqlToAGUI(gqlMsg);
70
+ const gqlMsgs2 = aguiToGQL(aguiMsgs);
71
+ expect(gqlMsgs2[0].id).toBe(gqlMsg.id);
72
+ expect((gqlMsgs2[0] as any).result).toBe(gqlMsg.result);
73
+ expect((gqlMsgs2[0] as any).actionExecutionId).toBe(gqlMsg.actionExecutionId);
74
+ });
75
+
76
+ test("action execution AGUI -> GQL -> AGUI", () => {
77
+ const aguiMsg: agui.Message = {
78
+ id: "assistant-1",
79
+ role: "assistant",
80
+ content: "Running action",
81
+ toolCalls: [
82
+ {
83
+ id: "tool-call-1",
84
+ type: "function",
85
+ function: {
86
+ name: "doSomething",
87
+ arguments: JSON.stringify({ foo: "bar" }),
88
+ },
89
+ },
90
+ ],
91
+ };
92
+ const gqlMsgs = aguiToGQL(aguiMsg);
93
+ const aguiMsgs2 = gqlToAGUI(gqlMsgs);
94
+ // Should have an assistant message and an action execution message
95
+ expect(aguiMsgs2[0].role).toBe("assistant");
96
+ expect(aguiMsgs2[1].role).toBe("assistant");
97
+ // Only check toolCalls if present
98
+ if ("toolCalls" in aguiMsgs2[1]) {
99
+ expect((aguiMsgs2[1] as any).toolCalls[0].function.name).toBe("doSomething");
100
+ }
101
+ });
102
+
103
+ test("action execution GQL -> AGUI -> GQL", () => {
104
+ const actionExecMsg = new gql.ActionExecutionMessage({
105
+ id: "tool-call-1",
106
+ name: "doSomething",
107
+ arguments: { foo: "bar" },
108
+ parentMessageId: "assistant-1",
109
+ });
110
+ const aguiMsgs = gqlToAGUI([actionExecMsg]);
111
+ const gqlMsgs2 = aguiToGQL(aguiMsgs);
112
+ // The ActionExecutionMessage is at index 1, not index 0
113
+ expect(gqlMsgs2[1].id).toBe("tool-call-1");
114
+ // The name should be extracted from the toolCall function name
115
+ expect((gqlMsgs2[1] as any).name).toBe("doSomething");
116
+ expect((gqlMsgs2[1] as any).arguments).toEqual({ foo: "bar" });
117
+ });
118
+
119
+ test("agent state GQL -> AGUI -> GQL", () => {
120
+ const agentStateMsg = new gql.AgentStateMessage({
121
+ id: "agent-state-1",
122
+ agentName: "testAgent",
123
+ state: { status: "running" },
124
+ role: gql.Role.Assistant,
125
+ });
126
+ const aguiMsgs = gqlToAGUI([agentStateMsg]);
127
+ const gqlMsgs2 = aguiToGQL(aguiMsgs);
128
+ expect(gqlMsgs2[0].id).toBe("agent-state-1");
129
+ // The agentName should be preserved in the roundtrip
130
+ expect((gqlMsgs2[0] as any).agentName).toBe("testAgent");
131
+ });
132
+
133
+ test("action execution with render function roundtrip", () => {
134
+ const mockRender = vi.fn();
135
+ const aguiMsg: agui.Message = {
136
+ id: "assistant-1",
137
+ role: "assistant",
138
+ content: "Running action",
139
+ toolCalls: [
140
+ {
141
+ id: "tool-call-1",
142
+ type: "function",
143
+ function: {
144
+ name: "doSomething",
145
+ arguments: JSON.stringify({ foo: "bar" }),
146
+ },
147
+ },
148
+ ],
149
+ generativeUI: mockRender,
150
+ };
151
+ const actions: Record<string, any> = { doSomething: { name: "doSomething" } };
152
+ const gqlMsgs = aguiToGQL(aguiMsg, actions);
153
+ const aguiMsgs2 = gqlToAGUI(gqlMsgs, actions);
154
+ // The render function should be preserved in actions context
155
+ expect(typeof actions.doSomething.render).toBe("function");
156
+ // The roundtripped message should have the same tool call
157
+ if ("toolCalls" in aguiMsgs2[1]) {
158
+ expect((aguiMsgs2[1] as any).toolCalls[0].function.name).toBe("doSomething");
159
+ }
160
+ });
161
+
162
+ test("image message GQL -> AGUI -> GQL", () => {
163
+ const gqlMsg = new gql.ImageMessage({
164
+ id: "img-1",
165
+ format: "jpeg",
166
+ bytes: "somebase64string",
167
+ role: gql.Role.User,
168
+ });
169
+ const aguiMsgs = gqlToAGUI(gqlMsg);
170
+ const gqlMsgs2 = aguiToGQL(aguiMsgs);
171
+ expect(gqlMsgs2[0].id).toBe(gqlMsg.id);
172
+ expect((gqlMsgs2[0] as any).format).toBe(gqlMsg.format);
173
+ expect((gqlMsgs2[0] as any).bytes).toBe(gqlMsg.bytes);
174
+ expect((gqlMsgs2[0] as any).role).toBe(gqlMsg.role);
175
+ });
176
+
177
+ test("image message AGUI -> GQL -> AGUI (assistant and user)", () => {
178
+ // Assistant image message
179
+ const aguiAssistantImageMsg: agui.Message = {
180
+ id: "img-assistant-1",
181
+ role: "assistant",
182
+ image: {
183
+ format: "jpeg",
184
+ bytes: "assistantbase64data",
185
+ },
186
+ content: "", // required for type
187
+ };
188
+ const gqlAssistantMsgs = aguiToGQL(aguiAssistantImageMsg);
189
+ const aguiAssistantMsgs2 = gqlToAGUI(gqlAssistantMsgs);
190
+ expect(aguiAssistantMsgs2[0].id).toBe(aguiAssistantImageMsg.id);
191
+ expect(aguiAssistantMsgs2[0].role).toBe("assistant");
192
+ expect((aguiAssistantMsgs2[0] as any).image.format).toBe("jpeg");
193
+ expect((aguiAssistantMsgs2[0] as any).image.bytes).toBe("assistantbase64data");
194
+
195
+ // User image message
196
+ const aguiUserImageMsg: agui.Message = {
197
+ id: "img-user-1",
198
+ role: "user",
199
+ image: {
200
+ format: "png",
201
+ bytes: "userbase64data",
202
+ },
203
+ content: "", // required for type
204
+ };
205
+ const gqlUserMsgs = aguiToGQL(aguiUserImageMsg);
206
+ const aguiUserMsgs2 = gqlToAGUI(gqlUserMsgs);
207
+ expect(aguiUserMsgs2[0].id).toBe(aguiUserImageMsg.id);
208
+ expect(aguiUserMsgs2[0].role).toBe("user");
209
+ expect((aguiUserMsgs2[0] as any).image.format).toBe("png");
210
+ expect((aguiUserMsgs2[0] as any).image.bytes).toBe("userbase64data");
211
+ });
212
+ });