@wingman-ai/gateway 0.2.2 → 0.2.4

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 (160) hide show
  1. package/.wingman/agents/README.md +7 -1
  2. package/.wingman/agents/coding/agent.md +299 -201
  3. package/.wingman/agents/coding-v2/agent.md +127 -0
  4. package/.wingman/agents/coding-v2/implementor.md +89 -0
  5. package/.wingman/agents/main/agent.md +4 -0
  6. package/README.md +1 -0
  7. package/dist/agent/config/agentConfig.cjs +31 -17
  8. package/dist/agent/config/agentConfig.d.ts +23 -1
  9. package/dist/agent/config/agentConfig.js +30 -19
  10. package/dist/agent/config/agentLoader.cjs +26 -8
  11. package/dist/agent/config/agentLoader.d.ts +4 -2
  12. package/dist/agent/config/agentLoader.js +26 -8
  13. package/dist/agent/config/modelFactory.cjs +95 -25
  14. package/dist/agent/config/modelFactory.d.ts +13 -1
  15. package/dist/agent/config/modelFactory.js +95 -25
  16. package/dist/agent/config/toolRegistry.cjs +19 -6
  17. package/dist/agent/config/toolRegistry.d.ts +5 -2
  18. package/dist/agent/config/toolRegistry.js +19 -6
  19. package/dist/agent/middleware/hooks/types.cjs +13 -13
  20. package/dist/agent/middleware/hooks/types.d.ts +1 -1
  21. package/dist/agent/middleware/hooks/types.js +14 -14
  22. package/dist/agent/tests/agentConfig.test.cjs +22 -2
  23. package/dist/agent/tests/agentConfig.test.js +22 -2
  24. package/dist/agent/tests/agentLoader.test.cjs +38 -1
  25. package/dist/agent/tests/agentLoader.test.js +38 -1
  26. package/dist/agent/tests/backgroundTerminal.test.cjs +70 -0
  27. package/dist/agent/tests/backgroundTerminal.test.d.ts +1 -0
  28. package/dist/agent/tests/backgroundTerminal.test.js +64 -0
  29. package/dist/agent/tests/commandExecuteTool.test.cjs +29 -0
  30. package/dist/agent/tests/commandExecuteTool.test.d.ts +1 -0
  31. package/dist/agent/tests/commandExecuteTool.test.js +23 -0
  32. package/dist/agent/tests/modelFactory.test.cjs +47 -5
  33. package/dist/agent/tests/modelFactory.test.js +47 -5
  34. package/dist/agent/tests/terminalSessionManager.test.cjs +121 -0
  35. package/dist/agent/tests/terminalSessionManager.test.d.ts +1 -0
  36. package/dist/agent/tests/terminalSessionManager.test.js +115 -0
  37. package/dist/agent/tests/toolRegistry.test.cjs +14 -2
  38. package/dist/agent/tests/toolRegistry.test.js +14 -2
  39. package/dist/agent/tools/background_terminal.cjs +128 -0
  40. package/dist/agent/tools/background_terminal.d.ts +41 -0
  41. package/dist/agent/tools/background_terminal.js +94 -0
  42. package/dist/agent/tools/code_search.cjs +6 -6
  43. package/dist/agent/tools/code_search.d.ts +1 -1
  44. package/dist/agent/tools/code_search.js +7 -7
  45. package/dist/agent/tools/command_execute.cjs +22 -7
  46. package/dist/agent/tools/command_execute.d.ts +3 -2
  47. package/dist/agent/tools/command_execute.js +23 -8
  48. package/dist/agent/tools/git_status.cjs +3 -3
  49. package/dist/agent/tools/git_status.d.ts +1 -1
  50. package/dist/agent/tools/git_status.js +4 -4
  51. package/dist/agent/tools/internet_search.cjs +6 -6
  52. package/dist/agent/tools/internet_search.d.ts +1 -1
  53. package/dist/agent/tools/internet_search.js +7 -7
  54. package/dist/agent/tools/terminal_session_manager.cjs +321 -0
  55. package/dist/agent/tools/terminal_session_manager.d.ts +77 -0
  56. package/dist/agent/tools/terminal_session_manager.js +284 -0
  57. package/dist/agent/tools/think.cjs +4 -4
  58. package/dist/agent/tools/think.d.ts +1 -1
  59. package/dist/agent/tools/think.js +5 -5
  60. package/dist/agent/tools/ui_registry.cjs +13 -13
  61. package/dist/agent/tools/ui_registry.d.ts +4 -4
  62. package/dist/agent/tools/ui_registry.js +14 -14
  63. package/dist/agent/tools/web_crawler.cjs +4 -4
  64. package/dist/agent/tools/web_crawler.d.ts +1 -1
  65. package/dist/agent/tools/web_crawler.js +5 -5
  66. package/dist/agent/utils.cjs +2 -1
  67. package/dist/agent/utils.js +2 -1
  68. package/dist/cli/commands/init.cjs +7 -6
  69. package/dist/cli/commands/init.js +7 -6
  70. package/dist/cli/commands/provider.cjs +17 -3
  71. package/dist/cli/commands/provider.js +17 -3
  72. package/dist/cli/config/loader.cjs +27 -0
  73. package/dist/cli/config/loader.js +27 -0
  74. package/dist/cli/config/schema.cjs +146 -68
  75. package/dist/cli/config/schema.d.ts +89 -1
  76. package/dist/cli/config/schema.js +134 -68
  77. package/dist/cli/core/agentInvoker.cjs +344 -17
  78. package/dist/cli/core/agentInvoker.d.ts +63 -3
  79. package/dist/cli/core/agentInvoker.js +303 -12
  80. package/dist/cli/core/sessionManager.cjs +32 -5
  81. package/dist/cli/core/sessionManager.js +32 -5
  82. package/dist/cli/core/streamParser.cjs +15 -0
  83. package/dist/cli/core/streamParser.js +15 -0
  84. package/dist/cli/index.cjs +6 -5
  85. package/dist/cli/index.js +6 -5
  86. package/dist/cli/types.d.ts +32 -0
  87. package/dist/cli/ui/toolDisplayHelpers.cjs +2 -0
  88. package/dist/cli/ui/toolDisplayHelpers.js +2 -0
  89. package/dist/gateway/hooks/registry.cjs +2 -1
  90. package/dist/gateway/hooks/registry.d.ts +1 -1
  91. package/dist/gateway/hooks/registry.js +2 -1
  92. package/dist/gateway/hooks/types.cjs +11 -11
  93. package/dist/gateway/hooks/types.d.ts +1 -1
  94. package/dist/gateway/hooks/types.js +12 -12
  95. package/dist/gateway/http/agents.cjs +67 -4
  96. package/dist/gateway/http/agents.js +67 -4
  97. package/dist/gateway/http/sessions.cjs +7 -7
  98. package/dist/gateway/http/sessions.js +7 -7
  99. package/dist/gateway/http/types.d.ts +5 -3
  100. package/dist/gateway/http/webhooks.cjs +6 -5
  101. package/dist/gateway/http/webhooks.js +6 -5
  102. package/dist/gateway/server.cjs +198 -41
  103. package/dist/gateway/server.d.ts +9 -1
  104. package/dist/gateway/server.js +198 -41
  105. package/dist/gateway/types.d.ts +1 -0
  106. package/dist/gateway/validation.cjs +39 -39
  107. package/dist/gateway/validation.d.ts +1 -1
  108. package/dist/gateway/validation.js +40 -40
  109. package/dist/providers/codex.cjs +167 -0
  110. package/dist/providers/codex.d.ts +15 -0
  111. package/dist/providers/codex.js +127 -0
  112. package/dist/providers/credentials.cjs +8 -0
  113. package/dist/providers/credentials.js +8 -0
  114. package/dist/providers/registry.cjs +11 -0
  115. package/dist/providers/registry.d.ts +1 -1
  116. package/dist/providers/registry.js +11 -0
  117. package/dist/tests/additionalMessageMiddleware.test.cjs +3 -0
  118. package/dist/tests/additionalMessageMiddleware.test.js +3 -0
  119. package/dist/tests/agentInvokerSummarization.test.cjs +455 -0
  120. package/dist/tests/agentInvokerSummarization.test.d.ts +1 -0
  121. package/dist/tests/agentInvokerSummarization.test.js +449 -0
  122. package/dist/tests/agents-api.test.cjs +45 -5
  123. package/dist/tests/agents-api.test.js +45 -5
  124. package/dist/tests/cli-config-loader.test.cjs +88 -0
  125. package/dist/tests/cli-config-loader.test.js +88 -0
  126. package/dist/tests/cli-init.test.cjs +27 -3
  127. package/dist/tests/cli-init.test.js +27 -3
  128. package/dist/tests/codex-credentials-precedence.test.cjs +94 -0
  129. package/dist/tests/codex-credentials-precedence.test.d.ts +1 -0
  130. package/dist/tests/codex-credentials-precedence.test.js +88 -0
  131. package/dist/tests/codex-provider.test.cjs +210 -0
  132. package/dist/tests/codex-provider.test.d.ts +1 -0
  133. package/dist/tests/codex-provider.test.js +204 -0
  134. package/dist/tests/gateway.test.cjs +115 -8
  135. package/dist/tests/gateway.test.js +115 -8
  136. package/dist/tests/provider-command-codex.test.cjs +57 -0
  137. package/dist/tests/provider-command-codex.test.d.ts +1 -0
  138. package/dist/tests/provider-command-codex.test.js +51 -0
  139. package/dist/tests/sessionStateMessages.test.cjs +38 -0
  140. package/dist/tests/sessionStateMessages.test.js +38 -0
  141. package/dist/tests/toolDisplayHelpers.test.cjs +3 -0
  142. package/dist/tests/toolDisplayHelpers.test.js +3 -0
  143. package/dist/tools/mcp-finance.cjs +48 -48
  144. package/dist/tools/mcp-finance.js +48 -48
  145. package/dist/types/mcp.cjs +15 -15
  146. package/dist/types/mcp.d.ts +1 -1
  147. package/dist/types/mcp.js +16 -16
  148. package/dist/types/voice.cjs +21 -21
  149. package/dist/types/voice.d.ts +1 -1
  150. package/dist/types/voice.js +22 -22
  151. package/dist/webui/assets/index-DVWQluit.css +11 -0
  152. package/dist/webui/assets/index-Dlyzwalc.js +270 -0
  153. package/dist/webui/favicon-32x32.png +0 -0
  154. package/dist/webui/favicon-64x64.png +0 -0
  155. package/dist/webui/favicon.webp +0 -0
  156. package/dist/webui/index.html +4 -2
  157. package/package.json +13 -12
  158. package/.wingman/agents/coding/implementor.md +0 -79
  159. package/dist/webui/assets/index-CPhfGPHc.js +0 -182
  160. package/dist/webui/assets/index-DDsMIOTX.css +0 -11
