@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,594 +0,0 @@
|
|
|
1
|
-
import type {
|
|
2
|
-
WorkflowDefinition,
|
|
3
|
-
NodeDefinition,
|
|
4
|
-
WorkflowDefinitionExtended
|
|
5
|
-
} from '../types';
|
|
6
|
-
|
|
7
|
-
/**
|
|
8
|
-
* DAG Analysis Result
|
|
9
|
-
*/
|
|
10
|
-
export interface DAGAnalysis {
|
|
11
|
-
/** 入口节点(无依赖) */
|
|
12
|
-
entryNodes: string[];
|
|
13
|
-
/** 出口节点(无下游) */
|
|
14
|
-
exitNodes: string[];
|
|
15
|
-
/** 按层级组织的节点,用于并行执行 */
|
|
16
|
-
levels: string[][];
|
|
17
|
-
/** 每个节点的依赖 */
|
|
18
|
-
dependencies: Map<string, string[]>;
|
|
19
|
-
/** 每个节点的下游 */
|
|
20
|
-
dependents: Map<string, string[]>;
|
|
21
|
-
/** 关键路径 */
|
|
22
|
-
criticalPath: string[];
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
/**
|
|
26
|
-
* DAG Manager
|
|
27
|
-
*
|
|
28
|
-
* Manages Directed Acyclic Graph (DAG) operations for workflow execution.
|
|
29
|
-
* Provides analysis, topological sorting, and validation of workflow structures.
|
|
30
|
-
*/
|
|
31
|
-
export class DAGManager {
|
|
32
|
-
private workflow: WorkflowDefinition;
|
|
33
|
-
private nodeMap: Map<string, NodeDefinition>;
|
|
34
|
-
private extendedDefinition: WorkflowDefinitionExtended | null = null;
|
|
35
|
-
private cachedAnalysis: DAGAnalysis | null = null;
|
|
36
|
-
|
|
37
|
-
constructor(workflow: WorkflowDefinition) {
|
|
38
|
-
// Validate basic structure
|
|
39
|
-
if (!workflow.nodes || workflow.nodes.length === 0) {
|
|
40
|
-
throw new Error('At least one node is required');
|
|
41
|
-
}
|
|
42
|
-
|
|
43
|
-
this.workflow = workflow;
|
|
44
|
-
this.nodeMap = new Map();
|
|
45
|
-
|
|
46
|
-
// Build node map
|
|
47
|
-
for (const node of workflow.nodes) {
|
|
48
|
-
if (node.id === '') {
|
|
49
|
-
throw new Error('Node ID cannot be empty');
|
|
50
|
-
}
|
|
51
|
-
if (this.nodeMap.has(node.id)) {
|
|
52
|
-
throw new Error(`Duplicate node ID: ${node.id}`);
|
|
53
|
-
}
|
|
54
|
-
this.nodeMap.set(node.id, node);
|
|
55
|
-
}
|
|
56
|
-
}
|
|
57
|
-
|
|
58
|
-
/**
|
|
59
|
-
* Get extended workflow definition with computed properties
|
|
60
|
-
*/
|
|
61
|
-
getExtendedDefinition(): WorkflowDefinitionExtended {
|
|
62
|
-
if (this.extendedDefinition) {
|
|
63
|
-
return this.extendedDefinition;
|
|
64
|
-
}
|
|
65
|
-
|
|
66
|
-
const analysis = this.analyze();
|
|
67
|
-
|
|
68
|
-
this.extendedDefinition = {
|
|
69
|
-
...this.workflow,
|
|
70
|
-
computedEntry: analysis.entryNodes,
|
|
71
|
-
nodeMap: this.nodeMap,
|
|
72
|
-
topologicalOrder: this.getTopologicalOrder(),
|
|
73
|
-
};
|
|
74
|
-
|
|
75
|
-
return this.extendedDefinition;
|
|
76
|
-
}
|
|
77
|
-
|
|
78
|
-
/**
|
|
79
|
-
* Analyze the DAG structure
|
|
80
|
-
*/
|
|
81
|
-
analyze(): DAGAnalysis {
|
|
82
|
-
if (this.cachedAnalysis) {
|
|
83
|
-
return this.cachedAnalysis;
|
|
84
|
-
}
|
|
85
|
-
|
|
86
|
-
// Compute dependencies map
|
|
87
|
-
const dependencies = this.buildDependenciesMap();
|
|
88
|
-
|
|
89
|
-
// Compute dependents map (reverse of dependencies)
|
|
90
|
-
const dependents = this.buildDependentsMap(dependencies);
|
|
91
|
-
|
|
92
|
-
// Find entry nodes (no incoming edges)
|
|
93
|
-
const entryNodes = this.findEntryNodes(dependencies);
|
|
94
|
-
|
|
95
|
-
// Find exit nodes (no outgoing edges)
|
|
96
|
-
const exitNodes = this.findExitNodes(dependents);
|
|
97
|
-
|
|
98
|
-
// Compute levels for parallel execution
|
|
99
|
-
const levels = this.computeLevels(dependencies);
|
|
100
|
-
|
|
101
|
-
// Find critical path
|
|
102
|
-
const criticalPath = this.findCriticalPath(levels, dependencies);
|
|
103
|
-
|
|
104
|
-
this.cachedAnalysis = {
|
|
105
|
-
entryNodes,
|
|
106
|
-
exitNodes,
|
|
107
|
-
levels,
|
|
108
|
-
dependencies,
|
|
109
|
-
dependents,
|
|
110
|
-
criticalPath,
|
|
111
|
-
};
|
|
112
|
-
|
|
113
|
-
return this.cachedAnalysis;
|
|
114
|
-
}
|
|
115
|
-
|
|
116
|
-
/**
|
|
117
|
-
* Build a map of node dependencies
|
|
118
|
-
*/
|
|
119
|
-
private buildDependenciesMap(): Map<string, string[]> {
|
|
120
|
-
const dependencies = new Map<string, string[]>();
|
|
121
|
-
|
|
122
|
-
for (const node of this.workflow.nodes) {
|
|
123
|
-
const deps = node.depends_on || [];
|
|
124
|
-
dependencies.set(node.id, [...deps]);
|
|
125
|
-
}
|
|
126
|
-
|
|
127
|
-
return dependencies;
|
|
128
|
-
}
|
|
129
|
-
|
|
130
|
-
/**
|
|
131
|
-
* Build a map of node dependents (reverse of dependencies)
|
|
132
|
-
*/
|
|
133
|
-
private buildDependentsMap(dependencies: Map<string, string[]>): Map<string, string[]> {
|
|
134
|
-
const dependentsMap = new Map<string, string[]>();
|
|
135
|
-
|
|
136
|
-
// Initialize all nodes with empty dependents
|
|
137
|
-
for (const nodeId of dependencies.keys()) {
|
|
138
|
-
dependentsMap.set(nodeId, []);
|
|
139
|
-
}
|
|
140
|
-
|
|
141
|
-
// Populate dependents
|
|
142
|
-
for (const [nodeId, deps] of dependencies.entries()) {
|
|
143
|
-
for (const dep of deps) {
|
|
144
|
-
const nodeDependents = dependentsMap.get(dep) || [];
|
|
145
|
-
nodeDependents.push(nodeId);
|
|
146
|
-
dependentsMap.set(dep, nodeDependents);
|
|
147
|
-
}
|
|
148
|
-
}
|
|
149
|
-
|
|
150
|
-
return dependentsMap;
|
|
151
|
-
}
|
|
152
|
-
|
|
153
|
-
/**
|
|
154
|
-
* Find entry nodes (nodes with no dependencies)
|
|
155
|
-
*/
|
|
156
|
-
private findEntryNodes(dependencies: Map<string, string[]>): string[] {
|
|
157
|
-
const entryNodes: string[] = [];
|
|
158
|
-
|
|
159
|
-
for (const [nodeId, deps] of dependencies.entries()) {
|
|
160
|
-
if (deps.length === 0) {
|
|
161
|
-
entryNodes.push(nodeId);
|
|
162
|
-
}
|
|
163
|
-
}
|
|
164
|
-
|
|
165
|
-
return entryNodes;
|
|
166
|
-
}
|
|
167
|
-
|
|
168
|
-
/**
|
|
169
|
-
* Find exit nodes (nodes with no dependents)
|
|
170
|
-
*/
|
|
171
|
-
private findExitNodes(dependents: Map<string, string[]>): string[] {
|
|
172
|
-
const exitNodes: string[] = [];
|
|
173
|
-
|
|
174
|
-
for (const [nodeId, deps] of dependents.entries()) {
|
|
175
|
-
if (deps.length === 0) {
|
|
176
|
-
exitNodes.push(nodeId);
|
|
177
|
-
}
|
|
178
|
-
}
|
|
179
|
-
|
|
180
|
-
return exitNodes;
|
|
181
|
-
}
|
|
182
|
-
|
|
183
|
-
/**
|
|
184
|
-
* Compute levels for parallel execution
|
|
185
|
-
* Nodes at the same level can be executed in parallel
|
|
186
|
-
*/
|
|
187
|
-
private computeLevels(dependencies: Map<string, string[]>): string[][] {
|
|
188
|
-
const levels: string[][] = [];
|
|
189
|
-
const assigned = new Set<string>();
|
|
190
|
-
const nodeLevels = new Map<string, number>();
|
|
191
|
-
|
|
192
|
-
// Build dependents map for efficient lookup
|
|
193
|
-
const dependentsMap = this.buildDependentsMap(dependencies);
|
|
194
|
-
|
|
195
|
-
// Calculate level for each node (BFS from entry nodes)
|
|
196
|
-
const entryNodes = this.findEntryNodes(dependencies);
|
|
197
|
-
|
|
198
|
-
// Initialize entry nodes at level 0
|
|
199
|
-
const queue: Array<{ nodeId: string; level: number }> = [];
|
|
200
|
-
for (const entry of entryNodes) {
|
|
201
|
-
queue.push({ nodeId: entry, level: 0 });
|
|
202
|
-
nodeLevels.set(entry, 0);
|
|
203
|
-
}
|
|
204
|
-
|
|
205
|
-
// BFS to compute levels
|
|
206
|
-
while (queue.length > 0) {
|
|
207
|
-
const { nodeId, level } = queue.shift()!;
|
|
208
|
-
|
|
209
|
-
if (assigned.has(nodeId)) {
|
|
210
|
-
// Update to maximum level if already assigned
|
|
211
|
-
const currentLevel = nodeLevels.get(nodeId) || 0;
|
|
212
|
-
if (level > currentLevel) {
|
|
213
|
-
nodeLevels.set(nodeId, level);
|
|
214
|
-
}
|
|
215
|
-
continue;
|
|
216
|
-
}
|
|
217
|
-
|
|
218
|
-
assigned.add(nodeId);
|
|
219
|
-
nodeLevels.set(nodeId, level);
|
|
220
|
-
|
|
221
|
-
// Add dependents to queue with level + 1
|
|
222
|
-
const dependentsOfNode = dependentsMap.get(nodeId) || [];
|
|
223
|
-
for (const dependent of dependentsOfNode) {
|
|
224
|
-
if (!assigned.has(dependent)) {
|
|
225
|
-
queue.push({ nodeId: dependent, level: level + 1 });
|
|
226
|
-
}
|
|
227
|
-
}
|
|
228
|
-
}
|
|
229
|
-
|
|
230
|
-
// Handle nodes not reachable from entry nodes (shouldn't happen in valid DAG)
|
|
231
|
-
for (const nodeId of dependencies.keys()) {
|
|
232
|
-
if (!nodeLevels.has(nodeId)) {
|
|
233
|
-
nodeLevels.set(nodeId, 0);
|
|
234
|
-
}
|
|
235
|
-
}
|
|
236
|
-
|
|
237
|
-
// Group nodes by level
|
|
238
|
-
const maxLevel = Math.max(...nodeLevels.values(), 0);
|
|
239
|
-
for (let i = 0; i <= maxLevel; i++) {
|
|
240
|
-
levels.push([]);
|
|
241
|
-
}
|
|
242
|
-
|
|
243
|
-
for (const [nodeId, level] of nodeLevels.entries()) {
|
|
244
|
-
levels[level].push(nodeId);
|
|
245
|
-
}
|
|
246
|
-
|
|
247
|
-
return levels;
|
|
248
|
-
}
|
|
249
|
-
|
|
250
|
-
/**
|
|
251
|
-
* Find critical path (longest path through the DAG)
|
|
252
|
-
*/
|
|
253
|
-
private findCriticalPath(levels: string[][], dependencies: Map<string, string[]>): string[] {
|
|
254
|
-
if (levels.length === 0) {
|
|
255
|
-
return [];
|
|
256
|
-
}
|
|
257
|
-
|
|
258
|
-
// DP: longest path ending at each node
|
|
259
|
-
const longestPath = new Map<string, { length: number; path: string[] }>();
|
|
260
|
-
|
|
261
|
-
// Process nodes in reverse topological order
|
|
262
|
-
const allNodes: string[] = [];
|
|
263
|
-
for (const level of levels) {
|
|
264
|
-
allNodes.push(...level);
|
|
265
|
-
}
|
|
266
|
-
|
|
267
|
-
for (const nodeId of allNodes) {
|
|
268
|
-
const deps = dependencies.get(nodeId) || [];
|
|
269
|
-
|
|
270
|
-
if (deps.length === 0) {
|
|
271
|
-
longestPath.set(nodeId, { length: 1, path: [nodeId] });
|
|
272
|
-
} else {
|
|
273
|
-
let maxLength = 0;
|
|
274
|
-
let maxPath: string[] = [];
|
|
275
|
-
|
|
276
|
-
for (const dep of deps) {
|
|
277
|
-
const depInfo = longestPath.get(dep);
|
|
278
|
-
if (depInfo && depInfo.length > maxLength) {
|
|
279
|
-
maxLength = depInfo.length;
|
|
280
|
-
maxPath = depInfo.path;
|
|
281
|
-
}
|
|
282
|
-
}
|
|
283
|
-
|
|
284
|
-
longestPath.set(nodeId, {
|
|
285
|
-
length: maxLength + 1,
|
|
286
|
-
path: [...maxPath, nodeId]
|
|
287
|
-
});
|
|
288
|
-
}
|
|
289
|
-
}
|
|
290
|
-
|
|
291
|
-
// Find the longest path
|
|
292
|
-
let maxLength = 0;
|
|
293
|
-
let criticalPath: string[] = [];
|
|
294
|
-
|
|
295
|
-
for (const [, info] of longestPath.entries()) {
|
|
296
|
-
if (info.length > maxLength) {
|
|
297
|
-
maxLength = info.length;
|
|
298
|
-
criticalPath = info.path;
|
|
299
|
-
}
|
|
300
|
-
}
|
|
301
|
-
|
|
302
|
-
return criticalPath;
|
|
303
|
-
}
|
|
304
|
-
|
|
305
|
-
/**
|
|
306
|
-
* Get topological order using Kahn's algorithm
|
|
307
|
-
*/
|
|
308
|
-
getTopologicalOrder(): string[] {
|
|
309
|
-
const dependencies = this.buildDependenciesMap();
|
|
310
|
-
const dependents = this.buildDependentsMap(dependencies);
|
|
311
|
-
|
|
312
|
-
// Calculate in-degree for each node
|
|
313
|
-
const inDegree = new Map<string, number>();
|
|
314
|
-
for (const nodeId of dependencies.keys()) {
|
|
315
|
-
inDegree.set(nodeId, dependencies.get(nodeId)?.length || 0);
|
|
316
|
-
}
|
|
317
|
-
|
|
318
|
-
// Start with nodes that have no dependencies
|
|
319
|
-
const queue: string[] = [];
|
|
320
|
-
for (const [nodeId, degree] of inDegree.entries()) {
|
|
321
|
-
if (degree === 0) {
|
|
322
|
-
queue.push(nodeId);
|
|
323
|
-
}
|
|
324
|
-
}
|
|
325
|
-
|
|
326
|
-
const result: string[] = [];
|
|
327
|
-
|
|
328
|
-
while (queue.length > 0) {
|
|
329
|
-
const nodeId = queue.shift()!;
|
|
330
|
-
result.push(nodeId);
|
|
331
|
-
|
|
332
|
-
// Reduce in-degree for dependents
|
|
333
|
-
const nodeDependents = dependents.get(nodeId) || [];
|
|
334
|
-
for (const dependent of nodeDependents) {
|
|
335
|
-
const currentDegree = inDegree.get(dependent) || 0;
|
|
336
|
-
inDegree.set(dependent, currentDegree - 1);
|
|
337
|
-
|
|
338
|
-
if (currentDegree - 1 === 0) {
|
|
339
|
-
queue.push(dependent);
|
|
340
|
-
}
|
|
341
|
-
}
|
|
342
|
-
}
|
|
343
|
-
|
|
344
|
-
return result;
|
|
345
|
-
}
|
|
346
|
-
|
|
347
|
-
/**
|
|
348
|
-
* Get nodes that are ready to execute (dependencies met and not completed)
|
|
349
|
-
*/
|
|
350
|
-
getReadyNodes(completedNodes: Set<string>): string[] {
|
|
351
|
-
const dependencies = this.buildDependenciesMap();
|
|
352
|
-
const ready: string[] = [];
|
|
353
|
-
|
|
354
|
-
for (const nodeId of dependencies.keys()) {
|
|
355
|
-
// Skip already completed nodes
|
|
356
|
-
if (completedNodes.has(nodeId)) {
|
|
357
|
-
continue;
|
|
358
|
-
}
|
|
359
|
-
|
|
360
|
-
// Check if all dependencies are completed
|
|
361
|
-
if (this.areDependenciesMet(nodeId, completedNodes)) {
|
|
362
|
-
ready.push(nodeId);
|
|
363
|
-
}
|
|
364
|
-
}
|
|
365
|
-
|
|
366
|
-
return ready;
|
|
367
|
-
}
|
|
368
|
-
|
|
369
|
-
/**
|
|
370
|
-
* Check if all dependencies for a node are completed
|
|
371
|
-
*/
|
|
372
|
-
areDependenciesMet(nodeId: string, completedNodes: Set<string>): boolean {
|
|
373
|
-
const node = this.nodeMap.get(nodeId);
|
|
374
|
-
if (!node) {
|
|
375
|
-
return false;
|
|
376
|
-
}
|
|
377
|
-
|
|
378
|
-
const dependencies = node.depends_on || [];
|
|
379
|
-
|
|
380
|
-
for (const dep of dependencies) {
|
|
381
|
-
if (!completedNodes.has(dep)) {
|
|
382
|
-
return false;
|
|
383
|
-
}
|
|
384
|
-
}
|
|
385
|
-
|
|
386
|
-
return true;
|
|
387
|
-
}
|
|
388
|
-
|
|
389
|
-
/**
|
|
390
|
-
* Check if the workflow contains a cycle
|
|
391
|
-
*/
|
|
392
|
-
isCyclic(): boolean {
|
|
393
|
-
const dependencies = this.buildDependenciesMap();
|
|
394
|
-
|
|
395
|
-
// Try to get topological order
|
|
396
|
-
const order = this.getTopologicalOrderInternal(dependencies);
|
|
397
|
-
|
|
398
|
-
// If we can't include all nodes, there's a cycle
|
|
399
|
-
return order.length !== this.workflow.nodes.length;
|
|
400
|
-
}
|
|
401
|
-
|
|
402
|
-
/**
|
|
403
|
-
* Internal topological sort that detects cycles
|
|
404
|
-
*/
|
|
405
|
-
private getTopologicalOrderInternal(dependencies: Map<string, string[]>): string[] {
|
|
406
|
-
const dependents = this.buildDependentsMap(dependencies);
|
|
407
|
-
|
|
408
|
-
// Calculate in-degree for each node
|
|
409
|
-
const inDegree = new Map<string, number>();
|
|
410
|
-
for (const nodeId of dependencies.keys()) {
|
|
411
|
-
inDegree.set(nodeId, dependencies.get(nodeId)?.length || 0);
|
|
412
|
-
}
|
|
413
|
-
|
|
414
|
-
// Start with nodes that have no dependencies
|
|
415
|
-
const queue: string[] = [];
|
|
416
|
-
for (const [nodeId, degree] of inDegree.entries()) {
|
|
417
|
-
if (degree === 0) {
|
|
418
|
-
queue.push(nodeId);
|
|
419
|
-
}
|
|
420
|
-
}
|
|
421
|
-
|
|
422
|
-
const result: string[] = [];
|
|
423
|
-
|
|
424
|
-
while (queue.length > 0) {
|
|
425
|
-
const nodeId = queue.shift()!;
|
|
426
|
-
result.push(nodeId);
|
|
427
|
-
|
|
428
|
-
// Reduce in-degree for dependents
|
|
429
|
-
const nodeDependents = dependents.get(nodeId) || [];
|
|
430
|
-
for (const dependent of nodeDependents) {
|
|
431
|
-
const currentDegree = inDegree.get(dependent) || 0;
|
|
432
|
-
inDegree.set(dependent, currentDegree - 1);
|
|
433
|
-
|
|
434
|
-
if (currentDegree - 1 === 0) {
|
|
435
|
-
queue.push(dependent);
|
|
436
|
-
}
|
|
437
|
-
}
|
|
438
|
-
}
|
|
439
|
-
|
|
440
|
-
return result;
|
|
441
|
-
}
|
|
442
|
-
|
|
443
|
-
/**
|
|
444
|
-
* Validate the workflow
|
|
445
|
-
*/
|
|
446
|
-
validate(): { valid: boolean; errors: string[] } {
|
|
447
|
-
const errors: string[] = [];
|
|
448
|
-
|
|
449
|
-
// Check for duplicate node IDs
|
|
450
|
-
const nodeIds = new Set<string>();
|
|
451
|
-
for (const node of this.workflow.nodes) {
|
|
452
|
-
if (node.id === '') {
|
|
453
|
-
errors.push('Node ID cannot be empty');
|
|
454
|
-
continue;
|
|
455
|
-
}
|
|
456
|
-
|
|
457
|
-
if (nodeIds.has(node.id)) {
|
|
458
|
-
errors.push(`Duplicate node ID: ${node.id}`);
|
|
459
|
-
}
|
|
460
|
-
nodeIds.add(node.id);
|
|
461
|
-
}
|
|
462
|
-
|
|
463
|
-
// Check for missing dependencies
|
|
464
|
-
for (const node of this.workflow.nodes) {
|
|
465
|
-
const deps = node.depends_on || [];
|
|
466
|
-
for (const dep of deps) {
|
|
467
|
-
if (!nodeIds.has(dep)) {
|
|
468
|
-
errors.push(`Node '${node.id}' depends on non-existent node '${dep}'`);
|
|
469
|
-
}
|
|
470
|
-
}
|
|
471
|
-
}
|
|
472
|
-
|
|
473
|
-
// Check for cycles
|
|
474
|
-
if (this.isCyclic()) {
|
|
475
|
-
errors.push('Workflow contains a cycle');
|
|
476
|
-
}
|
|
477
|
-
|
|
478
|
-
return {
|
|
479
|
-
valid: errors.length === 0,
|
|
480
|
-
errors,
|
|
481
|
-
};
|
|
482
|
-
}
|
|
483
|
-
|
|
484
|
-
/**
|
|
485
|
-
* Add a new node to the workflow dynamically
|
|
486
|
-
*
|
|
487
|
-
* @param node - The node definition to add
|
|
488
|
-
* @throws Error if a node with the same ID already exists
|
|
489
|
-
* @throws Error if the node depends on a non-existent node
|
|
490
|
-
* @throws Error if adding this node would create a cycle
|
|
491
|
-
*/
|
|
492
|
-
addNode(node: NodeDefinition): void {
|
|
493
|
-
// Validate node ID
|
|
494
|
-
if (!node.id || node.id.trim() === '') {
|
|
495
|
-
throw new Error('Node ID cannot be empty');
|
|
496
|
-
}
|
|
497
|
-
|
|
498
|
-
// Check for duplicate node ID
|
|
499
|
-
if (this.nodeMap.has(node.id)) {
|
|
500
|
-
throw new Error(`Node with ID '${node.id}' already exists`);
|
|
501
|
-
}
|
|
502
|
-
|
|
503
|
-
// Check for missing dependencies
|
|
504
|
-
const deps = node.depends_on || [];
|
|
505
|
-
for (const dep of deps) {
|
|
506
|
-
if (!this.nodeMap.has(dep)) {
|
|
507
|
-
throw new Error(`Cannot add node '${node.id}': depends on non-existent node '${dep}'`);
|
|
508
|
-
}
|
|
509
|
-
}
|
|
510
|
-
|
|
511
|
-
// Temporarily add the node and check for cycles
|
|
512
|
-
const originalNodes = [...this.workflow.nodes];
|
|
513
|
-
this.workflow.nodes.push(node);
|
|
514
|
-
|
|
515
|
-
if (this.isCyclic()) {
|
|
516
|
-
// Revert if it creates a cycle
|
|
517
|
-
this.workflow.nodes = originalNodes;
|
|
518
|
-
throw new Error(`Cannot add node '${node.id}': would create a cycle in the workflow`);
|
|
519
|
-
}
|
|
520
|
-
|
|
521
|
-
// Add to node map
|
|
522
|
-
this.nodeMap.set(node.id, node);
|
|
523
|
-
|
|
524
|
-
// Clear cached analysis since the DAG has changed
|
|
525
|
-
this.cachedAnalysis = null;
|
|
526
|
-
this.extendedDefinition = null;
|
|
527
|
-
}
|
|
528
|
-
|
|
529
|
-
/**
|
|
530
|
-
* Remove a node from the workflow dynamically
|
|
531
|
-
*
|
|
532
|
-
* @param nodeId - The ID of the node to remove
|
|
533
|
-
* @throws Error if the node doesn't exist
|
|
534
|
-
* @throws Error if other nodes depend on this node
|
|
535
|
-
*/
|
|
536
|
-
removeNode(nodeId: string): void {
|
|
537
|
-
// Check if node exists
|
|
538
|
-
if (!this.nodeMap.has(nodeId)) {
|
|
539
|
-
throw new Error(`Node '${nodeId}' does not exist`);
|
|
540
|
-
}
|
|
541
|
-
|
|
542
|
-
// Check if any nodes depend on this node
|
|
543
|
-
const dependents = this.getDependents(nodeId);
|
|
544
|
-
if (dependents.length > 0) {
|
|
545
|
-
throw new Error(`Cannot remove node '${nodeId}': other nodes depend on it: ${dependents.join(', ')}`);
|
|
546
|
-
}
|
|
547
|
-
|
|
548
|
-
// Remove from workflow nodes array
|
|
549
|
-
this.workflow.nodes = this.workflow.nodes.filter(n => n.id !== nodeId);
|
|
550
|
-
|
|
551
|
-
// Remove from node map
|
|
552
|
-
this.nodeMap.delete(nodeId);
|
|
553
|
-
|
|
554
|
-
// Clear cached analysis since the DAG has changed
|
|
555
|
-
this.cachedAnalysis = null;
|
|
556
|
-
this.extendedDefinition = null;
|
|
557
|
-
}
|
|
558
|
-
|
|
559
|
-
/**
|
|
560
|
-
* Get direct dependents of a node (nodes that depend on this node)
|
|
561
|
-
*/
|
|
562
|
-
getDependents(nodeId: string): string[] {
|
|
563
|
-
const analysis = this.analyze();
|
|
564
|
-
return analysis.dependents.get(nodeId) || [];
|
|
565
|
-
}
|
|
566
|
-
|
|
567
|
-
/**
|
|
568
|
-
* Get the workflow definition (current state)
|
|
569
|
-
*/
|
|
570
|
-
getDefinition(): WorkflowDefinition {
|
|
571
|
-
return this.workflow;
|
|
572
|
-
}
|
|
573
|
-
|
|
574
|
-
/**
|
|
575
|
-
* Get a node by ID
|
|
576
|
-
*/
|
|
577
|
-
getNode(nodeId: string): NodeDefinition | undefined {
|
|
578
|
-
return this.nodeMap.get(nodeId);
|
|
579
|
-
}
|
|
580
|
-
|
|
581
|
-
/**
|
|
582
|
-
* Get all node IDs
|
|
583
|
-
*/
|
|
584
|
-
getNodeIds(): string[] {
|
|
585
|
-
return Array.from(this.nodeMap.keys());
|
|
586
|
-
}
|
|
587
|
-
|
|
588
|
-
/**
|
|
589
|
-
* Get the number of nodes in the workflow
|
|
590
|
-
*/
|
|
591
|
-
getNodeCount(): number {
|
|
592
|
-
return this.nodeMap.size;
|
|
593
|
-
}
|
|
594
|
-
}
|