@wingman-ai/gateway 0.2.3 → 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.
- package/.wingman/agents/README.md +7 -1
- package/.wingman/agents/coding/agent.md +295 -202
- package/.wingman/agents/coding-v2/agent.md +127 -0
- package/.wingman/agents/coding-v2/implementor.md +89 -0
- package/dist/agent/config/agentConfig.cjs +31 -17
- package/dist/agent/config/agentConfig.d.ts +23 -1
- package/dist/agent/config/agentConfig.js +30 -19
- package/dist/agent/config/agentLoader.cjs +26 -8
- package/dist/agent/config/agentLoader.d.ts +4 -2
- package/dist/agent/config/agentLoader.js +26 -8
- package/dist/agent/config/modelFactory.cjs +77 -27
- package/dist/agent/config/modelFactory.d.ts +11 -1
- package/dist/agent/config/modelFactory.js +77 -27
- package/dist/agent/config/toolRegistry.cjs +19 -6
- package/dist/agent/config/toolRegistry.d.ts +5 -2
- package/dist/agent/config/toolRegistry.js +19 -6
- package/dist/agent/middleware/hooks/types.cjs +13 -13
- package/dist/agent/middleware/hooks/types.d.ts +1 -1
- package/dist/agent/middleware/hooks/types.js +14 -14
- package/dist/agent/tests/agentConfig.test.cjs +22 -2
- package/dist/agent/tests/agentConfig.test.js +22 -2
- package/dist/agent/tests/agentLoader.test.cjs +38 -1
- package/dist/agent/tests/agentLoader.test.js +38 -1
- package/dist/agent/tests/backgroundTerminal.test.cjs +70 -0
- package/dist/agent/tests/backgroundTerminal.test.d.ts +1 -0
- package/dist/agent/tests/backgroundTerminal.test.js +64 -0
- package/dist/agent/tests/commandExecuteTool.test.cjs +29 -0
- package/dist/agent/tests/commandExecuteTool.test.d.ts +1 -0
- package/dist/agent/tests/commandExecuteTool.test.js +23 -0
- package/dist/agent/tests/modelFactory.test.cjs +35 -0
- package/dist/agent/tests/modelFactory.test.js +35 -0
- package/dist/agent/tests/terminalSessionManager.test.cjs +121 -0
- package/dist/agent/tests/terminalSessionManager.test.d.ts +1 -0
- package/dist/agent/tests/terminalSessionManager.test.js +115 -0
- package/dist/agent/tests/toolRegistry.test.cjs +14 -2
- package/dist/agent/tests/toolRegistry.test.js +14 -2
- package/dist/agent/tools/background_terminal.cjs +128 -0
- package/dist/agent/tools/background_terminal.d.ts +41 -0
- package/dist/agent/tools/background_terminal.js +94 -0
- package/dist/agent/tools/code_search.cjs +6 -6
- package/dist/agent/tools/code_search.d.ts +1 -1
- package/dist/agent/tools/code_search.js +7 -7
- package/dist/agent/tools/command_execute.cjs +22 -7
- package/dist/agent/tools/command_execute.d.ts +3 -2
- package/dist/agent/tools/command_execute.js +23 -8
- package/dist/agent/tools/git_status.cjs +3 -3
- package/dist/agent/tools/git_status.d.ts +1 -1
- package/dist/agent/tools/git_status.js +4 -4
- package/dist/agent/tools/internet_search.cjs +6 -6
- package/dist/agent/tools/internet_search.d.ts +1 -1
- package/dist/agent/tools/internet_search.js +7 -7
- package/dist/agent/tools/terminal_session_manager.cjs +321 -0
- package/dist/agent/tools/terminal_session_manager.d.ts +77 -0
- package/dist/agent/tools/terminal_session_manager.js +284 -0
- package/dist/agent/tools/think.cjs +4 -4
- package/dist/agent/tools/think.d.ts +1 -1
- package/dist/agent/tools/think.js +5 -5
- package/dist/agent/tools/ui_registry.cjs +13 -13
- package/dist/agent/tools/ui_registry.d.ts +4 -4
- package/dist/agent/tools/ui_registry.js +14 -14
- package/dist/agent/tools/web_crawler.cjs +4 -4
- package/dist/agent/tools/web_crawler.d.ts +1 -1
- package/dist/agent/tools/web_crawler.js +5 -5
- package/dist/agent/utils.cjs +2 -1
- package/dist/agent/utils.js +2 -1
- package/dist/cli/config/schema.cjs +89 -89
- package/dist/cli/config/schema.d.ts +1 -1
- package/dist/cli/config/schema.js +90 -90
- package/dist/cli/core/agentInvoker.cjs +170 -21
- package/dist/cli/core/agentInvoker.d.ts +25 -4
- package/dist/cli/core/agentInvoker.js +157 -20
- package/dist/cli/core/streamParser.cjs +15 -0
- package/dist/cli/core/streamParser.js +15 -0
- package/dist/cli/ui/toolDisplayHelpers.cjs +2 -0
- package/dist/cli/ui/toolDisplayHelpers.js +2 -0
- package/dist/gateway/hooks/registry.cjs +2 -1
- package/dist/gateway/hooks/registry.d.ts +1 -1
- package/dist/gateway/hooks/registry.js +2 -1
- package/dist/gateway/hooks/types.cjs +11 -11
- package/dist/gateway/hooks/types.d.ts +1 -1
- package/dist/gateway/hooks/types.js +12 -12
- package/dist/gateway/http/agents.cjs +67 -4
- package/dist/gateway/http/agents.js +67 -4
- package/dist/gateway/http/types.d.ts +5 -3
- package/dist/gateway/http/webhooks.cjs +6 -5
- package/dist/gateway/http/webhooks.js +6 -5
- package/dist/gateway/server.cjs +7 -0
- package/dist/gateway/server.d.ts +1 -0
- package/dist/gateway/server.js +7 -0
- package/dist/gateway/validation.cjs +39 -39
- package/dist/gateway/validation.d.ts +1 -1
- package/dist/gateway/validation.js +40 -40
- package/dist/tests/additionalMessageMiddleware.test.cjs +3 -0
- package/dist/tests/additionalMessageMiddleware.test.js +3 -0
- package/dist/tests/agentInvokerSummarization.test.cjs +171 -12
- package/dist/tests/agentInvokerSummarization.test.js +172 -13
- package/dist/tests/agents-api.test.cjs +45 -5
- package/dist/tests/agents-api.test.js +45 -5
- package/dist/tests/cli-init.test.cjs +27 -3
- package/dist/tests/cli-init.test.js +27 -3
- package/dist/tests/codex-provider.test.cjs +24 -0
- package/dist/tests/codex-provider.test.js +24 -0
- package/dist/tests/gateway.test.cjs +7 -7
- package/dist/tests/gateway.test.js +7 -7
- package/dist/tests/toolDisplayHelpers.test.cjs +3 -0
- package/dist/tests/toolDisplayHelpers.test.js +3 -0
- package/dist/tools/mcp-finance.cjs +48 -48
- package/dist/tools/mcp-finance.js +48 -48
- package/dist/types/mcp.cjs +15 -15
- package/dist/types/mcp.d.ts +1 -1
- package/dist/types/mcp.js +16 -16
- package/dist/types/voice.cjs +21 -21
- package/dist/types/voice.d.ts +1 -1
- package/dist/types/voice.js +22 -22
- package/dist/webui/assets/index-DVWQluit.css +11 -0
- package/dist/webui/assets/index-Dlyzwalc.js +270 -0
- package/dist/webui/favicon-32x32.png +0 -0
- package/dist/webui/favicon-64x64.png +0 -0
- package/dist/webui/favicon.webp +0 -0
- package/dist/webui/index.html +4 -2
- package/package.json +13 -12
- package/.wingman/agents/coding/implementor.md +0 -103
- package/dist/webui/assets/index-BVMavpud.css +0 -11
- package/dist/webui/assets/index-DCB2aVVf.js +0 -182
|
@@ -208,6 +208,16 @@ const parseConfig = (input)=>{
|
|
|
208
208
|
toolName: "glob"
|
|
209
209
|
});
|
|
210
210
|
});
|
|
211
|
+
(0, external_vitest_namespaceObject.it)("extracts tool context for tool error events", ()=>{
|
|
212
|
+
const detected = (0, agentInvoker_cjs_namespaceObject.detectToolEventContext)({
|
|
213
|
+
event: "on_tool_error",
|
|
214
|
+
name: "grep"
|
|
215
|
+
});
|
|
216
|
+
(0, external_vitest_namespaceObject.expect)(detected).toEqual({
|
|
217
|
+
event: "on_tool_error",
|
|
218
|
+
toolName: "grep"
|
|
219
|
+
});
|
|
220
|
+
});
|
|
211
221
|
(0, external_vitest_namespaceObject.it)("returns null for non-tool events", ()=>{
|
|
212
222
|
const detected = (0, agentInvoker_cjs_namespaceObject.detectToolEventContext)({
|
|
213
223
|
event: "on_chat_model_stream",
|
|
@@ -254,41 +264,190 @@ const parseConfig = (input)=>{
|
|
|
254
264
|
});
|
|
255
265
|
});
|
|
256
266
|
(0, external_vitest_namespaceObject.describe)("selectStreamingFallbackText", ()=>{
|
|
257
|
-
(0, external_vitest_namespaceObject.it)("returns
|
|
267
|
+
(0, external_vitest_namespaceObject.it)("returns assistant text introduced during the current invocation", ()=>{
|
|
258
268
|
const selected = (0, agentInvoker_cjs_namespaceObject.selectStreamingFallbackText)([
|
|
259
269
|
{
|
|
260
270
|
role: "assistant",
|
|
261
|
-
|
|
271
|
+
content: "stale message"
|
|
272
|
+
}
|
|
273
|
+
], [
|
|
274
|
+
{
|
|
275
|
+
role: "assistant",
|
|
262
276
|
content: "stale message"
|
|
263
277
|
},
|
|
264
278
|
{
|
|
265
279
|
role: "assistant",
|
|
266
|
-
createdAt: 4500,
|
|
267
280
|
content: "fresh fallback"
|
|
268
281
|
}
|
|
269
|
-
]
|
|
282
|
+
]);
|
|
270
283
|
(0, external_vitest_namespaceObject.expect)(selected).toBe("fresh fallback");
|
|
271
284
|
});
|
|
272
|
-
(0, external_vitest_namespaceObject.it)("
|
|
285
|
+
(0, external_vitest_namespaceObject.it)("returns undefined when no new assistant text is introduced", ()=>{
|
|
273
286
|
const selected = (0, agentInvoker_cjs_namespaceObject.selectStreamingFallbackText)([
|
|
274
287
|
{
|
|
275
288
|
role: "assistant",
|
|
276
|
-
|
|
277
|
-
|
|
278
|
-
|
|
289
|
+
content: "existing response"
|
|
290
|
+
}
|
|
291
|
+
], [
|
|
279
292
|
{
|
|
280
293
|
role: "user",
|
|
281
|
-
createdAt: 4900,
|
|
282
294
|
content: "not assistant"
|
|
283
295
|
},
|
|
284
296
|
{
|
|
285
297
|
role: "assistant",
|
|
286
|
-
|
|
287
|
-
content: " "
|
|
298
|
+
content: "existing response"
|
|
288
299
|
}
|
|
289
|
-
]
|
|
300
|
+
]);
|
|
290
301
|
(0, external_vitest_namespaceObject.expect)(selected).toBeUndefined();
|
|
291
302
|
});
|
|
303
|
+
(0, external_vitest_namespaceObject.it)("handles repeated assistant messages by count", ()=>{
|
|
304
|
+
const selected = (0, agentInvoker_cjs_namespaceObject.selectStreamingFallbackText)([
|
|
305
|
+
{
|
|
306
|
+
role: "assistant",
|
|
307
|
+
content: "repeat"
|
|
308
|
+
}
|
|
309
|
+
], [
|
|
310
|
+
{
|
|
311
|
+
role: "assistant",
|
|
312
|
+
content: "repeat"
|
|
313
|
+
},
|
|
314
|
+
{
|
|
315
|
+
role: "assistant",
|
|
316
|
+
content: "repeat"
|
|
317
|
+
}
|
|
318
|
+
]);
|
|
319
|
+
(0, external_vitest_namespaceObject.expect)(selected).toBe("repeat");
|
|
320
|
+
});
|
|
321
|
+
});
|
|
322
|
+
(0, external_vitest_namespaceObject.describe)("detectStreamErrorMessage", ()=>{
|
|
323
|
+
(0, external_vitest_namespaceObject.it)("extracts non-tool stream error messages", ()=>{
|
|
324
|
+
const detected = (0, agentInvoker_cjs_namespaceObject.detectStreamErrorMessage)({
|
|
325
|
+
event: "on_chat_model_error",
|
|
326
|
+
data: {
|
|
327
|
+
error: {
|
|
328
|
+
message: "Model call failed after retries"
|
|
329
|
+
}
|
|
330
|
+
}
|
|
331
|
+
});
|
|
332
|
+
(0, external_vitest_namespaceObject.expect)(detected).toBe("Model call failed after retries");
|
|
333
|
+
});
|
|
334
|
+
(0, external_vitest_namespaceObject.it)("ignores tool errors and non-error events", ()=>{
|
|
335
|
+
(0, external_vitest_namespaceObject.expect)((0, agentInvoker_cjs_namespaceObject.detectStreamErrorMessage)({
|
|
336
|
+
event: "on_tool_error",
|
|
337
|
+
data: {
|
|
338
|
+
error: "tool failed"
|
|
339
|
+
}
|
|
340
|
+
})).toBeUndefined();
|
|
341
|
+
(0, external_vitest_namespaceObject.expect)((0, agentInvoker_cjs_namespaceObject.detectStreamErrorMessage)({
|
|
342
|
+
event: "on_chat_model_stream",
|
|
343
|
+
data: {
|
|
344
|
+
chunk: {
|
|
345
|
+
text: "hello"
|
|
346
|
+
}
|
|
347
|
+
}
|
|
348
|
+
})).toBeUndefined();
|
|
349
|
+
});
|
|
350
|
+
});
|
|
351
|
+
(0, external_vitest_namespaceObject.describe)("evaluateStreamingCompletion", ()=>{
|
|
352
|
+
(0, external_vitest_namespaceObject.it)("blocks with stream_error when no assistant text and stream error exists", ()=>{
|
|
353
|
+
const result = (0, agentInvoker_cjs_namespaceObject.evaluateStreamingCompletion)({
|
|
354
|
+
sawAssistantText: false,
|
|
355
|
+
fallbackText: void 0,
|
|
356
|
+
streamErrorMessage: "provider timeout"
|
|
357
|
+
});
|
|
358
|
+
(0, external_vitest_namespaceObject.expect)(result).toEqual({
|
|
359
|
+
status: "blocked",
|
|
360
|
+
reason: "stream_error",
|
|
361
|
+
message: "Model call failed: provider timeout"
|
|
362
|
+
});
|
|
363
|
+
});
|
|
364
|
+
(0, external_vitest_namespaceObject.it)("blocks with empty_stream_response when no text or fallback is present", ()=>{
|
|
365
|
+
const result = (0, agentInvoker_cjs_namespaceObject.evaluateStreamingCompletion)({
|
|
366
|
+
sawAssistantText: false,
|
|
367
|
+
fallbackText: void 0,
|
|
368
|
+
streamErrorMessage: void 0
|
|
369
|
+
});
|
|
370
|
+
(0, external_vitest_namespaceObject.expect)(result).toEqual({
|
|
371
|
+
status: "blocked",
|
|
372
|
+
reason: "empty_stream_response",
|
|
373
|
+
message: "Model completed without a response. Check provider logs for request errors."
|
|
374
|
+
});
|
|
375
|
+
});
|
|
376
|
+
(0, external_vitest_namespaceObject.it)("returns ok when assistant text exists", ()=>{
|
|
377
|
+
const result = (0, agentInvoker_cjs_namespaceObject.evaluateStreamingCompletion)({
|
|
378
|
+
sawAssistantText: true,
|
|
379
|
+
fallbackText: void 0,
|
|
380
|
+
streamErrorMessage: void 0
|
|
381
|
+
});
|
|
382
|
+
(0, external_vitest_namespaceObject.expect)(result).toEqual({
|
|
383
|
+
status: "ok"
|
|
384
|
+
});
|
|
385
|
+
});
|
|
386
|
+
});
|
|
387
|
+
(0, external_vitest_namespaceObject.describe)("LangGraph lifecycle termination", ()=>{
|
|
388
|
+
(0, external_vitest_namespaceObject.it)("tracks root LangGraph run id from parentless on_chain_start", ()=>{
|
|
389
|
+
const rootRunId = (0, agentInvoker_cjs_namespaceObject.trackRootLangGraphRunId)(void 0, {
|
|
390
|
+
event: "on_chain_start",
|
|
391
|
+
name: "LangGraph",
|
|
392
|
+
run_id: "root-run",
|
|
393
|
+
parent_ids: []
|
|
394
|
+
});
|
|
395
|
+
(0, external_vitest_namespaceObject.expect)(rootRunId).toBe("root-run");
|
|
396
|
+
});
|
|
397
|
+
(0, external_vitest_namespaceObject.it)("ignores non-root or non-LangGraph start events", ()=>{
|
|
398
|
+
const nestedRunId = (0, agentInvoker_cjs_namespaceObject.trackRootLangGraphRunId)(void 0, {
|
|
399
|
+
event: "on_chain_start",
|
|
400
|
+
name: "LangGraph",
|
|
401
|
+
run_id: "child-run",
|
|
402
|
+
parent_ids: [
|
|
403
|
+
"root-run"
|
|
404
|
+
]
|
|
405
|
+
});
|
|
406
|
+
(0, external_vitest_namespaceObject.expect)(nestedRunId).toBeUndefined();
|
|
407
|
+
const otherChainRunId = (0, agentInvoker_cjs_namespaceObject.trackRootLangGraphRunId)(void 0, {
|
|
408
|
+
event: "on_chain_start",
|
|
409
|
+
name: "CustomChain",
|
|
410
|
+
run_id: "other-run",
|
|
411
|
+
parent_ids: []
|
|
412
|
+
});
|
|
413
|
+
(0, external_vitest_namespaceObject.expect)(otherChainRunId).toBeUndefined();
|
|
414
|
+
});
|
|
415
|
+
(0, external_vitest_namespaceObject.it)("detects terminal root LangGraph on_chain_end by run id", ()=>{
|
|
416
|
+
(0, external_vitest_namespaceObject.expect)((0, agentInvoker_cjs_namespaceObject.isRootLangGraphTerminalEvent)({
|
|
417
|
+
event: "on_chain_end",
|
|
418
|
+
name: "LangGraph",
|
|
419
|
+
run_id: "root-run",
|
|
420
|
+
parent_ids: []
|
|
421
|
+
}, "root-run")).toBe(true);
|
|
422
|
+
});
|
|
423
|
+
(0, external_vitest_namespaceObject.it)("does not treat nested, mismatched, or non-LangGraph chain end as terminal", ()=>{
|
|
424
|
+
(0, external_vitest_namespaceObject.expect)((0, agentInvoker_cjs_namespaceObject.isRootLangGraphTerminalEvent)({
|
|
425
|
+
event: "on_chain_end",
|
|
426
|
+
name: "LangGraph",
|
|
427
|
+
run_id: "child-run",
|
|
428
|
+
parent_ids: [
|
|
429
|
+
"root-run"
|
|
430
|
+
]
|
|
431
|
+
}, "root-run")).toBe(false);
|
|
432
|
+
(0, external_vitest_namespaceObject.expect)((0, agentInvoker_cjs_namespaceObject.isRootLangGraphTerminalEvent)({
|
|
433
|
+
event: "on_chain_end",
|
|
434
|
+
name: "LangGraph",
|
|
435
|
+
run_id: "other-run",
|
|
436
|
+
parent_ids: []
|
|
437
|
+
}, "root-run")).toBe(false);
|
|
438
|
+
(0, external_vitest_namespaceObject.expect)((0, agentInvoker_cjs_namespaceObject.isRootLangGraphTerminalEvent)({
|
|
439
|
+
event: "on_chain_end",
|
|
440
|
+
name: "CustomChain",
|
|
441
|
+
run_id: "root-run",
|
|
442
|
+
parent_ids: []
|
|
443
|
+
}, "root-run")).toBe(false);
|
|
444
|
+
(0, external_vitest_namespaceObject.expect)((0, agentInvoker_cjs_namespaceObject.isRootLangGraphTerminalEvent)({
|
|
445
|
+
event: "on_chain_end",
|
|
446
|
+
name: "LangGraph",
|
|
447
|
+
run_id: "root-run",
|
|
448
|
+
parent_ids: []
|
|
449
|
+
}, void 0)).toBe(false);
|
|
450
|
+
});
|
|
292
451
|
});
|
|
293
452
|
for(var __rspack_i in __webpack_exports__)exports[__rspack_i] = __webpack_exports__[__rspack_i];
|
|
294
453
|
Object.defineProperty(exports, '__esModule', {
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { describe, expect, it } from "vitest";
|
|
2
2
|
import { validateConfig } from "../cli/config/schema.js";
|
|
3
|
-
import { chunkHasAssistantText, configureDeepAgentSummarizationMiddleware, detectToolEventContext, resolveHumanInTheLoopSettings, resolveModelRetryMiddlewareSettings, resolveSummarizationMiddlewareSettings, resolveToolRetryMiddlewareSettings, selectStreamingFallbackText } from "../cli/core/agentInvoker.js";
|
|
3
|
+
import { chunkHasAssistantText, configureDeepAgentSummarizationMiddleware, detectStreamErrorMessage, detectToolEventContext, evaluateStreamingCompletion, isRootLangGraphTerminalEvent, resolveHumanInTheLoopSettings, resolveModelRetryMiddlewareSettings, resolveSummarizationMiddlewareSettings, resolveToolRetryMiddlewareSettings, selectStreamingFallbackText, trackRootLangGraphRunId } from "../cli/core/agentInvoker.js";
|
|
4
4
|
const parseConfig = (input)=>{
|
|
5
5
|
const result = validateConfig(input);
|
|
6
6
|
if (!result.success || !result.data) throw new Error(result.error || "Expected config validation to succeed");
|
|
@@ -206,6 +206,16 @@ describe("detectToolEventContext", ()=>{
|
|
|
206
206
|
toolName: "glob"
|
|
207
207
|
});
|
|
208
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
|
+
});
|
|
209
219
|
it("returns null for non-tool events", ()=>{
|
|
210
220
|
const detected = detectToolEventContext({
|
|
211
221
|
event: "on_chat_model_stream",
|
|
@@ -252,39 +262,188 @@ describe("chunkHasAssistantText", ()=>{
|
|
|
252
262
|
});
|
|
253
263
|
});
|
|
254
264
|
describe("selectStreamingFallbackText", ()=>{
|
|
255
|
-
it("returns
|
|
265
|
+
it("returns assistant text introduced during the current invocation", ()=>{
|
|
256
266
|
const selected = selectStreamingFallbackText([
|
|
257
267
|
{
|
|
258
268
|
role: "assistant",
|
|
259
|
-
|
|
269
|
+
content: "stale message"
|
|
270
|
+
}
|
|
271
|
+
], [
|
|
272
|
+
{
|
|
273
|
+
role: "assistant",
|
|
260
274
|
content: "stale message"
|
|
261
275
|
},
|
|
262
276
|
{
|
|
263
277
|
role: "assistant",
|
|
264
|
-
createdAt: 4500,
|
|
265
278
|
content: "fresh fallback"
|
|
266
279
|
}
|
|
267
|
-
]
|
|
280
|
+
]);
|
|
268
281
|
expect(selected).toBe("fresh fallback");
|
|
269
282
|
});
|
|
270
|
-
it("
|
|
283
|
+
it("returns undefined when no new assistant text is introduced", ()=>{
|
|
271
284
|
const selected = selectStreamingFallbackText([
|
|
272
285
|
{
|
|
273
286
|
role: "assistant",
|
|
274
|
-
|
|
275
|
-
|
|
276
|
-
|
|
287
|
+
content: "existing response"
|
|
288
|
+
}
|
|
289
|
+
], [
|
|
277
290
|
{
|
|
278
291
|
role: "user",
|
|
279
|
-
createdAt: 4900,
|
|
280
292
|
content: "not assistant"
|
|
281
293
|
},
|
|
282
294
|
{
|
|
283
295
|
role: "assistant",
|
|
284
|
-
|
|
285
|
-
content: " "
|
|
296
|
+
content: "existing response"
|
|
286
297
|
}
|
|
287
|
-
]
|
|
298
|
+
]);
|
|
288
299
|
expect(selected).toBeUndefined();
|
|
289
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
|
+
});
|
|
290
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', {
|