@copilotkit/runtime 1.55.3 → 1.56.1

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 (169) hide show
  1. package/dist/agent/converters/tanstack.cjs.map +1 -1
  2. package/dist/agent/converters/tanstack.d.cts +6 -19
  3. package/dist/agent/converters/tanstack.d.cts.map +1 -1
  4. package/dist/agent/converters/tanstack.d.mts +6 -19
  5. package/dist/agent/converters/tanstack.d.mts.map +1 -1
  6. package/dist/agent/converters/tanstack.mjs.map +1 -1
  7. package/dist/agent/index.cjs +16 -2
  8. package/dist/agent/index.cjs.map +1 -1
  9. package/dist/agent/index.d.cts +12 -1
  10. package/dist/agent/index.d.cts.map +1 -1
  11. package/dist/agent/index.d.mts +12 -1
  12. package/dist/agent/index.d.mts.map +1 -1
  13. package/dist/agent/index.mjs +16 -2
  14. package/dist/agent/index.mjs.map +1 -1
  15. package/dist/index.cjs +1 -1
  16. package/dist/index.d.cts +3 -2
  17. package/dist/index.d.mts +3 -2
  18. package/dist/index.mjs +1 -1
  19. package/dist/lib/index.cjs +1 -1
  20. package/dist/lib/index.d.cts +2 -1
  21. package/dist/lib/index.d.cts.map +1 -1
  22. package/dist/lib/index.d.mts +2 -1
  23. package/dist/lib/index.d.mts.map +1 -1
  24. package/dist/lib/index.mjs +1 -1
  25. package/dist/lib/integrations/node-http/index.cjs +4 -1
  26. package/dist/lib/integrations/node-http/index.cjs.map +1 -1
  27. package/dist/lib/integrations/node-http/index.d.cts.map +1 -1
  28. package/dist/lib/integrations/node-http/index.d.mts.map +1 -1
  29. package/dist/lib/integrations/node-http/index.mjs +4 -1
  30. package/dist/lib/integrations/node-http/index.mjs.map +1 -1
  31. package/dist/lib/integrations/shared.cjs +1 -1
  32. package/dist/lib/integrations/shared.d.cts +1 -1
  33. package/dist/lib/integrations/shared.d.mts +1 -1
  34. package/dist/lib/integrations/shared.mjs +1 -1
  35. package/dist/lib/runtime/copilot-runtime.cjs +25 -5
  36. package/dist/lib/runtime/copilot-runtime.cjs.map +1 -1
  37. package/dist/lib/runtime/copilot-runtime.d.cts +15 -3
  38. package/dist/lib/runtime/copilot-runtime.d.cts.map +1 -1
  39. package/dist/lib/runtime/copilot-runtime.d.mts +15 -3
  40. package/dist/lib/runtime/copilot-runtime.d.mts.map +1 -1
  41. package/dist/lib/runtime/copilot-runtime.mjs +25 -5
  42. package/dist/lib/runtime/copilot-runtime.mjs.map +1 -1
  43. package/dist/lib/runtime/mcp-tools-utils.cjs +21 -4
  44. package/dist/lib/runtime/mcp-tools-utils.cjs.map +1 -1
  45. package/dist/lib/runtime/mcp-tools-utils.d.cts.map +1 -1
  46. package/dist/lib/runtime/mcp-tools-utils.d.mts.map +1 -1
  47. package/dist/lib/runtime/mcp-tools-utils.mjs +21 -4
  48. package/dist/lib/runtime/mcp-tools-utils.mjs.map +1 -1
  49. package/dist/package.cjs +6 -5
  50. package/dist/package.mjs +6 -5
  51. package/dist/service-adapters/anthropic/anthropic-adapter.cjs +11 -3
  52. package/dist/service-adapters/anthropic/anthropic-adapter.cjs.map +1 -1
  53. package/dist/service-adapters/anthropic/anthropic-adapter.d.cts +6 -0
  54. package/dist/service-adapters/anthropic/anthropic-adapter.d.cts.map +1 -1
  55. package/dist/service-adapters/anthropic/anthropic-adapter.d.mts +6 -0
  56. package/dist/service-adapters/anthropic/anthropic-adapter.d.mts.map +1 -1
  57. package/dist/service-adapters/anthropic/anthropic-adapter.mjs +11 -3
  58. package/dist/service-adapters/anthropic/anthropic-adapter.mjs.map +1 -1
  59. package/dist/service-adapters/anthropic/utils.cjs +27 -1
  60. package/dist/service-adapters/anthropic/utils.cjs.map +1 -1
  61. package/dist/service-adapters/anthropic/utils.mjs +27 -1
  62. package/dist/service-adapters/anthropic/utils.mjs.map +1 -1
  63. package/dist/service-adapters/langchain/utils.cjs +1 -1
  64. package/dist/service-adapters/langchain/utils.cjs.map +1 -1
  65. package/dist/service-adapters/langchain/utils.mjs +1 -1
  66. package/dist/service-adapters/langchain/utils.mjs.map +1 -1
  67. package/dist/service-adapters/openai/openai-adapter.cjs +3 -2
  68. package/dist/service-adapters/openai/openai-adapter.cjs.map +1 -1
  69. package/dist/service-adapters/openai/openai-adapter.d.cts +6 -0
  70. package/dist/service-adapters/openai/openai-adapter.d.cts.map +1 -1
  71. package/dist/service-adapters/openai/openai-adapter.d.mts +6 -0
  72. package/dist/service-adapters/openai/openai-adapter.d.mts.map +1 -1
  73. package/dist/service-adapters/openai/openai-adapter.mjs +4 -3
  74. package/dist/service-adapters/openai/openai-adapter.mjs.map +1 -1
  75. package/dist/service-adapters/openai/openai-assistant-adapter.cjs +8 -9
  76. package/dist/service-adapters/openai/openai-assistant-adapter.cjs.map +1 -1
  77. package/dist/service-adapters/openai/openai-assistant-adapter.d.cts.map +1 -1
  78. package/dist/service-adapters/openai/openai-assistant-adapter.d.mts.map +1 -1
  79. package/dist/service-adapters/openai/openai-assistant-adapter.mjs +9 -10
  80. package/dist/service-adapters/openai/openai-assistant-adapter.mjs.map +1 -1
  81. package/dist/service-adapters/openai/utils.cjs +53 -0
  82. package/dist/service-adapters/openai/utils.cjs.map +1 -1
  83. package/dist/service-adapters/openai/utils.mjs +51 -1
  84. package/dist/service-adapters/openai/utils.mjs.map +1 -1
  85. package/dist/v2/index.cjs +1 -0
  86. package/dist/v2/index.d.cts +3 -3
  87. package/dist/v2/index.d.mts +3 -3
  88. package/dist/v2/index.mjs +2 -2
  89. package/dist/v2/runtime/core/middleware-sse-parser.cjs +5 -2
  90. package/dist/v2/runtime/core/middleware-sse-parser.cjs.map +1 -1
  91. package/dist/v2/runtime/core/middleware-sse-parser.mjs +5 -2
  92. package/dist/v2/runtime/core/middleware-sse-parser.mjs.map +1 -1
  93. package/dist/v2/runtime/core/runtime.cjs +25 -0
  94. package/dist/v2/runtime/core/runtime.cjs.map +1 -1
  95. package/dist/v2/runtime/core/runtime.d.cts +53 -4
  96. package/dist/v2/runtime/core/runtime.d.cts.map +1 -1
  97. package/dist/v2/runtime/core/runtime.d.mts +53 -4
  98. package/dist/v2/runtime/core/runtime.d.mts.map +1 -1
  99. package/dist/v2/runtime/core/runtime.mjs +26 -2
  100. package/dist/v2/runtime/core/runtime.mjs.map +1 -1
  101. package/dist/v2/runtime/handlers/get-runtime-info.cjs +18 -10
  102. package/dist/v2/runtime/handlers/get-runtime-info.cjs.map +1 -1
  103. package/dist/v2/runtime/handlers/get-runtime-info.mjs +19 -11
  104. package/dist/v2/runtime/handlers/get-runtime-info.mjs.map +1 -1
  105. package/dist/v2/runtime/handlers/handle-connect.cjs +1 -1
  106. package/dist/v2/runtime/handlers/handle-connect.cjs.map +1 -1
  107. package/dist/v2/runtime/handlers/handle-connect.mjs +1 -1
  108. package/dist/v2/runtime/handlers/handle-connect.mjs.map +1 -1
  109. package/dist/v2/runtime/handlers/handle-run.cjs +8 -2
  110. package/dist/v2/runtime/handlers/handle-run.cjs.map +1 -1
  111. package/dist/v2/runtime/handlers/handle-run.mjs +8 -2
  112. package/dist/v2/runtime/handlers/handle-run.mjs.map +1 -1
  113. package/dist/v2/runtime/handlers/handle-stop.cjs +2 -1
  114. package/dist/v2/runtime/handlers/handle-stop.cjs.map +1 -1
  115. package/dist/v2/runtime/handlers/handle-stop.mjs +2 -1
  116. package/dist/v2/runtime/handlers/handle-stop.mjs.map +1 -1
  117. package/dist/v2/runtime/handlers/intelligence/thread-names.cjs +1 -1
  118. package/dist/v2/runtime/handlers/intelligence/thread-names.cjs.map +1 -1
  119. package/dist/v2/runtime/handlers/intelligence/thread-names.mjs +1 -1
  120. package/dist/v2/runtime/handlers/intelligence/thread-names.mjs.map +1 -1
  121. package/dist/v2/runtime/handlers/shared/agent-utils.cjs +3 -2
  122. package/dist/v2/runtime/handlers/shared/agent-utils.cjs.map +1 -1
  123. package/dist/v2/runtime/handlers/shared/agent-utils.mjs +3 -2
  124. package/dist/v2/runtime/handlers/shared/agent-utils.mjs.map +1 -1
  125. package/dist/v2/runtime/handlers/shared/sse-response.cjs +40 -1
  126. package/dist/v2/runtime/handlers/shared/sse-response.cjs.map +1 -1
  127. package/dist/v2/runtime/handlers/shared/sse-response.mjs +40 -1
  128. package/dist/v2/runtime/handlers/shared/sse-response.mjs.map +1 -1
  129. package/dist/v2/runtime/handlers/sse/run.cjs +3 -1
  130. package/dist/v2/runtime/handlers/sse/run.cjs.map +1 -1
  131. package/dist/v2/runtime/handlers/sse/run.mjs +3 -1
  132. package/dist/v2/runtime/handlers/sse/run.mjs.map +1 -1
  133. package/dist/v2/runtime/index.d.cts +1 -1
  134. package/dist/v2/runtime/index.d.mts +1 -1
  135. package/package.json +7 -6
  136. package/src/agent/__tests__/capabilities.test.ts +81 -0
  137. package/src/agent/__tests__/provider-id-collision.test.ts +195 -0
  138. package/src/agent/converters/tanstack.ts +15 -7
  139. package/src/agent/index.ts +52 -11
  140. package/src/lib/integrations/node-http/__tests__/request-duck-type.test.ts +66 -0
  141. package/src/lib/integrations/node-http/index.ts +15 -1
  142. package/src/lib/runtime/__tests__/mcp-tools-utils.test.ts +30 -1
  143. package/src/lib/runtime/__tests__/on-after-request.test.ts +122 -0
  144. package/src/lib/runtime/__tests__/v1-agent-factory.test.ts +109 -0
  145. package/src/lib/runtime/copilot-runtime.ts +54 -5
  146. package/src/lib/runtime/mcp-tools-utils.ts +41 -6
  147. package/src/service-adapters/anthropic/anthropic-adapter.ts +22 -2
  148. package/src/service-adapters/anthropic/utils.ts +60 -1
  149. package/src/service-adapters/langchain/utils.ts +1 -1
  150. package/src/service-adapters/openai/__tests__/openai-v5-compat.test.ts +177 -0
  151. package/src/service-adapters/openai/openai-adapter.ts +17 -2
  152. package/src/service-adapters/openai/openai-assistant-adapter.ts +7 -9
  153. package/src/service-adapters/openai/utils.ts +100 -0
  154. package/src/v2/runtime/__tests__/agents-factory.test.ts +136 -0
  155. package/src/v2/runtime/__tests__/debug-sse-response.test.ts +302 -0
  156. package/src/v2/runtime/__tests__/get-runtime-info.test.ts +134 -1
  157. package/src/v2/runtime/__tests__/middleware-sse-parser.test.ts +50 -0
  158. package/src/v2/runtime/core/middleware-sse-parser.ts +12 -2
  159. package/src/v2/runtime/core/runtime.ts +90 -2
  160. package/src/v2/runtime/handlers/get-runtime-info.ts +33 -8
  161. package/src/v2/runtime/handlers/handle-connect.ts +1 -1
  162. package/src/v2/runtime/handlers/handle-run.ts +16 -2
  163. package/src/v2/runtime/handlers/handle-stop.ts +2 -1
  164. package/src/v2/runtime/handlers/intelligence/thread-names.ts +1 -1
  165. package/src/v2/runtime/handlers/shared/agent-utils.ts +3 -2
  166. package/src/v2/runtime/handlers/shared/sse-response.ts +69 -0
  167. package/src/v2/runtime/handlers/sse/run.ts +9 -0
  168. package/tests/service-adapters/anthropic/anthropic-adapter.test.ts +268 -0
  169. package/tests/service-adapters/anthropic/utils-token-trimming.test.ts +301 -0
