@ai-setting/roy-agent-core 1.3.10 → 1.3.14
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/dist/config/index.js +1647 -0
- package/dist/index.js +12579 -89691
- package/package.json +19 -56
- package/src/config/config-component.test.ts +0 -627
- package/src/config/config-component.ts +0 -906
- package/src/config/config-parser.test.ts +0 -319
- package/src/config/config-parser.ts +0 -203
- package/src/config/decentralized-config.test.ts +0 -740
- package/src/config/env-key.ts +0 -210
- package/src/config/env-source.test.ts +0 -252
- package/src/config/env-source.ts +0 -301
- package/src/config/file-source.test.ts +0 -357
- package/src/config/file-source.ts +0 -421
- package/src/config/index.ts +0 -24
- package/src/config/protocol-resolver.test.ts +0 -217
- package/src/config/protocol-resolver.ts +0 -228
- package/src/env/agent/agent-component.abort.test.ts +0 -511
- package/src/env/agent/agent-component.record-session.test.ts +0 -349
- package/src/env/agent/agent-component.test.ts +0 -1389
- package/src/env/agent/agent-component.tool-error.test.ts +0 -327
- package/src/env/agent/agent-component.ts +0 -1711
- package/src/env/agent/agent-config-registration.test.ts +0 -226
- package/src/env/agent/agent-config-registration.ts +0 -46
- package/src/env/agent/agent-reminder-plugin.integration.test.ts +0 -243
- package/src/env/agent/index.ts +0 -10
- package/src/env/agent/summary-agent.parse-hint.test.ts +0 -360
- package/src/env/agent/summary-agent.ts +0 -508
- package/src/env/agent/types.ts +0 -536
- package/src/env/commands/commands-component.test.ts +0 -364
- package/src/env/commands/commands-component.ts +0 -604
- package/src/env/commands/commands-config-registration.test.ts +0 -198
- package/src/env/commands/commands-config-registration.ts +0 -38
- package/src/env/commands/index.ts +0 -21
- package/src/env/commands/parser.test.ts +0 -203
- package/src/env/commands/parser.ts +0 -115
- package/src/env/commands/types.ts +0 -184
- package/src/env/commands-prompt-integration.test.ts +0 -243
- package/src/env/component-env.test.ts +0 -119
- package/src/env/component.ts +0 -335
- package/src/env/constants.test.ts +0 -72
- package/src/env/constants.ts +0 -123
- package/src/env/debug/debug-component.test.ts +0 -114
- package/src/env/debug/debug-component.ts +0 -547
- package/src/env/debug/formatters/index.ts +0 -9
- package/src/env/debug/formatters/repl-formatter.test.ts +0 -139
- package/src/env/debug/formatters/repl-formatter.ts +0 -358
- package/src/env/debug/formatters/trace-formatter.test.ts +0 -119
- package/src/env/debug/formatters/trace-formatter.ts +0 -191
- package/src/env/debug/formatters/tree-formatter.test.ts +0 -107
- package/src/env/debug/formatters/tree-formatter.ts +0 -325
- package/src/env/debug/index.ts +0 -38
- package/src/env/debug/parser/regex-parser.test.ts +0 -201
- package/src/env/debug/parser/regex-parser.ts +0 -196
- package/src/env/debug/parser/span-builder.test.ts +0 -241
- package/src/env/debug/parser/span-builder.ts +0 -386
- package/src/env/debug/reader/log-reader.test.ts +0 -170
- package/src/env/debug/reader/log-reader.ts +0 -186
- package/src/env/debug/reader/span-db-reader.test.ts +0 -118
- package/src/env/debug/reader/span-db-reader.ts +0 -201
- package/src/env/debug/types.test.ts +0 -187
- package/src/env/debug/types.ts +0 -171
- package/src/env/environment-init.test.ts +0 -183
- package/src/env/environment-lifecycle.test.ts +0 -516
- package/src/env/environment-service.test.ts +0 -332
- package/src/env/environment.handle-query.test.ts +0 -96
- package/src/env/environment.test.ts +0 -232
- package/src/env/environment.ts +0 -708
- package/src/env/errors.test.ts +0 -165
- package/src/env/errors.ts +0 -157
- package/src/env/event-source/event-source-agent-handler.test.ts +0 -193
- package/src/env/event-source/event-source-agent-handler.ts +0 -111
- package/src/env/event-source/event-source-component.process-cleanup.test.ts +0 -236
- package/src/env/event-source/event-source-component.stop.test.ts +0 -346
- package/src/env/event-source/event-source-component.test.ts +0 -1207
- package/src/env/event-source/event-source-component.ts +0 -1379
- package/src/env/event-source/event-source-config-registration.test.ts +0 -242
- package/src/env/event-source/event-source-config-registration.ts +0 -37
- package/src/env/event-source/event-source-integration.test.ts +0 -320
- package/src/env/event-source/event-source-platform.test.ts +0 -630
- package/src/env/event-source/types.ts +0 -298
- package/src/env/hook/global-hook-manager.ts +0 -162
- package/src/env/hook/hook-manager.test.ts +0 -374
- package/src/env/hook/hook-manager.ts +0 -309
- package/src/env/hook/index.ts +0 -38
- package/src/env/hook/types.ts +0 -138
- package/src/env/index.ts +0 -144
- package/src/env/interface.ts +0 -203
- package/src/env/llm/hooks.test.ts +0 -293
- package/src/env/llm/hooks.ts +0 -316
- package/src/env/llm/index.ts +0 -61
- package/src/env/llm/invoke-threshold-check.test.ts +0 -88
- package/src/env/llm/invoke-timeout.test.ts +0 -54
- package/src/env/llm/invoke.test.ts +0 -71
- package/src/env/llm/invoke.ts +0 -1039
- package/src/env/llm/llm-config.test.ts +0 -523
- package/src/env/llm/llm.test.ts +0 -233
- package/src/env/llm/llm.ts +0 -568
- package/src/env/llm/provider.test.ts +0 -182
- package/src/env/llm/provider.ts +0 -108
- package/src/env/llm/transform.test.ts +0 -251
- package/src/env/llm/transform.ts +0 -286
- package/src/env/llm/types.test.ts +0 -580
- package/src/env/llm/types.ts +0 -424
- package/src/env/log-trace/decorator-otel.test.ts +0 -182
- package/src/env/log-trace/decorator.ts +0 -230
- package/src/env/log-trace/index.ts +0 -79
- package/src/env/log-trace/log-trace-component.test.ts +0 -242
- package/src/env/log-trace/log-trace-component.ts +0 -497
- package/src/env/log-trace/log-trace-config-registration.test.ts +0 -348
- package/src/env/log-trace/log-trace-config-registration.ts +0 -45
- package/src/env/log-trace/logger.test.ts +0 -149
- package/src/env/log-trace/logger.ts +0 -522
- package/src/env/log-trace/opentelemetry/cli-propagation.test.ts +0 -147
- package/src/env/log-trace/opentelemetry/cli-propagation.ts +0 -194
- package/src/env/log-trace/opentelemetry/integration.test.ts +0 -668
- package/src/env/log-trace/opentelemetry/mod.ts +0 -25
- package/src/env/log-trace/opentelemetry/propagation-env.test.ts +0 -181
- package/src/env/log-trace/opentelemetry/propagation-env.ts +0 -136
- package/src/env/log-trace/opentelemetry/propagation.test.ts +0 -259
- package/src/env/log-trace/opentelemetry/propagation.ts +0 -215
- package/src/env/log-trace/opentelemetry/tracer-provider-context.test.ts +0 -166
- package/src/env/log-trace/opentelemetry/tracer-provider.test.ts +0 -379
- package/src/env/log-trace/opentelemetry/tracer-provider.ts +0 -612
- package/src/env/log-trace/span-storage.test.ts +0 -145
- package/src/env/log-trace/span-storage.ts +0 -230
- package/src/env/log-trace/trace-context.test.ts +0 -187
- package/src/env/log-trace/trace-context.ts +0 -162
- package/src/env/log-trace/types.test.ts +0 -63
- package/src/env/log-trace/types.ts +0 -172
- package/src/env/mcp/README.md +0 -244
- package/src/env/mcp/__integration__/mcp-component.integration.test.ts +0 -373
- package/src/env/mcp/config.test.ts +0 -74
- package/src/env/mcp/config.ts +0 -116
- package/src/env/mcp/index.ts +0 -41
- package/src/env/mcp/loader.test.ts +0 -161
- package/src/env/mcp/loader.ts +0 -209
- package/src/env/mcp/mcp-component.test.ts +0 -111
- package/src/env/mcp/mcp-component.ts +0 -358
- package/src/env/mcp/mcp-config-registration.test.ts +0 -304
- package/src/env/mcp/mcp-config-registration.ts +0 -50
- package/src/env/mcp/scanner.test.ts +0 -170
- package/src/env/mcp/scanner.ts +0 -246
- package/src/env/mcp/tool/adapter.test.ts +0 -520
- package/src/env/mcp/tool/adapter.ts +0 -521
- package/src/env/mcp/tool/index.ts +0 -5
- package/src/env/mcp/types.test.ts +0 -171
- package/src/env/mcp/types.ts +0 -79
- package/src/env/memory/README.md +0 -177
- package/src/env/memory/built-in/index.ts +0 -59
- package/src/env/memory/built-in/recall-memory.ts +0 -103
- package/src/env/memory/built-in/record-memory.ts +0 -148
- package/src/env/memory/index.ts +0 -20
- package/src/env/memory/memory-component.test.ts +0 -239
- package/src/env/memory/memory-component.ts +0 -503
- package/src/env/memory/memory-config-registration.test.ts +0 -67
- package/src/env/memory/memory-config-registration.ts +0 -48
- package/src/env/memory/memory-config.ts +0 -45
- package/src/env/memory/memory-file.test.ts +0 -268
- package/src/env/memory/plugin/index.ts +0 -48
- package/src/env/memory/plugin/memory-agent.test.ts +0 -249
- package/src/env/memory/plugin/memory-agent.ts +0 -365
- package/src/env/memory/plugin/memory-manager.ts +0 -198
- package/src/env/memory/plugin/memory-plugin-agent.test.ts +0 -145
- package/src/env/memory/plugin/memory-plugin.ts +0 -210
- package/src/env/memory/plugin/plugin-simplified.test.ts +0 -51
- package/src/env/memory/plugin/recall-memory.test.ts +0 -106
- package/src/env/memory/plugin/recall-memory.ts +0 -53
- package/src/env/memory/plugin/types.ts +0 -101
- package/src/env/memory/tools/memory-agent-tools.ts +0 -228
- package/src/env/memory/types.ts +0 -85
- package/src/env/paths.ts +0 -118
- package/src/env/prompt/index.ts +0 -18
- package/src/env/prompt/memory-prompts.test.ts +0 -91
- package/src/env/prompt/prompt-component.test.ts +0 -491
- package/src/env/prompt/prompt-component.ts +0 -619
- package/src/env/prompt/prompt-config-registration.test.ts +0 -213
- package/src/env/prompt/prompt-config-registration.ts +0 -39
- package/src/env/prompt/prompts-index.ts +0 -504
- package/src/env/prompt/renderer.ts +0 -67
- package/src/env/prompt/types.ts +0 -136
- package/src/env/session/hooks.ts +0 -18
- package/src/env/session/index.ts +0 -37
- package/src/env/session/search-query-parser.test.ts +0 -425
- package/src/env/session/search-query-parser.ts +0 -171
- package/src/env/session/session-checkpoint.test.ts +0 -523
- package/src/env/session/session-component.extract-recent-messages.test.ts +0 -209
- package/src/env/session/session-component.test.ts +0 -132
- package/src/env/session/session-component.ts +0 -1249
- package/src/env/session/session-config-registration.test.ts +0 -138
- package/src/env/session/session-config-registration.ts +0 -52
- package/src/env/session/session-message-converter.test.ts +0 -763
- package/src/env/session/session-message-converter.ts +0 -415
- package/src/env/session/session-message-e2e.test.ts +0 -448
- package/src/env/session/session-search.test.ts +0 -391
- package/src/env/session/session-store.test.ts +0 -362
- package/src/env/session/session-store.ts +0 -141
- package/src/env/session/storage/index.ts +0 -6
- package/src/env/session/storage/memory.ts +0 -502
- package/src/env/session/storage/sqlite.ts +0 -794
- package/src/env/session/types.ts +0 -742
- package/src/env/skill/config.ts +0 -39
- package/src/env/skill/index.ts +0 -6
- package/src/env/skill/parser.test.ts +0 -116
- package/src/env/skill/parser.ts +0 -77
- package/src/env/skill/scanner.test.ts +0 -211
- package/src/env/skill/scanner.ts +0 -119
- package/src/env/skill/skill-component.test.ts +0 -234
- package/src/env/skill/skill-component.ts +0 -352
- package/src/env/skill/skill-config-registration.test.ts +0 -60
- package/src/env/skill/skill-config-registration.ts +0 -43
- package/src/env/skill/tool/index.ts +0 -1
- package/src/env/skill/tool/skill-tool.test.ts +0 -100
- package/src/env/skill/tool/skill-tool.ts +0 -72
- package/src/env/skill/types.ts +0 -64
- package/src/env/task/delegate/delegate-tool.test.ts +0 -498
- package/src/env/task/delegate/delegate-tool.ts +0 -1014
- package/src/env/task/delegate/index.ts +0 -18
- package/src/env/task/delegate/stop-tool.test.ts +0 -140
- package/src/env/task/delegate/stop-tool.ts +0 -119
- package/src/env/task/delegate/task-events.test.ts +0 -178
- package/src/env/task/delegate/task-events.ts +0 -143
- package/src/env/task/hooks/contexts.test.ts +0 -92
- package/src/env/task/hooks/contexts.ts +0 -192
- package/src/env/task/hooks/index.ts +0 -23
- package/src/env/task/hooks/task-hook-points.test.ts +0 -32
- package/src/env/task/hooks/task-hook-points.ts +0 -54
- package/src/env/task/index.ts +0 -7
- package/src/env/task/plugins/index.ts +0 -13
- package/src/env/task/plugins/task-plugin.test.ts +0 -74
- package/src/env/task/plugins/task-plugin.ts +0 -89
- package/src/env/task/plugins/task-tag-plugin.test.ts +0 -377
- package/src/env/task/plugins/task-tag-plugin.ts +0 -319
- package/src/env/task/plugins/task-workflow-extractor.integration.test.ts +0 -226
- package/src/env/task/plugins/workflow-extractor-agent.test.ts +0 -107
- package/src/env/task/plugins/workflow-extractor-agent.ts +0 -225
- package/src/env/task/storage/index.ts +0 -6
- package/src/env/task/storage/sqlite-task-store.test.ts +0 -283
- package/src/env/task/storage/sqlite-task-store.ts +0 -903
- package/src/env/task/storage/task-search.test.ts +0 -291
- package/src/env/task/tag-service.test.ts +0 -198
- package/src/env/task/tag-service.ts +0 -264
- package/src/env/task/task-component.test.ts +0 -193
- package/src/env/task/task-component.ts +0 -658
- package/src/env/task/task-config-registration.test.ts +0 -57
- package/src/env/task/task-config-registration.ts +0 -37
- package/src/env/task/task-types.test.ts +0 -137
- package/src/env/task/tools/complete-tool.ts +0 -44
- package/src/env/task/tools/create-tool.ts +0 -49
- package/src/env/task/tools/delete-tool.ts +0 -43
- package/src/env/task/tools/get-tool.ts +0 -59
- package/src/env/task/tools/index.ts +0 -10
- package/src/env/task/tools/list-tool.ts +0 -40
- package/src/env/task/tools/operation/create-tool.ts +0 -48
- package/src/env/task/tools/operation/delete-tool.ts +0 -43
- package/src/env/task/tools/operation/get-tool.ts +0 -43
- package/src/env/task/tools/operation/index.ts +0 -9
- package/src/env/task/tools/operation/list-tool.ts +0 -40
- package/src/env/task/tools/operation/operation-tools.test.ts +0 -274
- package/src/env/task/tools/operation/operation-types.ts +0 -75
- package/src/env/task/tools/operation/update-tool.ts +0 -47
- package/src/env/task/tools/task-tools.test.ts +0 -203
- package/src/env/task/tools/task-types.test.ts +0 -75
- package/src/env/task/tools/task-types.ts +0 -68
- package/src/env/task/tools/update-tool.ts +0 -70
- package/src/env/task/types.ts +0 -160
- package/src/env/tool/built-in/bash.ts +0 -201
- package/src/env/tool/built-in/echo.ts +0 -29
- package/src/env/tool/built-in/edit-file.test.ts +0 -136
- package/src/env/tool/built-in/edit-file.ts +0 -92
- package/src/env/tool/built-in/glob.test.ts +0 -94
- package/src/env/tool/built-in/glob.ts +0 -65
- package/src/env/tool/built-in/grep.test.ts +0 -122
- package/src/env/tool/built-in/grep.ts +0 -108
- package/src/env/tool/built-in/index.ts +0 -44
- package/src/env/tool/built-in/read-file.test.ts +0 -84
- package/src/env/tool/built-in/read-file.ts +0 -75
- package/src/env/tool/built-in/write-file.test.ts +0 -119
- package/src/env/tool/built-in/write-file.ts +0 -68
- package/src/env/tool/index.ts +0 -24
- package/src/env/tool/registry.test.ts +0 -257
- package/src/env/tool/registry.ts +0 -167
- package/src/env/tool/tool-component.test.ts +0 -559
- package/src/env/tool/tool-component.ts +0 -563
- package/src/env/tool/tool-config-registration.test.ts +0 -249
- package/src/env/tool/tool-config-registration.ts +0 -46
- package/src/env/tool/types.ts +0 -267
- package/src/env/tool/validator.test.ts +0 -143
- package/src/env/tool/validator.ts +0 -44
- package/src/env/types.ts +0 -180
- package/src/env/workflow/ask-user-tool-registration.test.ts +0 -216
- package/src/env/workflow/complex-workflow.integration.test.ts +0 -1900
- package/src/env/workflow/decorators/decorator-node.ts +0 -229
- package/src/env/workflow/decorators/decorator.test.ts +0 -196
- package/src/env/workflow/decorators/edge.ts +0 -82
- package/src/env/workflow/decorators/index.ts +0 -31
- package/src/env/workflow/decorators/node-as.ts +0 -98
- package/src/env/workflow/decorators/workflow.ts +0 -54
- package/src/env/workflow/engine/dag-manager.test.ts +0 -570
- package/src/env/workflow/engine/dag-manager.ts +0 -594
- package/src/env/workflow/engine/engine.ts +0 -1422
- package/src/env/workflow/engine/event-bus.test.ts +0 -359
- package/src/env/workflow/engine/event-bus.ts +0 -156
- package/src/env/workflow/engine/executor-agent-session.test.ts +0 -84
- package/src/env/workflow/engine/executor.test.ts +0 -619
- package/src/env/workflow/engine/executor.ts +0 -593
- package/src/env/workflow/engine/index.ts +0 -24
- package/src/env/workflow/engine/node-registry.test.ts +0 -560
- package/src/env/workflow/engine/node-registry.ts +0 -289
- package/src/env/workflow/engine/resume-removed.test.ts +0 -22
- package/src/env/workflow/engine/scheduler.test.ts +0 -715
- package/src/env/workflow/engine/scheduler.ts +0 -318
- package/src/env/workflow/engine/workflow-engine.test.ts +0 -815
- package/src/env/workflow/extractor/workflow-converter.ts +0 -306
- package/src/env/workflow/fixtures.ts +0 -380
- package/src/env/workflow/index.ts +0 -38
- package/src/env/workflow/integration/run-resume-unified.test.ts +0 -186
- package/src/env/workflow/integration/service-integration.test.ts +0 -267
- package/src/env/workflow/metadata/keys.ts +0 -12
- package/src/env/workflow/nodes/agent-component-adapter.test.ts +0 -318
- package/src/env/workflow/nodes/agent-component-adapter.ts +0 -448
- package/src/env/workflow/nodes/agent-node.test.ts +0 -371
- package/src/env/workflow/nodes/agent-node.ts +0 -598
- package/src/env/workflow/nodes/ask-user-node.ts +0 -113
- package/src/env/workflow/nodes/condition-node.ts +0 -200
- package/src/env/workflow/nodes/index.ts +0 -9
- package/src/env/workflow/nodes/merge-node.ts +0 -141
- package/src/env/workflow/nodes/skill-node.test.ts +0 -253
- package/src/env/workflow/nodes/skill-node.ts +0 -393
- package/src/env/workflow/nodes/tool-node.test.ts +0 -251
- package/src/env/workflow/nodes/tool-node.ts +0 -493
- package/src/env/workflow/nodes/workflow-llm-history.test.ts +0 -455
- package/src/env/workflow/nodes/workflow-node.test.ts +0 -315
- package/src/env/workflow/nodes/workflow-node.ts +0 -311
- package/src/env/workflow/service/index.ts +0 -27
- package/src/env/workflow/service/registry.test.ts +0 -133
- package/src/env/workflow/service/registry.ts +0 -71
- package/src/env/workflow/service/workflow-service.test.ts +0 -310
- package/src/env/workflow/service/workflow-service.ts +0 -393
- package/src/env/workflow/storage/index.ts +0 -28
- package/src/env/workflow/storage/mock-repositories.ts +0 -385
- package/src/env/workflow/storage/sqlite.test.ts +0 -179
- package/src/env/workflow/storage/sqlite.ts +0 -163
- package/src/env/workflow/storage/workflow-repo.test.ts +0 -780
- package/src/env/workflow/storage/workflow-repo.ts +0 -342
- package/src/env/workflow/tools/ask-user-tool.ts +0 -82
- package/src/env/workflow/tools/index.ts +0 -26
- package/src/env/workflow/tools/run-workflow.test.ts +0 -352
- package/src/env/workflow/tools/run-workflow.ts +0 -214
- package/src/env/workflow/types/context.ts +0 -18
- package/src/env/workflow/types/decorators-types.ts +0 -198
- package/src/env/workflow/types/event.test.ts +0 -515
- package/src/env/workflow/types/event.ts +0 -193
- package/src/env/workflow/types/index.ts +0 -49
- package/src/env/workflow/types/run.test.ts +0 -437
- package/src/env/workflow/types/run.ts +0 -173
- package/src/env/workflow/types/workflow-hil.ts +0 -114
- package/src/env/workflow/types/workflow-message.test.ts +0 -138
- package/src/env/workflow/types/workflow-message.ts +0 -196
- package/src/env/workflow/types/workflow-session.test.ts +0 -95
- package/src/env/workflow/types/workflow-session.ts +0 -59
- package/src/env/workflow/types/workflow.test.ts +0 -495
- package/src/env/workflow/types/workflow.ts +0 -195
- package/src/env/workflow/types_compat.ts +0 -51
- package/src/env/workflow/utils/create-workflow.ts +0 -47
- package/src/env/workflow/utils/execution-state.ts +0 -245
- package/src/env/workflow/utils/index.ts +0 -18
- package/src/env/workflow/utils/node-registry-helper.ts +0 -58
- package/src/env/workflow/utils/recovery-validator.test.ts +0 -460
- package/src/env/workflow/utils/recovery-validator.ts +0 -377
- package/src/env/workflow/utils/session-parser.test.ts +0 -111
- package/src/env/workflow/utils/session-parser.ts +0 -94
- package/src/env/workflow/utils/session-recovery.test.ts +0 -334
- package/src/env/workflow/utils/session-recovery.ts +0 -188
- package/src/env/workflow/utils/template-resolver.test.ts +0 -258
- package/src/env/workflow/utils/template-resolver.ts +0 -436
- package/src/env/workflow/utils/validation-rules.ts +0 -149
- package/src/env/workflow/workflow-component.ts +0 -544
- package/src/index.ts +0 -422
- package/src/utils/id.ts +0 -21
|
@@ -1,315 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect, beforeEach, jest } from 'vitest';
|
|
2
|
-
import { WorkflowNode, WorkflowRunner, WorkflowRunResult } from './workflow-node';
|
|
3
|
-
import type { NodeDefinition, NodeExecutionContext } from '../types';
|
|
4
|
-
|
|
5
|
-
// Mock EventBus
|
|
6
|
-
const mockEventBus = {
|
|
7
|
-
emit: jest.fn(),
|
|
8
|
-
on: jest.fn(),
|
|
9
|
-
off: jest.fn(),
|
|
10
|
-
} as any;
|
|
11
|
-
|
|
12
|
-
// Mock WorkflowRunner
|
|
13
|
-
const createMockRunner = () => ({
|
|
14
|
-
run: jest.fn<(workflowName: string, input?: any) => Promise<WorkflowRunResult>>(),
|
|
15
|
-
getRun: jest.fn<(runId: string) => Promise<WorkflowRunResult | null>>(),
|
|
16
|
-
});
|
|
17
|
-
|
|
18
|
-
describe('WorkflowNode', () => {
|
|
19
|
-
let nodeDefinition: NodeDefinition;
|
|
20
|
-
let executionContext: NodeExecutionContext;
|
|
21
|
-
let mockRunner: ReturnType<typeof createMockRunner>;
|
|
22
|
-
|
|
23
|
-
beforeEach(() => {
|
|
24
|
-
jest.clearAllMocks();
|
|
25
|
-
|
|
26
|
-
nodeDefinition = {
|
|
27
|
-
id: 'workflow-node-1',
|
|
28
|
-
type: 'workflow',
|
|
29
|
-
name: 'Nested Workflow Node',
|
|
30
|
-
config: {
|
|
31
|
-
workflow_name: 'sub-workflow',
|
|
32
|
-
input: { message: 'hello from parent' },
|
|
33
|
-
wait_complete: true,
|
|
34
|
-
},
|
|
35
|
-
};
|
|
36
|
-
|
|
37
|
-
executionContext = {
|
|
38
|
-
sessionId: 'session-1',
|
|
39
|
-
nodeOutputs: {},
|
|
40
|
-
input: {},
|
|
41
|
-
metadata: {},
|
|
42
|
-
};
|
|
43
|
-
|
|
44
|
-
mockRunner = createMockRunner();
|
|
45
|
-
});
|
|
46
|
-
|
|
47
|
-
describe('constructor', () => {
|
|
48
|
-
it('should create a WorkflowNode with correct type', () => {
|
|
49
|
-
const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
|
|
50
|
-
expect(workflowNode.type).toBe('workflow');
|
|
51
|
-
expect(workflowNode.id).toBe('workflow-node-1');
|
|
52
|
-
});
|
|
53
|
-
|
|
54
|
-
it('should create WorkflowNode with minimal config', () => {
|
|
55
|
-
const minimalDef: NodeDefinition = {
|
|
56
|
-
id: 'minimal-node',
|
|
57
|
-
type: 'workflow',
|
|
58
|
-
config: {
|
|
59
|
-
workflow_name: 'simple-workflow',
|
|
60
|
-
},
|
|
61
|
-
};
|
|
62
|
-
|
|
63
|
-
const workflowNode = new WorkflowNode(minimalDef, mockRunner);
|
|
64
|
-
expect(workflowNode.type).toBe('workflow');
|
|
65
|
-
expect(workflowNode.id).toBe('minimal-node');
|
|
66
|
-
});
|
|
67
|
-
});
|
|
68
|
-
|
|
69
|
-
describe('execute', () => {
|
|
70
|
-
it('should run sub-workflow successfully', async () => {
|
|
71
|
-
const completedResult: WorkflowRunResult = {
|
|
72
|
-
runId: 'run_123',
|
|
73
|
-
status: 'completed',
|
|
74
|
-
output: { result: 'nested workflow done' },
|
|
75
|
-
};
|
|
76
|
-
|
|
77
|
-
mockRunner.run.mockResolvedValue(completedResult);
|
|
78
|
-
|
|
79
|
-
const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
|
|
80
|
-
const result = await workflowNode.execute(executionContext);
|
|
81
|
-
|
|
82
|
-
expect(result.success).toBe(true);
|
|
83
|
-
expect(result.output).toEqual({ result: 'nested workflow done' });
|
|
84
|
-
expect(result.error).toBeUndefined();
|
|
85
|
-
expect(mockRunner.run).toHaveBeenCalledWith('sub-workflow', {
|
|
86
|
-
message: 'hello from parent',
|
|
87
|
-
});
|
|
88
|
-
});
|
|
89
|
-
|
|
90
|
-
it('should pass input to sub-workflow', async () => {
|
|
91
|
-
const customInput = { param1: 'value1', param2: 123, nested: { data: 'test' } };
|
|
92
|
-
nodeDefinition.config.input = customInput;
|
|
93
|
-
|
|
94
|
-
mockRunner.run.mockResolvedValue({
|
|
95
|
-
runId: 'run_456',
|
|
96
|
-
status: 'completed',
|
|
97
|
-
output: { processed: true },
|
|
98
|
-
});
|
|
99
|
-
|
|
100
|
-
const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
|
|
101
|
-
await workflowNode.execute(executionContext);
|
|
102
|
-
|
|
103
|
-
expect(mockRunner.run).toHaveBeenCalledWith('sub-workflow', customInput);
|
|
104
|
-
});
|
|
105
|
-
|
|
106
|
-
it('should wait for completion when wait_complete=true', async () => {
|
|
107
|
-
nodeDefinition.config.wait_complete = true;
|
|
108
|
-
|
|
109
|
-
mockRunner.run.mockResolvedValue({
|
|
110
|
-
runId: 'run_wait',
|
|
111
|
-
status: 'completed',
|
|
112
|
-
output: { waited: true },
|
|
113
|
-
});
|
|
114
|
-
|
|
115
|
-
const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
|
|
116
|
-
const result = await workflowNode.execute(executionContext);
|
|
117
|
-
|
|
118
|
-
expect(result.success).toBe(true);
|
|
119
|
-
expect(result.output).toEqual({ waited: true });
|
|
120
|
-
// When wait_complete=true, run should return completed status
|
|
121
|
-
expect(mockRunner.run).toHaveBeenCalled();
|
|
122
|
-
});
|
|
123
|
-
|
|
124
|
-
it('should not wait for completion when wait_complete=false', async () => {
|
|
125
|
-
nodeDefinition.config.wait_complete = false;
|
|
126
|
-
|
|
127
|
-
mockRunner.run.mockResolvedValue({
|
|
128
|
-
runId: 'run_fire',
|
|
129
|
-
status: 'completed', // Fire-and-forget returns immediately
|
|
130
|
-
output: { runId: 'run_fire' },
|
|
131
|
-
});
|
|
132
|
-
|
|
133
|
-
const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
|
|
134
|
-
const result = await workflowNode.execute(executionContext);
|
|
135
|
-
|
|
136
|
-
expect(result.success).toBe(true);
|
|
137
|
-
expect(result.output).toBeDefined();
|
|
138
|
-
// Should have called run
|
|
139
|
-
expect(mockRunner.run).toHaveBeenCalled();
|
|
140
|
-
});
|
|
141
|
-
|
|
142
|
-
it('should handle sub-workflow failure', async () => {
|
|
143
|
-
mockRunner.run.mockResolvedValue({
|
|
144
|
-
runId: 'run_fail',
|
|
145
|
-
status: 'failed',
|
|
146
|
-
error: 'Sub-workflow execution failed',
|
|
147
|
-
});
|
|
148
|
-
|
|
149
|
-
const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
|
|
150
|
-
const result = await workflowNode.execute(executionContext);
|
|
151
|
-
|
|
152
|
-
expect(result.success).toBe(false);
|
|
153
|
-
expect(result.error).toBeDefined();
|
|
154
|
-
expect(result.error).toContain('Sub-workflow execution failed');
|
|
155
|
-
});
|
|
156
|
-
|
|
157
|
-
it('should handle sub-workflow not found', async () => {
|
|
158
|
-
mockRunner.run.mockRejectedValue(new Error('Workflow not found: non-existent-workflow'));
|
|
159
|
-
|
|
160
|
-
nodeDefinition.config.workflow_name = 'non-existent-workflow';
|
|
161
|
-
|
|
162
|
-
const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
|
|
163
|
-
const result = await workflowNode.execute(executionContext);
|
|
164
|
-
|
|
165
|
-
expect(result.success).toBe(false);
|
|
166
|
-
expect(result.error).toBeDefined();
|
|
167
|
-
expect(result.error.message).toContain('Workflow not found');
|
|
168
|
-
});
|
|
169
|
-
|
|
170
|
-
it('should record execution duration', async () => {
|
|
171
|
-
mockRunner.run.mockImplementation(async () => {
|
|
172
|
-
await new Promise(resolve => setTimeout(resolve, 10));
|
|
173
|
-
return {
|
|
174
|
-
runId: 'run_duration',
|
|
175
|
-
status: 'completed',
|
|
176
|
-
output: { done: true },
|
|
177
|
-
};
|
|
178
|
-
});
|
|
179
|
-
|
|
180
|
-
const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
|
|
181
|
-
const result = await workflowNode.execute(executionContext);
|
|
182
|
-
|
|
183
|
-
expect(result.duration).toBeGreaterThanOrEqual(0);
|
|
184
|
-
});
|
|
185
|
-
|
|
186
|
-
it('should resolve template references from previous outputs', async () => {
|
|
187
|
-
// Set up previous outputs from other nodes
|
|
188
|
-
// Note: The Map format stores raw outputs directly, so path is {{nodeId.property}}
|
|
189
|
-
executionContext.previousOutputs = new Map([
|
|
190
|
-
['node-0', { value: 'from-node-0' }]
|
|
191
|
-
]);
|
|
192
|
-
|
|
193
|
-
// Create node with template reference input
|
|
194
|
-
const templateNodeDef: NodeDefinition = {
|
|
195
|
-
id: 'workflow-node-1',
|
|
196
|
-
type: 'workflow',
|
|
197
|
-
name: 'Nested Workflow Node',
|
|
198
|
-
config: {
|
|
199
|
-
workflow_name: 'sub-workflow',
|
|
200
|
-
input: {
|
|
201
|
-
message: '{{node-0.value}}',
|
|
202
|
-
},
|
|
203
|
-
},
|
|
204
|
-
};
|
|
205
|
-
|
|
206
|
-
mockRunner.run.mockResolvedValue({
|
|
207
|
-
runId: 'run_template',
|
|
208
|
-
status: 'completed',
|
|
209
|
-
output: { received: 'from-node-0' },
|
|
210
|
-
});
|
|
211
|
-
|
|
212
|
-
const workflowNode = new WorkflowNode(templateNodeDef, mockRunner);
|
|
213
|
-
await workflowNode.execute(executionContext);
|
|
214
|
-
|
|
215
|
-
// The input should have template references resolved
|
|
216
|
-
expect(mockRunner.run).toHaveBeenCalledWith('sub-workflow', {
|
|
217
|
-
message: 'from-node-0',
|
|
218
|
-
});
|
|
219
|
-
});
|
|
220
|
-
|
|
221
|
-
it('should handle unresolved template references gracefully', async () => {
|
|
222
|
-
nodeDefinition.config.input = {
|
|
223
|
-
message: '{{non-existent.value}}',
|
|
224
|
-
};
|
|
225
|
-
|
|
226
|
-
mockRunner.run.mockResolvedValue({
|
|
227
|
-
runId: 'run_unresolved',
|
|
228
|
-
status: 'completed',
|
|
229
|
-
output: {},
|
|
230
|
-
});
|
|
231
|
-
|
|
232
|
-
const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
|
|
233
|
-
await workflowNode.execute(executionContext);
|
|
234
|
-
|
|
235
|
-
// Should pass through unresolved template as-is
|
|
236
|
-
expect(mockRunner.run).toHaveBeenCalledWith('sub-workflow', {
|
|
237
|
-
message: '{{non-existent.value}}',
|
|
238
|
-
});
|
|
239
|
-
});
|
|
240
|
-
|
|
241
|
-
it('should throw error when workflow_name is missing', async () => {
|
|
242
|
-
nodeDefinition.config = {}; // No workflow_name specified
|
|
243
|
-
|
|
244
|
-
const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
|
|
245
|
-
const result = await workflowNode.execute(executionContext);
|
|
246
|
-
|
|
247
|
-
expect(result.success).toBe(false);
|
|
248
|
-
expect(result.error).toBeDefined();
|
|
249
|
-
expect(result.error.message).toContain('workflow_name');
|
|
250
|
-
});
|
|
251
|
-
|
|
252
|
-
it('should pass metadata to sub-workflow', async () => {
|
|
253
|
-
executionContext.metadata = {
|
|
254
|
-
parentRunId: 'parent-123',
|
|
255
|
-
correlationId: 'corr-456',
|
|
256
|
-
};
|
|
257
|
-
|
|
258
|
-
mockRunner.run.mockResolvedValue({
|
|
259
|
-
runId: 'run_meta',
|
|
260
|
-
status: 'completed',
|
|
261
|
-
output: {},
|
|
262
|
-
});
|
|
263
|
-
|
|
264
|
-
const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
|
|
265
|
-
await workflowNode.execute(executionContext);
|
|
266
|
-
|
|
267
|
-
expect(mockRunner.run).toHaveBeenCalledWith('sub-workflow', {
|
|
268
|
-
message: 'hello from parent',
|
|
269
|
-
});
|
|
270
|
-
});
|
|
271
|
-
|
|
272
|
-
it('should handle stopped sub-workflow', async () => {
|
|
273
|
-
mockRunner.run.mockResolvedValue({
|
|
274
|
-
runId: 'run_stopped',
|
|
275
|
-
status: 'stopped',
|
|
276
|
-
error: 'Workflow was stopped',
|
|
277
|
-
});
|
|
278
|
-
|
|
279
|
-
const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
|
|
280
|
-
const result = await workflowNode.execute(executionContext);
|
|
281
|
-
|
|
282
|
-
expect(result.success).toBe(false);
|
|
283
|
-
expect(result.error).toBeDefined();
|
|
284
|
-
});
|
|
285
|
-
});
|
|
286
|
-
|
|
287
|
-
describe('WorkflowRunner interface', () => {
|
|
288
|
-
it('should use run method to start workflows', async () => {
|
|
289
|
-
mockRunner.run.mockResolvedValue({
|
|
290
|
-
runId: 'run_interface',
|
|
291
|
-
status: 'completed',
|
|
292
|
-
output: { test: true },
|
|
293
|
-
});
|
|
294
|
-
|
|
295
|
-
const workflowNode = new WorkflowNode(nodeDefinition, mockRunner);
|
|
296
|
-
await workflowNode.execute(executionContext);
|
|
297
|
-
|
|
298
|
-
expect(mockRunner.run).toHaveBeenCalledTimes(1);
|
|
299
|
-
});
|
|
300
|
-
|
|
301
|
-
it('should use getRun method for status checking', async () => {
|
|
302
|
-
// This test verifies the interface contract
|
|
303
|
-
mockRunner.getRun.mockResolvedValue({
|
|
304
|
-
runId: 'run_status',
|
|
305
|
-
status: 'completed',
|
|
306
|
-
output: {},
|
|
307
|
-
});
|
|
308
|
-
|
|
309
|
-
// The interface supports getRun for status checking
|
|
310
|
-
const status = await mockRunner.getRun('run_status');
|
|
311
|
-
expect(status).toBeDefined();
|
|
312
|
-
expect(status?.runId).toBe('run_status');
|
|
313
|
-
});
|
|
314
|
-
});
|
|
315
|
-
});
|
|
@@ -1,311 +0,0 @@
|
|
|
1
|
-
import type { Node, NodeDefinition, NodeExecutionContext, NodeExecutionResult } from '../types';
|
|
2
|
-
|
|
3
|
-
/**
|
|
4
|
-
* WorkflowRunner interface for running sub-workflows
|
|
5
|
-
*/
|
|
6
|
-
export interface WorkflowRunner {
|
|
7
|
-
/**
|
|
8
|
-
* Run a workflow by name
|
|
9
|
-
* @param workflowName - Name of the workflow to run
|
|
10
|
-
* @param input - Optional input to pass to the workflow
|
|
11
|
-
* @returns Promise resolving to the workflow run result
|
|
12
|
-
*/
|
|
13
|
-
run(workflowName: string, input?: any): Promise<WorkflowRunResult>;
|
|
14
|
-
|
|
15
|
-
/**
|
|
16
|
-
* Get the status of a workflow run
|
|
17
|
-
* @param runId - The run ID to check
|
|
18
|
-
* @returns Promise resolving to the run result or null if not found
|
|
19
|
-
*/
|
|
20
|
-
getRun(runId: string): Promise<WorkflowRunResult | null>;
|
|
21
|
-
}
|
|
22
|
-
|
|
23
|
-
/**
|
|
24
|
-
* Result from a workflow run
|
|
25
|
-
*/
|
|
26
|
-
export interface WorkflowRunResult {
|
|
27
|
-
/** Unique run ID */
|
|
28
|
-
runId: string;
|
|
29
|
-
/** Final status of the run */
|
|
30
|
-
status: 'completed' | 'failed' | 'stopped';
|
|
31
|
-
/** Output from the workflow (if completed) */
|
|
32
|
-
output?: any;
|
|
33
|
-
/** Error message (if failed or stopped) */
|
|
34
|
-
error?: string;
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
/**
|
|
38
|
-
* WorkflowNode configuration
|
|
39
|
-
*/
|
|
40
|
-
export interface WorkflowNodeConfig {
|
|
41
|
-
/** Name of the workflow to invoke */
|
|
42
|
-
workflow_name: string;
|
|
43
|
-
/** Input to pass to the sub-workflow (supports template strings) */
|
|
44
|
-
input?: Record<string, unknown>;
|
|
45
|
-
/** Whether to wait for completion (default: true) */
|
|
46
|
-
wait_complete?: boolean;
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
/**
|
|
50
|
-
* WorkflowNode executes a nested workflow as a node.
|
|
51
|
-
*
|
|
52
|
-
* This allows one workflow to invoke another workflow as a single node,
|
|
53
|
-
* enabling workflow composition and reusability.
|
|
54
|
-
*
|
|
55
|
-
* Features:
|
|
56
|
-
* - Pass input to sub-workflow (with template resolution)
|
|
57
|
-
* - Wait for completion or fire-and-forget
|
|
58
|
-
* - Handle sub-workflow errors
|
|
59
|
-
* - Track execution duration
|
|
60
|
-
*/
|
|
61
|
-
export class WorkflowNode implements Node {
|
|
62
|
-
readonly type = 'workflow';
|
|
63
|
-
readonly id: string;
|
|
64
|
-
|
|
65
|
-
private readonly config: WorkflowNodeConfig;
|
|
66
|
-
|
|
67
|
-
constructor(
|
|
68
|
-
definition: NodeDefinition,
|
|
69
|
-
private workflowRunner: WorkflowRunner
|
|
70
|
-
) {
|
|
71
|
-
this.id = definition.id;
|
|
72
|
-
this.config = (definition.config ?? {}) as any as WorkflowNodeConfig;
|
|
73
|
-
}
|
|
74
|
-
|
|
75
|
-
async execute(context: NodeExecutionContext): Promise<NodeExecutionResult> {
|
|
76
|
-
const startTime = Date.now();
|
|
77
|
-
|
|
78
|
-
try {
|
|
79
|
-
// 1. Get workflow name from config
|
|
80
|
-
const workflowName = this.config.workflow_name;
|
|
81
|
-
|
|
82
|
-
if (!workflowName) {
|
|
83
|
-
throw new Error('workflow_name is required in config. Please specify config.workflow_name in the node definition.');
|
|
84
|
-
}
|
|
85
|
-
|
|
86
|
-
// 2. Resolve input template (may reference previous outputs)
|
|
87
|
-
// Support both Map (previousOutputs) and Record (nodeOutputs) formats
|
|
88
|
-
const previousOutputs = context.previousOutputs ?? this.convertToMap((context.nodeOutputs ?? {}) as any);
|
|
89
|
-
const resolvedInput = this.resolveInput(
|
|
90
|
-
this.config.input ?? {},
|
|
91
|
-
previousOutputs
|
|
92
|
-
);
|
|
93
|
-
|
|
94
|
-
// 3. Run sub-workflow via workflowRunner
|
|
95
|
-
const result = await this.workflowRunner.run(workflowName, resolvedInput);
|
|
96
|
-
|
|
97
|
-
// 4. Determine success based on status
|
|
98
|
-
const success = result.status === 'completed';
|
|
99
|
-
|
|
100
|
-
// 5. Return result
|
|
101
|
-
return {
|
|
102
|
-
success,
|
|
103
|
-
output: result.output,
|
|
104
|
-
error: success ? undefined : result.error,
|
|
105
|
-
duration: Date.now() - startTime,
|
|
106
|
-
};
|
|
107
|
-
} catch (error) {
|
|
108
|
-
return {
|
|
109
|
-
success: false,
|
|
110
|
-
output: undefined,
|
|
111
|
-
error: error instanceof Error ? error : new Error(String(error)),
|
|
112
|
-
duration: Date.now() - startTime,
|
|
113
|
-
};
|
|
114
|
-
}
|
|
115
|
-
}
|
|
116
|
-
|
|
117
|
-
/**
|
|
118
|
-
* Convert Record<string, NodeExecutionResult> to Map<string, any>
|
|
119
|
-
* for compatibility with resolvePath that expects Map format.
|
|
120
|
-
*
|
|
121
|
-
* In the Record format, the value is NodeExecutionResult (with output field).
|
|
122
|
-
* In the Map format, the value is the raw output directly.
|
|
123
|
-
*/
|
|
124
|
-
private convertToMap(nodeOutputs: Record<string, NodeExecutionResult>): Map<string, any> {
|
|
125
|
-
const map = new Map<string, any>();
|
|
126
|
-
for (const [nodeId, result] of Object.entries(nodeOutputs)) {
|
|
127
|
-
// Store the entire result for access to both output and success
|
|
128
|
-
map.set(nodeId, result.output);
|
|
129
|
-
}
|
|
130
|
-
return map;
|
|
131
|
-
}
|
|
132
|
-
|
|
133
|
-
/**
|
|
134
|
-
* Resolve input template references.
|
|
135
|
-
* Templates are in the format {{nodeId.output.path}}
|
|
136
|
-
*
|
|
137
|
-
* @param input - The input object that may contain template strings
|
|
138
|
-
* @param previousOutputs - Map of previous node outputs (value is raw output)
|
|
139
|
-
* @returns The input with template references resolved
|
|
140
|
-
*/
|
|
141
|
-
private resolveInput(
|
|
142
|
-
input: Record<string, unknown>,
|
|
143
|
-
previousOutputs: Map<string, any>
|
|
144
|
-
): Record<string, unknown> {
|
|
145
|
-
const resolved: Record<string, unknown> = {};
|
|
146
|
-
|
|
147
|
-
for (const [key, value] of Object.entries(input)) {
|
|
148
|
-
resolved[key] = this.resolveValue(value, previousOutputs);
|
|
149
|
-
}
|
|
150
|
-
|
|
151
|
-
return resolved;
|
|
152
|
-
}
|
|
153
|
-
|
|
154
|
-
/**
|
|
155
|
-
* Resolve a value that may contain template references.
|
|
156
|
-
*/
|
|
157
|
-
private resolveValue(value: unknown, previousOutputs: Map<string, any>): unknown {
|
|
158
|
-
if (typeof value === 'string') {
|
|
159
|
-
return this.resolveStringTemplate(value, previousOutputs);
|
|
160
|
-
}
|
|
161
|
-
|
|
162
|
-
if (Array.isArray(value)) {
|
|
163
|
-
return value.map(item => this.resolveValue(item, previousOutputs));
|
|
164
|
-
}
|
|
165
|
-
|
|
166
|
-
if (value !== null && typeof value === 'object') {
|
|
167
|
-
const resolved: Record<string, unknown> = {};
|
|
168
|
-
for (const [k, v] of Object.entries(value as Record<string, unknown>)) {
|
|
169
|
-
resolved[k] = this.resolveValue(v, previousOutputs);
|
|
170
|
-
}
|
|
171
|
-
return resolved;
|
|
172
|
-
}
|
|
173
|
-
|
|
174
|
-
// Return primitive values as-is
|
|
175
|
-
return value;
|
|
176
|
-
}
|
|
177
|
-
|
|
178
|
-
/**
|
|
179
|
-
* Resolve template references in a single string.
|
|
180
|
-
* Supports format: {{nodeId.output.path}}
|
|
181
|
-
*/
|
|
182
|
-
private resolveStringTemplate(
|
|
183
|
-
template: string,
|
|
184
|
-
previousOutputs: Map<string, any>
|
|
185
|
-
): string {
|
|
186
|
-
// Match template patterns like {{nodeId}} or {{nodeId.output.value}}
|
|
187
|
-
const templatePattern = /\{\{([^}]+)\}\}/g;
|
|
188
|
-
|
|
189
|
-
return template.replace(templatePattern, (match, path) => {
|
|
190
|
-
const trimmedPath = path.trim();
|
|
191
|
-
|
|
192
|
-
// Try to resolve the path from previous outputs
|
|
193
|
-
const value = this.resolvePath(trimmedPath, previousOutputs);
|
|
194
|
-
|
|
195
|
-
if (value !== undefined) {
|
|
196
|
-
return this.stringifyValue(value, match);
|
|
197
|
-
}
|
|
198
|
-
|
|
199
|
-
// If not found, return the original template string
|
|
200
|
-
return match;
|
|
201
|
-
});
|
|
202
|
-
}
|
|
203
|
-
|
|
204
|
-
/**
|
|
205
|
-
* Resolve a dot-notation path from previous outputs.
|
|
206
|
-
* First segment is the node ID, rest is the path within that node's output.
|
|
207
|
-
*/
|
|
208
|
-
private resolvePath(
|
|
209
|
-
path: string,
|
|
210
|
-
previousOutputs: Map<string, any>
|
|
211
|
-
): unknown {
|
|
212
|
-
const segments = path.split('.');
|
|
213
|
-
|
|
214
|
-
if (segments.length === 0) {
|
|
215
|
-
return undefined;
|
|
216
|
-
}
|
|
217
|
-
|
|
218
|
-
const nodeId = segments[0];
|
|
219
|
-
let nodeOutput = previousOutputs.get(nodeId);
|
|
220
|
-
|
|
221
|
-
if (nodeOutput === undefined) {
|
|
222
|
-
return undefined;
|
|
223
|
-
}
|
|
224
|
-
|
|
225
|
-
// Extract from wrapper formats (DecoratorNode: {success, output} and AgentNode: {result, metadata})
|
|
226
|
-
nodeOutput = this.extractFromWrapper(nodeOutput);
|
|
227
|
-
|
|
228
|
-
// If no path beyond node ID, return the extracted value
|
|
229
|
-
if (segments.length === 1) {
|
|
230
|
-
return nodeOutput;
|
|
231
|
-
}
|
|
232
|
-
|
|
233
|
-
// Navigate the path within the node's output
|
|
234
|
-
// Skip first segment if it's a wrapper property since extractFromWrapper already extracted it
|
|
235
|
-
let current: any = nodeOutput;
|
|
236
|
-
let startIndex = 1;
|
|
237
|
-
if (segments.length > 1 &&
|
|
238
|
-
(segments[1] === 'output' || segments[1] === 'result' || segments[1] === 'metadata')) {
|
|
239
|
-
startIndex = 2;
|
|
240
|
-
}
|
|
241
|
-
|
|
242
|
-
for (let i = startIndex; i < segments.length; i++) {
|
|
243
|
-
if (current === null || current === undefined) {
|
|
244
|
-
return undefined;
|
|
245
|
-
}
|
|
246
|
-
current = current[segments[i]];
|
|
247
|
-
}
|
|
248
|
-
|
|
249
|
-
return current;
|
|
250
|
-
}
|
|
251
|
-
|
|
252
|
-
/**
|
|
253
|
-
* Extract the actual output value from wrapper formats.
|
|
254
|
-
*
|
|
255
|
-
* Handles:
|
|
256
|
-
* - AgentNode format: { result, metadata } -> returns result
|
|
257
|
-
* - DecoratorNode format: { success, output, ... } -> returns output
|
|
258
|
-
* - Other formats: returns value as-is
|
|
259
|
-
*
|
|
260
|
-
* @param value - The value to extract from
|
|
261
|
-
* @returns The extracted value
|
|
262
|
-
*/
|
|
263
|
-
private extractFromWrapper(value: unknown): unknown {
|
|
264
|
-
if (value === null || value === undefined) {
|
|
265
|
-
return value;
|
|
266
|
-
}
|
|
267
|
-
|
|
268
|
-
if (typeof value !== 'object') {
|
|
269
|
-
return value;
|
|
270
|
-
}
|
|
271
|
-
|
|
272
|
-
// Check for AgentNode wrapper: { result, metadata }
|
|
273
|
-
// This is the signature of AgentNode output
|
|
274
|
-
if ('result' in value && 'metadata' in value) {
|
|
275
|
-
return (value as { result: unknown }).result;
|
|
276
|
-
}
|
|
277
|
-
|
|
278
|
-
// Check for DecoratorNode wrapper: { success, output, ... }
|
|
279
|
-
// This has 'output' but NOT 'result' (that's AgentNode's signature)
|
|
280
|
-
if ('output' in value) {
|
|
281
|
-
return (value as { output: unknown }).output;
|
|
282
|
-
}
|
|
283
|
-
|
|
284
|
-
return value;
|
|
285
|
-
}
|
|
286
|
-
|
|
287
|
-
/**
|
|
288
|
-
* Convert a value to string for template replacement.
|
|
289
|
-
* - If value is undefined, return the original template
|
|
290
|
-
* - If value is a string, return it directly
|
|
291
|
-
* - If value is null, return 'null'
|
|
292
|
-
* - If value is an object or array, serialize to JSON
|
|
293
|
-
*
|
|
294
|
-
* @param value - The value to stringify
|
|
295
|
-
* @param originalTemplate - The original template string (returned if value is undefined)
|
|
296
|
-
* @returns String representation of the value
|
|
297
|
-
*/
|
|
298
|
-
private stringifyValue(value: unknown, originalTemplate: string): string {
|
|
299
|
-
if (value === undefined) {
|
|
300
|
-
return originalTemplate;
|
|
301
|
-
}
|
|
302
|
-
if (value === null) {
|
|
303
|
-
return 'null';
|
|
304
|
-
}
|
|
305
|
-
if (typeof value === 'string') {
|
|
306
|
-
return value;
|
|
307
|
-
}
|
|
308
|
-
// For objects and arrays, serialize to JSON
|
|
309
|
-
return JSON.stringify(value);
|
|
310
|
-
}
|
|
311
|
-
}
|
|
@@ -1,27 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Service Index - Session-based Workflow Service
|
|
3
|
-
*
|
|
4
|
-
* After refactoring, RunRepository is removed. Workflow runs are managed via SessionComponent.
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
export { WorkflowService } from './workflow-service';
|
|
8
|
-
export type {
|
|
9
|
-
WorkflowRepository,
|
|
10
|
-
WorkflowEngine,
|
|
11
|
-
CreateOptions,
|
|
12
|
-
ListOptions,
|
|
13
|
-
WorkflowUpdates,
|
|
14
|
-
WorkflowEngineFactory,
|
|
15
|
-
SessionInfo,
|
|
16
|
-
WorkflowSessionMetadata,
|
|
17
|
-
AgentSessionRef,
|
|
18
|
-
} from './workflow-service';
|
|
19
|
-
|
|
20
|
-
export {
|
|
21
|
-
WorkflowServiceRegistry,
|
|
22
|
-
getWorkflowService,
|
|
23
|
-
registerWorkflowService,
|
|
24
|
-
} from './registry';
|
|
25
|
-
|
|
26
|
-
// Re-export WorkflowDefinition from types
|
|
27
|
-
export type { WorkflowDefinition } from '../types/workflow';
|