prjct-cli 1.22.0 → 1.24.0
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/CHANGELOG.md +230 -0
- package/bin/prjct +30 -13
- package/dist/bin/prjct-core.mjs +1748 -0
- package/dist/bin/prjct.mjs +17 -36672
- package/dist/cli/linear.mjs +14 -0
- package/dist/daemon/entry.mjs +1429 -0
- package/dist/templates.json +1 -0
- package/package.json +4 -5
- package/bin/prjct.ts +0 -342
- package/core/__tests__/agentic/analysis-injection.test.ts +0 -377
- package/core/__tests__/agentic/cache-eviction.test.ts +0 -294
- package/core/__tests__/agentic/command-context.test.ts +0 -281
- package/core/__tests__/agentic/command-executor.test.ts +0 -659
- package/core/__tests__/agentic/domain-classifier.test.ts +0 -330
- package/core/__tests__/agentic/injection-validator.test.ts +0 -255
- package/core/__tests__/agentic/memory-system.test.ts +0 -281
- package/core/__tests__/agentic/plan-mode.test.ts +0 -386
- package/core/__tests__/agentic/prompt-assembly.test.ts +0 -298
- package/core/__tests__/agentic/prompt-builder.test.ts +0 -243
- package/core/__tests__/agentic/response-validator.test.ts +0 -263
- package/core/__tests__/agentic/semantic-matching.test.ts +0 -131
- package/core/__tests__/agentic/smart-context.test.ts +0 -372
- package/core/__tests__/agentic/tech-normalizer.test.ts +0 -136
- package/core/__tests__/agentic/token-budget.test.ts +0 -294
- package/core/__tests__/ai-tools/formatters.test.ts +0 -476
- package/core/__tests__/domain/bm25.test.ts +0 -225
- package/core/__tests__/domain/change-propagator.test.ts +0 -100
- package/core/__tests__/domain/fibonacci.test.ts +0 -113
- package/core/__tests__/domain/file-hasher.test.ts +0 -146
- package/core/__tests__/domain/file-ranker.test.ts +0 -169
- package/core/__tests__/domain/git-cochange.test.ts +0 -121
- package/core/__tests__/domain/import-graph.test.ts +0 -156
- package/core/__tests__/domain/velocity.test.ts +0 -623
- package/core/__tests__/infrastructure/performance-tracker.test.ts +0 -328
- package/core/__tests__/schemas/model.test.ts +0 -272
- package/core/__tests__/services/dependency-validator.test.ts +0 -175
- package/core/__tests__/services/hierarchical-agent-resolver.test.ts +0 -359
- package/core/__tests__/services/nested-context-resolver.test.ts +0 -443
- package/core/__tests__/services/project-index.test.ts +0 -355
- package/core/__tests__/services/staleness-checker.test.ts +0 -204
- package/core/__tests__/storage/analysis-storage.test.ts +0 -641
- package/core/__tests__/storage/archive-storage.test.ts +0 -455
- package/core/__tests__/storage/safe-reader.test.ts +0 -262
- package/core/__tests__/storage/sqlite-migration.test.ts +0 -1016
- package/core/__tests__/storage/state-storage-feedback.test.ts +0 -463
- package/core/__tests__/storage/state-storage-history.test.ts +0 -469
- package/core/__tests__/storage/storage-manager.test.ts +0 -383
- package/core/__tests__/storage/subtask-handoff.test.ts +0 -237
- package/core/__tests__/types/fs.test.ts +0 -125
- package/core/__tests__/utils/date-helper.test.ts +0 -449
- package/core/__tests__/utils/output.test.ts +0 -278
- package/core/__tests__/utils/preserve-sections.test.ts +0 -216
- package/core/__tests__/utils/project-commands.test.ts +0 -71
- package/core/__tests__/utils/retry.test.ts +0 -381
- package/core/__tests__/workflow/state-machine.test.ts +0 -216
- package/core/agentic/agent-router.ts +0 -150
- package/core/agentic/anti-hallucination.ts +0 -141
- package/core/agentic/chain-of-thought.ts +0 -234
- package/core/agentic/command-classifier.ts +0 -141
- package/core/agentic/command-context.ts +0 -168
- package/core/agentic/command-executor.ts +0 -471
- package/core/agentic/context-builder.ts +0 -285
- package/core/agentic/domain-classifier.ts +0 -525
- package/core/agentic/environment-block.ts +0 -102
- package/core/agentic/ground-truth.ts +0 -706
- package/core/agentic/index.ts +0 -193
- package/core/agentic/injection-validator.ts +0 -208
- package/core/agentic/loop-detector.ts +0 -451
- package/core/agentic/memory-system.ts +0 -1547
- package/core/agentic/orchestrator-executor.ts +0 -579
- package/core/agentic/plan-mode.ts +0 -525
- package/core/agentic/prompt-builder.ts +0 -1069
- package/core/agentic/response-validator.ts +0 -98
- package/core/agentic/services.ts +0 -167
- package/core/agentic/skill-loader.ts +0 -106
- package/core/agentic/smart-context.ts +0 -393
- package/core/agentic/tech-normalizer.ts +0 -167
- package/core/agentic/template-executor.ts +0 -272
- package/core/agentic/template-loader.ts +0 -109
- package/core/agentic/token-budget.ts +0 -226
- package/core/agentic/tool-registry.ts +0 -146
- package/core/agents/index.ts +0 -28
- package/core/agents/performance.ts +0 -429
- package/core/ai-tools/formatters.ts +0 -341
- package/core/ai-tools/generator.ts +0 -144
- package/core/ai-tools/index.ts +0 -15
- package/core/ai-tools/registry.ts +0 -201
- package/core/bus/bus.ts +0 -314
- package/core/bus/index.ts +0 -8
- package/core/cli/linear.ts +0 -500
- package/core/cli/lint-meta-commentary.ts +0 -177
- package/core/cli/start.ts +0 -386
- package/core/commands/analysis.ts +0 -1274
- package/core/commands/analytics.ts +0 -342
- package/core/commands/base.ts +0 -118
- package/core/commands/cleanup.ts +0 -157
- package/core/commands/command-data.ts +0 -463
- package/core/commands/commands.ts +0 -306
- package/core/commands/context.ts +0 -238
- package/core/commands/design.ts +0 -77
- package/core/commands/index.ts +0 -19
- package/core/commands/maintenance.ts +0 -77
- package/core/commands/performance.ts +0 -114
- package/core/commands/planning.ts +0 -662
- package/core/commands/register.ts +0 -127
- package/core/commands/registry.ts +0 -444
- package/core/commands/setup.ts +0 -280
- package/core/commands/shipping.ts +0 -267
- package/core/commands/snapshots.ts +0 -297
- package/core/commands/uninstall.ts +0 -542
- package/core/commands/velocity.ts +0 -149
- package/core/commands/workflow.ts +0 -505
- package/core/config/command-context.config.json +0 -66
- package/core/constants/index.ts +0 -379
- package/core/context/generator.ts +0 -368
- package/core/context-tools/files-tool.ts +0 -577
- package/core/context-tools/imports-tool.ts +0 -400
- package/core/context-tools/index.ts +0 -434
- package/core/context-tools/recent-tool.ts +0 -301
- package/core/context-tools/signatures-tool.ts +0 -495
- package/core/context-tools/summary-tool.ts +0 -301
- package/core/context-tools/token-counter.ts +0 -273
- package/core/context-tools/types.ts +0 -253
- package/core/domain/agent-generator.ts +0 -186
- package/core/domain/agent-loader.ts +0 -419
- package/core/domain/analyzer.ts +0 -387
- package/core/domain/architecture-generator.ts +0 -108
- package/core/domain/bm25.ts +0 -525
- package/core/domain/change-propagator.ts +0 -162
- package/core/domain/context-estimator.ts +0 -175
- package/core/domain/fibonacci.ts +0 -128
- package/core/domain/file-hasher.ts +0 -296
- package/core/domain/file-ranker.ts +0 -151
- package/core/domain/git-cochange.ts +0 -250
- package/core/domain/import-graph.ts +0 -315
- package/core/domain/snapshot-manager.ts +0 -415
- package/core/domain/task-stack.ts +0 -578
- package/core/domain/velocity.ts +0 -470
- package/core/errors.ts +0 -335
- package/core/events/events.ts +0 -85
- package/core/events/index.ts +0 -8
- package/core/index.ts +0 -481
- package/core/infrastructure/agent-detector.ts +0 -135
- package/core/infrastructure/ai-provider.ts +0 -578
- package/core/infrastructure/author-detector.ts +0 -133
- package/core/infrastructure/capability-installer.ts +0 -76
- package/core/infrastructure/claude-agent.ts +0 -297
- package/core/infrastructure/command-installer.ts +0 -752
- package/core/infrastructure/config-manager.ts +0 -364
- package/core/infrastructure/editors-config.ts +0 -172
- package/core/infrastructure/path-manager.ts +0 -571
- package/core/infrastructure/performance-tracker.ts +0 -326
- package/core/infrastructure/permission-manager.ts +0 -289
- package/core/infrastructure/setup.ts +0 -1061
- package/core/infrastructure/update-checker.ts +0 -246
- package/core/integrations/issue-tracker/enricher.ts +0 -271
- package/core/integrations/issue-tracker/index.ts +0 -8
- package/core/integrations/issue-tracker/manager.ts +0 -286
- package/core/integrations/issue-tracker/types.ts +0 -310
- package/core/integrations/jira/cache.ts +0 -57
- package/core/integrations/jira/client.ts +0 -688
- package/core/integrations/jira/index.ts +0 -23
- package/core/integrations/jira/service.ts +0 -244
- package/core/integrations/linear/cache.ts +0 -68
- package/core/integrations/linear/client.ts +0 -436
- package/core/integrations/linear/index.ts +0 -20
- package/core/integrations/linear/service.ts +0 -260
- package/core/integrations/linear/sync.ts +0 -314
- package/core/outcomes/analyzer.ts +0 -286
- package/core/outcomes/index.ts +0 -34
- package/core/outcomes/recorder.ts +0 -195
- package/core/plugin/builtin/webhook.ts +0 -148
- package/core/plugin/hooks.ts +0 -315
- package/core/plugin/index.ts +0 -50
- package/core/plugin/loader.ts +0 -354
- package/core/plugin/registry.ts +0 -326
- package/core/schemas/agents.ts +0 -27
- package/core/schemas/analysis.ts +0 -530
- package/core/schemas/classification.ts +0 -91
- package/core/schemas/command-context.ts +0 -29
- package/core/schemas/enriched-task.ts +0 -291
- package/core/schemas/ideas.ts +0 -114
- package/core/schemas/index.ts +0 -53
- package/core/schemas/issues.ts +0 -159
- package/core/schemas/llm-output.ts +0 -170
- package/core/schemas/metrics.ts +0 -143
- package/core/schemas/model.ts +0 -153
- package/core/schemas/outcomes.ts +0 -487
- package/core/schemas/performance.ts +0 -128
- package/core/schemas/permissions.ts +0 -180
- package/core/schemas/prd.ts +0 -450
- package/core/schemas/project.ts +0 -57
- package/core/schemas/roadmap.ts +0 -322
- package/core/schemas/schemas.ts +0 -38
- package/core/schemas/shipped.ts +0 -109
- package/core/schemas/state.ts +0 -284
- package/core/schemas/velocity.ts +0 -103
- package/core/server/index.ts +0 -21
- package/core/server/routes-extended.ts +0 -566
- package/core/server/routes.ts +0 -176
- package/core/server/server.ts +0 -149
- package/core/server/sse.ts +0 -192
- package/core/services/agent-generator.ts +0 -385
- package/core/services/agent-service.ts +0 -168
- package/core/services/breakdown-service.ts +0 -124
- package/core/services/context-generator.ts +0 -445
- package/core/services/context-selector.ts +0 -429
- package/core/services/dependency-validator.ts +0 -318
- package/core/services/diff-generator.ts +0 -313
- package/core/services/doctor-service.ts +0 -423
- package/core/services/file-categorizer.ts +0 -448
- package/core/services/file-scorer.ts +0 -270
- package/core/services/git-analyzer.ts +0 -293
- package/core/services/hierarchical-agent-resolver.ts +0 -236
- package/core/services/hooks-service.ts +0 -685
- package/core/services/index.ts +0 -46
- package/core/services/local-state-generator.ts +0 -158
- package/core/services/memory-service.ts +0 -181
- package/core/services/nested-context-resolver.ts +0 -842
- package/core/services/project-index.ts +0 -911
- package/core/services/project-service.ts +0 -155
- package/core/services/session-tracker.ts +0 -287
- package/core/services/skill-installer.ts +0 -447
- package/core/services/skill-lock.ts +0 -132
- package/core/services/skill-service.ts +0 -306
- package/core/services/stack-detector.ts +0 -229
- package/core/services/staleness-checker.ts +0 -327
- package/core/services/sync-service.ts +0 -1515
- package/core/services/sync-verifier.ts +0 -253
- package/core/services/watch-service.ts +0 -312
- package/core/session/compaction.ts +0 -248
- package/core/session/index.ts +0 -35
- package/core/session/log-migration.ts +0 -88
- package/core/session/metrics.ts +0 -323
- package/core/session/session-log-manager.ts +0 -307
- package/core/session/task-session-manager.ts +0 -404
- package/core/session/utils.ts +0 -51
- package/core/storage/analysis-storage.ts +0 -373
- package/core/storage/archive-storage.ts +0 -205
- package/core/storage/database.ts +0 -575
- package/core/storage/ideas-storage.ts +0 -298
- package/core/storage/index-storage.ts +0 -523
- package/core/storage/index.ts +0 -79
- package/core/storage/metrics-storage.ts +0 -321
- package/core/storage/migrate-json.ts +0 -720
- package/core/storage/queue-storage.ts +0 -336
- package/core/storage/safe-reader.ts +0 -105
- package/core/storage/shipped-storage.ts +0 -253
- package/core/storage/state-storage.ts +0 -1035
- package/core/storage/storage-manager.ts +0 -205
- package/core/storage/storage.ts +0 -177
- package/core/storage/velocity-storage.ts +0 -149
- package/core/sync/auth-config.ts +0 -138
- package/core/sync/index.ts +0 -31
- package/core/sync/oauth-handler.ts +0 -143
- package/core/sync/sync-client.ts +0 -251
- package/core/sync/sync-manager.ts +0 -327
- package/core/tsconfig.json +0 -22
- package/core/types/agentic.ts +0 -760
- package/core/types/agents.ts +0 -150
- package/core/types/bus.ts +0 -193
- package/core/types/citations.ts +0 -22
- package/core/types/commands.ts +0 -399
- package/core/types/config.ts +0 -92
- package/core/types/core.ts +0 -96
- package/core/types/diff.ts +0 -41
- package/core/types/domain.ts +0 -71
- package/core/types/errors.ts +0 -111
- package/core/types/events.ts +0 -42
- package/core/types/fs.ts +0 -72
- package/core/types/index.ts +0 -510
- package/core/types/infrastructure.ts +0 -210
- package/core/types/integrations.ts +0 -31
- package/core/types/jira.ts +0 -51
- package/core/types/logger.ts +0 -17
- package/core/types/memory.ts +0 -313
- package/core/types/outcomes.ts +0 -190
- package/core/types/output.ts +0 -47
- package/core/types/plugin.ts +0 -25
- package/core/types/project-sync.ts +0 -129
- package/core/types/provider.ts +0 -163
- package/core/types/server.ts +0 -71
- package/core/types/services.ts +0 -84
- package/core/types/session.ts +0 -135
- package/core/types/stack.ts +0 -19
- package/core/types/storage.ts +0 -318
- package/core/types/sync-verifier.ts +0 -33
- package/core/types/sync.ts +0 -121
- package/core/types/task.ts +0 -72
- package/core/types/template.ts +0 -24
- package/core/types/utils.ts +0 -92
- package/core/types/workflow.ts +0 -23
- package/core/utils/agent-stream.ts +0 -140
- package/core/utils/animations.ts +0 -251
- package/core/utils/branding.ts +0 -88
- package/core/utils/cache.ts +0 -187
- package/core/utils/citations.ts +0 -39
- package/core/utils/collection-filters.ts +0 -209
- package/core/utils/date-helper.ts +0 -176
- package/core/utils/error-messages.ts +0 -38
- package/core/utils/file-helper.ts +0 -277
- package/core/utils/fs-helpers.ts +0 -14
- package/core/utils/help.ts +0 -314
- package/core/utils/jsonl-helper.ts +0 -290
- package/core/utils/keychain.ts +0 -127
- package/core/utils/logger.ts +0 -77
- package/core/utils/markdown-builder.ts +0 -280
- package/core/utils/next-steps.ts +0 -95
- package/core/utils/output.ts +0 -403
- package/core/utils/preserve-sections.ts +0 -218
- package/core/utils/project-commands.ts +0 -126
- package/core/utils/project-credentials.ts +0 -143
- package/core/utils/provider-cache.ts +0 -49
- package/core/utils/retry.ts +0 -318
- package/core/utils/runtime.ts +0 -108
- package/core/utils/session-helper.ts +0 -278
- package/core/utils/subtask-table.ts +0 -227
- package/core/utils/version.ts +0 -128
- package/core/wizard/index.ts +0 -13
- package/core/wizard/onboarding.ts +0 -633
- package/core/workflow/index.ts +0 -7
- package/core/workflow/state-machine.ts +0 -198
- package/core/workflow/workflow-preferences.ts +0 -294
- package/dist/core/infrastructure/command-installer.js +0 -1141
- package/dist/core/infrastructure/editors-config.js +0 -177
- package/dist/core/infrastructure/setup.js +0 -2244
- package/dist/core/utils/version.js +0 -141
- package/templates/agentic/agent-routing.md +0 -45
- package/templates/agentic/agents/uxui.md +0 -63
- package/templates/agentic/checklist-routing.md +0 -98
- package/templates/agentic/orchestrator.md +0 -68
- package/templates/agentic/task-fragmentation.md +0 -89
- package/templates/agents/AGENTS.md +0 -68
- package/templates/analysis/analyze.md +0 -84
- package/templates/analysis/patterns.md +0 -60
- package/templates/antigravity/SKILL.md +0 -39
- package/templates/architect/discovery.md +0 -67
- package/templates/architect/phases.md +0 -59
- package/templates/checklists/architecture.md +0 -28
- package/templates/checklists/code-quality.md +0 -28
- package/templates/checklists/data.md +0 -33
- package/templates/checklists/documentation.md +0 -33
- package/templates/checklists/infrastructure.md +0 -33
- package/templates/checklists/performance.md +0 -33
- package/templates/checklists/security.md +0 -33
- package/templates/checklists/testing.md +0 -33
- package/templates/checklists/ux-ui.md +0 -37
- package/templates/commands/analyze.md +0 -56
- package/templates/commands/auth.md +0 -234
- package/templates/commands/bug.md +0 -163
- package/templates/commands/cleanup.md +0 -19
- package/templates/commands/dash.md +0 -99
- package/templates/commands/design.md +0 -15
- package/templates/commands/done.md +0 -291
- package/templates/commands/enrich.md +0 -174
- package/templates/commands/git.md +0 -295
- package/templates/commands/history.md +0 -389
- package/templates/commands/idea.md +0 -88
- package/templates/commands/impact.md +0 -864
- package/templates/commands/init.md +0 -54
- package/templates/commands/jira.md +0 -278
- package/templates/commands/linear.md +0 -288
- package/templates/commands/merge.md +0 -206
- package/templates/commands/next.md +0 -80
- package/templates/commands/p.md +0 -67
- package/templates/commands/p.toml +0 -37
- package/templates/commands/pause.md +0 -136
- package/templates/commands/plan.md +0 -696
- package/templates/commands/prd.md +0 -356
- package/templates/commands/resume.md +0 -171
- package/templates/commands/review.md +0 -276
- package/templates/commands/serve.md +0 -118
- package/templates/commands/setup.md +0 -91
- package/templates/commands/ship.md +0 -475
- package/templates/commands/skill.md +0 -259
- package/templates/commands/spec.md +0 -218
- package/templates/commands/status.md +0 -207
- package/templates/commands/sync.md +0 -104
- package/templates/commands/task.md +0 -312
- package/templates/commands/test.md +0 -93
- package/templates/commands/update.md +0 -63
- package/templates/commands/verify.md +0 -204
- package/templates/commands/workflow.md +0 -150
- package/templates/config/skill-mappings.json +0 -82
- package/templates/context/dashboard.md +0 -256
- package/templates/context/roadmap.md +0 -221
- package/templates/cursor/commands/bug.md +0 -8
- package/templates/cursor/commands/done.md +0 -4
- package/templates/cursor/commands/pause.md +0 -6
- package/templates/cursor/commands/resume.md +0 -4
- package/templates/cursor/commands/ship.md +0 -8
- package/templates/cursor/commands/sync.md +0 -4
- package/templates/cursor/commands/task.md +0 -8
- package/templates/cursor/p.md +0 -29
- package/templates/cursor/router.mdc +0 -28
- package/templates/design/api.md +0 -95
- package/templates/design/architecture.md +0 -77
- package/templates/design/component.md +0 -89
- package/templates/design/database.md +0 -78
- package/templates/design/flow.md +0 -94
- package/templates/global/ANTIGRAVITY.md +0 -254
- package/templates/global/CLAUDE.md +0 -497
- package/templates/global/CURSOR.mdc +0 -266
- package/templates/global/GEMINI.md +0 -293
- package/templates/global/STORAGE-SPEC.md +0 -391
- package/templates/global/WINDSURF.md +0 -266
- package/templates/global/modules/CLAUDE-commands.md +0 -70
- package/templates/global/modules/CLAUDE-core.md +0 -105
- package/templates/global/modules/CLAUDE-git.md +0 -50
- package/templates/global/modules/CLAUDE-intelligence.md +0 -92
- package/templates/global/modules/CLAUDE-storage.md +0 -50
- package/templates/global/modules/module-config.json +0 -36
- package/templates/mcp-config.json +0 -19
- package/templates/permissions/default.jsonc +0 -60
- package/templates/permissions/permissive.jsonc +0 -49
- package/templates/permissions/strict.jsonc +0 -58
- package/templates/planning-methodology.md +0 -195
- package/templates/skills/code-review.md +0 -47
- package/templates/skills/debug.md +0 -61
- package/templates/skills/refactor.md +0 -47
- package/templates/subagents/agent-base.md +0 -20
- package/templates/subagents/domain/backend.md +0 -109
- package/templates/subagents/domain/database.md +0 -121
- package/templates/subagents/domain/devops.md +0 -152
- package/templates/subagents/domain/frontend.md +0 -103
- package/templates/subagents/domain/testing.md +0 -169
- package/templates/subagents/pm-expert.md +0 -366
- package/templates/subagents/workflow/chief-architect.md +0 -657
- package/templates/subagents/workflow/prjct-planner.md +0 -159
- package/templates/subagents/workflow/prjct-shipper.md +0 -188
- package/templates/subagents/workflow/prjct-workflow.md +0 -98
- package/templates/tools/bash.txt +0 -22
- package/templates/tools/edit.txt +0 -18
- package/templates/tools/glob.txt +0 -19
- package/templates/tools/grep.txt +0 -21
- package/templates/tools/read.txt +0 -14
- package/templates/tools/task.txt +0 -20
- package/templates/tools/webfetch.txt +0 -16
- package/templates/tools/websearch.txt +0 -18
- package/templates/tools/write.txt +0 -17
- package/templates/windsurf/router.md +0 -28
- package/templates/windsurf/workflows/bug.md +0 -8
- package/templates/windsurf/workflows/done.md +0 -4
- package/templates/windsurf/workflows/pause.md +0 -4
- package/templates/windsurf/workflows/resume.md +0 -4
- package/templates/windsurf/workflows/ship.md +0 -8
- package/templates/windsurf/workflows/sync.md +0 -4
- package/templates/windsurf/workflows/task.md +0 -8
|
@@ -1,578 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Task Stack
|
|
3
|
-
* Manages task breakdown and hierarchical task tracking.
|
|
4
|
-
*/
|
|
5
|
-
|
|
6
|
-
import { exec } from 'node:child_process'
|
|
7
|
-
import fs from 'node:fs/promises'
|
|
8
|
-
import path from 'node:path'
|
|
9
|
-
import { promisify } from 'node:util'
|
|
10
|
-
import type {
|
|
11
|
-
ParsedNowFile,
|
|
12
|
-
TaskStackEntry,
|
|
13
|
-
TaskStackMigrationResult,
|
|
14
|
-
TaskStackSummary,
|
|
15
|
-
TaskSwitchResult,
|
|
16
|
-
} from '../types'
|
|
17
|
-
import { getErrorMessage, isNotFoundError } from '../types/fs'
|
|
18
|
-
import log from '../utils/logger'
|
|
19
|
-
|
|
20
|
-
const execAsync = promisify(exec)
|
|
21
|
-
|
|
22
|
-
// =============================================================================
|
|
23
|
-
// Parser
|
|
24
|
-
// =============================================================================
|
|
25
|
-
|
|
26
|
-
/**
|
|
27
|
-
* Parse legacy now.md format
|
|
28
|
-
*/
|
|
29
|
-
export function parseNowFile(content: string): ParsedNowFile {
|
|
30
|
-
const result: ParsedNowFile = {
|
|
31
|
-
description: '',
|
|
32
|
-
started: null,
|
|
33
|
-
agent: null,
|
|
34
|
-
complexity: null,
|
|
35
|
-
dev: null,
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
// Check for frontmatter
|
|
39
|
-
if (content.startsWith('---')) {
|
|
40
|
-
const frontmatterEnd = content.indexOf('---', 3)
|
|
41
|
-
if (frontmatterEnd > 0) {
|
|
42
|
-
const frontmatter = content.substring(3, frontmatterEnd)
|
|
43
|
-
const lines = frontmatter.split('\n')
|
|
44
|
-
|
|
45
|
-
for (const line of lines) {
|
|
46
|
-
if (line.includes('task:')) {
|
|
47
|
-
result.description = line.split('task:')[1].trim().replace(/['"]/g, '')
|
|
48
|
-
}
|
|
49
|
-
if (line.includes('started:')) {
|
|
50
|
-
result.started = line.split('started:')[1].trim()
|
|
51
|
-
}
|
|
52
|
-
if (line.includes('agent:')) {
|
|
53
|
-
result.agent = line.split('agent:')[1].trim()
|
|
54
|
-
}
|
|
55
|
-
if (line.includes('complexity:')) {
|
|
56
|
-
result.complexity = line.split('complexity:')[1].trim()
|
|
57
|
-
}
|
|
58
|
-
if (line.includes('dev:')) {
|
|
59
|
-
result.dev = line.split('dev:')[1].trim()
|
|
60
|
-
}
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
// Get description from content if not in frontmatter
|
|
64
|
-
if (!result.description) {
|
|
65
|
-
const contentBody = content.substring(frontmatterEnd + 3).trim()
|
|
66
|
-
const firstLine = contentBody.split('\n')[0]
|
|
67
|
-
if (firstLine && !firstLine.startsWith('#')) {
|
|
68
|
-
result.description = firstLine.replace(/^[*-]\s*/, '').trim()
|
|
69
|
-
}
|
|
70
|
-
}
|
|
71
|
-
}
|
|
72
|
-
} else {
|
|
73
|
-
// No frontmatter, try to extract task from content
|
|
74
|
-
const lines = content.split('\n')
|
|
75
|
-
for (const line of lines) {
|
|
76
|
-
if (line.trim() && !line.startsWith('#') && !line.startsWith('---')) {
|
|
77
|
-
result.description = line.replace(/^[*-]\s*/, '').trim()
|
|
78
|
-
break
|
|
79
|
-
}
|
|
80
|
-
}
|
|
81
|
-
}
|
|
82
|
-
|
|
83
|
-
return result
|
|
84
|
-
}
|
|
85
|
-
|
|
86
|
-
/**
|
|
87
|
-
* Format duration in human-readable format
|
|
88
|
-
*/
|
|
89
|
-
export function formatDuration(ms: number): string {
|
|
90
|
-
const seconds = Math.floor(ms / 1000)
|
|
91
|
-
const minutes = Math.floor(seconds / 60)
|
|
92
|
-
const hours = Math.floor(minutes / 60)
|
|
93
|
-
const days = Math.floor(hours / 24)
|
|
94
|
-
|
|
95
|
-
if (days > 0) {
|
|
96
|
-
return `${days}d ${hours % 24}h`
|
|
97
|
-
} else if (hours > 0) {
|
|
98
|
-
return `${hours}h ${minutes % 60}m`
|
|
99
|
-
} else if (minutes > 0) {
|
|
100
|
-
return `${minutes}m`
|
|
101
|
-
} else {
|
|
102
|
-
return `${seconds}s`
|
|
103
|
-
}
|
|
104
|
-
}
|
|
105
|
-
|
|
106
|
-
// =============================================================================
|
|
107
|
-
// Storage
|
|
108
|
-
// =============================================================================
|
|
109
|
-
|
|
110
|
-
/**
|
|
111
|
-
* Ensure stack file exists
|
|
112
|
-
*/
|
|
113
|
-
export async function ensureStackFile(stackPath: string): Promise<void> {
|
|
114
|
-
try {
|
|
115
|
-
await fs.access(stackPath)
|
|
116
|
-
} catch (error) {
|
|
117
|
-
if (isNotFoundError(error)) {
|
|
118
|
-
// Create empty file
|
|
119
|
-
await fs.writeFile(stackPath, '')
|
|
120
|
-
} else {
|
|
121
|
-
throw error
|
|
122
|
-
}
|
|
123
|
-
}
|
|
124
|
-
}
|
|
125
|
-
|
|
126
|
-
/**
|
|
127
|
-
* Append entry to stack
|
|
128
|
-
*/
|
|
129
|
-
export async function appendToStack(stackPath: string, entry: TaskStackEntry): Promise<void> {
|
|
130
|
-
await ensureStackFile(stackPath)
|
|
131
|
-
const line = `${JSON.stringify(entry)}\n`
|
|
132
|
-
await fs.appendFile(stackPath, line)
|
|
133
|
-
}
|
|
134
|
-
|
|
135
|
-
/**
|
|
136
|
-
* Read all stack entries
|
|
137
|
-
*/
|
|
138
|
-
export async function readStack(stackPath: string): Promise<TaskStackEntry[]> {
|
|
139
|
-
await ensureStackFile(stackPath)
|
|
140
|
-
const content = await fs.readFile(stackPath, 'utf8')
|
|
141
|
-
|
|
142
|
-
if (!content.trim()) {
|
|
143
|
-
return []
|
|
144
|
-
}
|
|
145
|
-
|
|
146
|
-
const entries: TaskStackEntry[] = []
|
|
147
|
-
const lines = content.split('\n').filter((line) => line.trim())
|
|
148
|
-
|
|
149
|
-
for (const line of lines) {
|
|
150
|
-
try {
|
|
151
|
-
entries.push(JSON.parse(line))
|
|
152
|
-
} catch (error) {
|
|
153
|
-
log.error('Error parsing stack line:', getErrorMessage(error))
|
|
154
|
-
}
|
|
155
|
-
}
|
|
156
|
-
|
|
157
|
-
return entries
|
|
158
|
-
}
|
|
159
|
-
|
|
160
|
-
/**
|
|
161
|
-
* Write full stack to file
|
|
162
|
-
*/
|
|
163
|
-
export async function writeStack(stackPath: string, stack: TaskStackEntry[]): Promise<void> {
|
|
164
|
-
const content = `${stack.map((task) => JSON.stringify(task)).join('\n')}\n`
|
|
165
|
-
await fs.writeFile(stackPath, content)
|
|
166
|
-
}
|
|
167
|
-
|
|
168
|
-
/**
|
|
169
|
-
* Generate now.md content for a task
|
|
170
|
-
*/
|
|
171
|
-
export function generateNowContent(
|
|
172
|
-
task: TaskStackEntry | null,
|
|
173
|
-
customContent: string | null,
|
|
174
|
-
formatDurationFn: (ms: number) => string
|
|
175
|
-
): string {
|
|
176
|
-
if (customContent !== undefined && customContent !== null) {
|
|
177
|
-
return customContent
|
|
178
|
-
}
|
|
179
|
-
|
|
180
|
-
if (!task) {
|
|
181
|
-
return `# Current Task
|
|
182
|
-
|
|
183
|
-
**No active task**
|
|
184
|
-
|
|
185
|
-
Use \`/p:work\` or \`/p:resume\` to start working.
|
|
186
|
-
|
|
187
|
-
---
|
|
188
|
-
|
|
189
|
-
_Track your focus with \`/p:work [task]\`_
|
|
190
|
-
`
|
|
191
|
-
}
|
|
192
|
-
|
|
193
|
-
const started = new Date(task.started)
|
|
194
|
-
const now = new Date()
|
|
195
|
-
const elapsed = formatDurationFn(now.getTime() - started.getTime() - (task.pausedDuration || 0))
|
|
196
|
-
|
|
197
|
-
return `---
|
|
198
|
-
task: "${task.task}"
|
|
199
|
-
started: ${task.started}
|
|
200
|
-
agent: ${task.agent}
|
|
201
|
-
complexity: ${task.complexity}
|
|
202
|
-
dev: ${task.dev}
|
|
203
|
-
---
|
|
204
|
-
|
|
205
|
-
# Current Task
|
|
206
|
-
|
|
207
|
-
**${task.task}**
|
|
208
|
-
|
|
209
|
-
- Started: ${started.toLocaleTimeString()} (${elapsed} ago)
|
|
210
|
-
- Agent: ${task.agent}
|
|
211
|
-
- Complexity: ${task.complexity}
|
|
212
|
-
|
|
213
|
-
---
|
|
214
|
-
|
|
215
|
-
When done: \`/p:done\`
|
|
216
|
-
Need to pause: \`/p:pause\`
|
|
217
|
-
`
|
|
218
|
-
}
|
|
219
|
-
|
|
220
|
-
/**
|
|
221
|
-
* Update now.md file
|
|
222
|
-
*/
|
|
223
|
-
export async function updateNowFile(
|
|
224
|
-
nowPath: string,
|
|
225
|
-
task: TaskStackEntry | null,
|
|
226
|
-
customContent: string | null,
|
|
227
|
-
formatDurationFn: (ms: number) => string
|
|
228
|
-
): Promise<void> {
|
|
229
|
-
const content = generateNowContent(task, customContent, formatDurationFn)
|
|
230
|
-
await fs.writeFile(nowPath, content)
|
|
231
|
-
}
|
|
232
|
-
|
|
233
|
-
// =============================================================================
|
|
234
|
-
// Task Stack Class
|
|
235
|
-
// =============================================================================
|
|
236
|
-
|
|
237
|
-
export class TaskStack {
|
|
238
|
-
projectPath: string
|
|
239
|
-
stackPath: string
|
|
240
|
-
nowPath: string
|
|
241
|
-
|
|
242
|
-
constructor(projectPath: string) {
|
|
243
|
-
this.projectPath = projectPath
|
|
244
|
-
this.stackPath = path.join(projectPath, 'core', 'stack.jsonl')
|
|
245
|
-
this.nowPath = path.join(projectPath, 'core', 'now.md')
|
|
246
|
-
}
|
|
247
|
-
|
|
248
|
-
/**
|
|
249
|
-
* Initialize stack system - migrate from legacy now.md if needed
|
|
250
|
-
*/
|
|
251
|
-
async initialize(): Promise<TaskStackMigrationResult> {
|
|
252
|
-
try {
|
|
253
|
-
// Check if stack already exists
|
|
254
|
-
await fs.access(this.stackPath)
|
|
255
|
-
return { migrated: false }
|
|
256
|
-
} catch (error) {
|
|
257
|
-
if (isNotFoundError(error)) {
|
|
258
|
-
// Stack doesn't exist, check for legacy now.md
|
|
259
|
-
return await this.migrateFromLegacy()
|
|
260
|
-
}
|
|
261
|
-
throw error
|
|
262
|
-
}
|
|
263
|
-
}
|
|
264
|
-
|
|
265
|
-
/**
|
|
266
|
-
* Migrate from legacy now.md to stack system
|
|
267
|
-
*/
|
|
268
|
-
async migrateFromLegacy(): Promise<TaskStackMigrationResult> {
|
|
269
|
-
try {
|
|
270
|
-
const nowContent = await fs.readFile(this.nowPath, 'utf8')
|
|
271
|
-
|
|
272
|
-
if (!nowContent.trim() || nowContent.includes('No active task')) {
|
|
273
|
-
// Empty or no task, just create empty stack
|
|
274
|
-
await ensureStackFile(this.stackPath)
|
|
275
|
-
return { migrated: true, hadTask: false }
|
|
276
|
-
}
|
|
277
|
-
|
|
278
|
-
// Parse task from now.md
|
|
279
|
-
const task = parseNowFile(nowContent)
|
|
280
|
-
|
|
281
|
-
// Create initial stack entry
|
|
282
|
-
const entry: TaskStackEntry = {
|
|
283
|
-
id: `task-${Date.now()}`,
|
|
284
|
-
task: task.description || 'Migrated task',
|
|
285
|
-
agent: task.agent || 'unknown',
|
|
286
|
-
status: 'active',
|
|
287
|
-
started: task.started || new Date().toISOString(),
|
|
288
|
-
paused: null,
|
|
289
|
-
resumed: null,
|
|
290
|
-
completed: null,
|
|
291
|
-
duration: null,
|
|
292
|
-
complexity: task.complexity || 'moderate',
|
|
293
|
-
dev: task.dev || 'unknown',
|
|
294
|
-
}
|
|
295
|
-
|
|
296
|
-
// Write to stack
|
|
297
|
-
await appendToStack(this.stackPath, entry)
|
|
298
|
-
|
|
299
|
-
return { migrated: true, hadTask: true, task: entry }
|
|
300
|
-
} catch (error) {
|
|
301
|
-
// No now.md or error reading, just create empty stack
|
|
302
|
-
await ensureStackFile(this.stackPath)
|
|
303
|
-
return { migrated: true, hadTask: false, error: getErrorMessage(error) }
|
|
304
|
-
}
|
|
305
|
-
}
|
|
306
|
-
|
|
307
|
-
// Re-expose parseNowFile for compatibility
|
|
308
|
-
parseNowFile(content: string) {
|
|
309
|
-
return parseNowFile(content)
|
|
310
|
-
}
|
|
311
|
-
|
|
312
|
-
// Re-expose formatDuration for compatibility
|
|
313
|
-
formatDuration(ms: number): string {
|
|
314
|
-
return formatDuration(ms)
|
|
315
|
-
}
|
|
316
|
-
|
|
317
|
-
/**
|
|
318
|
-
* Get active task
|
|
319
|
-
*/
|
|
320
|
-
async getActiveTask(): Promise<TaskStackEntry | null> {
|
|
321
|
-
const stack = await readStack(this.stackPath)
|
|
322
|
-
return stack.find((task) => task.status === 'active') || null
|
|
323
|
-
}
|
|
324
|
-
|
|
325
|
-
/**
|
|
326
|
-
* Get paused tasks
|
|
327
|
-
*/
|
|
328
|
-
async getPausedTasks(): Promise<TaskStackEntry[]> {
|
|
329
|
-
const stack = await readStack(this.stackPath)
|
|
330
|
-
return stack
|
|
331
|
-
.filter((task) => task.status === 'paused')
|
|
332
|
-
.sort((a, b) => new Date(b.paused!).getTime() - new Date(a.paused!).getTime())
|
|
333
|
-
}
|
|
334
|
-
|
|
335
|
-
/**
|
|
336
|
-
* Get all incomplete tasks
|
|
337
|
-
*/
|
|
338
|
-
async getIncompleteTasks(): Promise<TaskStackEntry[]> {
|
|
339
|
-
const stack = await readStack(this.stackPath)
|
|
340
|
-
return stack.filter((task) => task.status !== 'completed')
|
|
341
|
-
}
|
|
342
|
-
|
|
343
|
-
/**
|
|
344
|
-
* Start a new task
|
|
345
|
-
*/
|
|
346
|
-
async startTask(
|
|
347
|
-
description: string,
|
|
348
|
-
agent: string = 'general',
|
|
349
|
-
complexity: string = 'moderate'
|
|
350
|
-
): Promise<TaskStackEntry> {
|
|
351
|
-
// Check if there's already an active task
|
|
352
|
-
const active = await this.getActiveTask()
|
|
353
|
-
if (active) {
|
|
354
|
-
throw new Error(`Already working on: ${active.task}. Use /p:pause to pause it first.`)
|
|
355
|
-
}
|
|
356
|
-
|
|
357
|
-
const entry: TaskStackEntry = {
|
|
358
|
-
id: `task-${Date.now()}`,
|
|
359
|
-
task: description,
|
|
360
|
-
agent,
|
|
361
|
-
status: 'active',
|
|
362
|
-
started: new Date().toISOString(),
|
|
363
|
-
paused: null,
|
|
364
|
-
resumed: null,
|
|
365
|
-
completed: null,
|
|
366
|
-
duration: null,
|
|
367
|
-
complexity,
|
|
368
|
-
dev: await this.getCurrentDev(),
|
|
369
|
-
}
|
|
370
|
-
|
|
371
|
-
await appendToStack(this.stackPath, entry)
|
|
372
|
-
await updateNowFile(this.nowPath, entry, null, formatDuration)
|
|
373
|
-
|
|
374
|
-
return entry
|
|
375
|
-
}
|
|
376
|
-
|
|
377
|
-
/**
|
|
378
|
-
* Pause the active task
|
|
379
|
-
*/
|
|
380
|
-
async pauseTask(reason: string = ''): Promise<TaskStackEntry> {
|
|
381
|
-
const active = await this.getActiveTask()
|
|
382
|
-
if (!active) {
|
|
383
|
-
throw new Error('No active task to pause')
|
|
384
|
-
}
|
|
385
|
-
|
|
386
|
-
// Update the task
|
|
387
|
-
active.status = 'paused'
|
|
388
|
-
active.paused = new Date().toISOString()
|
|
389
|
-
if (reason) {
|
|
390
|
-
active.pauseReason = reason
|
|
391
|
-
}
|
|
392
|
-
|
|
393
|
-
// Rewrite stack with updated task
|
|
394
|
-
await this.updateTask(active)
|
|
395
|
-
|
|
396
|
-
// Update now.md to show paused state
|
|
397
|
-
await updateNowFile(this.nowPath, null, `Paused: ${active.task}`, formatDuration)
|
|
398
|
-
|
|
399
|
-
return active
|
|
400
|
-
}
|
|
401
|
-
|
|
402
|
-
/**
|
|
403
|
-
* Resume a paused task
|
|
404
|
-
*/
|
|
405
|
-
async resumeTask(taskId: string | null = null): Promise<TaskStackEntry> {
|
|
406
|
-
// Check if there's an active task
|
|
407
|
-
const active = await this.getActiveTask()
|
|
408
|
-
if (active) {
|
|
409
|
-
throw new Error(`Already working on: ${active.task}. Complete or pause it first.`)
|
|
410
|
-
}
|
|
411
|
-
|
|
412
|
-
const paused = await this.getPausedTasks()
|
|
413
|
-
if (paused.length === 0) {
|
|
414
|
-
throw new Error('No paused tasks to resume')
|
|
415
|
-
}
|
|
416
|
-
|
|
417
|
-
let taskToResume: TaskStackEntry | undefined
|
|
418
|
-
if (taskId) {
|
|
419
|
-
taskToResume = paused.find((t) => t.id === taskId)
|
|
420
|
-
if (!taskToResume) {
|
|
421
|
-
throw new Error(`Task ${taskId} not found or not paused`)
|
|
422
|
-
}
|
|
423
|
-
} else {
|
|
424
|
-
// Resume most recently paused
|
|
425
|
-
taskToResume = paused[0]
|
|
426
|
-
}
|
|
427
|
-
|
|
428
|
-
// Update the task
|
|
429
|
-
taskToResume.status = 'active'
|
|
430
|
-
taskToResume.resumed = new Date().toISOString()
|
|
431
|
-
|
|
432
|
-
// Calculate paused duration
|
|
433
|
-
if (taskToResume.paused) {
|
|
434
|
-
const pausedMs = Date.now() - new Date(taskToResume.paused).getTime()
|
|
435
|
-
taskToResume.pausedDuration = (taskToResume.pausedDuration || 0) + pausedMs
|
|
436
|
-
}
|
|
437
|
-
|
|
438
|
-
// Rewrite stack with updated task
|
|
439
|
-
await this.updateTask(taskToResume)
|
|
440
|
-
|
|
441
|
-
// Update now.md
|
|
442
|
-
await updateNowFile(this.nowPath, taskToResume, null, formatDuration)
|
|
443
|
-
|
|
444
|
-
return taskToResume
|
|
445
|
-
}
|
|
446
|
-
|
|
447
|
-
/**
|
|
448
|
-
* Complete the active task
|
|
449
|
-
*/
|
|
450
|
-
async completeTask(): Promise<TaskStackEntry> {
|
|
451
|
-
const active = await this.getActiveTask()
|
|
452
|
-
if (!active) {
|
|
453
|
-
throw new Error('No active task to complete')
|
|
454
|
-
}
|
|
455
|
-
|
|
456
|
-
// Update the task
|
|
457
|
-
active.status = 'completed'
|
|
458
|
-
active.completed = new Date().toISOString()
|
|
459
|
-
|
|
460
|
-
// Calculate duration (excluding paused time)
|
|
461
|
-
const totalMs = Date.now() - new Date(active.started).getTime()
|
|
462
|
-
const pausedMs = active.pausedDuration || 0
|
|
463
|
-
active.duration = totalMs - pausedMs
|
|
464
|
-
active.durationFormatted = formatDuration(active.duration)
|
|
465
|
-
|
|
466
|
-
// Rewrite stack with updated task
|
|
467
|
-
await this.updateTask(active)
|
|
468
|
-
|
|
469
|
-
// Clear now.md
|
|
470
|
-
await updateNowFile(this.nowPath, null, '', formatDuration)
|
|
471
|
-
|
|
472
|
-
return active
|
|
473
|
-
}
|
|
474
|
-
|
|
475
|
-
/**
|
|
476
|
-
* Switch tasks (atomic pause + resume/start)
|
|
477
|
-
*/
|
|
478
|
-
async switchTask(targetTaskOrDescription: string): Promise<TaskSwitchResult> {
|
|
479
|
-
const active = await this.getActiveTask()
|
|
480
|
-
let pausedTask: TaskStackEntry | null = null
|
|
481
|
-
|
|
482
|
-
// Pause current if exists
|
|
483
|
-
if (active) {
|
|
484
|
-
pausedTask = await this.pauseTask('Switched to another task')
|
|
485
|
-
}
|
|
486
|
-
|
|
487
|
-
try {
|
|
488
|
-
// Check if target is a task ID or description
|
|
489
|
-
const paused = await this.getPausedTasks()
|
|
490
|
-
const existingTask = paused.find((t) => t.id === targetTaskOrDescription)
|
|
491
|
-
|
|
492
|
-
if (existingTask) {
|
|
493
|
-
// Resume existing task
|
|
494
|
-
return {
|
|
495
|
-
paused: pausedTask,
|
|
496
|
-
resumed: await this.resumeTask(targetTaskOrDescription),
|
|
497
|
-
type: 'resumed',
|
|
498
|
-
}
|
|
499
|
-
} else {
|
|
500
|
-
// Start new task
|
|
501
|
-
return {
|
|
502
|
-
paused: pausedTask,
|
|
503
|
-
started: await this.startTask(targetTaskOrDescription),
|
|
504
|
-
type: 'started',
|
|
505
|
-
}
|
|
506
|
-
}
|
|
507
|
-
} catch (error) {
|
|
508
|
-
// If switch fails, resume the original task
|
|
509
|
-
if (pausedTask) {
|
|
510
|
-
await this.resumeTask(pausedTask.id)
|
|
511
|
-
}
|
|
512
|
-
throw error
|
|
513
|
-
}
|
|
514
|
-
}
|
|
515
|
-
|
|
516
|
-
/**
|
|
517
|
-
* Update a task in the stack
|
|
518
|
-
*/
|
|
519
|
-
async updateTask(updatedTask: TaskStackEntry): Promise<void> {
|
|
520
|
-
const stack = await readStack(this.stackPath)
|
|
521
|
-
const index = stack.findIndex((t) => t.id === updatedTask.id)
|
|
522
|
-
|
|
523
|
-
if (index === -1) {
|
|
524
|
-
throw new Error(`Task ${updatedTask.id} not found`)
|
|
525
|
-
}
|
|
526
|
-
|
|
527
|
-
stack[index] = updatedTask
|
|
528
|
-
await writeStack(this.stackPath, stack)
|
|
529
|
-
}
|
|
530
|
-
|
|
531
|
-
/**
|
|
532
|
-
* Update now.md to reflect current state
|
|
533
|
-
*/
|
|
534
|
-
async updateNowFile(
|
|
535
|
-
task: TaskStackEntry | null,
|
|
536
|
-
customContent: string | null = null
|
|
537
|
-
): Promise<void> {
|
|
538
|
-
await updateNowFile(this.nowPath, task, customContent, formatDuration)
|
|
539
|
-
}
|
|
540
|
-
|
|
541
|
-
/**
|
|
542
|
-
* Get current developer from git or system
|
|
543
|
-
*/
|
|
544
|
-
async getCurrentDev(): Promise<string> {
|
|
545
|
-
try {
|
|
546
|
-
const { stdout } = await execAsync('git config user.name')
|
|
547
|
-
return stdout.trim()
|
|
548
|
-
} catch (_error) {
|
|
549
|
-
// Git not available or not configured - return unknown (expected)
|
|
550
|
-
return 'unknown'
|
|
551
|
-
}
|
|
552
|
-
}
|
|
553
|
-
|
|
554
|
-
/**
|
|
555
|
-
* Get stack summary for display
|
|
556
|
-
*/
|
|
557
|
-
async getStackSummary(): Promise<TaskStackSummary> {
|
|
558
|
-
const active = await this.getActiveTask()
|
|
559
|
-
const paused = await this.getPausedTasks()
|
|
560
|
-
const stack = await readStack(this.stackPath)
|
|
561
|
-
const completed = stack.filter((t) => t.status === 'completed')
|
|
562
|
-
|
|
563
|
-
return {
|
|
564
|
-
active,
|
|
565
|
-
paused,
|
|
566
|
-
pausedCount: paused.length,
|
|
567
|
-
completed,
|
|
568
|
-
completedCount: completed.length,
|
|
569
|
-
totalTasks: stack.length,
|
|
570
|
-
}
|
|
571
|
-
}
|
|
572
|
-
}
|
|
573
|
-
|
|
574
|
-
// =============================================================================
|
|
575
|
-
// Exports
|
|
576
|
-
// =============================================================================
|
|
577
|
-
|
|
578
|
-
export default TaskStack
|