@@ -0,0 +1,301 @@
1
+ import { describe, it, expect } from "vitest";
2
+ import { limitMessagesToTokenCount } from "../../../src/service-adapters/anthropic/utils";
3
+
4
+ // Helper to build messages for testing. The token counter is length/3,
5
+ // so we can control token counts via string length.
6
+
7
+ function textUser(text: string) {
8
+ return { role: "user", content: [{ type: "text", text }] };
9
+ }
10
+
11
+ function textAssistant(text: string) {
12
+ return { role: "assistant", content: [{ type: "text", text }] };
13
+ }
14
+
15
+ function toolUseAssistant(id: string, name = "my_tool", input = {}) {
16
+ return {
17
+ role: "assistant",
18
+ content: [{ type: "tool_use", id, name, input }],
19
+ };
20
+ }
21
+
22
+ function toolResultUser(toolUseId: string, content = "result") {
23
+ return {
24
+ role: "user",
25
+ content: [{ type: "tool_result", tool_use_id: toolUseId, content }],
26
+ };
27
+ }
28
+
29
+ function mixedAssistant(blocks: any[]) {
30
+ return { role: "assistant", content: blocks };
31
+ }
32
+
33
+ function mixedUser(blocks: any[]) {
34
+ return { role: "user", content: blocks };
35
+ }
36
+
37
+ describe("limitMessagesToTokenCount - orphan handling", () => {
38
+ // Use a high token limit so trimming doesn't kick in for these tests
39
+ const HIGH_LIMIT = 999999;
40
+
41
+ it("preserves matched tool_use / tool_result pairs", () => {
42
+ const messages = [
43
+ textUser("hello"),
44
+ toolUseAssistant("t1", "tool_a"),
45
+ toolResultUser("t1", "done"),
46
+ textAssistant("ok"),
47
+ ];
48
+
49
+ const result = limitMessagesToTokenCount(
50
+ messages,
51
+ [],
52
+ "claude-3",
53
+ HIGH_LIMIT,
54
+ );
55
+
56
+ // All four messages should survive
57
+ expect(result).toHaveLength(4);
58
+ // The tool_use and tool_result should still be present
59
+ const toolUse = result.find(
60
+ (m: any) =>
61
+ m.role === "assistant" &&
62
+ Array.isArray(m.content) &&
63
+ m.content.some((b: any) => b.type === "tool_use"),
64
+ );
65
+ const toolResult = result.find(
66
+ (m: any) =>
67
+ m.role === "user" &&
68
+ Array.isArray(m.content) &&
69
+ m.content.some((b: any) => b.type === "tool_result"),
70
+ );
71
+ expect(toolUse).toBeDefined();
72
+ expect(toolResult).toBeDefined();
73
+ });
74
+
75
+ it("removes orphaned tool_result when tool_use was trimmed", () => {
76
+ // Simulate: tool_use message was removed by token trimming, leaving
77
+ // a tool_result without a matching tool_use.
78
+ const messages = [
79
+ textUser("hello"),
80
+ // no toolUseAssistant for "t1"
81
+ toolResultUser("t1", "orphaned result"),
82
+ textAssistant("ok"),
83
+ ];
84
+
85
+ const result = limitMessagesToTokenCount(
86
+ messages,
87
+ [],
88
+ "claude-3",
89
+ HIGH_LIMIT,
90
+ );
91
+
92
+ // The orphaned tool_result message should be gone
93
+ const hasToolResult = result.some(
94
+ (m: any) =>
95
+ m.role === "user" &&
96
+ Array.isArray(m.content) &&
97
+ m.content.some((b: any) => b.type === "tool_result"),
98
+ );
99
+ expect(hasToolResult).toBe(false);
100
+ expect(result).toHaveLength(2); // textUser + textAssistant
101
+ });
102
+
103
+ it("removes orphaned tool_use when tool_result was trimmed", () => {
104
+ // Simulate: tool_result message was removed by token trimming, leaving
105
+ // a tool_use without a matching tool_result.
106
+ const messages = [
107
+ textUser("hello"),
108
+ toolUseAssistant("t1", "tool_a"),
109
+ // no toolResultUser for "t1"
110
+ textAssistant("ok"),
111
+ ];
112
+
113
+ const result = limitMessagesToTokenCount(
114
+ messages,
115
+ [],
116
+ "claude-3",
117
+ HIGH_LIMIT,
118
+ );
119
+
120
+ // The orphaned tool_use message should be gone
121
+ const hasToolUse = result.some(
122
+ (m: any) =>
123
+ m.role === "assistant" &&
124
+ Array.isArray(m.content) &&
125
+ m.content.some((b: any) => b.type === "tool_use"),
126
+ );
127
+ expect(hasToolUse).toBe(false);
128
+ expect(result).toHaveLength(2); // textUser + textAssistant
129
+ });
130
+
131
+ it("retains non-orphaned blocks in mixed-content messages", () => {
132
+ // Assistant message has both a text block and an orphaned tool_use
133
+ const messages = [
134
+ textUser("hello"),
135
+ mixedAssistant([
136
+ { type: "text", text: "thinking..." },
137
+ { type: "tool_use", id: "t1", name: "tool_a", input: {} },
138
+ ]),
139
+ // no tool_result for t1
140
+ textAssistant("done"),
141
+ ];
142
+
143
+ const result = limitMessagesToTokenCount(
144
+ messages,
145
+ [],
146
+ "claude-3",
147
+ HIGH_LIMIT,
148
+ );
149
+
150
+ // The assistant message should survive with only the text block
151
+ const assistantMixed = result.find(
152
+ (m: any) =>
153
+ m.role === "assistant" &&
154
+ Array.isArray(m.content) &&
155
+ m.content.some(
156
+ (b: any) => b.type === "text" && b.text === "thinking...",
157
+ ),
158
+ );
159
+ expect(assistantMixed).toBeDefined();
160
+ expect(assistantMixed.content).toHaveLength(1);
161
+ expect(assistantMixed.content[0].type).toBe("text");
162
+ });
163
+
164
+ it("retains non-orphaned blocks in mixed user messages", () => {
165
+ // User message has both a text block and an orphaned tool_result
166
+ const messages = [
167
+ textUser("hello"),
168
+ mixedUser([
169
+ { type: "text", text: "here is context" },
170
+ { type: "tool_result", tool_use_id: "t_missing", content: "orphan" },
171
+ ]),
172
+ textAssistant("ok"),
173
+ ];
174
+
175
+ const result = limitMessagesToTokenCount(
176
+ messages,
177
+ [],
178
+ "claude-3",
179
+ HIGH_LIMIT,
180
+ );
181
+
182
+ const userMixed = result.find(
183
+ (m: any) =>
184
+ m.role === "user" &&
185
+ Array.isArray(m.content) &&
186
+ m.content.some(
187
+ (b: any) => b.type === "text" && b.text === "here is context",
188
+ ),
189
+ );
190
+ expect(userMixed).toBeDefined();
191
+ expect(userMixed.content).toHaveLength(1);
192
+ expect(userMixed.content[0].type).toBe("text");
193
+ });
194
+
195
+ it("drops message entirely when all blocks are orphaned", () => {
196
+ const messages = [
197
+ textUser("hello"),
198
+ mixedUser([
199
+ { type: "tool_result", tool_use_id: "t_a", content: "orphan a" },
200
+ { type: "tool_result", tool_use_id: "t_b", content: "orphan b" },
201
+ ]),
202
+ textAssistant("ok"),
203
+ ];
204
+
205
+ const result = limitMessagesToTokenCount(
206
+ messages,
207
+ [],
208
+ "claude-3",
209
+ HIGH_LIMIT,
210
+ );
211
+
212
+ expect(result).toHaveLength(2);
213
+ expect(result[0].role).toBe("user");
214
+ expect(result[1].role).toBe("assistant");
215
+ });
216
+
217
+ it("drops assistant message entirely when all tool_use blocks are orphaned", () => {
218
+ const messages = [
219
+ textUser("hello"),
220
+ mixedAssistant([
221
+ { type: "tool_use", id: "t_x", name: "tool_x", input: {} },
222
+ { type: "tool_use", id: "t_y", name: "tool_y", input: {} },
223
+ ]),
224
+ // no tool_results for either
225
+ textAssistant("done"),
226
+ ];
227
+
228
+ const result = limitMessagesToTokenCount(
229
+ messages,
230
+ [],
231
+ "claude-3",
232
+ HIGH_LIMIT,
233
+ );
234
+
235
+ expect(result).toHaveLength(2);
236
+ });
237
+
238
+ it("does not mutate the original messages array or message objects", () => {
239
+ const originalContent = [
240
+ { type: "text", text: "context" },
241
+ { type: "tool_result", tool_use_id: "t_orphan", content: "orphan" },
242
+ ];
243
+ const userMsg = { role: "user", content: [...originalContent] };
244
+ const messages = [textUser("hello"), userMsg, textAssistant("ok")];
245
+
246
+ const result = limitMessagesToTokenCount(
247
+ messages,
248
+ [],
249
+ "claude-3",
250
+ HIGH_LIMIT,
251
+ );
252
+
253
+ // Original message should still have both blocks
254
+ expect(userMsg.content).toHaveLength(2);
255
+ expect(userMsg.content[1].type).toBe("tool_result");
256
+
257
+ // Original messages array should still have 3 entries
258
+ expect(messages).toHaveLength(3);
259
+
260
+ // Result should have the filtered version
261
+ const filtered = result.find(
262
+ (m: any) =>
263
+ m.role === "user" &&
264
+ Array.isArray(m.content) &&
265
+ m.content.some((b: any) => b.text === "context"),
266
+ );
267
+ expect(filtered).toBeDefined();
268
+ expect(filtered.content).toHaveLength(1);
269
+ });
270
+
271
+ it("handles token trimming that creates orphans via cutoff", () => {
272
+ // Build messages where token trimming will cut off early messages,
273
+ // leaving orphaned tool_result for a tool_use that got trimmed.
274
+ // Each char ~0.33 tokens, so 300 chars ~ 100 tokens
275
+ const longText = "x".repeat(300);
276
+
277
+ const messages = [
278
+ toolUseAssistant("t_old"),
279
+ toolResultUser("t_old", "old result"),
280
+ textUser(longText),
281
+ textAssistant(longText),
282
+ toolUseAssistant("t_new"),
283
+ toolResultUser("t_new", "new result"),
284
+ ];
285
+
286
+ // Set a limit that keeps only the last few messages, trimming t_old's tool_use
287
+ const result = limitMessagesToTokenCount(messages, [], "claude-3", 300);
288
+
289
+ // t_old's tool_use should have been trimmed by the token limit,
290
+ // and then t_old's tool_result should be cleaned up as orphaned
291
+ const hasOldResult = result.some(
292
+ (m: any) =>
293
+ m.role === "user" &&
294
+ Array.isArray(m.content) &&
295
+ m.content.some(
296
+ (b: any) => b.type === "tool_result" && b.tool_use_id === "t_old",
297
+ ),
298
+ );
299
+ expect(hasOldResult).toBe(false);
300
+ });
301
+ });