@ai-setting/roy-agent-core 1.3.10 → 1.3.11
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,92 +0,0 @@
|
|
|
1
|
-
import { z } from "zod";
|
|
2
|
-
import { readFile, writeFile } from "fs/promises";
|
|
3
|
-
import type { Tool, ToolContext, ToolResult } from "../types";
|
|
4
|
-
|
|
5
|
-
/**
|
|
6
|
-
* 编辑文件工具
|
|
7
|
-
* 通过 oldString/newString 替换来编辑文件
|
|
8
|
-
*/
|
|
9
|
-
export const editFileTool: Tool = {
|
|
10
|
-
name: "edit_file",
|
|
11
|
-
description: "Edit a file by replacing specific text. Use this to make targeted changes to files by specifying what to find and what to replace.",
|
|
12
|
-
parameters: z.object({
|
|
13
|
-
path: z.string().describe("The file path to edit"),
|
|
14
|
-
oldString: z.string().describe("The text to find and replace"),
|
|
15
|
-
newString: z.string().describe("The replacement text"),
|
|
16
|
-
replaceAll: z.boolean().default(false).optional().describe("Replace all occurrences instead of just the first"),
|
|
17
|
-
}),
|
|
18
|
-
sandbox: {
|
|
19
|
-
enabled: true,
|
|
20
|
-
config: {
|
|
21
|
-
filesystem: {
|
|
22
|
-
denyWrite: ["/etc", "/usr/bin", "/usr/sbin", "/bin", "/sbin"],
|
|
23
|
-
},
|
|
24
|
-
},
|
|
25
|
-
},
|
|
26
|
-
permission: {
|
|
27
|
-
requireConfirmation: true,
|
|
28
|
-
level: "dangerous",
|
|
29
|
-
},
|
|
30
|
-
metadata: {
|
|
31
|
-
category: "file",
|
|
32
|
-
tags: ["edit", "file", "replace", "modify"],
|
|
33
|
-
version: "1.0.0",
|
|
34
|
-
},
|
|
35
|
-
execute: async (args, ctx: ToolContext): Promise<ToolResult> => {
|
|
36
|
-
const startTime = Date.now();
|
|
37
|
-
const { path: filePath, oldString, newString, replaceAll = false } = args;
|
|
38
|
-
|
|
39
|
-
try {
|
|
40
|
-
// 读取文件内容
|
|
41
|
-
const content = await readFile(filePath, "utf-8");
|
|
42
|
-
|
|
43
|
-
// 检查是否找到要替换的内容
|
|
44
|
-
const searchPattern = replaceAll ? new RegExp(escapeRegExp(oldString), "g") : new RegExp(escapeRegExp(oldString));
|
|
45
|
-
if (!searchPattern.test(content)) {
|
|
46
|
-
return {
|
|
47
|
-
success: false,
|
|
48
|
-
output: "",
|
|
49
|
-
error: `Could not find "${oldString}" in file ${filePath}`,
|
|
50
|
-
metadata: {
|
|
51
|
-
execution_time_ms: Date.now() - startTime,
|
|
52
|
-
},
|
|
53
|
-
};
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
// 执行替换
|
|
57
|
-
const newContent = replaceAll
|
|
58
|
-
? content.replace(new RegExp(escapeRegExp(oldString), "g"), newString)
|
|
59
|
-
: content.replace(oldString, newString);
|
|
60
|
-
|
|
61
|
-
// 写回文件
|
|
62
|
-
await writeFile(filePath, newContent, "utf-8");
|
|
63
|
-
|
|
64
|
-
return {
|
|
65
|
-
success: true,
|
|
66
|
-
output: replaceAll
|
|
67
|
-
? `Replaced all occurrences in: ${filePath}`
|
|
68
|
-
: `Replaced text in: ${filePath}`,
|
|
69
|
-
metadata: {
|
|
70
|
-
execution_time_ms: Date.now() - startTime,
|
|
71
|
-
file_path: filePath,
|
|
72
|
-
},
|
|
73
|
-
};
|
|
74
|
-
} catch (error) {
|
|
75
|
-
return {
|
|
76
|
-
success: false,
|
|
77
|
-
output: "",
|
|
78
|
-
error: error instanceof Error ? error.message : String(error),
|
|
79
|
-
metadata: {
|
|
80
|
-
execution_time_ms: Date.now() - startTime,
|
|
81
|
-
},
|
|
82
|
-
};
|
|
83
|
-
}
|
|
84
|
-
},
|
|
85
|
-
};
|
|
86
|
-
|
|
87
|
-
/**
|
|
88
|
-
* 转义正则表达式特殊字符
|
|
89
|
-
*/
|
|
90
|
-
function escapeRegExp(string: string): string {
|
|
91
|
-
return string.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
|
|
92
|
-
}
|
|
@@ -1,94 +0,0 @@
|
|
|
1
|
-
import { describe, test, expect, beforeEach } from "bun:test";
|
|
2
|
-
import { globTool } from "./glob";
|
|
3
|
-
import { mkdir, rm } from "fs/promises";
|
|
4
|
-
import { join } from "path";
|
|
5
|
-
|
|
6
|
-
describe("globTool", () => {
|
|
7
|
-
const testDir = join("/tmp", `glob-test-${Date.now()}`);
|
|
8
|
-
|
|
9
|
-
beforeEach(async () => {
|
|
10
|
-
// 创建测试目录结构
|
|
11
|
-
await mkdir(join(testDir, "src"), { recursive: true });
|
|
12
|
-
await mkdir(join(testDir, "docs"), { recursive: true });
|
|
13
|
-
await mkdir(join(testDir, "node_modules", "pkg"), { recursive: true });
|
|
14
|
-
|
|
15
|
-
// 创建测试文件
|
|
16
|
-
await Bun.write(join(testDir, "src", "index.ts"), "");
|
|
17
|
-
await Bun.write(join(testDir, "src", "util.ts"), "");
|
|
18
|
-
await Bun.write(join(testDir, "docs", "readme.md"), "");
|
|
19
|
-
await Bun.write(join(testDir, "package.json"), "");
|
|
20
|
-
});
|
|
21
|
-
|
|
22
|
-
describe("parameters", () => {
|
|
23
|
-
test("should validate pattern parameter", async () => {
|
|
24
|
-
const result = await globTool.execute(
|
|
25
|
-
{ pattern: "**/*.ts", cwd: testDir },
|
|
26
|
-
{}
|
|
27
|
-
);
|
|
28
|
-
|
|
29
|
-
expect(result.success).toBe(true);
|
|
30
|
-
});
|
|
31
|
-
});
|
|
32
|
-
|
|
33
|
-
describe("execution", () => {
|
|
34
|
-
test("should find files by pattern", async () => {
|
|
35
|
-
const result = await globTool.execute(
|
|
36
|
-
{ pattern: "**/*.ts", cwd: testDir },
|
|
37
|
-
{}
|
|
38
|
-
);
|
|
39
|
-
|
|
40
|
-
expect(result.success).toBe(true);
|
|
41
|
-
expect(result.output).toContain("index.ts");
|
|
42
|
-
expect(result.output).toContain("util.ts");
|
|
43
|
-
});
|
|
44
|
-
|
|
45
|
-
test("should exclude node_modules by default", async () => {
|
|
46
|
-
const result = await globTool.execute(
|
|
47
|
-
{ pattern: "**/*.js", cwd: testDir },
|
|
48
|
-
{}
|
|
49
|
-
);
|
|
50
|
-
|
|
51
|
-
expect(result.success).toBe(true);
|
|
52
|
-
expect(result.output).not.toContain("node_modules");
|
|
53
|
-
});
|
|
54
|
-
|
|
55
|
-
test("should respect maxResults limit", async () => {
|
|
56
|
-
const result = await globTool.execute(
|
|
57
|
-
{ pattern: "**/*", cwd: testDir, maxResults: 2 },
|
|
58
|
-
{}
|
|
59
|
-
);
|
|
60
|
-
|
|
61
|
-
expect(result.success).toBe(true);
|
|
62
|
-
const files = result.output.split("\n").filter(f => f.length > 0);
|
|
63
|
-
expect(files.length).toBeLessThanOrEqual(2);
|
|
64
|
-
});
|
|
65
|
-
|
|
66
|
-
test("should use context workdir", async () => {
|
|
67
|
-
const result = await globTool.execute(
|
|
68
|
-
{ pattern: "**/*.ts" },
|
|
69
|
-
{ workdir: testDir }
|
|
70
|
-
);
|
|
71
|
-
|
|
72
|
-
expect(result.success).toBe(true);
|
|
73
|
-
expect(result.output).toContain("index.ts");
|
|
74
|
-
});
|
|
75
|
-
|
|
76
|
-
test("should return metadata with counts", async () => {
|
|
77
|
-
const result = await globTool.execute(
|
|
78
|
-
{ pattern: "**/*", cwd: testDir },
|
|
79
|
-
{}
|
|
80
|
-
);
|
|
81
|
-
|
|
82
|
-
expect(result.metadata?.total_matches).toBeDefined();
|
|
83
|
-
expect(result.metadata?.returned_matches).toBeDefined();
|
|
84
|
-
});
|
|
85
|
-
});
|
|
86
|
-
|
|
87
|
-
describe("metadata", () => {
|
|
88
|
-
test("should have correct metadata", () => {
|
|
89
|
-
expect(globTool.name).toBe("glob");
|
|
90
|
-
expect(globTool.metadata?.category).toBe("file");
|
|
91
|
-
expect(globTool.metadata?.tags).toContain("search");
|
|
92
|
-
});
|
|
93
|
-
});
|
|
94
|
-
});
|
|
@@ -1,65 +0,0 @@
|
|
|
1
|
-
import { z } from "zod";
|
|
2
|
-
import { glob as globAsync } from "glob";
|
|
3
|
-
import path from "path";
|
|
4
|
-
import type { Tool, ToolContext, ToolResult } from "../types";
|
|
5
|
-
|
|
6
|
-
/**
|
|
7
|
-
* Glob 工具
|
|
8
|
-
* 文件模式匹配搜索
|
|
9
|
-
*/
|
|
10
|
-
export const globTool: Tool = {
|
|
11
|
-
name: "glob",
|
|
12
|
-
description: "Search for files matching a glob pattern. Use this to find files by name patterns like **/*.ts or **/*.md.",
|
|
13
|
-
parameters: z.object({
|
|
14
|
-
pattern: z.string().describe("Glob pattern to match (e.g., **/*.ts, **/*.md)"),
|
|
15
|
-
cwd: z.string().optional().describe("Working directory to search in"),
|
|
16
|
-
maxResults: z.number().int().positive().optional().describe("Maximum number of results to return"),
|
|
17
|
-
ignore: z.array(z.string()).optional().describe("Patterns to ignore"),
|
|
18
|
-
}),
|
|
19
|
-
sandbox: {
|
|
20
|
-
enabled: true,
|
|
21
|
-
},
|
|
22
|
-
metadata: {
|
|
23
|
-
category: "file",
|
|
24
|
-
tags: ["search", "file", "glob", "pattern"],
|
|
25
|
-
version: "1.0.0",
|
|
26
|
-
},
|
|
27
|
-
execute: async (args, ctx: ToolContext): Promise<ToolResult> => {
|
|
28
|
-
const startTime = Date.now();
|
|
29
|
-
const { pattern, cwd, maxResults = 100, ignore } = args;
|
|
30
|
-
|
|
31
|
-
const searchDir = cwd || ctx.workdir || process.cwd();
|
|
32
|
-
|
|
33
|
-
try {
|
|
34
|
-
const files = await globAsync(pattern, {
|
|
35
|
-
cwd: searchDir,
|
|
36
|
-
ignore: ignore || ["**/node_modules/**", "**/.git/**"],
|
|
37
|
-
maxDepth: 20,
|
|
38
|
-
});
|
|
39
|
-
|
|
40
|
-
// 限制结果数量
|
|
41
|
-
const limitedFiles = files.slice(0, maxResults);
|
|
42
|
-
const output = limitedFiles.join("\n");
|
|
43
|
-
|
|
44
|
-
return {
|
|
45
|
-
success: true,
|
|
46
|
-
output,
|
|
47
|
-
metadata: {
|
|
48
|
-
execution_time_ms: Date.now() - startTime,
|
|
49
|
-
output_size: output.length,
|
|
50
|
-
total_matches: files.length,
|
|
51
|
-
returned_matches: limitedFiles.length,
|
|
52
|
-
},
|
|
53
|
-
};
|
|
54
|
-
} catch (error) {
|
|
55
|
-
return {
|
|
56
|
-
success: false,
|
|
57
|
-
output: "",
|
|
58
|
-
error: error instanceof Error ? error.message : String(error),
|
|
59
|
-
metadata: {
|
|
60
|
-
execution_time_ms: Date.now() - startTime,
|
|
61
|
-
},
|
|
62
|
-
};
|
|
63
|
-
}
|
|
64
|
-
},
|
|
65
|
-
};
|
|
@@ -1,122 +0,0 @@
|
|
|
1
|
-
import { describe, test, expect, beforeEach } from "bun:test";
|
|
2
|
-
import { grepTool } from "./grep";
|
|
3
|
-
import { writeFile, mkdir } from "fs/promises";
|
|
4
|
-
import { join } from "path";
|
|
5
|
-
|
|
6
|
-
describe("grepTool", () => {
|
|
7
|
-
const testDir = join("/tmp", `grep-test-${Date.now()}`);
|
|
8
|
-
const testFile1 = join(testDir, "file1.ts");
|
|
9
|
-
const testFile2 = join(testDir, "file2.ts");
|
|
10
|
-
|
|
11
|
-
beforeEach(async () => {
|
|
12
|
-
await mkdir(testDir, { recursive: true });
|
|
13
|
-
await writeFile(testFile1, "function hello() {\n return 'Hello';\n}\nconsole.log(hello());");
|
|
14
|
-
await writeFile(testFile2, "class Greeter {\n greet() {\n return 'Hi';\n }\n}");
|
|
15
|
-
});
|
|
16
|
-
|
|
17
|
-
describe("parameters", () => {
|
|
18
|
-
test("should validate pattern parameter", async () => {
|
|
19
|
-
const result = await grepTool.execute(
|
|
20
|
-
{ pattern: "hello", path: testDir, include: "*.ts" },
|
|
21
|
-
{}
|
|
22
|
-
);
|
|
23
|
-
|
|
24
|
-
expect(result.success).toBe(true);
|
|
25
|
-
});
|
|
26
|
-
});
|
|
27
|
-
|
|
28
|
-
describe("execution", () => {
|
|
29
|
-
test("should find pattern in files", async () => {
|
|
30
|
-
const result = await grepTool.execute(
|
|
31
|
-
{ pattern: "hello", path: testDir, include: "*.ts" },
|
|
32
|
-
{}
|
|
33
|
-
);
|
|
34
|
-
|
|
35
|
-
expect(result.success).toBe(true);
|
|
36
|
-
expect(result.output).toContain("file1.ts");
|
|
37
|
-
expect(result.output).toContain("hello");
|
|
38
|
-
});
|
|
39
|
-
|
|
40
|
-
test("should show line numbers by default", async () => {
|
|
41
|
-
const result = await grepTool.execute(
|
|
42
|
-
{ pattern: "function", path: testDir, include: "*.ts" },
|
|
43
|
-
{}
|
|
44
|
-
);
|
|
45
|
-
|
|
46
|
-
expect(result.success).toBe(true);
|
|
47
|
-
expect(result.output).toContain(":1:");
|
|
48
|
-
});
|
|
49
|
-
|
|
50
|
-
test("should hide line numbers when disabled", async () => {
|
|
51
|
-
const result = await grepTool.execute(
|
|
52
|
-
{ pattern: "function", path: testDir, include: "*.ts", showLineNumbers: false },
|
|
53
|
-
{}
|
|
54
|
-
);
|
|
55
|
-
|
|
56
|
-
expect(result.success).toBe(true);
|
|
57
|
-
expect(result.output).not.toContain(":1:");
|
|
58
|
-
});
|
|
59
|
-
|
|
60
|
-
test("should be case sensitive by default", async () => {
|
|
61
|
-
const result = await grepTool.execute(
|
|
62
|
-
{ pattern: "Hello", path: testDir, include: "*.ts" },
|
|
63
|
-
{}
|
|
64
|
-
);
|
|
65
|
-
|
|
66
|
-
expect(result.success).toBe(true);
|
|
67
|
-
expect(result.output).toContain("Hello");
|
|
68
|
-
});
|
|
69
|
-
|
|
70
|
-
test("should be case insensitive when specified", async () => {
|
|
71
|
-
const result = await grepTool.execute(
|
|
72
|
-
{ pattern: "hello", path: testDir, include: "*.ts", caseSensitive: false },
|
|
73
|
-
{}
|
|
74
|
-
);
|
|
75
|
-
|
|
76
|
-
expect(result.success).toBe(true);
|
|
77
|
-
expect(result.output).toContain("Hello");
|
|
78
|
-
});
|
|
79
|
-
|
|
80
|
-
test("should respect maxResults limit", async () => {
|
|
81
|
-
await writeFile(join(testDir, "file3.ts"), "hello\nhello\nhello\nhello\nhello");
|
|
82
|
-
|
|
83
|
-
const result = await grepTool.execute(
|
|
84
|
-
{ pattern: "hello", path: testDir, include: "*.ts", maxResults: 2 },
|
|
85
|
-
{}
|
|
86
|
-
);
|
|
87
|
-
|
|
88
|
-
expect(result.success).toBe(true);
|
|
89
|
-
const matches = result.output.split("\n").filter(m => m.includes("hello"));
|
|
90
|
-
expect(matches.length).toBeLessThanOrEqual(2);
|
|
91
|
-
});
|
|
92
|
-
|
|
93
|
-
test("should return no matches message", async () => {
|
|
94
|
-
const result = await grepTool.execute(
|
|
95
|
-
{ pattern: "nonexistent", path: testDir, include: "*.ts" },
|
|
96
|
-
{}
|
|
97
|
-
);
|
|
98
|
-
|
|
99
|
-
expect(result.success).toBe(true);
|
|
100
|
-
expect(result.output).toContain("(no matches found)");
|
|
101
|
-
});
|
|
102
|
-
|
|
103
|
-
test("should include metadata", async () => {
|
|
104
|
-
const result = await grepTool.execute(
|
|
105
|
-
{ pattern: "hello", path: testDir, include: "*.ts" },
|
|
106
|
-
{}
|
|
107
|
-
);
|
|
108
|
-
|
|
109
|
-
expect(result.metadata?.execution_time_ms).toBeDefined();
|
|
110
|
-
expect(result.metadata?.matches_count).toBeDefined();
|
|
111
|
-
expect(result.metadata?.files_searched).toBeDefined();
|
|
112
|
-
});
|
|
113
|
-
});
|
|
114
|
-
|
|
115
|
-
describe("metadata", () => {
|
|
116
|
-
test("should have correct metadata", () => {
|
|
117
|
-
expect(grepTool.name).toBe("grep");
|
|
118
|
-
expect(grepTool.metadata?.category).toBe("file");
|
|
119
|
-
expect(grepTool.metadata?.tags).toContain("search");
|
|
120
|
-
});
|
|
121
|
-
});
|
|
122
|
-
});
|
|
@@ -1,108 +0,0 @@
|
|
|
1
|
-
import { z } from "zod";
|
|
2
|
-
import { readFile } from "fs/promises";
|
|
3
|
-
import { glob as globAsync } from "glob";
|
|
4
|
-
import type { Tool, ToolContext, ToolResult } from "../types";
|
|
5
|
-
|
|
6
|
-
/**
|
|
7
|
-
* Grep 工具
|
|
8
|
-
* 在文件中搜索内容
|
|
9
|
-
*/
|
|
10
|
-
export const grepTool: Tool = {
|
|
11
|
-
name: "grep",
|
|
12
|
-
description: "Search for text patterns in files. Use this to find specific strings, code patterns, or content across multiple files.",
|
|
13
|
-
parameters: z.object({
|
|
14
|
-
pattern: z.string().describe("The pattern to search for"),
|
|
15
|
-
path: z.string().optional().describe("File or directory path to search in"),
|
|
16
|
-
include: z.string().optional().describe("Glob pattern for files to include (e.g., *.ts, *.js)"),
|
|
17
|
-
caseSensitive: z.boolean().default(true).optional().describe("Whether the search should be case sensitive"),
|
|
18
|
-
maxResults: z.number().int().positive().default(50).optional().describe("Maximum number of results"),
|
|
19
|
-
showLineNumbers: z.boolean().default(true).optional().describe("Show line numbers in results"),
|
|
20
|
-
}),
|
|
21
|
-
sandbox: {
|
|
22
|
-
enabled: true,
|
|
23
|
-
},
|
|
24
|
-
metadata: {
|
|
25
|
-
category: "file",
|
|
26
|
-
tags: ["search", "grep", "find", "pattern"],
|
|
27
|
-
version: "1.0.0",
|
|
28
|
-
},
|
|
29
|
-
execute: async (args, ctx: ToolContext): Promise<ToolResult> => {
|
|
30
|
-
const startTime = Date.now();
|
|
31
|
-
const {
|
|
32
|
-
pattern,
|
|
33
|
-
path: searchPath,
|
|
34
|
-
include,
|
|
35
|
-
caseSensitive = true,
|
|
36
|
-
maxResults = 50,
|
|
37
|
-
showLineNumbers = true,
|
|
38
|
-
} = args;
|
|
39
|
-
|
|
40
|
-
const basePath = searchPath || ctx.workdir || process.cwd();
|
|
41
|
-
|
|
42
|
-
try {
|
|
43
|
-
// 如果指定了 include,先找到匹配的文件
|
|
44
|
-
let files: string[] = [];
|
|
45
|
-
if (include) {
|
|
46
|
-
files = await globAsync(include, {
|
|
47
|
-
cwd: basePath,
|
|
48
|
-
ignore: ["**/node_modules/**", "**/.git/**"],
|
|
49
|
-
maxDepth: 10,
|
|
50
|
-
});
|
|
51
|
-
} else {
|
|
52
|
-
// 否则搜索目录
|
|
53
|
-
files = await globAsync("**/*", {
|
|
54
|
-
cwd: basePath,
|
|
55
|
-
ignore: ["**/node_modules/**", "**/.git/**"],
|
|
56
|
-
maxDepth: 10,
|
|
57
|
-
});
|
|
58
|
-
}
|
|
59
|
-
|
|
60
|
-
const results: string[] = [];
|
|
61
|
-
// 不使用全局标志,避免 lastIndex 问题
|
|
62
|
-
const regexFlags = caseSensitive ? "" : "i";
|
|
63
|
-
const regex = new RegExp(pattern, regexFlags);
|
|
64
|
-
|
|
65
|
-
for (const file of files) {
|
|
66
|
-
if (results.length >= maxResults) break;
|
|
67
|
-
|
|
68
|
-
try {
|
|
69
|
-
const content = await readFile(`${basePath}/${file}`, "utf-8");
|
|
70
|
-
const lines = content.split("\n");
|
|
71
|
-
|
|
72
|
-
for (let i = 0; i < lines.length; i++) {
|
|
73
|
-
if (regex.test(lines[i])) {
|
|
74
|
-
const lineNum = showLineNumbers ? `${i + 1}:` : "";
|
|
75
|
-
results.push(`${file}:${lineNum}${lines[i]}`);
|
|
76
|
-
|
|
77
|
-
if (results.length >= maxResults) break;
|
|
78
|
-
}
|
|
79
|
-
}
|
|
80
|
-
} catch {
|
|
81
|
-
// 跳过无法读取的文件(如二进制)
|
|
82
|
-
}
|
|
83
|
-
}
|
|
84
|
-
|
|
85
|
-
const output = results.join("\n") || "(no matches found)";
|
|
86
|
-
|
|
87
|
-
return {
|
|
88
|
-
success: true,
|
|
89
|
-
output,
|
|
90
|
-
metadata: {
|
|
91
|
-
execution_time_ms: Date.now() - startTime,
|
|
92
|
-
output_size: output.length,
|
|
93
|
-
matches_count: results.length,
|
|
94
|
-
files_searched: files.length,
|
|
95
|
-
},
|
|
96
|
-
};
|
|
97
|
-
} catch (error) {
|
|
98
|
-
return {
|
|
99
|
-
success: false,
|
|
100
|
-
output: "",
|
|
101
|
-
error: error instanceof Error ? error.message : String(error),
|
|
102
|
-
metadata: {
|
|
103
|
-
execution_time_ms: Date.now() - startTime,
|
|
104
|
-
},
|
|
105
|
-
};
|
|
106
|
-
}
|
|
107
|
-
},
|
|
108
|
-
};
|
|
@@ -1,44 +0,0 @@
|
|
|
1
|
-
import type { Tool } from "../types";
|
|
2
|
-
import { bashTool } from "./bash";
|
|
3
|
-
import { echoTool } from "./echo";
|
|
4
|
-
import { globTool } from "./glob";
|
|
5
|
-
import { readFileTool } from "./read-file";
|
|
6
|
-
import { writeFileTool } from "./write-file";
|
|
7
|
-
import { editFileTool } from "./edit-file";
|
|
8
|
-
import { grepTool } from "./grep";
|
|
9
|
-
|
|
10
|
-
// 重新导出所有内置工具
|
|
11
|
-
export { bashTool };
|
|
12
|
-
export { echoTool };
|
|
13
|
-
export { globTool };
|
|
14
|
-
export { readFileTool };
|
|
15
|
-
export { writeFileTool };
|
|
16
|
-
export { editFileTool };
|
|
17
|
-
export { grepTool };
|
|
18
|
-
|
|
19
|
-
/**
|
|
20
|
-
* 获取所有内置工具
|
|
21
|
-
*
|
|
22
|
-
* Note: run-workflow tool is not included here because it requires
|
|
23
|
-
* a WorkflowService instance which needs to be initialized at runtime.
|
|
24
|
-
* Use initRunWorkflowTool(workflowService) to initialize it.
|
|
25
|
-
*/
|
|
26
|
-
export function getAllBuiltInTools(): Tool[] {
|
|
27
|
-
return [
|
|
28
|
-
bashTool,
|
|
29
|
-
echoTool,
|
|
30
|
-
globTool,
|
|
31
|
-
readFileTool,
|
|
32
|
-
writeFileTool,
|
|
33
|
-
editFileTool,
|
|
34
|
-
grepTool,
|
|
35
|
-
];
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
/**
|
|
39
|
-
* 获取指定内置工具
|
|
40
|
-
*/
|
|
41
|
-
export function getBuiltInTool(name: string): Tool | undefined {
|
|
42
|
-
const tools = getAllBuiltInTools();
|
|
43
|
-
return tools.find((t) => t.name === name);
|
|
44
|
-
}
|
|
@@ -1,84 +0,0 @@
|
|
|
1
|
-
import { describe, test, expect, beforeEach } from "bun:test";
|
|
2
|
-
import { readFileTool } from "./read-file";
|
|
3
|
-
import { writeFile, mkdir } from "fs/promises";
|
|
4
|
-
import { join } from "path";
|
|
5
|
-
|
|
6
|
-
describe("readFileTool", () => {
|
|
7
|
-
const testDir = join("/tmp", `read-test-${Date.now()}`);
|
|
8
|
-
const testFile = join(testDir, "test.txt");
|
|
9
|
-
const multiLineFile = join(testDir, "multi.txt");
|
|
10
|
-
|
|
11
|
-
beforeEach(async () => {
|
|
12
|
-
await mkdir(testDir, { recursive: true });
|
|
13
|
-
await writeFile(testFile, "Hello, World!");
|
|
14
|
-
await writeFile(multiLineFile, "Line 1\nLine 2\nLine 3\nLine 4\nLine 5");
|
|
15
|
-
});
|
|
16
|
-
|
|
17
|
-
describe("parameters", () => {
|
|
18
|
-
test("should validate path parameter", async () => {
|
|
19
|
-
const result = await readFileTool.execute({ path: testFile }, {});
|
|
20
|
-
|
|
21
|
-
expect(result.success).toBe(true);
|
|
22
|
-
expect(result.output).toBe("Hello, World!");
|
|
23
|
-
});
|
|
24
|
-
|
|
25
|
-
test("should handle missing path gracefully", async () => {
|
|
26
|
-
// Zod 会使用 undefined 值执行,可能导致文件系统错误而不是验证错误
|
|
27
|
-
const result = await readFileTool.execute({} as any, {});
|
|
28
|
-
expect(result).toBeDefined();
|
|
29
|
-
});
|
|
30
|
-
});
|
|
31
|
-
|
|
32
|
-
describe("execution", () => {
|
|
33
|
-
test("should read file content", async () => {
|
|
34
|
-
const result = await readFileTool.execute({ path: testFile }, {});
|
|
35
|
-
|
|
36
|
-
expect(result.success).toBe(true);
|
|
37
|
-
expect(result.output).toBe("Hello, World!");
|
|
38
|
-
});
|
|
39
|
-
|
|
40
|
-
test("should read with offset and limit", async () => {
|
|
41
|
-
const result = await readFileTool.execute(
|
|
42
|
-
{ path: multiLineFile, offset: 1, limit: 2 },
|
|
43
|
-
{}
|
|
44
|
-
);
|
|
45
|
-
|
|
46
|
-
expect(result.success).toBe(true);
|
|
47
|
-
expect(result.output).toBe("Line 2\nLine 3");
|
|
48
|
-
});
|
|
49
|
-
|
|
50
|
-
test("should return error for non-existent file", async () => {
|
|
51
|
-
const result = await readFileTool.execute(
|
|
52
|
-
{ path: join(testDir, "non-existent.txt") },
|
|
53
|
-
{}
|
|
54
|
-
);
|
|
55
|
-
|
|
56
|
-
expect(result.success).toBe(false);
|
|
57
|
-
expect(result.error).toBeDefined();
|
|
58
|
-
});
|
|
59
|
-
|
|
60
|
-
test("should include metadata", async () => {
|
|
61
|
-
const result = await readFileTool.execute({ path: testFile }, {});
|
|
62
|
-
|
|
63
|
-
expect(result.metadata?.execution_time_ms).toBeDefined();
|
|
64
|
-
expect(result.metadata?.output_size).toBeDefined();
|
|
65
|
-
expect(result.metadata?.file_path).toBe(testFile);
|
|
66
|
-
});
|
|
67
|
-
});
|
|
68
|
-
|
|
69
|
-
describe("metadata", () => {
|
|
70
|
-
test("should have correct metadata", () => {
|
|
71
|
-
expect(readFileTool.name).toBe("read_file");
|
|
72
|
-
expect(readFileTool.metadata?.category).toBe("file");
|
|
73
|
-
expect(readFileTool.metadata?.tags).toContain("read");
|
|
74
|
-
});
|
|
75
|
-
|
|
76
|
-
test("should have sandbox config", () => {
|
|
77
|
-
expect(readFileTool.sandbox?.enabled).toBe(true);
|
|
78
|
-
});
|
|
79
|
-
|
|
80
|
-
test("should be safe permission", () => {
|
|
81
|
-
expect(readFileTool.permission?.level).toBe("safe");
|
|
82
|
-
});
|
|
83
|
-
});
|
|
84
|
-
});
|
|
@@ -1,75 +0,0 @@
|
|
|
1
|
-
import { z } from "zod";
|
|
2
|
-
import { readFile } from "fs/promises";
|
|
3
|
-
import type { Tool, ToolContext, ToolResult } from "../types";
|
|
4
|
-
|
|
5
|
-
/**
|
|
6
|
-
* 读取文件工具
|
|
7
|
-
*/
|
|
8
|
-
export const readFileTool: Tool = {
|
|
9
|
-
name: "read_file",
|
|
10
|
-
description: "Read the contents of a file. Use this to read source code, configuration files, or any text files.",
|
|
11
|
-
parameters: z.object({
|
|
12
|
-
path: z.string().describe("The file path to read"),
|
|
13
|
-
offset: z.number().int().nonnegative().optional().describe("Line offset to start reading from"),
|
|
14
|
-
limit: z.number().int().positive().optional().describe("Number of lines to read"),
|
|
15
|
-
encoding: z.enum(["utf-8", "base64"]).default("utf-8").optional().describe("File encoding"),
|
|
16
|
-
}),
|
|
17
|
-
sandbox: {
|
|
18
|
-
enabled: true,
|
|
19
|
-
config: {
|
|
20
|
-
filesystem: {
|
|
21
|
-
denyRead: ["/etc/shadow", "/etc/sudoers", "/root/.ssh"],
|
|
22
|
-
},
|
|
23
|
-
},
|
|
24
|
-
},
|
|
25
|
-
permission: {
|
|
26
|
-
level: "safe",
|
|
27
|
-
},
|
|
28
|
-
metadata: {
|
|
29
|
-
category: "file",
|
|
30
|
-
tags: ["read", "file", "io"],
|
|
31
|
-
version: "1.0.0",
|
|
32
|
-
},
|
|
33
|
-
execute: async (args, ctx: ToolContext): Promise<ToolResult> => {
|
|
34
|
-
const startTime = Date.now();
|
|
35
|
-
const { path: filePath, offset, limit, encoding = "utf-8" } = args;
|
|
36
|
-
|
|
37
|
-
try {
|
|
38
|
-
const content = await readFile(filePath, encoding as BufferEncoding);
|
|
39
|
-
|
|
40
|
-
// 如果是二进制或指定了 offset/limit,需要特殊处理
|
|
41
|
-
let output: string;
|
|
42
|
-
const contentBuffer = content as Buffer | string;
|
|
43
|
-
const contentStr = typeof contentBuffer === "string"
|
|
44
|
-
? contentBuffer
|
|
45
|
-
: (contentBuffer as Buffer).toString(encoding as BufferEncoding);
|
|
46
|
-
if (offset !== undefined || limit !== undefined) {
|
|
47
|
-
const lines = contentStr.split("\n");
|
|
48
|
-
const startLine = offset || 0;
|
|
49
|
-
const endLine = limit ? startLine + limit : lines.length;
|
|
50
|
-
output = lines.slice(startLine, endLine).join("\n");
|
|
51
|
-
} else {
|
|
52
|
-
output = contentStr;
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
return {
|
|
56
|
-
success: true,
|
|
57
|
-
output,
|
|
58
|
-
metadata: {
|
|
59
|
-
execution_time_ms: Date.now() - startTime,
|
|
60
|
-
output_size: output.length,
|
|
61
|
-
file_path: filePath,
|
|
62
|
-
},
|
|
63
|
-
};
|
|
64
|
-
} catch (error) {
|
|
65
|
-
return {
|
|
66
|
-
success: false,
|
|
67
|
-
output: "",
|
|
68
|
-
error: error instanceof Error ? error.message : String(error),
|
|
69
|
-
metadata: {
|
|
70
|
-
execution_time_ms: Date.now() - startTime,
|
|
71
|
-
},
|
|
72
|
-
};
|
|
73
|
-
}
|
|
74
|
-
},
|
|
75
|
-
};
|