@@ -0,0 +1,449 @@
1
+ import { describe, expect, it } from "vitest";
2
+ import { validateConfig } from "../cli/config/schema.js";
3
+ import { chunkHasAssistantText, configureDeepAgentSummarizationMiddleware, detectStreamErrorMessage, detectToolEventContext, evaluateStreamingCompletion, isRootLangGraphTerminalEvent, resolveHumanInTheLoopSettings, resolveModelRetryMiddlewareSettings, resolveSummarizationMiddlewareSettings, resolveToolRetryMiddlewareSettings, selectStreamingFallbackText, trackRootLangGraphRunId } from "../cli/core/agentInvoker.js";
4
+ const parseConfig = (input)=>{
5
+ const result = validateConfig(input);
6
+ if (!result.success || !result.data) throw new Error(result.error || "Expected config validation to succeed");
7
+ return result.data;
8
+ };
9
+ describe("resolveSummarizationMiddlewareSettings", ()=>{
10
+ it("returns default summarization settings from config defaults", ()=>{
11
+ const config = parseConfig({});
12
+ expect(resolveSummarizationMiddlewareSettings(config)).toEqual({
13
+ maxTokensBeforeSummary: 12000,
14
+ messagesToKeep: 8
15
+ });
16
+ });
17
+ it("returns null when summarization is disabled", ()=>{
18
+ const config = parseConfig({
19
+ summarization: {
20
+ enabled: false
21
+ }
22
+ });
23
+ expect(resolveSummarizationMiddlewareSettings(config)).toBeNull();
24
+ });
25
+ it("returns custom summarization settings when configured", ()=>{
26
+ const config = parseConfig({
27
+ summarization: {
28
+ enabled: true,
29
+ maxTokensBeforeSummary: 20000,
30
+ messagesToKeep: 10
31
+ }
32
+ });
33
+ expect(resolveSummarizationMiddlewareSettings(config)).toEqual({
34
+ maxTokensBeforeSummary: 20000,
35
+ messagesToKeep: 10
36
+ });
37
+ });
38
+ });
39
+ describe("configureDeepAgentSummarizationMiddleware", ()=>{
40
+ it("replaces built-in summarization middleware with configured settings", ()=>{
41
+ const agent = {
42
+ options: {
43
+ middleware: [
44
+ {
45
+ name: "todoListMiddleware"
46
+ },
47
+ {
48
+ name: "SummarizationMiddleware",
49
+ marker: "old"
50
+ }
51
+ ]
52
+ }
53
+ };
54
+ configureDeepAgentSummarizationMiddleware(agent, {
55
+ maxTokensBeforeSummary: 9000,
56
+ messagesToKeep: 5
57
+ }, "openai:gpt-4o-mini");
58
+ expect(agent.options.middleware).toHaveLength(2);
59
+ expect(agent.options.middleware[1]?.name).toBe("SummarizationMiddleware");
60
+ expect(agent.options.middleware[1]?.marker).toBeUndefined();
61
+ });
62
+ it("removes built-in summarization middleware when disabled", ()=>{
63
+ const agent = {
64
+ options: {
65
+ middleware: [
66
+ {
67
+ name: "todoListMiddleware"
68
+ },
69
+ {
70
+ name: "SummarizationMiddleware"
71
+ },
72
+ {
73
+ name: "patchToolCallsMiddleware"
74
+ }
75
+ ]
76
+ }
77
+ };
78
+ configureDeepAgentSummarizationMiddleware(agent, null);
79
+ expect(agent.options.middleware.map((m)=>m.name)).toEqual([
80
+ "todoListMiddleware",
81
+ "patchToolCallsMiddleware"
82
+ ]);
83
+ });
84
+ });
85
+ describe("resolveModelRetryMiddlewareSettings", ()=>{
86
+ it("returns default retry settings when config defaults are used", ()=>{
87
+ const config = parseConfig({});
88
+ expect(resolveModelRetryMiddlewareSettings(config)).toEqual({
89
+ maxRetries: 2,
90
+ backoffFactor: 2,
91
+ initialDelayMs: 1000,
92
+ maxDelayMs: 60000,
93
+ jitter: true,
94
+ onFailure: "continue"
95
+ });
96
+ });
97
+ it("returns configured retry settings when enabled", ()=>{
98
+ const config = parseConfig({
99
+ modelRetry: {
100
+ enabled: true,
101
+ maxRetries: 3,
102
+ backoffFactor: 1.5,
103
+ initialDelayMs: 250,
104
+ maxDelayMs: 5000,
105
+ jitter: false,
106
+ onFailure: "error"
107
+ }
108
+ });
109
+ expect(resolveModelRetryMiddlewareSettings(config)).toEqual({
110
+ maxRetries: 3,
111
+ backoffFactor: 1.5,
112
+ initialDelayMs: 250,
113
+ maxDelayMs: 5000,
114
+ jitter: false,
115
+ onFailure: "error"
116
+ });
117
+ });
118
+ });
119
+ describe("resolveToolRetryMiddlewareSettings", ()=>{
120
+ it("returns null by default", ()=>{
121
+ const config = parseConfig({});
122
+ expect(resolveToolRetryMiddlewareSettings(config)).toBeNull();
123
+ });
124
+ it("returns configured retry settings and tool filter when enabled", ()=>{
125
+ const config = parseConfig({
126
+ toolRetry: {
127
+ enabled: true,
128
+ maxRetries: 4,
129
+ backoffFactor: 2,
130
+ initialDelayMs: 500,
131
+ maxDelayMs: 10000,
132
+ jitter: true,
133
+ onFailure: "continue",
134
+ tools: [
135
+ "internet_search",
136
+ "web_crawler"
137
+ ]
138
+ }
139
+ });
140
+ expect(resolveToolRetryMiddlewareSettings(config)).toEqual({
141
+ maxRetries: 4,
142
+ backoffFactor: 2,
143
+ initialDelayMs: 500,
144
+ maxDelayMs: 10000,
145
+ jitter: true,
146
+ onFailure: "continue",
147
+ tools: [
148
+ "internet_search",
149
+ "web_crawler"
150
+ ]
151
+ });
152
+ });
153
+ });
154
+ describe("resolveHumanInTheLoopSettings", ()=>{
155
+ it("returns null by default", ()=>{
156
+ const config = parseConfig({});
157
+ expect(resolveHumanInTheLoopSettings(config)).toBeNull();
158
+ });
159
+ it("returns null when enabled but no tool policy is defined", ()=>{
160
+ const config = parseConfig({
161
+ humanInTheLoop: {
162
+ enabled: true,
163
+ interruptOn: {}
164
+ }
165
+ });
166
+ expect(resolveHumanInTheLoopSettings(config)).toBeNull();
167
+ });
168
+ it("returns interrupt map when enabled with tool policies", ()=>{
169
+ const config = parseConfig({
170
+ humanInTheLoop: {
171
+ enabled: true,
172
+ interruptOn: {
173
+ command_execute: {
174
+ allowedDecisions: [
175
+ "approve",
176
+ "reject"
177
+ ],
178
+ description: "Command execution requires approval"
179
+ },
180
+ internet_search: false
181
+ }
182
+ }
183
+ });
184
+ expect(resolveHumanInTheLoopSettings(config)).toEqual({
185
+ interruptOn: {
186
+ command_execute: {
187
+ allowedDecisions: [
188
+ "approve",
189
+ "reject"
190
+ ],
191
+ description: "Command execution requires approval"
192
+ },
193
+ internet_search: false
194
+ }
195
+ });
196
+ });
197
+ });
198
+ describe("detectToolEventContext", ()=>{
199
+ it("extracts tool context for tool start events", ()=>{
200
+ const detected = detectToolEventContext({
201
+ event: "on_tool_start",
202
+ name: "glob"
203
+ });
204
+ expect(detected).toEqual({
205
+ event: "on_tool_start",
206
+ toolName: "glob"
207
+ });
208
+ });
209
+ it("extracts tool context for tool error events", ()=>{
210
+ const detected = detectToolEventContext({
211
+ event: "on_tool_error",
212
+ name: "grep"
213
+ });
214
+ expect(detected).toEqual({
215
+ event: "on_tool_error",
216
+ toolName: "grep"
217
+ });
218
+ });
219
+ it("returns null for non-tool events", ()=>{
220
+ const detected = detectToolEventContext({
221
+ event: "on_chat_model_stream",
222
+ data: {
223
+ chunk: {
224
+ text: "hello"
225
+ }
226
+ }
227
+ });
228
+ expect(detected).toBeNull();
229
+ });
230
+ });
231
+ describe("chunkHasAssistantText", ()=>{
232
+ it("detects text in on_chat_model_stream chunks", ()=>{
233
+ expect(chunkHasAssistantText({
234
+ event: "on_chat_model_stream",
235
+ data: {
236
+ chunk: {
237
+ content: [
238
+ {
239
+ type: "text",
240
+ text: "hello"
241
+ }
242
+ ]
243
+ }
244
+ }
245
+ })).toBe(true);
246
+ });
247
+ it("detects text in on_llm_stream chunks", ()=>{
248
+ expect(chunkHasAssistantText({
249
+ event: "on_llm_stream",
250
+ data: {
251
+ chunk: {
252
+ text: "delta"
253
+ }
254
+ }
255
+ })).toBe(true);
256
+ });
257
+ it("returns false when chunk has no assistant text", ()=>{
258
+ expect(chunkHasAssistantText({
259
+ event: "on_tool_start",
260
+ name: "glob"
261
+ })).toBe(false);
262
+ });
263
+ });
264
+ describe("selectStreamingFallbackText", ()=>{
265
+ it("returns assistant text introduced during the current invocation", ()=>{
266
+ const selected = selectStreamingFallbackText([
267
+ {
268
+ role: "assistant",
269
+ content: "stale message"
270
+ }
271
+ ], [
272
+ {
273
+ role: "assistant",
274
+ content: "stale message"
275
+ },
276
+ {
277
+ role: "assistant",
278
+ content: "fresh fallback"
279
+ }
280
+ ]);
281
+ expect(selected).toBe("fresh fallback");
282
+ });
283
+ it("returns undefined when no new assistant text is introduced", ()=>{
284
+ const selected = selectStreamingFallbackText([
285
+ {
286
+ role: "assistant",
287
+ content: "existing response"
288
+ }
289
+ ], [
290
+ {
291
+ role: "user",
292
+ content: "not assistant"
293
+ },
294
+ {
295
+ role: "assistant",
296
+ content: "existing response"
297
+ }
298
+ ]);
299
+ expect(selected).toBeUndefined();
300
+ });
301
+ it("handles repeated assistant messages by count", ()=>{
302
+ const selected = selectStreamingFallbackText([
303
+ {
304
+ role: "assistant",
305
+ content: "repeat"
306
+ }
307
+ ], [
308
+ {
309
+ role: "assistant",
310
+ content: "repeat"
311
+ },
312
+ {
313
+ role: "assistant",
314
+ content: "repeat"
315
+ }
316
+ ]);
317
+ expect(selected).toBe("repeat");
318
+ });
319
+ });
320
+ describe("detectStreamErrorMessage", ()=>{
321
+ it("extracts non-tool stream error messages", ()=>{
322
+ const detected = detectStreamErrorMessage({
323
+ event: "on_chat_model_error",
324
+ data: {
325
+ error: {
326
+ message: "Model call failed after retries"
327
+ }
328
+ }
329
+ });
330
+ expect(detected).toBe("Model call failed after retries");
331
+ });
332
+ it("ignores tool errors and non-error events", ()=>{
333
+ expect(detectStreamErrorMessage({
334
+ event: "on_tool_error",
335
+ data: {
336
+ error: "tool failed"
337
+ }
338
+ })).toBeUndefined();
339
+ expect(detectStreamErrorMessage({
340
+ event: "on_chat_model_stream",
341
+ data: {
342
+ chunk: {
343
+ text: "hello"
344
+ }
345
+ }
346
+ })).toBeUndefined();
347
+ });
348
+ });
349
+ describe("evaluateStreamingCompletion", ()=>{
350
+ it("blocks with stream_error when no assistant text and stream error exists", ()=>{
351
+ const result = evaluateStreamingCompletion({
352
+ sawAssistantText: false,
353
+ fallbackText: void 0,
354
+ streamErrorMessage: "provider timeout"
355
+ });
356
+ expect(result).toEqual({
357
+ status: "blocked",
358
+ reason: "stream_error",
359
+ message: "Model call failed: provider timeout"
360
+ });
361
+ });
362
+ it("blocks with empty_stream_response when no text or fallback is present", ()=>{
363
+ const result = evaluateStreamingCompletion({
364
+ sawAssistantText: false,
365
+ fallbackText: void 0,
366
+ streamErrorMessage: void 0
367
+ });
368
+ expect(result).toEqual({
369
+ status: "blocked",
370
+ reason: "empty_stream_response",
371
+ message: "Model completed without a response. Check provider logs for request errors."
372
+ });
373
+ });
374
+ it("returns ok when assistant text exists", ()=>{
375
+ const result = evaluateStreamingCompletion({
376
+ sawAssistantText: true,
377
+ fallbackText: void 0,
378
+ streamErrorMessage: void 0
379
+ });
380
+ expect(result).toEqual({
381
+ status: "ok"
382
+ });
383
+ });
384
+ });
385
+ describe("LangGraph lifecycle termination", ()=>{
386
+ it("tracks root LangGraph run id from parentless on_chain_start", ()=>{
387
+ const rootRunId = trackRootLangGraphRunId(void 0, {
388
+ event: "on_chain_start",
389
+ name: "LangGraph",
390
+ run_id: "root-run",
391
+ parent_ids: []
392
+ });
393
+ expect(rootRunId).toBe("root-run");
394
+ });
395
+ it("ignores non-root or non-LangGraph start events", ()=>{
396
+ const nestedRunId = trackRootLangGraphRunId(void 0, {
397
+ event: "on_chain_start",
398
+ name: "LangGraph",
399
+ run_id: "child-run",
400
+ parent_ids: [
401
+ "root-run"
402
+ ]
403
+ });
404
+ expect(nestedRunId).toBeUndefined();
405
+ const otherChainRunId = trackRootLangGraphRunId(void 0, {
406
+ event: "on_chain_start",
407
+ name: "CustomChain",
408
+ run_id: "other-run",
409
+ parent_ids: []
410
+ });
411
+ expect(otherChainRunId).toBeUndefined();
412
+ });
413
+ it("detects terminal root LangGraph on_chain_end by run id", ()=>{
414
+ expect(isRootLangGraphTerminalEvent({
415
+ event: "on_chain_end",
416
+ name: "LangGraph",
417
+ run_id: "root-run",
418
+ parent_ids: []
419
+ }, "root-run")).toBe(true);
420
+ });
421
+ it("does not treat nested, mismatched, or non-LangGraph chain end as terminal", ()=>{
422
+ expect(isRootLangGraphTerminalEvent({
423
+ event: "on_chain_end",
424
+ name: "LangGraph",
425
+ run_id: "child-run",
426
+ parent_ids: [
427
+ "root-run"
428
+ ]
429
+ }, "root-run")).toBe(false);
430
+ expect(isRootLangGraphTerminalEvent({
431
+ event: "on_chain_end",
432
+ name: "LangGraph",
433
+ run_id: "other-run",
434
+ parent_ids: []
435
+ }, "root-run")).toBe(false);
436
+ expect(isRootLangGraphTerminalEvent({
437
+ event: "on_chain_end",
438
+ name: "CustomChain",
439
+ run_id: "root-run",
440
+ parent_ids: []
441
+ }, "root-run")).toBe(false);
442
+ expect(isRootLangGraphTerminalEvent({
443
+ event: "on_chain_end",
444
+ name: "LangGraph",
445
+ run_id: "root-run",
446
+ parent_ids: []
447
+ }, void 0)).toBe(false);
448
+ });
449
+ });
@@ -1,11 +1,11 @@
1
1
  "use strict";
