claude-code-workflow 6.1.4 → 6.2.2
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/.claude/CLAUDE.md +10 -0
- package/.claude/agents/action-planning-agent.md +857 -778
- package/.claude/agents/cli-execution-agent.md +266 -269
- package/.claude/agents/cli-explore-agent.md +2 -2
- package/.claude/agents/cli-lite-planning-agent.md +142 -92
- package/.claude/agents/cli-planning-agent.md +4 -4
- package/.claude/agents/code-developer.md +7 -6
- package/.claude/agents/conceptual-planning-agent.md +2 -2
- package/.claude/agents/context-search-agent.md +31 -32
- package/.claude/agents/doc-generator.md +4 -4
- package/.claude/agents/memory-bridge.md +93 -93
- package/.claude/agents/test-context-search-agent.md +8 -7
- package/.claude/agents/test-fix-agent.md +7 -6
- package/.claude/commands/clean.md +516 -0
- package/.claude/commands/memory/compact.md +383 -0
- package/.claude/commands/memory/docs-full-cli.md +471 -471
- package/.claude/commands/memory/docs-related-cli.md +386 -386
- package/.claude/commands/memory/docs.md +615 -615
- package/.claude/commands/memory/load.md +5 -5
- package/.claude/commands/memory/tech-research-rules.md +310 -0
- package/.claude/commands/memory/update-full.md +332 -332
- package/.claude/commands/memory/workflow-skill-memory.md +4 -4
- package/.claude/commands/task/create.md +151 -151
- package/.claude/commands/version.md +254 -254
- package/.claude/commands/workflow/brainstorm/api-designer.md +587 -585
- package/.claude/commands/workflow/brainstorm/artifacts.md +1 -0
- package/.claude/commands/workflow/brainstorm/auto-parallel.md +443 -443
- package/.claude/commands/workflow/brainstorm/data-architect.md +220 -220
- package/.claude/commands/workflow/brainstorm/product-manager.md +200 -200
- package/.claude/commands/workflow/brainstorm/product-owner.md +200 -200
- package/.claude/commands/workflow/brainstorm/scrum-master.md +200 -200
- package/.claude/commands/workflow/brainstorm/subject-matter-expert.md +200 -200
- package/.claude/commands/workflow/brainstorm/system-architect.md +389 -387
- package/.claude/commands/workflow/brainstorm/ui-designer.md +221 -221
- package/.claude/commands/workflow/brainstorm/ux-expert.md +221 -221
- package/.claude/commands/workflow/debug.md +321 -0
- package/.claude/commands/workflow/execute.md +13 -0
- package/.claude/commands/workflow/init.md +165 -164
- package/.claude/commands/workflow/lite-execute.md +119 -13
- package/.claude/commands/workflow/lite-fix.md +623 -621
- package/.claude/commands/workflow/lite-plan.md +610 -592
- package/.claude/commands/workflow/plan.md +5 -5
- package/.claude/commands/workflow/review-module-cycle.md +2 -0
- package/.claude/commands/workflow/review-session-cycle.md +2 -0
- package/.claude/commands/workflow/review.md +297 -291
- package/.claude/commands/workflow/session/complete.md +153 -500
- package/.claude/commands/workflow/session/list.md +95 -95
- package/.claude/commands/workflow/session/resume.md +60 -60
- package/.claude/commands/workflow/session/start.md +199 -199
- package/.claude/commands/workflow/tdd-plan.md +3 -3
- package/.claude/commands/workflow/tdd-verify.md +23 -9
- package/.claude/commands/workflow/test-cycle-execute.md +2 -0
- package/.claude/commands/workflow/test-fix-gen.md +699 -699
- package/.claude/commands/workflow/tools/conflict-resolution.md +104 -18
- package/.claude/commands/workflow/tools/context-gather.md +436 -434
- package/.claude/commands/workflow/tools/task-generate-agent.md +490 -291
- package/.claude/commands/workflow/tools/task-generate-tdd.md +18 -10
- package/.claude/commands/workflow/tools/test-concept-enhanced.md +2 -1
- package/.claude/commands/workflow/tools/test-context-gather.md +1 -0
- package/.claude/commands/workflow/tools/test-task-generate.md +1 -0
- package/.claude/commands/workflow/ui-design/import-from-code.md +9 -6
- package/.claude/skills/command-guide/SKILL.md +5 -5
- package/.claude/skills/command-guide/index/all-commands.json +1 -1
- package/.claude/skills/command-guide/index/by-category.json +1 -1
- package/.claude/skills/command-guide/index/by-use-case.json +1 -1
- package/.claude/skills/command-guide/reference/agents/action-planning-agent.md +857 -778
- package/.claude/skills/command-guide/reference/agents/cli-execution-agent.md +266 -269
- package/.claude/skills/command-guide/reference/agents/cli-explore-agent.md +2 -2
- package/.claude/skills/command-guide/reference/agents/cli-lite-planning-agent.md +142 -92
- package/.claude/skills/command-guide/reference/agents/cli-planning-agent.md +4 -4
- package/.claude/skills/command-guide/reference/agents/code-developer.md +7 -6
- package/.claude/skills/command-guide/reference/agents/conceptual-planning-agent.md +2 -2
- package/.claude/skills/command-guide/reference/agents/context-search-agent.md +31 -32
- package/.claude/skills/command-guide/reference/agents/doc-generator.md +4 -4
- package/.claude/skills/command-guide/reference/agents/memory-bridge.md +93 -93
- package/.claude/skills/command-guide/reference/agents/test-context-search-agent.md +8 -7
- package/.claude/skills/command-guide/reference/agents/test-fix-agent.md +7 -6
- package/.claude/skills/command-guide/reference/commands/memory/docs-full-cli.md +471 -471
- package/.claude/skills/command-guide/reference/commands/memory/docs-related-cli.md +386 -386
- package/.claude/skills/command-guide/reference/commands/memory/docs.md +17 -16
- package/.claude/skills/command-guide/reference/commands/memory/load.md +5 -5
- package/.claude/skills/command-guide/reference/commands/memory/tech-research.md +194 -357
- package/.claude/skills/command-guide/reference/commands/memory/update-full.md +332 -332
- package/.claude/skills/command-guide/reference/commands/memory/workflow-skill-memory.md +4 -4
- package/.claude/skills/command-guide/reference/commands/task/create.md +151 -151
- package/.claude/skills/command-guide/reference/commands/version.md +254 -254
- package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/api-designer.md +585 -585
- package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/auto-parallel.md +443 -443
- package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/data-architect.md +220 -220
- package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/product-manager.md +200 -200
- package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/product-owner.md +200 -200
- package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/scrum-master.md +200 -200
- package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/subject-matter-expert.md +200 -200
- package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/system-architect.md +387 -387
- package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/ui-designer.md +221 -221
- package/.claude/skills/command-guide/reference/commands/workflow/brainstorm/ux-expert.md +221 -221
- package/.claude/skills/command-guide/reference/commands/workflow/execute.md +25 -20
- package/.claude/skills/command-guide/reference/commands/workflow/init.md +164 -164
- package/.claude/skills/command-guide/reference/commands/workflow/lite-execute.md +748 -686
- package/.claude/skills/command-guide/reference/commands/workflow/lite-fix.md +664 -621
- package/.claude/skills/command-guide/reference/commands/workflow/lite-plan.md +645 -592
- package/.claude/skills/command-guide/reference/commands/workflow/plan.md +5 -5
- package/.claude/skills/command-guide/reference/commands/workflow/review.md +25 -18
- package/.claude/skills/command-guide/reference/commands/workflow/session/complete.md +547 -500
- package/.claude/skills/command-guide/reference/commands/workflow/session/list.md +45 -27
- package/.claude/skills/command-guide/reference/commands/workflow/session/resume.md +35 -19
- package/.claude/skills/command-guide/reference/commands/workflow/session/start.md +90 -33
- package/.claude/skills/command-guide/reference/commands/workflow/tdd-plan.md +3 -3
- package/.claude/skills/command-guide/reference/commands/workflow/tdd-verify.md +23 -9
- package/.claude/skills/command-guide/reference/commands/workflow/test-fix-gen.md +699 -699
- package/.claude/skills/command-guide/reference/commands/workflow/tools/conflict-resolution.md +103 -17
- package/.claude/skills/command-guide/reference/commands/workflow/tools/context-gather.md +434 -434
- package/.claude/skills/command-guide/reference/commands/workflow/tools/task-generate-agent.md +487 -291
- package/.claude/skills/command-guide/reference/commands/workflow/tools/task-generate-tdd.md +17 -10
- package/.claude/skills/command-guide/reference/commands/workflow/tools/test-concept-enhanced.md +1 -1
- package/.claude/skills/command-guide/reference/commands/workflow/ui-design/import-from-code.md +6 -6
- package/.claude/workflows/chinese-response.md +38 -0
- package/.claude/workflows/cli-templates/prompts/rules/rule-api.txt +122 -0
- package/.claude/workflows/cli-templates/prompts/rules/rule-components.txt +122 -0
- package/.claude/workflows/cli-templates/prompts/rules/rule-config.txt +89 -0
- package/.claude/workflows/cli-templates/prompts/rules/rule-core.txt +60 -0
- package/.claude/workflows/cli-templates/prompts/rules/rule-patterns.txt +70 -0
- package/.claude/workflows/cli-templates/prompts/rules/rule-testing.txt +81 -0
- package/.claude/workflows/cli-templates/prompts/rules/tech-rules-agent-prompt.txt +89 -0
- package/.claude/workflows/cli-templates/prompts/workflow/gemini-solution-design.txt +131 -131
- package/.claude/workflows/cli-templates/prompts/workflow/skill-conflict-patterns.txt +5 -9
- package/.claude/workflows/cli-templates/prompts/workflow/skill-lessons-learned.txt +5 -9
- package/.claude/workflows/cli-templates/protocols/analysis-protocol.md +112 -0
- package/.claude/workflows/cli-templates/protocols/write-protocol.md +201 -0
- package/.claude/workflows/cli-templates/schemas/conflict-resolution-schema.json +137 -0
- package/.claude/workflows/cli-templates/schemas/debug-log-json-schema.json +127 -0
- package/.claude/workflows/cli-templates/schemas/fix-plan-json-schema.json +25 -0
- package/.claude/workflows/cli-templates/schemas/plan-json-schema.json +25 -0
- package/.claude/workflows/cli-tools-usage.md +526 -0
- package/{CLAUDE.md → .claude/workflows/coding-philosophy.md} +24 -45
- package/.claude/workflows/context-tools.md +84 -0
- package/.claude/workflows/file-modification.md +64 -0
- package/.claude/workflows/tool-strategy.md +216 -79
- package/.claude/workflows/windows-platform.md +16 -0
- package/.claude/workflows/workflow-architecture.md +942 -942
- package/.codex/AGENTS.md +63 -330
- package/.codex/prompts/debug.md +318 -0
- package/.codex/prompts/execute.md +273 -0
- package/.codex/prompts/lite-execute.md +164 -0
- package/.codex/prompts/lite-plan.md +469 -0
- package/.codex/prompts.zip +0 -0
- package/.gemini/GEMINI.md +25 -164
- package/.qwen/QWEN.md +0 -139
- package/README.md +29 -9
- package/ccw/README.md +30 -6
- package/ccw/bin/ccw-mcp.js +7 -0
- package/ccw/bin/ccw.js +9 -9
- package/ccw/package.json +65 -47
- package/ccw/src/.workflow/.cli-history/history.db +0 -0
- package/ccw/src/.workflow/.cli-history/history.db-shm +0 -0
- package/ccw/src/.workflow/.cli-history/history.db-wal +0 -0
- package/ccw/src/cli.ts +244 -0
- package/ccw/src/commands/cli.ts +740 -0
- package/ccw/src/commands/core-memory.ts +770 -0
- package/ccw/src/commands/hook.ts +315 -0
- package/ccw/src/commands/install.ts +519 -0
- package/ccw/src/commands/{list.js → list.ts} +1 -1
- package/ccw/src/commands/memory.ts +1090 -0
- package/ccw/src/commands/{serve.js → serve.ts} +14 -5
- package/ccw/src/commands/session-path-resolver.ts +372 -0
- package/ccw/src/commands/session.ts +1141 -0
- package/ccw/src/commands/{stop.js → stop.ts} +16 -6
- package/ccw/src/commands/tool.ts +201 -0
- package/ccw/src/commands/{uninstall.js → uninstall.ts} +89 -40
- package/ccw/src/commands/{upgrade.js → upgrade.ts} +68 -23
- package/ccw/src/commands/{view.js → view.ts} +22 -8
- package/ccw/src/config/storage-paths.ts +670 -0
- package/ccw/src/core/cache-manager.ts +294 -0
- package/ccw/src/core/claude-freshness.ts +319 -0
- package/ccw/src/core/core-memory-store.ts +1528 -0
- package/ccw/src/core/{dashboard-generator-patch.js → dashboard-generator-patch.ts} +18 -0
- package/ccw/src/core/{dashboard-generator.js → dashboard-generator.ts} +69 -12
- package/ccw/src/core/data-aggregator.ts +584 -0
- package/ccw/src/core/history-importer.ts +625 -0
- package/ccw/src/core/{lite-scanner.js → lite-scanner-complete.ts} +162 -66
- package/ccw/src/core/lite-scanner.ts +469 -0
- package/ccw/src/core/{manifest.js → manifest.ts} +104 -34
- package/ccw/src/core/memory-embedder-bridge.ts +262 -0
- package/ccw/src/core/memory-store.ts +978 -0
- package/ccw/src/core/routes/ccw-routes.ts +96 -0
- package/ccw/src/core/routes/claude-routes.ts +1183 -0
- package/ccw/src/core/routes/cli-routes.ts +561 -0
- package/ccw/src/core/routes/codexlens-routes.ts +806 -0
- package/ccw/src/core/routes/core-memory-routes.ts +605 -0
- package/ccw/src/core/routes/files-routes.ts +428 -0
- package/ccw/src/core/routes/graph-routes.md +164 -0
- package/ccw/src/core/routes/graph-routes.ts +626 -0
- package/ccw/src/core/routes/help-routes.ts +308 -0
- package/ccw/src/core/routes/hooks-routes.ts +405 -0
- package/ccw/src/core/routes/mcp-routes.ts +1271 -0
- package/ccw/src/core/routes/mcp-routes.ts.backup +550 -0
- package/ccw/src/core/routes/mcp-templates-db.ts +268 -0
- package/ccw/src/core/routes/memory-routes.ts +1206 -0
- package/ccw/src/core/routes/rules-routes.ts +526 -0
- package/ccw/src/core/routes/session-routes.ts +467 -0
- package/ccw/src/core/routes/skills-routes.ts +599 -0
- package/ccw/src/core/routes/status-routes.ts +57 -0
- package/ccw/src/core/routes/system-routes.ts +427 -0
- package/ccw/src/core/server.ts +431 -0
- package/ccw/src/core/session-clustering-service.ts +1258 -0
- package/ccw/src/core/session-scanner.ts +283 -0
- package/ccw/src/core/websocket.ts +190 -0
- package/ccw/src/{index.js → index.ts} +1 -0
- package/ccw/src/mcp-server/index.ts +186 -0
- package/ccw/src/templates/assets/css/github-dark.min.css +10 -0
- package/ccw/src/templates/assets/css/github.min.css +10 -0
- package/ccw/src/templates/assets/js/cytoscape.min.js +32 -0
- package/ccw/src/templates/assets/js/d3.min.js +2 -0
- package/ccw/src/templates/assets/js/highlight.min.js +1244 -0
- package/ccw/src/templates/assets/js/lucide.min.js +12 -0
- package/ccw/src/templates/assets/js/marked.min.js +69 -0
- package/ccw/src/templates/assets/js/tailwind.js +83 -0
- package/ccw/src/templates/dashboard-css/01-base.css +11 -0
- package/ccw/src/templates/dashboard-css/02-session.css +22 -0
- package/ccw/src/templates/dashboard-css/04-lite-tasks.css +10 -0
- package/ccw/src/templates/dashboard-css/06-cards.css +10 -4
- package/ccw/src/templates/dashboard-css/07-managers.css +1178 -7
- package/ccw/src/templates/dashboard-css/09-explorer.css +23 -12
- package/ccw/src/templates/dashboard-css/10-cli-status.css +337 -0
- package/ccw/src/templates/dashboard-css/11-cli-history.css +271 -0
- package/ccw/src/templates/dashboard-css/12-cli-legacy.css +796 -0
- package/ccw/src/templates/dashboard-css/13-cli-ccw.css +199 -0
- package/ccw/src/templates/dashboard-css/14-cli-modals.css +258 -0
- package/ccw/src/templates/dashboard-css/15-cli-endpoints.css +305 -0
- package/ccw/src/templates/dashboard-css/16-cli-session.css +241 -0
- package/ccw/src/templates/dashboard-css/17-cli-conversation.css +283 -0
- package/ccw/src/templates/dashboard-css/18-cli-settings.css +160 -0
- package/ccw/src/templates/dashboard-css/19-cli-native-session.css +496 -0
- package/ccw/src/templates/dashboard-css/20-cli-taskqueue.css +188 -0
- package/ccw/src/templates/dashboard-css/21-cli-toolmgmt.css +310 -0
- package/ccw/src/templates/dashboard-css/22-cli-semantic.css +240 -0
- package/ccw/src/templates/dashboard-css/23-memory.css +2390 -0
- package/ccw/src/templates/dashboard-css/24-prompt-history.css +1089 -0
- package/ccw/src/templates/dashboard-css/25-skills-rules.css +326 -0
- package/ccw/src/templates/dashboard-css/26-claude-manager.css +908 -0
- package/ccw/src/templates/dashboard-css/27-graph-explorer.css +1678 -0
- package/ccw/src/templates/dashboard-css/28-mcp-manager.css +748 -0
- package/ccw/src/templates/dashboard-css/29-help.css +264 -0
- package/ccw/src/templates/dashboard-css/30-core-memory.css +1700 -0
- package/ccw/src/templates/dashboard-js/api.js +162 -142
- package/ccw/src/templates/dashboard-js/components/carousel.js +4 -4
- package/ccw/src/templates/dashboard-js/components/cli-history.js +876 -0
- package/ccw/src/templates/dashboard-js/components/cli-status.js +978 -0
- package/ccw/src/templates/dashboard-js/components/global-notifications.js +508 -219
- package/ccw/src/templates/dashboard-js/components/hook-manager.js +1277 -282
- package/ccw/src/templates/dashboard-js/components/index-manager.js +302 -0
- package/ccw/src/templates/dashboard-js/components/mcp-manager.js +718 -27
- package/ccw/src/templates/dashboard-js/components/modals.js +66 -0
- package/ccw/src/templates/dashboard-js/components/navigation.js +80 -12
- package/ccw/src/templates/dashboard-js/components/notifications.js +758 -194
- package/ccw/src/templates/dashboard-js/components/storage-manager.js +478 -0
- package/ccw/src/templates/dashboard-js/components/tabs-other.js +157 -6
- package/ccw/src/templates/dashboard-js/components/task-queue-sidebar.js +716 -0
- package/ccw/src/templates/dashboard-js/help-i18n.js +272 -0
- package/ccw/src/templates/dashboard-js/i18n.js +2807 -0
- package/ccw/src/templates/dashboard-js/main.js +15 -0
- package/ccw/src/templates/dashboard-js/state.js +243 -42
- package/ccw/src/templates/dashboard-js/utils.js +47 -1
- package/ccw/src/templates/dashboard-js/views/claude-manager.js +912 -0
- package/ccw/src/templates/dashboard-js/views/cli-manager.js +2272 -0
- package/ccw/src/templates/dashboard-js/views/codexlens-manager.js +964 -0
- package/ccw/src/templates/dashboard-js/views/core-memory-clusters.js +503 -0
- package/ccw/src/templates/dashboard-js/views/core-memory.js +782 -0
- package/ccw/src/templates/dashboard-js/views/explorer.js +888 -852
- package/ccw/src/templates/dashboard-js/views/graph-explorer.js +1157 -0
- package/ccw/src/templates/dashboard-js/views/help.js +856 -0
- package/ccw/src/templates/dashboard-js/views/history.js +337 -0
- package/ccw/src/templates/dashboard-js/views/home.js +61 -15
- package/ccw/src/templates/dashboard-js/views/hook-manager.js +311 -43
- package/ccw/src/templates/dashboard-js/views/lite-tasks.js +204 -28
- package/ccw/src/templates/dashboard-js/views/mcp-manager.js +2187 -411
- package/ccw/src/templates/dashboard-js/views/mcp-manager.js.backup +1729 -0
- package/ccw/src/templates/dashboard-js/views/mcp-manager.js.new +928 -0
- package/ccw/src/templates/dashboard-js/views/memory.js +1221 -0
- package/ccw/src/templates/dashboard-js/views/prompt-history.js +713 -0
- package/ccw/src/templates/dashboard-js/views/rules-manager.js +828 -0
- package/ccw/src/templates/dashboard-js/views/session-detail.js +54 -53
- package/ccw/src/templates/dashboard-js/views/skills-manager.js +819 -0
- package/ccw/src/templates/dashboard.html +185 -85
- package/ccw/src/templates/hooks-config-example.json +60 -0
- package/ccw/src/tools/classify-folders.ts +245 -0
- package/ccw/src/tools/cli-config-manager.ts +268 -0
- package/ccw/src/tools/cli-executor.ts +2014 -0
- package/ccw/src/tools/cli-history-store.ts +1195 -0
- package/ccw/src/tools/codex-lens.ts +1141 -0
- package/ccw/src/tools/{convert-tokens-to-css.js → convert-tokens-to-css.ts} +73 -23
- package/ccw/src/tools/core-memory.ts +444 -0
- package/ccw/src/tools/detect-changed-modules.ts +325 -0
- package/ccw/src/tools/{discover-design-files.js → discover-design-files.ts} +74 -24
- package/ccw/src/tools/edit-file.ts +568 -0
- package/ccw/src/tools/{generate-module-docs.js → generate-module-docs.ts} +207 -185
- package/ccw/src/tools/{get-modules-by-depth.js → get-modules-by-depth.ts} +120 -79
- package/ccw/src/tools/index.ts +370 -0
- package/ccw/src/tools/native-session-discovery.ts +795 -0
- package/ccw/src/tools/notifier.ts +129 -0
- package/ccw/src/tools/read-file.ts +410 -0
- package/ccw/src/tools/resume-strategy.ts +345 -0
- package/ccw/src/tools/session-content-parser.ts +619 -0
- package/ccw/src/tools/session-manager.ts +1026 -0
- package/ccw/src/tools/smart-context.ts +228 -0
- package/ccw/src/tools/smart-search.ts +2065 -0
- package/ccw/src/tools/smart-search.ts.backup +1233 -0
- package/ccw/src/tools/storage-manager.ts +455 -0
- package/ccw/src/tools/write-file.ts +222 -0
- package/ccw/src/types/config.ts +11 -0
- package/ccw/src/types/index.ts +3 -0
- package/ccw/src/types/session.ts +25 -0
- package/ccw/src/types/tool.ts +41 -0
- package/ccw/src/utils/{browser-launcher.js → browser-launcher.ts} +10 -8
- package/ccw/src/utils/file-utils.ts +48 -0
- package/ccw/src/utils/{path-resolver.js → path-resolver.ts} +114 -78
- package/ccw/src/utils/path-validator.ts +153 -0
- package/ccw/src/utils/{ui.js → ui.ts} +32 -25
- package/codex-lens/pyproject.toml +48 -0
- package/codex-lens/src/codexlens/.workflow/.cli-history/history.db +0 -0
- package/codex-lens/src/codexlens/__init__.py +28 -0
- package/codex-lens/src/codexlens/__main__.py +14 -0
- package/codex-lens/src/codexlens/__pycache__/__init__.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/__pycache__/__main__.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/__pycache__/config.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/__pycache__/entities.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/__pycache__/errors.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/cli/__init__.py +27 -0
- package/codex-lens/src/codexlens/cli/__pycache__/__init__.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/cli/__pycache__/commands.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/cli/__pycache__/embedding_manager.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/cli/__pycache__/model_manager.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/cli/__pycache__/output.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/cli/commands.py +1931 -0
- package/codex-lens/src/codexlens/cli/embedding_manager.py +620 -0
- package/codex-lens/src/codexlens/cli/model_manager.py +289 -0
- package/codex-lens/src/codexlens/cli/output.py +124 -0
- package/codex-lens/src/codexlens/config.py +201 -0
- package/codex-lens/src/codexlens/entities.py +121 -0
- package/codex-lens/src/codexlens/errors.py +55 -0
- package/codex-lens/src/codexlens/indexing/README.md +77 -0
- package/codex-lens/src/codexlens/indexing/__init__.py +4 -0
- package/codex-lens/src/codexlens/indexing/__pycache__/__init__.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/indexing/__pycache__/symbol_extractor.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/indexing/symbol_extractor.py +243 -0
- package/codex-lens/src/codexlens/parsers/__init__.py +8 -0
- package/codex-lens/src/codexlens/parsers/__pycache__/__init__.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/parsers/__pycache__/encoding.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/parsers/__pycache__/factory.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/parsers/__pycache__/tokenizer.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/parsers/__pycache__/treesitter_parser.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/parsers/encoding.py +202 -0
- package/codex-lens/src/codexlens/parsers/factory.py +256 -0
- package/codex-lens/src/codexlens/parsers/tokenizer.py +98 -0
- package/codex-lens/src/codexlens/parsers/treesitter_parser.py +335 -0
- package/codex-lens/src/codexlens/search/__init__.py +15 -0
- package/codex-lens/src/codexlens/search/__pycache__/__init__.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/search/__pycache__/chain_search.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/search/__pycache__/enrichment.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/search/__pycache__/hybrid_search.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/search/__pycache__/query_parser.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/search/__pycache__/ranking.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/search/chain_search.py +647 -0
- package/codex-lens/src/codexlens/search/enrichment.py +150 -0
- package/codex-lens/src/codexlens/search/hybrid_search.py +313 -0
- package/codex-lens/src/codexlens/search/query_parser.py +242 -0
- package/codex-lens/src/codexlens/search/ranking.py +274 -0
- package/codex-lens/src/codexlens/semantic/__init__.py +39 -0
- package/codex-lens/src/codexlens/semantic/__pycache__/__init__.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/semantic/__pycache__/ann_index.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/semantic/__pycache__/chunker.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/semantic/__pycache__/code_extractor.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/semantic/__pycache__/embedder.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/semantic/__pycache__/graph_analyzer.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/semantic/__pycache__/llm_enhancer.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/semantic/__pycache__/vector_store.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/semantic/ann_index.py +414 -0
- package/codex-lens/src/codexlens/semantic/chunker.py +448 -0
- package/codex-lens/src/codexlens/semantic/code_extractor.py +274 -0
- package/codex-lens/src/codexlens/semantic/embedder.py +185 -0
- package/codex-lens/src/codexlens/semantic/vector_store.py +955 -0
- package/codex-lens/src/codexlens/storage/__init__.py +29 -0
- package/codex-lens/src/codexlens/storage/__pycache__/__init__.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/storage/__pycache__/dir_index.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/storage/__pycache__/file_cache.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/storage/__pycache__/index_tree.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/storage/__pycache__/migration_manager.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/storage/__pycache__/path_mapper.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/storage/__pycache__/registry.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/storage/__pycache__/sqlite_store.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/storage/__pycache__/sqlite_utils.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/storage/dir_index.py +1850 -0
- package/codex-lens/src/codexlens/storage/file_cache.py +32 -0
- package/codex-lens/src/codexlens/storage/index_tree.py +776 -0
- package/codex-lens/src/codexlens/storage/migration_manager.py +154 -0
- package/codex-lens/src/codexlens/storage/migrations/__init__.py +1 -0
- package/codex-lens/src/codexlens/storage/migrations/__pycache__/__init__.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/storage/migrations/__pycache__/migration_001_normalize_keywords.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/storage/migrations/__pycache__/migration_002_add_token_metadata.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/storage/migrations/__pycache__/migration_003_code_relationships.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/storage/migrations/__pycache__/migration_004_dual_fts.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/storage/migrations/__pycache__/migration_005_cleanup_unused_fields.cpython-313.pyc +0 -0
- package/codex-lens/src/codexlens/storage/migrations/migration_001_normalize_keywords.py +123 -0
- package/codex-lens/src/codexlens/storage/migrations/migration_002_add_token_metadata.py +48 -0
- package/codex-lens/src/codexlens/storage/migrations/migration_004_dual_fts.py +232 -0
- package/codex-lens/src/codexlens/storage/migrations/migration_005_cleanup_unused_fields.py +196 -0
- package/codex-lens/src/codexlens/storage/path_mapper.py +274 -0
- package/codex-lens/src/codexlens/storage/registry.py +670 -0
- package/codex-lens/src/codexlens/storage/sqlite_store.py +576 -0
- package/codex-lens/src/codexlens/storage/sqlite_utils.py +64 -0
- package/package.json +4 -1
- package/.claude/commands/memory/tech-research.md +0 -477
- package/.claude/scripts/classify-folders.sh +0 -39
- package/.claude/scripts/convert_tokens_to_css.sh +0 -229
- package/.claude/scripts/detect_changed_modules.sh +0 -161
- package/.claude/scripts/discover-design-files.sh +0 -87
- package/.claude/scripts/extract-animations.js +0 -243
- package/.claude/scripts/extract-computed-styles.js +0 -118
- package/.claude/scripts/extract-layout-structure.js +0 -411
- package/.claude/scripts/generate_module_docs.sh +0 -717
- package/.claude/scripts/get_modules_by_depth.sh +0 -170
- package/.claude/scripts/ui-generate-preview.sh +0 -395
- package/.claude/scripts/ui-instantiate-prototypes.sh +0 -815
- package/.claude/scripts/update_module_claude.sh +0 -337
- package/.claude/workflows/context-search-strategy.md +0 -77
- package/.claude/workflows/intelligent-tools-strategy.md +0 -662
- package/ccw/src/cli.js +0 -119
- package/ccw/src/commands/install.js +0 -324
- package/ccw/src/commands/tool.js +0 -138
- package/ccw/src/core/data-aggregator.js +0 -409
- package/ccw/src/core/server.js +0 -2063
- package/ccw/src/core/session-scanner.js +0 -235
- package/ccw/src/tools/classify-folders.js +0 -204
- package/ccw/src/tools/detect-changed-modules.js +0 -288
- package/ccw/src/tools/edit-file.js +0 -266
- package/ccw/src/tools/index.js +0 -176
- package/ccw/src/utils/file-utils.js +0 -48
|
@@ -0,0 +1,1026 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Session Manager Tool - Workflow session lifecycle management
|
|
3
|
+
* Operations: init, list, read, write, update, archive, mkdir, delete, stats
|
|
4
|
+
* Content routing via content_type + path_params
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
import { z } from 'zod';
|
|
8
|
+
import type { ToolSchema, ToolResult } from '../types/tool.js';
|
|
9
|
+
import {
|
|
10
|
+
readFileSync,
|
|
11
|
+
writeFileSync,
|
|
12
|
+
existsSync,
|
|
13
|
+
readdirSync,
|
|
14
|
+
mkdirSync,
|
|
15
|
+
renameSync,
|
|
16
|
+
rmSync,
|
|
17
|
+
statSync,
|
|
18
|
+
} from 'fs';
|
|
19
|
+
import { resolve, join, dirname } from 'path';
|
|
20
|
+
|
|
21
|
+
// Base paths for session storage
|
|
22
|
+
const WORKFLOW_BASE = '.workflow';
|
|
23
|
+
const ACTIVE_BASE = '.workflow/active';
|
|
24
|
+
const ARCHIVE_BASE = '.workflow/archives';
|
|
25
|
+
const LITE_PLAN_BASE = '.workflow/.lite-plan';
|
|
26
|
+
const LITE_FIX_BASE = '.workflow/.lite-fix';
|
|
27
|
+
|
|
28
|
+
// Session ID validation pattern (alphanumeric, hyphen, underscore)
|
|
29
|
+
const SESSION_ID_PATTERN = /^[a-zA-Z0-9_-]+$/;
|
|
30
|
+
|
|
31
|
+
// Zod schemas - using tuple syntax for z.enum
|
|
32
|
+
const ContentTypeEnum = z.enum([
|
|
33
|
+
'session', 'plan', 'task', 'summary', 'process', 'chat', 'brainstorm',
|
|
34
|
+
'review-dim', 'review-iter', 'review-fix', 'todo', 'context',
|
|
35
|
+
// Lite-specific content types
|
|
36
|
+
'lite-plan', 'lite-fix-plan', 'exploration', 'explorations-manifest',
|
|
37
|
+
'diagnosis', 'diagnoses-manifest', 'clarifications', 'execution-context', 'session-metadata'
|
|
38
|
+
]);
|
|
39
|
+
|
|
40
|
+
const OperationEnum = z.enum(['init', 'list', 'read', 'write', 'update', 'archive', 'mkdir', 'delete', 'stats']);
|
|
41
|
+
|
|
42
|
+
const LocationEnum = z.enum([
|
|
43
|
+
'active', 'archived', 'both',
|
|
44
|
+
'lite-plan', 'lite-fix', 'all'
|
|
45
|
+
]);
|
|
46
|
+
|
|
47
|
+
const ParamsSchema = z.object({
|
|
48
|
+
operation: OperationEnum,
|
|
49
|
+
session_id: z.string().optional(),
|
|
50
|
+
content_type: ContentTypeEnum.optional(),
|
|
51
|
+
content: z.union([z.string(), z.record(z.string(), z.any())]).optional(),
|
|
52
|
+
path_params: z.record(z.string(), z.string()).optional(),
|
|
53
|
+
metadata: z.record(z.string(), z.any()).optional(),
|
|
54
|
+
location: LocationEnum.optional(),
|
|
55
|
+
include_metadata: z.boolean().optional(),
|
|
56
|
+
dirs: z.array(z.string()).optional(),
|
|
57
|
+
update_status: z.boolean().optional(),
|
|
58
|
+
file_path: z.string().optional(),
|
|
59
|
+
});
|
|
60
|
+
|
|
61
|
+
type Params = z.infer<typeof ParamsSchema>;
|
|
62
|
+
type ContentType = z.infer<typeof ContentTypeEnum>;
|
|
63
|
+
type Operation = z.infer<typeof OperationEnum>;
|
|
64
|
+
type Location = z.infer<typeof LocationEnum>;
|
|
65
|
+
|
|
66
|
+
interface SessionInfo {
|
|
67
|
+
session_id: string;
|
|
68
|
+
location: string;
|
|
69
|
+
metadata?: any;
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
interface SessionLocation {
|
|
73
|
+
path: string;
|
|
74
|
+
location: string;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
interface TaskStats {
|
|
78
|
+
total: number;
|
|
79
|
+
pending: number;
|
|
80
|
+
in_progress: number;
|
|
81
|
+
completed: number;
|
|
82
|
+
blocked: number;
|
|
83
|
+
cancelled: number;
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
// Cached workflow root (computed once per execution)
|
|
87
|
+
let cachedWorkflowRoot: string | null = null;
|
|
88
|
+
|
|
89
|
+
/**
|
|
90
|
+
* Find project root by traversing up looking for .workflow directory
|
|
91
|
+
* Falls back to cwd if not found
|
|
92
|
+
*/
|
|
93
|
+
function findWorkflowRoot(): string {
|
|
94
|
+
if (cachedWorkflowRoot) return cachedWorkflowRoot;
|
|
95
|
+
|
|
96
|
+
let dir = process.cwd();
|
|
97
|
+
const root = dirname(dir) === dir ? dir : null; // filesystem root
|
|
98
|
+
|
|
99
|
+
while (dir && dir !== root) {
|
|
100
|
+
if (existsSync(join(dir, WORKFLOW_BASE))) {
|
|
101
|
+
cachedWorkflowRoot = dir;
|
|
102
|
+
return dir;
|
|
103
|
+
}
|
|
104
|
+
const parent = dirname(dir);
|
|
105
|
+
if (parent === dir) break; // reached filesystem root
|
|
106
|
+
dir = parent;
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
// Fallback to cwd (for init operation)
|
|
110
|
+
cachedWorkflowRoot = process.cwd();
|
|
111
|
+
return cachedWorkflowRoot;
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
/**
|
|
115
|
+
* Validate session ID format
|
|
116
|
+
*/
|
|
117
|
+
function validateSessionId(sessionId: string): void {
|
|
118
|
+
if (!sessionId || typeof sessionId !== 'string') {
|
|
119
|
+
throw new Error('session_id must be a non-empty string');
|
|
120
|
+
}
|
|
121
|
+
if (!SESSION_ID_PATTERN.test(sessionId)) {
|
|
122
|
+
throw new Error(
|
|
123
|
+
`Invalid session_id format: "${sessionId}". Only alphanumeric, hyphen, and underscore allowed.`
|
|
124
|
+
);
|
|
125
|
+
}
|
|
126
|
+
if (sessionId.length > 100) {
|
|
127
|
+
throw new Error('session_id must be 100 characters or less');
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
/**
|
|
132
|
+
* Validate path params to prevent path traversal
|
|
133
|
+
*/
|
|
134
|
+
function validatePathParams(pathParams: Record<string, unknown>): void {
|
|
135
|
+
for (const [key, value] of Object.entries(pathParams)) {
|
|
136
|
+
if (typeof value !== 'string') continue;
|
|
137
|
+
if (value.includes('..') || value.includes('/') || value.includes('\\')) {
|
|
138
|
+
throw new Error(`Invalid path_params.${key}: path traversal characters not allowed`);
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
/**
|
|
144
|
+
* Content type to file path routing
|
|
145
|
+
* {base} is replaced with session base path
|
|
146
|
+
* Dynamic params: {task_id}, {filename}, {dimension}, {iteration}
|
|
147
|
+
*/
|
|
148
|
+
const PATH_ROUTES: Record<ContentType, string> = {
|
|
149
|
+
// Standard WFS content types
|
|
150
|
+
session: '{base}/workflow-session.json',
|
|
151
|
+
plan: '{base}/IMPL_PLAN.md',
|
|
152
|
+
task: '{base}/.task/{task_id}.json',
|
|
153
|
+
summary: '{base}/.summaries/{task_id}-summary.md',
|
|
154
|
+
process: '{base}/.process/{filename}',
|
|
155
|
+
chat: '{base}/.chat/{filename}',
|
|
156
|
+
brainstorm: '{base}/.brainstorming/{filename}',
|
|
157
|
+
'review-dim': '{base}/.review/dimensions/{dimension}.json',
|
|
158
|
+
'review-iter': '{base}/.review/iterations/{iteration}.json',
|
|
159
|
+
'review-fix': '{base}/.review/fixes/{filename}',
|
|
160
|
+
todo: '{base}/TODO_LIST.md',
|
|
161
|
+
context: '{base}/context-package.json',
|
|
162
|
+
// Lite-specific content types
|
|
163
|
+
'lite-plan': '{base}/plan.json',
|
|
164
|
+
'lite-fix-plan': '{base}/fix-plan.json',
|
|
165
|
+
'exploration': '{base}/exploration-{angle}.json',
|
|
166
|
+
'explorations-manifest': '{base}/explorations-manifest.json',
|
|
167
|
+
'diagnosis': '{base}/diagnosis-{angle}.json',
|
|
168
|
+
'diagnoses-manifest': '{base}/diagnoses-manifest.json',
|
|
169
|
+
'clarifications': '{base}/clarifications.json',
|
|
170
|
+
'execution-context': '{base}/execution-context.json',
|
|
171
|
+
'session-metadata': '{base}/session-metadata.json',
|
|
172
|
+
};
|
|
173
|
+
|
|
174
|
+
/**
|
|
175
|
+
* Resolve path with base and parameters
|
|
176
|
+
*/
|
|
177
|
+
function resolvePath(
|
|
178
|
+
base: string,
|
|
179
|
+
contentType: ContentType,
|
|
180
|
+
pathParams: Record<string, string> = {}
|
|
181
|
+
): string {
|
|
182
|
+
const template = PATH_ROUTES[contentType];
|
|
183
|
+
if (!template) {
|
|
184
|
+
throw new Error(
|
|
185
|
+
`Unknown content_type: ${contentType}. Valid types: ${Object.keys(PATH_ROUTES).join(', ')}`
|
|
186
|
+
);
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
let path = template.replace('{base}', base);
|
|
190
|
+
|
|
191
|
+
// Replace dynamic parameters
|
|
192
|
+
for (const [key, value] of Object.entries(pathParams)) {
|
|
193
|
+
path = path.replace(`{${key}}`, value);
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
// Check for unreplaced placeholders
|
|
197
|
+
const unreplaced = path.match(/\{[^}]+\}/g);
|
|
198
|
+
if (unreplaced) {
|
|
199
|
+
throw new Error(
|
|
200
|
+
`Missing path_params: ${unreplaced.join(', ')} for content_type "${contentType}"`
|
|
201
|
+
);
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
return resolve(findWorkflowRoot(), path);
|
|
205
|
+
}
|
|
206
|
+
|
|
207
|
+
/**
|
|
208
|
+
* Get session base path
|
|
209
|
+
*/
|
|
210
|
+
function getSessionBase(
|
|
211
|
+
sessionId: string,
|
|
212
|
+
location: 'active' | 'archived' | 'lite-plan' | 'lite-fix' = 'active'
|
|
213
|
+
): string {
|
|
214
|
+
const locationMap: Record<string, string> = {
|
|
215
|
+
'active': ACTIVE_BASE,
|
|
216
|
+
'archived': ARCHIVE_BASE,
|
|
217
|
+
'lite-plan': LITE_PLAN_BASE,
|
|
218
|
+
'lite-fix': LITE_FIX_BASE,
|
|
219
|
+
};
|
|
220
|
+
const basePath = locationMap[location] || ACTIVE_BASE;
|
|
221
|
+
return resolve(findWorkflowRoot(), basePath, sessionId);
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
/**
|
|
225
|
+
* Auto-detect session location by searching all known paths
|
|
226
|
+
* Search order: active, archives, lite-plan, lite-fix
|
|
227
|
+
*/
|
|
228
|
+
function findSession(sessionId: string): SessionLocation | null {
|
|
229
|
+
const root = findWorkflowRoot();
|
|
230
|
+
const searchPaths = [
|
|
231
|
+
{ path: resolve(root, ACTIVE_BASE, sessionId), location: 'active' },
|
|
232
|
+
{ path: resolve(root, ARCHIVE_BASE, sessionId), location: 'archived' },
|
|
233
|
+
{ path: resolve(root, LITE_PLAN_BASE, sessionId), location: 'lite-plan' },
|
|
234
|
+
{ path: resolve(root, LITE_FIX_BASE, sessionId), location: 'lite-fix' },
|
|
235
|
+
];
|
|
236
|
+
|
|
237
|
+
for (const { path, location } of searchPaths) {
|
|
238
|
+
if (existsSync(path)) {
|
|
239
|
+
return { path, location };
|
|
240
|
+
}
|
|
241
|
+
}
|
|
242
|
+
return null;
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
/**
|
|
246
|
+
* Ensure directory exists
|
|
247
|
+
*/
|
|
248
|
+
function ensureDir(dirPath: string): void {
|
|
249
|
+
if (!existsSync(dirPath)) {
|
|
250
|
+
mkdirSync(dirPath, { recursive: true });
|
|
251
|
+
}
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
/**
|
|
255
|
+
* Read JSON file safely
|
|
256
|
+
*/
|
|
257
|
+
function readJsonFile(filePath: string): any {
|
|
258
|
+
if (!existsSync(filePath)) {
|
|
259
|
+
throw new Error(`File not found: ${filePath}`);
|
|
260
|
+
}
|
|
261
|
+
try {
|
|
262
|
+
const content = readFileSync(filePath, 'utf8');
|
|
263
|
+
return JSON.parse(content);
|
|
264
|
+
} catch (error) {
|
|
265
|
+
if (error instanceof SyntaxError) {
|
|
266
|
+
throw new Error(`Invalid JSON in ${filePath}: ${error.message}`);
|
|
267
|
+
}
|
|
268
|
+
throw new Error(`Failed to read ${filePath}: ${(error as Error).message}`);
|
|
269
|
+
}
|
|
270
|
+
}
|
|
271
|
+
|
|
272
|
+
/**
|
|
273
|
+
* Write JSON file with formatting
|
|
274
|
+
*/
|
|
275
|
+
function writeJsonFile(filePath: string, data: any): void {
|
|
276
|
+
ensureDir(dirname(filePath));
|
|
277
|
+
const content = JSON.stringify(data, null, 2);
|
|
278
|
+
writeFileSync(filePath, content, 'utf8');
|
|
279
|
+
}
|
|
280
|
+
|
|
281
|
+
/**
|
|
282
|
+
* Write text file
|
|
283
|
+
*/
|
|
284
|
+
function writeTextFile(filePath: string, content: string): void {
|
|
285
|
+
ensureDir(dirname(filePath));
|
|
286
|
+
writeFileSync(filePath, content, 'utf8');
|
|
287
|
+
}
|
|
288
|
+
|
|
289
|
+
// ============================================================
|
|
290
|
+
// Helper Functions
|
|
291
|
+
// ============================================================
|
|
292
|
+
|
|
293
|
+
/**
|
|
294
|
+
* List sessions in a specific directory
|
|
295
|
+
* @param dirPath - Directory to scan
|
|
296
|
+
* @param location - Location identifier for returned sessions
|
|
297
|
+
* @param prefix - Optional prefix filter (e.g., 'WFS-'), null means no filter
|
|
298
|
+
* @param includeMetadata - Whether to load metadata for each session
|
|
299
|
+
*/
|
|
300
|
+
function listSessionsInDir(
|
|
301
|
+
dirPath: string,
|
|
302
|
+
location: string,
|
|
303
|
+
prefix: string | null,
|
|
304
|
+
includeMetadata: boolean
|
|
305
|
+
): SessionInfo[] {
|
|
306
|
+
if (!existsSync(dirPath)) return [];
|
|
307
|
+
|
|
308
|
+
try {
|
|
309
|
+
const entries = readdirSync(dirPath, { withFileTypes: true });
|
|
310
|
+
return entries
|
|
311
|
+
.filter(e => e.isDirectory() && (prefix === null || e.name.startsWith(prefix)))
|
|
312
|
+
.map(e => {
|
|
313
|
+
const sessionInfo: SessionInfo = { session_id: e.name, location };
|
|
314
|
+
if (includeMetadata) {
|
|
315
|
+
// Try multiple metadata file locations
|
|
316
|
+
const metaPaths = [
|
|
317
|
+
join(dirPath, e.name, 'workflow-session.json'),
|
|
318
|
+
join(dirPath, e.name, 'session-metadata.json'),
|
|
319
|
+
join(dirPath, e.name, 'explorations-manifest.json'),
|
|
320
|
+
join(dirPath, e.name, 'diagnoses-manifest.json'),
|
|
321
|
+
];
|
|
322
|
+
for (const metaPath of metaPaths) {
|
|
323
|
+
if (existsSync(metaPath)) {
|
|
324
|
+
try {
|
|
325
|
+
sessionInfo.metadata = readJsonFile(metaPath);
|
|
326
|
+
break;
|
|
327
|
+
} catch { /* continue */ }
|
|
328
|
+
}
|
|
329
|
+
}
|
|
330
|
+
}
|
|
331
|
+
return sessionInfo;
|
|
332
|
+
});
|
|
333
|
+
} catch {
|
|
334
|
+
return [];
|
|
335
|
+
}
|
|
336
|
+
}
|
|
337
|
+
|
|
338
|
+
// ============================================================
|
|
339
|
+
// Operation Handlers
|
|
340
|
+
// ============================================================
|
|
341
|
+
|
|
342
|
+
/**
|
|
343
|
+
* Operation: init
|
|
344
|
+
* Create new session with directory structure
|
|
345
|
+
* Supports both WFS sessions and lite sessions (lite-plan, lite-fix)
|
|
346
|
+
*/
|
|
347
|
+
function executeInit(params: Params): any {
|
|
348
|
+
const { session_id, metadata, location } = params;
|
|
349
|
+
|
|
350
|
+
if (!session_id) {
|
|
351
|
+
throw new Error('Parameter "session_id" is required for init');
|
|
352
|
+
}
|
|
353
|
+
|
|
354
|
+
// Validate session_id format
|
|
355
|
+
validateSessionId(session_id);
|
|
356
|
+
|
|
357
|
+
// Auto-infer location from metadata.type if location not explicitly provided
|
|
358
|
+
// Priority: explicit location > metadata.type > default 'active'
|
|
359
|
+
const sessionLocation: 'active' | 'archived' | 'lite-plan' | 'lite-fix' =
|
|
360
|
+
(location === 'active' || location === 'archived' || location === 'lite-plan' || location === 'lite-fix')
|
|
361
|
+
? location
|
|
362
|
+
: (metadata?.type === 'lite-plan' ? 'lite-plan' :
|
|
363
|
+
metadata?.type === 'lite-fix' ? 'lite-fix' :
|
|
364
|
+
'active');
|
|
365
|
+
|
|
366
|
+
// Check if session already exists (auto-detect all locations)
|
|
367
|
+
const existing = findSession(session_id);
|
|
368
|
+
if (existing) {
|
|
369
|
+
throw new Error(`Session "${session_id}" already exists in ${existing.location}`);
|
|
370
|
+
}
|
|
371
|
+
|
|
372
|
+
const sessionPath = getSessionBase(session_id, sessionLocation);
|
|
373
|
+
|
|
374
|
+
// Create session directory structure based on type
|
|
375
|
+
ensureDir(sessionPath);
|
|
376
|
+
|
|
377
|
+
let directoriesCreated: string[] = [];
|
|
378
|
+
if (sessionLocation === 'lite-plan' || sessionLocation === 'lite-fix') {
|
|
379
|
+
// Lite sessions: minimal structure, files created by workflow
|
|
380
|
+
// No subdirectories needed initially
|
|
381
|
+
directoriesCreated = [];
|
|
382
|
+
} else {
|
|
383
|
+
// WFS sessions: standard structure
|
|
384
|
+
ensureDir(join(sessionPath, '.task'));
|
|
385
|
+
ensureDir(join(sessionPath, '.summaries'));
|
|
386
|
+
ensureDir(join(sessionPath, '.process'));
|
|
387
|
+
directoriesCreated = ['.task', '.summaries', '.process'];
|
|
388
|
+
}
|
|
389
|
+
|
|
390
|
+
// Create session metadata file if provided
|
|
391
|
+
let sessionMetadata = null;
|
|
392
|
+
if (metadata) {
|
|
393
|
+
const sessionFile = sessionLocation.startsWith('lite-')
|
|
394
|
+
? join(sessionPath, 'session-metadata.json') // Lite sessions
|
|
395
|
+
: join(sessionPath, 'workflow-session.json'); // WFS sessions
|
|
396
|
+
|
|
397
|
+
const sessionData = {
|
|
398
|
+
session_id,
|
|
399
|
+
type: metadata?.type || sessionLocation, // Preserve user-specified type if provided
|
|
400
|
+
status: 'initialized',
|
|
401
|
+
created_at: new Date().toISOString(),
|
|
402
|
+
...metadata,
|
|
403
|
+
};
|
|
404
|
+
writeJsonFile(sessionFile, sessionData);
|
|
405
|
+
sessionMetadata = sessionData;
|
|
406
|
+
}
|
|
407
|
+
|
|
408
|
+
return {
|
|
409
|
+
operation: 'init',
|
|
410
|
+
session_id,
|
|
411
|
+
location: sessionLocation,
|
|
412
|
+
path: sessionPath,
|
|
413
|
+
directories_created: directoriesCreated,
|
|
414
|
+
metadata: sessionMetadata,
|
|
415
|
+
message: `Session "${session_id}" initialized in ${sessionLocation}`,
|
|
416
|
+
};
|
|
417
|
+
}
|
|
418
|
+
|
|
419
|
+
/**
|
|
420
|
+
* Operation: list
|
|
421
|
+
* List sessions (active, archived, lite-plan, lite-fix, or all)
|
|
422
|
+
*/
|
|
423
|
+
function executeList(params: Params): any {
|
|
424
|
+
const { location = 'both', include_metadata = false } = params;
|
|
425
|
+
|
|
426
|
+
const result: {
|
|
427
|
+
operation: string;
|
|
428
|
+
active: SessionInfo[];
|
|
429
|
+
archived: SessionInfo[];
|
|
430
|
+
litePlan: SessionInfo[];
|
|
431
|
+
liteFix: SessionInfo[];
|
|
432
|
+
total: number;
|
|
433
|
+
} = {
|
|
434
|
+
operation: 'list',
|
|
435
|
+
active: [],
|
|
436
|
+
archived: [],
|
|
437
|
+
litePlan: [],
|
|
438
|
+
liteFix: [],
|
|
439
|
+
total: 0,
|
|
440
|
+
};
|
|
441
|
+
|
|
442
|
+
const root = findWorkflowRoot();
|
|
443
|
+
|
|
444
|
+
// Helper to check if location should be included
|
|
445
|
+
const shouldInclude = (loc: string) =>
|
|
446
|
+
location === 'all' || location === 'both' || location === loc;
|
|
447
|
+
|
|
448
|
+
// List active sessions (WFS-* prefix)
|
|
449
|
+
if (shouldInclude('active')) {
|
|
450
|
+
result.active = listSessionsInDir(
|
|
451
|
+
resolve(root, ACTIVE_BASE),
|
|
452
|
+
'active',
|
|
453
|
+
'WFS-',
|
|
454
|
+
include_metadata
|
|
455
|
+
);
|
|
456
|
+
}
|
|
457
|
+
|
|
458
|
+
// List archived sessions (WFS-* prefix)
|
|
459
|
+
if (shouldInclude('archived')) {
|
|
460
|
+
result.archived = listSessionsInDir(
|
|
461
|
+
resolve(root, ARCHIVE_BASE),
|
|
462
|
+
'archived',
|
|
463
|
+
'WFS-',
|
|
464
|
+
include_metadata
|
|
465
|
+
);
|
|
466
|
+
}
|
|
467
|
+
|
|
468
|
+
// List lite-plan sessions (no prefix filter)
|
|
469
|
+
if (location === 'all' || location === 'lite-plan') {
|
|
470
|
+
result.litePlan = listSessionsInDir(
|
|
471
|
+
resolve(root, LITE_PLAN_BASE),
|
|
472
|
+
'lite-plan',
|
|
473
|
+
null,
|
|
474
|
+
include_metadata
|
|
475
|
+
);
|
|
476
|
+
}
|
|
477
|
+
|
|
478
|
+
// List lite-fix sessions (no prefix filter)
|
|
479
|
+
if (location === 'all' || location === 'lite-fix') {
|
|
480
|
+
result.liteFix = listSessionsInDir(
|
|
481
|
+
resolve(root, LITE_FIX_BASE),
|
|
482
|
+
'lite-fix',
|
|
483
|
+
null,
|
|
484
|
+
include_metadata
|
|
485
|
+
);
|
|
486
|
+
}
|
|
487
|
+
|
|
488
|
+
result.total = result.active.length + result.archived.length +
|
|
489
|
+
result.litePlan.length + result.liteFix.length;
|
|
490
|
+
|
|
491
|
+
return result;
|
|
492
|
+
}
|
|
493
|
+
|
|
494
|
+
/**
|
|
495
|
+
* Operation: read
|
|
496
|
+
* Read file content by content_type
|
|
497
|
+
*/
|
|
498
|
+
function executeRead(params: Params): any {
|
|
499
|
+
const { session_id, content_type, path_params = {} } = params;
|
|
500
|
+
|
|
501
|
+
if (!session_id) {
|
|
502
|
+
throw new Error('Parameter "session_id" is required for read');
|
|
503
|
+
}
|
|
504
|
+
if (!content_type) {
|
|
505
|
+
throw new Error('Parameter "content_type" is required for read');
|
|
506
|
+
}
|
|
507
|
+
|
|
508
|
+
// Validate inputs
|
|
509
|
+
validateSessionId(session_id);
|
|
510
|
+
validatePathParams(path_params);
|
|
511
|
+
|
|
512
|
+
const session = findSession(session_id);
|
|
513
|
+
if (!session) {
|
|
514
|
+
throw new Error(`Session "${session_id}" not found`);
|
|
515
|
+
}
|
|
516
|
+
|
|
517
|
+
const filePath = resolvePath(session.path, content_type, path_params as Record<string, string>);
|
|
518
|
+
|
|
519
|
+
if (!existsSync(filePath)) {
|
|
520
|
+
throw new Error(`File not found: ${filePath}`);
|
|
521
|
+
}
|
|
522
|
+
|
|
523
|
+
// Read content
|
|
524
|
+
const rawContent = readFileSync(filePath, 'utf8');
|
|
525
|
+
|
|
526
|
+
// Parse JSON for JSON content types
|
|
527
|
+
const isJson = filePath.endsWith('.json');
|
|
528
|
+
const content = isJson ? JSON.parse(rawContent) : rawContent;
|
|
529
|
+
|
|
530
|
+
return {
|
|
531
|
+
operation: 'read',
|
|
532
|
+
session_id,
|
|
533
|
+
content_type,
|
|
534
|
+
path: filePath,
|
|
535
|
+
location: session.location,
|
|
536
|
+
content,
|
|
537
|
+
is_json: isJson,
|
|
538
|
+
};
|
|
539
|
+
}
|
|
540
|
+
|
|
541
|
+
/**
|
|
542
|
+
* Operation: write
|
|
543
|
+
* Write content to file by content_type
|
|
544
|
+
*/
|
|
545
|
+
function executeWrite(params: Params): any {
|
|
546
|
+
const { session_id, content_type, content, path_params = {} } = params;
|
|
547
|
+
|
|
548
|
+
if (!session_id) {
|
|
549
|
+
throw new Error('Parameter "session_id" is required for write');
|
|
550
|
+
}
|
|
551
|
+
if (!content_type) {
|
|
552
|
+
throw new Error('Parameter "content_type" is required for write');
|
|
553
|
+
}
|
|
554
|
+
if (content === undefined) {
|
|
555
|
+
throw new Error('Parameter "content" is required for write');
|
|
556
|
+
}
|
|
557
|
+
|
|
558
|
+
// Validate inputs
|
|
559
|
+
validateSessionId(session_id);
|
|
560
|
+
validatePathParams(path_params);
|
|
561
|
+
|
|
562
|
+
const session = findSession(session_id);
|
|
563
|
+
if (!session) {
|
|
564
|
+
throw new Error(`Session "${session_id}" not found. Use init operation first.`);
|
|
565
|
+
}
|
|
566
|
+
|
|
567
|
+
const filePath = resolvePath(session.path, content_type, path_params as Record<string, string>);
|
|
568
|
+
const isJson = filePath.endsWith('.json');
|
|
569
|
+
|
|
570
|
+
// Write content
|
|
571
|
+
if (isJson) {
|
|
572
|
+
writeJsonFile(filePath, content);
|
|
573
|
+
} else {
|
|
574
|
+
writeTextFile(filePath, typeof content === 'string' ? content : JSON.stringify(content, null, 2));
|
|
575
|
+
}
|
|
576
|
+
|
|
577
|
+
// Return written content for task/summary types
|
|
578
|
+
const returnContent =
|
|
579
|
+
content_type === 'task' || content_type === 'summary' ? content : undefined;
|
|
580
|
+
|
|
581
|
+
return {
|
|
582
|
+
operation: 'write',
|
|
583
|
+
session_id,
|
|
584
|
+
content_type,
|
|
585
|
+
written_content: returnContent,
|
|
586
|
+
path: filePath,
|
|
587
|
+
location: session.location,
|
|
588
|
+
message: `File written successfully`,
|
|
589
|
+
};
|
|
590
|
+
}
|
|
591
|
+
|
|
592
|
+
/**
|
|
593
|
+
* Operation: update
|
|
594
|
+
* Update existing JSON file with shallow merge
|
|
595
|
+
*/
|
|
596
|
+
function executeUpdate(params: Params): any {
|
|
597
|
+
const { session_id, content_type, content, path_params = {} } = params;
|
|
598
|
+
|
|
599
|
+
if (!session_id) {
|
|
600
|
+
throw new Error('Parameter "session_id" is required for update');
|
|
601
|
+
}
|
|
602
|
+
if (!content_type) {
|
|
603
|
+
throw new Error('Parameter "content_type" is required for update');
|
|
604
|
+
}
|
|
605
|
+
if (!content || typeof content !== 'object') {
|
|
606
|
+
throw new Error('Parameter "content" must be an object for update');
|
|
607
|
+
}
|
|
608
|
+
|
|
609
|
+
const session = findSession(session_id);
|
|
610
|
+
if (!session) {
|
|
611
|
+
throw new Error(`Session "${session_id}" not found`);
|
|
612
|
+
}
|
|
613
|
+
|
|
614
|
+
const filePath = resolvePath(session.path, content_type, path_params as Record<string, string>);
|
|
615
|
+
|
|
616
|
+
if (!filePath.endsWith('.json')) {
|
|
617
|
+
throw new Error('Update operation only supports JSON files');
|
|
618
|
+
}
|
|
619
|
+
|
|
620
|
+
// Read existing content or start with empty object
|
|
621
|
+
let existing: any = {};
|
|
622
|
+
if (existsSync(filePath)) {
|
|
623
|
+
existing = readJsonFile(filePath);
|
|
624
|
+
}
|
|
625
|
+
|
|
626
|
+
// Shallow merge
|
|
627
|
+
const merged = { ...existing, ...(content as object) };
|
|
628
|
+
writeJsonFile(filePath, merged);
|
|
629
|
+
|
|
630
|
+
return {
|
|
631
|
+
operation: 'update',
|
|
632
|
+
session_id,
|
|
633
|
+
content_type,
|
|
634
|
+
path: filePath,
|
|
635
|
+
location: session.location,
|
|
636
|
+
fields_updated: Object.keys(content as object),
|
|
637
|
+
merged_data: merged,
|
|
638
|
+
message: `File updated successfully`,
|
|
639
|
+
};
|
|
640
|
+
}
|
|
641
|
+
|
|
642
|
+
/**
|
|
643
|
+
* Operation: archive
|
|
644
|
+
* Move session from active to archives
|
|
645
|
+
*/
|
|
646
|
+
function executeArchive(params: Params): any {
|
|
647
|
+
const { session_id, update_status = true } = params;
|
|
648
|
+
|
|
649
|
+
if (!session_id) {
|
|
650
|
+
throw new Error('Parameter "session_id" is required for archive');
|
|
651
|
+
}
|
|
652
|
+
|
|
653
|
+
// Find session in any location
|
|
654
|
+
const session = findSession(session_id);
|
|
655
|
+
if (!session) {
|
|
656
|
+
throw new Error(`Session "${session_id}" not found`);
|
|
657
|
+
}
|
|
658
|
+
|
|
659
|
+
// Lite sessions do not support archiving
|
|
660
|
+
if (session.location === 'lite-plan' || session.location === 'lite-fix') {
|
|
661
|
+
throw new Error(`Lite sessions (${session.location}) do not support archiving. Use delete operation instead.`);
|
|
662
|
+
}
|
|
663
|
+
|
|
664
|
+
// Determine archive destination based on source location
|
|
665
|
+
let archivePath: string;
|
|
666
|
+
|
|
667
|
+
if (session.location === 'active') {
|
|
668
|
+
archivePath = getSessionBase(session_id, 'archived');
|
|
669
|
+
} else {
|
|
670
|
+
// Already archived
|
|
671
|
+
return {
|
|
672
|
+
operation: 'archive',
|
|
673
|
+
session_id,
|
|
674
|
+
status: 'already_archived',
|
|
675
|
+
path: session.path,
|
|
676
|
+
location: session.location,
|
|
677
|
+
message: `Session "${session_id}" is already archived`,
|
|
678
|
+
};
|
|
679
|
+
}
|
|
680
|
+
|
|
681
|
+
// Update status before archiving
|
|
682
|
+
if (update_status) {
|
|
683
|
+
const metadataFiles = [
|
|
684
|
+
join(session.path, 'workflow-session.json'),
|
|
685
|
+
join(session.path, 'session-metadata.json'),
|
|
686
|
+
join(session.path, 'explorations-manifest.json'),
|
|
687
|
+
];
|
|
688
|
+
for (const metaFile of metadataFiles) {
|
|
689
|
+
if (existsSync(metaFile)) {
|
|
690
|
+
try {
|
|
691
|
+
const data = readJsonFile(metaFile);
|
|
692
|
+
data.status = 'completed';
|
|
693
|
+
data.archived_at = new Date().toISOString();
|
|
694
|
+
writeJsonFile(metaFile, data);
|
|
695
|
+
break;
|
|
696
|
+
} catch { /* continue */ }
|
|
697
|
+
}
|
|
698
|
+
}
|
|
699
|
+
}
|
|
700
|
+
|
|
701
|
+
// Ensure archive directory exists
|
|
702
|
+
ensureDir(dirname(archivePath));
|
|
703
|
+
|
|
704
|
+
// Move session directory
|
|
705
|
+
renameSync(session.path, archivePath);
|
|
706
|
+
|
|
707
|
+
// Read session metadata after archiving
|
|
708
|
+
let sessionMetadata = null;
|
|
709
|
+
const metadataFiles = [
|
|
710
|
+
join(archivePath, 'workflow-session.json'),
|
|
711
|
+
join(archivePath, 'session-metadata.json'),
|
|
712
|
+
join(archivePath, 'explorations-manifest.json'),
|
|
713
|
+
];
|
|
714
|
+
for (const metaFile of metadataFiles) {
|
|
715
|
+
if (existsSync(metaFile)) {
|
|
716
|
+
try {
|
|
717
|
+
sessionMetadata = readJsonFile(metaFile);
|
|
718
|
+
break;
|
|
719
|
+
} catch { /* continue */ }
|
|
720
|
+
}
|
|
721
|
+
}
|
|
722
|
+
|
|
723
|
+
return {
|
|
724
|
+
operation: 'archive',
|
|
725
|
+
session_id,
|
|
726
|
+
status: 'archived',
|
|
727
|
+
source: session.path,
|
|
728
|
+
source_location: session.location,
|
|
729
|
+
destination: archivePath,
|
|
730
|
+
metadata: sessionMetadata,
|
|
731
|
+
message: `Session "${session_id}" archived from ${session.location}`,
|
|
732
|
+
};
|
|
733
|
+
}
|
|
734
|
+
|
|
735
|
+
/**
|
|
736
|
+
* Operation: mkdir
|
|
737
|
+
* Create directory structure within session
|
|
738
|
+
*/
|
|
739
|
+
function executeMkdir(params: Params): any {
|
|
740
|
+
const { session_id, dirs } = params;
|
|
741
|
+
|
|
742
|
+
if (!session_id) {
|
|
743
|
+
throw new Error('Parameter "session_id" is required for mkdir');
|
|
744
|
+
}
|
|
745
|
+
if (!dirs || !Array.isArray(dirs) || dirs.length === 0) {
|
|
746
|
+
throw new Error('Parameter "dirs" must be a non-empty array');
|
|
747
|
+
}
|
|
748
|
+
|
|
749
|
+
const session = findSession(session_id);
|
|
750
|
+
if (!session) {
|
|
751
|
+
throw new Error(`Session "${session_id}" not found`);
|
|
752
|
+
}
|
|
753
|
+
|
|
754
|
+
const created: string[] = [];
|
|
755
|
+
for (const dir of dirs) {
|
|
756
|
+
const dirPath = join(session.path, dir);
|
|
757
|
+
ensureDir(dirPath);
|
|
758
|
+
created.push(dir);
|
|
759
|
+
}
|
|
760
|
+
|
|
761
|
+
return {
|
|
762
|
+
operation: 'mkdir',
|
|
763
|
+
session_id,
|
|
764
|
+
location: session.location,
|
|
765
|
+
directories_created: created,
|
|
766
|
+
message: `Created ${created.length} directories`,
|
|
767
|
+
};
|
|
768
|
+
}
|
|
769
|
+
|
|
770
|
+
/**
|
|
771
|
+
* Operation: delete
|
|
772
|
+
* Delete a file within session (security: path traversal prevention)
|
|
773
|
+
*/
|
|
774
|
+
function executeDelete(params: Params): any {
|
|
775
|
+
const { session_id, file_path } = params;
|
|
776
|
+
|
|
777
|
+
if (!session_id) {
|
|
778
|
+
throw new Error('Parameter "session_id" is required for delete');
|
|
779
|
+
}
|
|
780
|
+
if (!file_path) {
|
|
781
|
+
throw new Error('Parameter "file_path" is required for delete');
|
|
782
|
+
}
|
|
783
|
+
|
|
784
|
+
// Validate session exists
|
|
785
|
+
const session = findSession(session_id);
|
|
786
|
+
if (!session) {
|
|
787
|
+
throw new Error(`Session "${session_id}" not found`);
|
|
788
|
+
}
|
|
789
|
+
|
|
790
|
+
// Security: Prevent path traversal
|
|
791
|
+
if (file_path.includes('..') || file_path.includes('\\')) {
|
|
792
|
+
throw new Error('Invalid file_path: path traversal characters not allowed');
|
|
793
|
+
}
|
|
794
|
+
|
|
795
|
+
// Construct absolute path
|
|
796
|
+
const absolutePath = resolve(session.path, file_path);
|
|
797
|
+
|
|
798
|
+
// Security: Verify path is within session directory
|
|
799
|
+
if (!absolutePath.startsWith(session.path)) {
|
|
800
|
+
throw new Error('Security error: file_path must be within session directory');
|
|
801
|
+
}
|
|
802
|
+
|
|
803
|
+
// Check file exists
|
|
804
|
+
if (!existsSync(absolutePath)) {
|
|
805
|
+
throw new Error(`File not found: ${file_path}`);
|
|
806
|
+
}
|
|
807
|
+
|
|
808
|
+
// Delete the file
|
|
809
|
+
rmSync(absolutePath, { force: true });
|
|
810
|
+
|
|
811
|
+
return {
|
|
812
|
+
operation: 'delete',
|
|
813
|
+
session_id,
|
|
814
|
+
deleted: file_path,
|
|
815
|
+
absolute_path: absolutePath,
|
|
816
|
+
message: `File deleted successfully`,
|
|
817
|
+
};
|
|
818
|
+
}
|
|
819
|
+
|
|
820
|
+
/**
|
|
821
|
+
* Operation: stats
|
|
822
|
+
* Get session statistics (tasks, summaries, plan)
|
|
823
|
+
*/
|
|
824
|
+
function executeStats(params: Params): any {
|
|
825
|
+
const { session_id } = params;
|
|
826
|
+
|
|
827
|
+
if (!session_id) {
|
|
828
|
+
throw new Error('Parameter "session_id" is required for stats');
|
|
829
|
+
}
|
|
830
|
+
|
|
831
|
+
// Validate session exists
|
|
832
|
+
const session = findSession(session_id);
|
|
833
|
+
if (!session) {
|
|
834
|
+
throw new Error(`Session "${session_id}" not found`);
|
|
835
|
+
}
|
|
836
|
+
|
|
837
|
+
const taskDir = join(session.path, '.task');
|
|
838
|
+
const summariesDir = join(session.path, '.summaries');
|
|
839
|
+
const planFile = join(session.path, 'IMPL_PLAN.md');
|
|
840
|
+
|
|
841
|
+
// Count tasks by status
|
|
842
|
+
const taskStats: TaskStats = {
|
|
843
|
+
total: 0,
|
|
844
|
+
pending: 0,
|
|
845
|
+
in_progress: 0,
|
|
846
|
+
completed: 0,
|
|
847
|
+
blocked: 0,
|
|
848
|
+
cancelled: 0,
|
|
849
|
+
};
|
|
850
|
+
|
|
851
|
+
if (existsSync(taskDir)) {
|
|
852
|
+
const taskFiles = readdirSync(taskDir).filter((f) => f.endsWith('.json'));
|
|
853
|
+
taskStats.total = taskFiles.length;
|
|
854
|
+
|
|
855
|
+
for (const taskFile of taskFiles) {
|
|
856
|
+
try {
|
|
857
|
+
const taskPath = join(taskDir, taskFile);
|
|
858
|
+
const taskData = readJsonFile(taskPath);
|
|
859
|
+
const status = taskData.status || 'unknown';
|
|
860
|
+
if (status in taskStats) {
|
|
861
|
+
(taskStats as any)[status]++;
|
|
862
|
+
}
|
|
863
|
+
} catch {
|
|
864
|
+
// Skip invalid task files
|
|
865
|
+
}
|
|
866
|
+
}
|
|
867
|
+
}
|
|
868
|
+
|
|
869
|
+
// Count summaries
|
|
870
|
+
let summariesCount = 0;
|
|
871
|
+
if (existsSync(summariesDir)) {
|
|
872
|
+
summariesCount = readdirSync(summariesDir).filter((f) => f.endsWith('.md')).length;
|
|
873
|
+
}
|
|
874
|
+
|
|
875
|
+
// Check for plan
|
|
876
|
+
const hasPlan = existsSync(planFile);
|
|
877
|
+
|
|
878
|
+
return {
|
|
879
|
+
operation: 'stats',
|
|
880
|
+
session_id,
|
|
881
|
+
location: session.location,
|
|
882
|
+
tasks: taskStats,
|
|
883
|
+
summaries: summariesCount,
|
|
884
|
+
has_plan: hasPlan,
|
|
885
|
+
message: `Session statistics retrieved`,
|
|
886
|
+
};
|
|
887
|
+
}
|
|
888
|
+
|
|
889
|
+
// ============================================================
|
|
890
|
+
// Main Execute Function
|
|
891
|
+
// ============================================================
|
|
892
|
+
|
|
893
|
+
/**
|
|
894
|
+
* Route to appropriate operation handler
|
|
895
|
+
*/
|
|
896
|
+
async function execute(params: Params): Promise<any> {
|
|
897
|
+
const { operation } = params;
|
|
898
|
+
|
|
899
|
+
if (!operation) {
|
|
900
|
+
throw new Error(
|
|
901
|
+
'Parameter "operation" is required. Valid operations: init, list, read, write, update, archive, mkdir, delete, stats'
|
|
902
|
+
);
|
|
903
|
+
}
|
|
904
|
+
|
|
905
|
+
switch (operation) {
|
|
906
|
+
case 'init':
|
|
907
|
+
return executeInit(params);
|
|
908
|
+
case 'list':
|
|
909
|
+
return executeList(params);
|
|
910
|
+
case 'read':
|
|
911
|
+
return executeRead(params);
|
|
912
|
+
case 'write':
|
|
913
|
+
return executeWrite(params);
|
|
914
|
+
case 'update':
|
|
915
|
+
return executeUpdate(params);
|
|
916
|
+
case 'archive':
|
|
917
|
+
return executeArchive(params);
|
|
918
|
+
case 'mkdir':
|
|
919
|
+
return executeMkdir(params);
|
|
920
|
+
case 'delete':
|
|
921
|
+
return executeDelete(params);
|
|
922
|
+
case 'stats':
|
|
923
|
+
return executeStats(params);
|
|
924
|
+
default:
|
|
925
|
+
throw new Error(
|
|
926
|
+
`Unknown operation: ${operation}. Valid operations: init, list, read, write, update, archive, mkdir, delete, stats`
|
|
927
|
+
);
|
|
928
|
+
}
|
|
929
|
+
}
|
|
930
|
+
|
|
931
|
+
// ============================================================
|
|
932
|
+
// Tool Definition
|
|
933
|
+
// ============================================================
|
|
934
|
+
|
|
935
|
+
export const schema: ToolSchema = {
|
|
936
|
+
name: 'session_manager',
|
|
937
|
+
description: `Workflow session management.
|
|
938
|
+
|
|
939
|
+
Usage:
|
|
940
|
+
session_manager(operation="init", type="workflow", description="...")
|
|
941
|
+
session_manager(operation="list", location="active|archived|both")
|
|
942
|
+
session_manager(operation="read", sessionId="WFS-xxx", contentType="plan|task|summary")
|
|
943
|
+
session_manager(operation="write", sessionId="WFS-xxx", contentType="plan", content={...})
|
|
944
|
+
session_manager(operation="archive", sessionId="WFS-xxx")
|
|
945
|
+
session_manager(operation="stats", sessionId="WFS-xxx")`,
|
|
946
|
+
inputSchema: {
|
|
947
|
+
type: 'object',
|
|
948
|
+
properties: {
|
|
949
|
+
operation: {
|
|
950
|
+
type: 'string',
|
|
951
|
+
enum: ['init', 'list', 'read', 'write', 'update', 'archive', 'mkdir', 'delete', 'stats'],
|
|
952
|
+
description: 'Operation to perform',
|
|
953
|
+
},
|
|
954
|
+
session_id: {
|
|
955
|
+
type: 'string',
|
|
956
|
+
description: 'Session identifier (e.g., WFS-my-session). Required for all operations except list.',
|
|
957
|
+
},
|
|
958
|
+
content_type: {
|
|
959
|
+
type: 'string',
|
|
960
|
+
enum: [
|
|
961
|
+
'session',
|
|
962
|
+
'plan',
|
|
963
|
+
'task',
|
|
964
|
+
'summary',
|
|
965
|
+
'process',
|
|
966
|
+
'chat',
|
|
967
|
+
'brainstorm',
|
|
968
|
+
'review-dim',
|
|
969
|
+
'review-iter',
|
|
970
|
+
'review-fix',
|
|
971
|
+
'todo',
|
|
972
|
+
'context',
|
|
973
|
+
],
|
|
974
|
+
description: 'Content type for read/write/update operations',
|
|
975
|
+
},
|
|
976
|
+
content: {
|
|
977
|
+
type: 'object',
|
|
978
|
+
description: 'Content for write/update operations (object for JSON, string for text)',
|
|
979
|
+
},
|
|
980
|
+
path_params: {
|
|
981
|
+
type: 'object',
|
|
982
|
+
description: 'Dynamic path parameters: task_id, filename, dimension, iteration',
|
|
983
|
+
},
|
|
984
|
+
metadata: {
|
|
985
|
+
type: 'object',
|
|
986
|
+
description: 'Session metadata for init operation (project, type, description, etc.)',
|
|
987
|
+
},
|
|
988
|
+
location: {
|
|
989
|
+
type: 'string',
|
|
990
|
+
enum: ['active', 'archived', 'both'],
|
|
991
|
+
description: 'Session location filter for list operation (default: both)',
|
|
992
|
+
},
|
|
993
|
+
include_metadata: {
|
|
994
|
+
type: 'boolean',
|
|
995
|
+
description: 'Include session metadata in list results (default: false)',
|
|
996
|
+
},
|
|
997
|
+
dirs: {
|
|
998
|
+
type: 'array',
|
|
999
|
+
description: 'Directory paths to create for mkdir operation',
|
|
1000
|
+
},
|
|
1001
|
+
update_status: {
|
|
1002
|
+
type: 'boolean',
|
|
1003
|
+
description: 'Update session status to completed when archiving (default: true)',
|
|
1004
|
+
},
|
|
1005
|
+
file_path: {
|
|
1006
|
+
type: 'string',
|
|
1007
|
+
description: 'Relative file path within session for delete operation',
|
|
1008
|
+
},
|
|
1009
|
+
},
|
|
1010
|
+
required: ['operation'],
|
|
1011
|
+
},
|
|
1012
|
+
};
|
|
1013
|
+
|
|
1014
|
+
export async function handler(params: Record<string, unknown>): Promise<ToolResult> {
|
|
1015
|
+
const parsed = ParamsSchema.safeParse(params);
|
|
1016
|
+
if (!parsed.success) {
|
|
1017
|
+
return { success: false, error: `Invalid params: ${parsed.error.message}` };
|
|
1018
|
+
}
|
|
1019
|
+
|
|
1020
|
+
try {
|
|
1021
|
+
const result = await execute(parsed.data);
|
|
1022
|
+
return { success: true, result };
|
|
1023
|
+
} catch (error) {
|
|
1024
|
+
return { success: false, error: (error as Error).message };
|
|
1025
|
+
}
|
|
1026
|
+
}
|