2
2
  var __webpack_exports__ = {};
3
- const external_vitest_namespaceObject = require("vitest");
4
- const agents_cjs_namespaceObject = require("../gateway/http/agents.cjs");
5
3
  const external_node_fs_namespaceObject = require("node:fs");
6
- const external_node_path_namespaceObject = require("node:path");
7
4
  const external_node_os_namespaceObject = require("node:os");
5
+ const external_node_path_namespaceObject = require("node:path");
8
6
  const external_js_yaml_namespaceObject = require("js-yaml");
7
+ const external_vitest_namespaceObject = require("vitest");
8
+ const agents_cjs_namespaceObject = require("../gateway/http/agents.cjs");
9
9
  const isBunRuntime = void 0 !== globalThis.Bun;
10
10
  const describeIfBun = isBunRuntime ? external_vitest_namespaceObject.describe : external_vitest_namespaceObject.describe.skip;
11
11
  const parseMarkdownAgent = (raw)=>{
@@ -65,6 +65,7 @@ describeIfBun("agents API", ()=>{
65
65
  "not_real_tool"
66
66
  ],
67
67
  prompt: "You are the orchestrator.",
68
+ reasoningEffort: "high",
68
69
  promptTraining: {
69
70
  enabled: true,
70
71
  instructionsPath: "/memories/agents/orchestrator/instructions.md"
@@ -78,6 +79,7 @@ describeIfBun("agents API", ()=>{
78
79
  "not_real_tool"
79
80
  ],
80
81
  prompt: "Plan tasks in detail.",
82
+ reasoningEffort: "low",
81
83
  promptTraining: true
82
84
  },
83
85
  {
@@ -86,7 +88,8 @@ describeIfBun("agents API", ()=>{
86
88
  tools: [
87
89
  "command_execute"
88
90
  ],
89
- prompt: "Execute planned tasks."
91
+ prompt: "Execute planned tasks.",
92
+ thinkingEffort: "minimal"
90
93
  }
91
94
  ]
92
95
  })
@@ -99,12 +102,15 @@ describeIfBun("agents API", ()=>{
99
102
  enabled: true,
100
103
  instructionsPath: "/memories/agents/orchestrator/instructions.md"
101
104
  });
105
+ (0, external_vitest_namespaceObject.expect)(created.reasoningEffort).toBe("high");
102
106
  (0, external_vitest_namespaceObject.expect)(created.promptRefinement).toEqual(created.promptTraining);
103
107
  (0, external_vitest_namespaceObject.expect)(created.subAgents).toHaveLength(2);
104
108
  (0, external_vitest_namespaceObject.expect)(created.subAgents[0].promptTraining).toBe(true);
105
109
  (0, external_vitest_namespaceObject.expect)(created.subAgents[0].tools).toEqual([
106
110
  "think"
107
111
  ]);
112
+ (0, external_vitest_namespaceObject.expect)(created.subAgents[0].reasoningEffort).toBe("low");
113
+ (0, external_vitest_namespaceObject.expect)(created.subAgents[1].reasoningEffort).toBe("minimal");
108
114
  const agentPath = (0, external_node_path_namespaceObject.join)(tempDir, "agents", "orchestrator", "agent.md");
109
115
  const parsed = parseMarkdownAgent((0, external_node_fs_namespaceObject.readFileSync)(agentPath, "utf-8"));
110
116
  (0, external_vitest_namespaceObject.expect)(parsed.prompt).toBe("You are the orchestrator.");
@@ -112,10 +118,13 @@ describeIfBun("agents API", ()=>{
112
118
  enabled: true,
113
119
  instructionsPath: "/memories/agents/orchestrator/instructions.md"
114
120
  });
121
+ (0, external_vitest_namespaceObject.expect)(parsed.metadata.reasoningEffort).toBe("high");
115
122
  (0, external_vitest_namespaceObject.expect)(parsed.metadata.subAgents).toHaveLength(2);
116
123
  (0, external_vitest_namespaceObject.expect)(parsed.metadata.subAgents[0].name).toBe("planner");
124
+ (0, external_vitest_namespaceObject.expect)(parsed.metadata.subAgents[0].reasoningEffort).toBe("low");
117
125
  (0, external_vitest_namespaceObject.expect)(parsed.metadata.subAgents[0].promptRefinement).toBe(true);
118
126
  (0, external_vitest_namespaceObject.expect)(parsed.metadata.subAgents[0].systemPrompt).toBe("Plan tasks in detail.");
127
+ (0, external_vitest_namespaceObject.expect)(parsed.metadata.subAgents[1].reasoningEffort).toBe("minimal");
119
128
  const detailReq = new Request("http://localhost/api/agents/orchestrator", {
120
129
  method: "GET"
121
130
  });
@@ -125,9 +134,11 @@ describeIfBun("agents API", ()=>{
125
134
  const detail = await detailRes.json();
126
135
  (0, external_vitest_namespaceObject.expect)(detail.promptTraining).toEqual(created.promptTraining);
127
136
  (0, external_vitest_namespaceObject.expect)(detail.promptRefinement).toEqual(created.promptTraining);
137
+ (0, external_vitest_namespaceObject.expect)(detail.reasoningEffort).toBe("high");
128
138
  (0, external_vitest_namespaceObject.expect)(detail.subAgents).toHaveLength(2);
129
139
  (0, external_vitest_namespaceObject.expect)(detail.subAgents[0].id).toBe("planner");
130
140
  (0, external_vitest_namespaceObject.expect)(detail.subAgents[0].prompt).toBe("Plan tasks in detail.");
141
+ (0, external_vitest_namespaceObject.expect)(detail.subAgents[0].reasoningEffort).toBe("low");
131
142
  });
132
143
  (0, external_vitest_namespaceObject.it)("updates promptTraining and subAgents in markdown", async ()=>{
133
144
  const createReq = new Request("http://localhost/api/agents", {
@@ -141,6 +152,7 @@ describeIfBun("agents API", ()=>{
141
152
  "think"
142
153
  ],
143
154
  prompt: "Original prompt",
155
+ reasoningEffort: "low",
144
156
  promptTraining: true,
145
157
  subAgents: [
146
158
  {
@@ -149,7 +161,8 @@ describeIfBun("agents API", ()=>{
149
161
  tools: [
150
162
  "think"
151
163
  ],
152
- prompt: "Original planner prompt"
164
+ prompt: "Original planner prompt",
165
+ reasoningEffort: "minimal"
153
166
  }
154
167
  ]
155
168
  })
@@ -162,6 +175,7 @@ describeIfBun("agents API", ()=>{
162
175
  "Content-Type": "application/json"
163
176
  },
164
177
  body: JSON.stringify({
178
+ reasoningEffort: "high",
165
179
  promptTraining: false,
166
180
  subAgents: [
167
181
  {
@@ -171,6 +185,7 @@ describeIfBun("agents API", ()=>{
171
185
  "git_status"
172
186
  ],
173
187
  prompt: "Review carefully.",
188
+ thinkingEffort: "medium",
174
189
  promptTraining: {
175
190
  enabled: true,
176
191
  instructionsPath: "/memories/reviewer.md"
@@ -186,8 +201,10 @@ describeIfBun("agents API", ()=>{
186
201
  const updated = await updateRes.json();
187
202
  (0, external_vitest_namespaceObject.expect)(updated.promptTraining).toBe(false);
188
203
  (0, external_vitest_namespaceObject.expect)(updated.promptRefinement).toBe(false);
204
+ (0, external_vitest_namespaceObject.expect)(updated.reasoningEffort).toBe("high");
189
205
  (0, external_vitest_namespaceObject.expect)(updated.subAgents).toHaveLength(1);
190
206
  (0, external_vitest_namespaceObject.expect)(updated.subAgents[0].id).toBe("reviewer");
207
+ (0, external_vitest_namespaceObject.expect)(updated.subAgents[0].reasoningEffort).toBe("medium");
191
208
  (0, external_vitest_namespaceObject.expect)(updated.subAgents[0].promptTraining).toEqual({
192
209
  enabled: true,
193
210
  instructionsPath: "/memories/reviewer.md"
@@ -195,8 +212,10 @@ describeIfBun("agents API", ()=>{
195
212
  const agentPath = (0, external_node_path_namespaceObject.join)(tempDir, "agents", "editor-agent", "agent.md");
196
213
  const parsed = parseMarkdownAgent((0, external_node_fs_namespaceObject.readFileSync)(agentPath, "utf-8"));
197
214
  (0, external_vitest_namespaceObject.expect)(parsed.metadata.promptRefinement).toBe(false);
215
+ (0, external_vitest_namespaceObject.expect)(parsed.metadata.reasoningEffort).toBe("high");
198
216
  (0, external_vitest_namespaceObject.expect)(parsed.metadata.subAgents).toHaveLength(1);
199
217
  (0, external_vitest_namespaceObject.expect)(parsed.metadata.subAgents[0].name).toBe("reviewer");
218
+ (0, external_vitest_namespaceObject.expect)(parsed.metadata.subAgents[0].reasoningEffort).toBe("medium");
200
219
  (0, external_vitest_namespaceObject.expect)(parsed.metadata.subAgents[0].promptRefinement).toEqual({
201
220
  enabled: true,
202
221
  instructionsPath: "/memories/reviewer.md"
@@ -225,6 +244,27 @@ describeIfBun("agents API", ()=>{
225
244
  (0, external_vitest_namespaceObject.expect)(created.promptTraining).toBe(true);
226
245
  (0, external_vitest_namespaceObject.expect)(created.promptRefinement).toBe(true);
227
246
  });
247
+ (0, external_vitest_namespaceObject.it)("accepts legacy thinkingEffort input as reasoningEffort", async ()=>{
248
+ const createReq = new Request("http://localhost/api/agents", {
249
+ method: "POST",
250
+ headers: {
251
+ "Content-Type": "application/json"
252
+ },
253
+ body: JSON.stringify({
254
+ id: "legacy-thinking-agent",
255
+ tools: [
256
+ "think"
257
+ ],
258
+ prompt: "Legacy thinking prompt",
259
+ thinkingEffort: "medium"
260
+ })
261
+ });
262
+ const createRes = await (0, agents_cjs_namespaceObject.handleAgentsApi)(ctx, createReq, new URL(createReq.url));
263
+ (0, external_vitest_namespaceObject.expect)(createRes).not.toBeNull();
264
+ (0, external_vitest_namespaceObject.expect)(createRes?.ok).toBe(true);
265
+ const created = await createRes.json();
266
+ (0, external_vitest_namespaceObject.expect)(created.reasoningEffort).toBe("medium");
267
+ });
228
268
  });
229
269
  for(var __rspack_i in __webpack_exports__)exports[__rspack_i] = __webpack_exports__[__rspack_i];
230
270
  Object.defineProperty(exports, '__esModule', {