prjct-cli 0.11.5 → 0.12.1
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 +58 -0
- package/README.md +81 -25
- package/bin/dev.js +1 -1
- package/bin/generate-views.js +209 -0
- package/bin/migrate-to-json.js +742 -0
- package/bin/prjct +5 -5
- package/bin/serve.js +226 -50
- package/core/__tests__/agentic/{memory-system.test.js → memory-system.test.ts} +12 -23
- package/core/__tests__/agentic/{plan-mode.test.js → plan-mode.test.ts} +26 -24
- package/core/__tests__/agentic/{prompt-builder.test.js → prompt-builder.test.ts} +3 -8
- package/core/__tests__/utils/{date-helper.test.js → date-helper.test.ts} +19 -30
- package/core/__tests__/utils/{output.test.js → output.test.ts} +12 -24
- package/core/agentic/agent-router.ts +137 -0
- package/core/agentic/chain-of-thought.ts +228 -0
- package/core/agentic/command-executor/command-executor.ts +384 -0
- package/core/agentic/command-executor/index.ts +16 -0
- package/core/agentic/command-executor/status-signal.ts +38 -0
- package/core/agentic/command-executor/types.ts +79 -0
- package/core/agentic/command-executor.ts +8 -0
- package/core/agentic/{context-builder.js → context-builder.ts} +92 -81
- package/core/agentic/context-filter.ts +365 -0
- package/core/agentic/ground-truth/index.ts +76 -0
- package/core/agentic/ground-truth/types.ts +33 -0
- package/core/agentic/ground-truth/utils.ts +48 -0
- package/core/agentic/ground-truth/verifiers/analyze.ts +54 -0
- package/core/agentic/ground-truth/verifiers/done.ts +75 -0
- package/core/agentic/ground-truth/verifiers/feature.ts +70 -0
- package/core/agentic/ground-truth/verifiers/index.ts +37 -0
- package/core/agentic/ground-truth/verifiers/init.ts +52 -0
- package/core/agentic/ground-truth/verifiers/now.ts +57 -0
- package/core/agentic/ground-truth/verifiers/ship.ts +85 -0
- package/core/agentic/ground-truth/verifiers/spec.ts +45 -0
- package/core/agentic/ground-truth/verifiers/sync.ts +47 -0
- package/core/agentic/ground-truth/verifiers.ts +6 -0
- package/core/agentic/ground-truth.ts +8 -0
- package/core/agentic/loop-detector/error-analysis.ts +97 -0
- package/core/agentic/loop-detector/hallucination.ts +71 -0
- package/core/agentic/loop-detector/index.ts +41 -0
- package/core/agentic/loop-detector/loop-detector.ts +222 -0
- package/core/agentic/loop-detector/types.ts +66 -0
- package/core/agentic/loop-detector.ts +8 -0
- package/core/agentic/memory-system/history.ts +53 -0
- package/core/agentic/memory-system/index.ts +192 -0
- package/core/agentic/memory-system/patterns.ts +156 -0
- package/core/agentic/memory-system/semantic-memories.ts +277 -0
- package/core/agentic/memory-system/session.ts +21 -0
- package/core/agentic/memory-system/types.ts +159 -0
- package/core/agentic/memory-system.ts +8 -0
- package/core/agentic/parallel-tools.ts +165 -0
- package/core/agentic/plan-mode/approval.ts +57 -0
- package/core/agentic/plan-mode/constants.ts +44 -0
- package/core/agentic/plan-mode/index.ts +28 -0
- package/core/agentic/plan-mode/plan-mode.ts +406 -0
- package/core/agentic/plan-mode/types.ts +193 -0
- package/core/agentic/plan-mode.ts +8 -0
- package/core/agentic/prompt-builder.ts +566 -0
- package/core/agentic/response-templates.ts +164 -0
- package/core/agentic/semantic-compression.ts +273 -0
- package/core/agentic/services.ts +206 -0
- package/core/agentic/smart-context.ts +476 -0
- package/core/agentic/{template-loader.js → template-loader.ts} +27 -16
- package/core/agentic/think-blocks.ts +202 -0
- package/core/agentic/tool-registry.ts +119 -0
- package/core/agentic/validation-rules.ts +313 -0
- package/core/agents/index.ts +28 -0
- package/core/agents/performance.ts +444 -0
- package/core/agents/types.ts +126 -0
- package/core/bus/{index.js → index.ts} +57 -61
- package/core/command-registry/categories.ts +23 -0
- package/core/command-registry/commands.ts +15 -0
- package/core/command-registry/core-commands.ts +319 -0
- package/core/command-registry/index.ts +158 -0
- package/core/command-registry/optional-commands.ts +119 -0
- package/core/command-registry/setup-commands.ts +53 -0
- package/core/command-registry/types.ts +59 -0
- package/core/command-registry.ts +9 -0
- package/core/commands/analysis.ts +298 -0
- package/core/commands/analytics.ts +288 -0
- package/core/commands/base.ts +273 -0
- package/core/commands/index.ts +211 -0
- package/core/commands/maintenance.ts +226 -0
- package/core/commands/planning.ts +311 -0
- package/core/commands/setup.ts +309 -0
- package/core/commands/shipping.ts +188 -0
- package/core/commands/types.ts +183 -0
- package/core/commands/workflow.ts +226 -0
- package/core/commands.ts +11 -0
- package/core/constants/formats.ts +187 -0
- package/core/constants/index.ts +7 -0
- package/core/{context-sync.js → context-sync.ts} +59 -26
- package/core/data/agents-manager.ts +76 -0
- package/core/data/analysis-manager.ts +83 -0
- package/core/data/base-manager.ts +156 -0
- package/core/data/ideas-manager.ts +81 -0
- package/core/data/index.ts +32 -0
- package/core/data/outcomes-manager.ts +96 -0
- package/core/data/project-manager.ts +75 -0
- package/core/data/roadmap-manager.ts +118 -0
- package/core/data/shipped-manager.ts +65 -0
- package/core/data/state-manager.ts +214 -0
- package/core/domain/{agent-generator.js → agent-generator.ts} +77 -57
- package/core/domain/{agent-loader.js → agent-loader.ts} +65 -56
- package/core/domain/{agent-matcher.js → agent-matcher.ts} +51 -24
- package/core/domain/{agent-validator.js → agent-validator.ts} +70 -37
- package/core/domain/{analyzer.js → analyzer.ts} +91 -85
- package/core/domain/{architect-session.js → architect-session.ts} +49 -34
- package/core/domain/{architecture-generator.js → architecture-generator.ts} +25 -13
- package/core/domain/{context-estimator.js → context-estimator.ts} +57 -36
- package/core/domain/{product-standards.js → product-standards.ts} +40 -26
- package/core/domain/{smart-cache.js → smart-cache.ts} +39 -30
- package/core/domain/{snapshot-manager.js → snapshot-manager.ts} +103 -100
- package/core/domain/{task-analyzer.js → task-analyzer.ts} +82 -43
- package/core/domain/task-stack/index.ts +19 -0
- package/core/domain/task-stack/parser.ts +86 -0
- package/core/domain/task-stack/storage.ts +123 -0
- package/core/domain/task-stack/task-stack.ts +340 -0
- package/core/domain/task-stack/types.ts +51 -0
- package/core/domain/task-stack.ts +8 -0
- package/core/{index.js → index.ts} +61 -18
- package/core/infrastructure/{agent-detector.js → agent-detector.ts} +55 -19
- package/core/infrastructure/agents/{claude-agent.js → claude-agent.ts} +61 -21
- package/core/infrastructure/{author-detector.js → author-detector.ts} +42 -49
- package/core/infrastructure/{capability-installer.js → capability-installer.ts} +51 -27
- package/core/infrastructure/{command-installer.js → command-installer/command-installer.ts} +43 -144
- package/core/infrastructure/command-installer/global-config.ts +106 -0
- package/core/infrastructure/command-installer/index.ts +25 -0
- package/core/infrastructure/command-installer/types.ts +41 -0
- package/core/infrastructure/command-installer.ts +8 -0
- package/core/infrastructure/{config-manager.js → config-manager.ts} +60 -80
- package/core/infrastructure/{editors-config.js → editors-config.ts} +33 -31
- package/core/infrastructure/legacy-installer-detector/cleanup.ts +216 -0
- package/core/infrastructure/legacy-installer-detector/detection.ts +95 -0
- package/core/infrastructure/legacy-installer-detector/index.ts +171 -0
- package/core/infrastructure/legacy-installer-detector/migration.ts +87 -0
- package/core/infrastructure/legacy-installer-detector/types.ts +42 -0
- package/core/infrastructure/legacy-installer-detector.ts +7 -0
- package/core/infrastructure/migrator/file-operations.ts +125 -0
- package/core/infrastructure/migrator/index.ts +288 -0
- package/core/infrastructure/migrator/project-scanner.ts +89 -0
- package/core/infrastructure/migrator/reports.ts +117 -0
- package/core/infrastructure/migrator/types.ts +124 -0
- package/core/infrastructure/migrator/validation.ts +94 -0
- package/core/infrastructure/migrator/version-migration.ts +117 -0
- package/core/infrastructure/migrator.ts +10 -0
- package/core/infrastructure/{path-manager.js → path-manager.ts} +51 -91
- package/core/infrastructure/session-manager/index.ts +23 -0
- package/core/infrastructure/session-manager/migration.ts +88 -0
- package/core/infrastructure/session-manager/session-manager.ts +307 -0
- package/core/infrastructure/session-manager/types.ts +45 -0
- package/core/infrastructure/session-manager.ts +8 -0
- package/core/infrastructure/{setup.js → setup.ts} +29 -21
- package/core/infrastructure/{update-checker.js → update-checker.ts} +40 -18
- package/core/outcomes/analyzer.ts +333 -0
- package/core/outcomes/index.ts +34 -0
- package/core/outcomes/recorder.ts +194 -0
- package/core/outcomes/types.ts +145 -0
- package/core/plugin/{hooks.js → hooks.ts} +56 -58
- package/core/plugin/{index.js → index.ts} +19 -8
- package/core/plugin/{loader.js → loader.ts} +87 -69
- package/core/plugin/{registry.js → registry.ts} +49 -45
- package/core/plugins/{webhook.js → webhook.ts} +43 -27
- package/core/schemas/agents.ts +27 -0
- package/core/schemas/analysis.ts +41 -0
- package/core/schemas/ideas.ts +83 -0
- package/core/schemas/index.ts +73 -0
- package/core/schemas/outcomes.ts +22 -0
- package/core/schemas/project.ts +26 -0
- package/core/schemas/roadmap.ts +90 -0
- package/core/schemas/shipped.ts +82 -0
- package/core/schemas/state.ts +107 -0
- package/core/session/index.ts +17 -0
- package/core/session/{metrics.js → metrics.ts} +64 -46
- package/core/session/{index.js → session-manager.ts} +51 -117
- package/core/session/types.ts +29 -0
- package/core/session/utils.ts +57 -0
- package/core/state/index.ts +25 -0
- package/core/state/manager.ts +376 -0
- package/core/state/types.ts +185 -0
- package/core/tsconfig.json +22 -0
- package/core/types/index.ts +506 -0
- package/core/utils/{animations.js → animations.ts} +74 -28
- package/core/utils/{branding.js → branding.ts} +29 -4
- package/core/utils/{date-helper.js → date-helper.ts} +31 -74
- package/core/utils/file-helper.ts +262 -0
- package/core/utils/{jsonl-helper.js → jsonl-helper.ts} +71 -107
- package/core/utils/{logger.js → logger.ts} +24 -12
- package/core/utils/{output.js → output.ts} +25 -13
- package/core/utils/{project-capabilities.js → project-capabilities.ts} +31 -18
- package/core/utils/{session-helper.js → session-helper.ts} +79 -66
- package/core/utils/{version.js → version.ts} +23 -31
- package/core/view-generator.ts +536 -0
- package/package.json +23 -17
- package/packages/shared/.turbo/turbo-build.log +14 -0
- package/packages/shared/dist/index.d.ts +8 -613
- package/packages/shared/dist/index.d.ts.map +1 -0
- package/packages/shared/dist/index.js +4110 -118
- package/packages/shared/dist/schemas.d.ts +408 -0
- package/packages/shared/dist/schemas.d.ts.map +1 -0
- package/packages/shared/dist/types.d.ts +144 -0
- package/packages/shared/dist/types.d.ts.map +1 -0
- package/packages/shared/dist/unified.d.ts +139 -0
- package/packages/shared/dist/unified.d.ts.map +1 -0
- package/packages/shared/dist/utils.d.ts +60 -0
- package/packages/shared/dist/utils.d.ts.map +1 -0
- package/packages/shared/package.json +4 -4
- package/packages/shared/src/index.ts +1 -0
- package/packages/shared/src/unified.ts +174 -0
- package/packages/web/app/api/claude/sessions/route.ts +1 -1
- package/packages/web/app/api/claude/status/route.ts +1 -1
- package/packages/web/app/api/migrate/route.ts +46 -0
- package/packages/web/app/api/projects/[id]/route.ts +1 -1
- package/packages/web/app/api/projects/[id]/stats/route.ts +30 -2
- package/packages/web/app/api/projects/[id]/status/route.ts +1 -1
- package/packages/web/app/api/projects/route.ts +1 -1
- package/packages/web/app/api/settings/route.ts +97 -0
- package/packages/web/app/api/v2/projects/[id]/unified/route.ts +57 -0
- package/packages/web/app/globals.css +38 -0
- package/packages/web/app/layout.tsx +10 -2
- package/packages/web/app/page.tsx +9 -224
- package/packages/web/app/project/[id]/page.tsx +191 -63
- package/packages/web/app/project/[id]/stats/loading.tsx +43 -0
- package/packages/web/app/project/[id]/stats/page.tsx +204 -163
- package/packages/web/app/settings/page.tsx +222 -2
- package/packages/web/components/ActivityTimeline/ActivityTimeline.constants.ts +2 -0
- package/packages/web/components/ActivityTimeline/ActivityTimeline.tsx +50 -0
- package/packages/web/components/ActivityTimeline/ActivityTimeline.types.ts +8 -0
- package/packages/web/components/ActivityTimeline/hooks/index.ts +2 -0
- package/packages/web/components/ActivityTimeline/hooks/useExpandable.ts +9 -0
- package/packages/web/components/ActivityTimeline/hooks/useGroupedEvents.ts +23 -0
- package/packages/web/components/ActivityTimeline/index.ts +2 -0
- package/packages/web/components/AgentsCard/AgentsCard.tsx +63 -0
- package/packages/web/components/AgentsCard/AgentsCard.types.ts +13 -0
- package/packages/web/components/AgentsCard/index.ts +2 -0
- package/packages/web/components/AppSidebar/AppSidebar.tsx +190 -0
- package/packages/web/components/AppSidebar/index.ts +1 -0
- package/packages/web/components/BackLink/BackLink.tsx +18 -0
- package/packages/web/components/BackLink/BackLink.types.ts +5 -0
- package/packages/web/components/BackLink/index.ts +2 -0
- package/packages/web/components/BentoCard/BentoCard.constants.ts +16 -0
- package/packages/web/components/BentoCard/BentoCard.tsx +47 -0
- package/packages/web/components/BentoCard/BentoCard.types.ts +15 -0
- package/packages/web/components/BentoCard/index.ts +2 -0
- package/packages/web/components/BentoCardSkeleton/BentoCardSkeleton.constants.ts +9 -0
- package/packages/web/components/BentoCardSkeleton/BentoCardSkeleton.tsx +18 -0
- package/packages/web/components/BentoCardSkeleton/BentoCardSkeleton.types.ts +5 -0
- package/packages/web/components/BentoCardSkeleton/index.ts +2 -0
- package/packages/web/components/{stats → BentoGrid}/BentoGrid.tsx +4 -8
- package/packages/web/components/BentoGrid/BentoGrid.types.ts +4 -0
- package/packages/web/components/BentoGrid/index.ts +2 -0
- package/packages/web/components/CommandButton/index.ts +1 -0
- package/packages/web/components/ConnectionStatus/index.ts +1 -0
- package/packages/web/components/DashboardContent/DashboardContent.tsx +254 -0
- package/packages/web/components/DashboardContent/index.ts +1 -0
- package/packages/web/components/DateGroup/DateGroup.tsx +18 -0
- package/packages/web/components/DateGroup/DateGroup.types.ts +6 -0
- package/packages/web/components/DateGroup/DateGroup.utils.ts +11 -0
- package/packages/web/components/DateGroup/index.ts +2 -0
- package/packages/web/components/{stats → EmptyState}/EmptyState.tsx +1 -10
- package/packages/web/components/EmptyState/EmptyState.types.ts +10 -0
- package/packages/web/components/EmptyState/index.ts +2 -0
- package/packages/web/components/EventRow/EventRow.constants.ts +10 -0
- package/packages/web/components/EventRow/EventRow.tsx +49 -0
- package/packages/web/components/EventRow/EventRow.types.ts +7 -0
- package/packages/web/components/EventRow/EventRow.utils.ts +49 -0
- package/packages/web/components/EventRow/index.ts +2 -0
- package/packages/web/components/ExpandButton/ExpandButton.tsx +18 -0
- package/packages/web/components/ExpandButton/ExpandButton.types.ts +6 -0
- package/packages/web/components/ExpandButton/index.ts +2 -0
- package/packages/web/components/HealthGradientBackground/HealthGradientBackground.tsx +14 -0
- package/packages/web/components/HealthGradientBackground/HealthGradientBackground.types.ts +5 -0
- package/packages/web/components/HealthGradientBackground/HealthGradientBackground.utils.ts +13 -0
- package/packages/web/components/HealthGradientBackground/index.ts +2 -0
- package/packages/web/components/HeroSection/HeroSection.tsx +55 -0
- package/packages/web/components/HeroSection/HeroSection.types.ts +14 -0
- package/packages/web/components/HeroSection/HeroSection.utils.ts +7 -0
- package/packages/web/components/HeroSection/hooks/index.ts +2 -0
- package/packages/web/components/HeroSection/hooks/useCountUp.ts +45 -0
- package/packages/web/components/HeroSection/hooks/useWeeklyActivity.ts +18 -0
- package/packages/web/components/HeroSection/index.ts +2 -0
- package/packages/web/components/{stats → IdeasCard}/IdeasCard.tsx +3 -14
- package/packages/web/components/IdeasCard/IdeasCard.types.ts +9 -0
- package/packages/web/components/IdeasCard/index.ts +2 -0
- package/packages/web/components/InsightMessage/InsightMessage.tsx +9 -0
- package/packages/web/components/InsightMessage/InsightMessage.types.ts +3 -0
- package/packages/web/components/InsightMessage/index.ts +2 -0
- package/packages/web/components/Logo/index.ts +1 -0
- package/packages/web/components/MarkdownContent/index.ts +1 -0
- package/packages/web/components/NowCard/NowCard.tsx +93 -0
- package/packages/web/components/NowCard/NowCard.types.ts +15 -0
- package/packages/web/components/NowCard/index.ts +2 -0
- package/packages/web/components/ProgressRing/ProgressRing.constants.ts +20 -0
- package/packages/web/components/{stats → ProgressRing}/ProgressRing.tsx +4 -27
- package/packages/web/components/ProgressRing/ProgressRing.types.ts +11 -0
- package/packages/web/components/ProgressRing/index.ts +2 -0
- package/packages/web/components/ProjectAvatar/index.ts +1 -0
- package/packages/web/components/Providers/index.ts +1 -0
- package/packages/web/components/QueueCard/QueueCard.tsx +72 -0
- package/packages/web/components/QueueCard/QueueCard.types.ts +11 -0
- package/packages/web/components/QueueCard/QueueCard.utils.ts +12 -0
- package/packages/web/components/QueueCard/index.ts +2 -0
- package/packages/web/components/{stats → RoadmapCard}/RoadmapCard.tsx +3 -23
- package/packages/web/components/RoadmapCard/RoadmapCard.types.ts +15 -0
- package/packages/web/components/RoadmapCard/index.ts +2 -0
- package/packages/web/components/{stats → ShipsCard}/ShipsCard.tsx +4 -22
- package/packages/web/components/ShipsCard/ShipsCard.types.ts +12 -0
- package/packages/web/components/ShipsCard/ShipsCard.utils.ts +4 -0
- package/packages/web/components/ShipsCard/index.ts +2 -0
- package/packages/web/components/{stats → SparklineChart}/SparklineChart.tsx +1 -7
- package/packages/web/components/SparklineChart/SparklineChart.types.ts +6 -0
- package/packages/web/components/SparklineChart/index.ts +2 -0
- package/packages/web/components/StreakCard/StreakCard.constants.ts +2 -0
- package/packages/web/components/{stats → StreakCard}/StreakCard.tsx +5 -11
- package/packages/web/components/StreakCard/StreakCard.types.ts +4 -0
- package/packages/web/components/StreakCard/index.ts +2 -0
- package/packages/web/components/TasksCounter/TasksCounter.tsx +14 -0
- package/packages/web/components/TasksCounter/TasksCounter.types.ts +3 -0
- package/packages/web/components/TasksCounter/index.ts +2 -0
- package/packages/web/components/TechStackBadges/index.ts +1 -0
- package/packages/web/components/{TerminalTab.tsx → TerminalTabs/TerminalTab.tsx} +11 -0
- package/packages/web/components/{TerminalTabs.tsx → TerminalTabs/TerminalTabs.tsx} +29 -28
- package/packages/web/components/TerminalTabs/index.ts +1 -0
- package/packages/web/components/VelocityBadge/VelocityBadge.tsx +27 -0
- package/packages/web/components/VelocityBadge/VelocityBadge.types.ts +3 -0
- package/packages/web/components/VelocityBadge/index.ts +2 -0
- package/packages/web/components/VelocityCard/VelocityCard.tsx +71 -0
- package/packages/web/components/VelocityCard/VelocityCard.types.ts +7 -0
- package/packages/web/components/VelocityCard/index.ts +2 -0
- package/packages/web/components/WeeklySparkline/WeeklySparkline.tsx +13 -0
- package/packages/web/components/WeeklySparkline/WeeklySparkline.types.ts +3 -0
- package/packages/web/components/WeeklySparkline/index.ts +2 -0
- package/packages/web/components/ui/input.tsx +21 -0
- package/packages/web/context/TerminalTabsContext.tsx +46 -1
- package/packages/web/hooks/useClaudeTerminal.ts +71 -21
- package/packages/web/hooks/useProjectStats.ts +55 -0
- package/packages/web/hooks/useProjects.ts +6 -6
- package/packages/web/lib/actions/projects.ts +15 -0
- package/packages/web/lib/json-loader.ts +630 -0
- package/packages/web/lib/services/index.ts +9 -0
- package/packages/web/lib/services/migration.server.ts +600 -0
- package/packages/web/lib/services/projects.server.ts +52 -0
- package/packages/web/lib/services/stats.server.ts +264 -0
- package/packages/web/lib/unified-loader.ts +396 -0
- package/packages/web/package.json +10 -7
- package/packages/web/server.ts +58 -8
- package/templates/commands/done.md +76 -32
- package/templates/commands/feature.md +121 -47
- package/templates/commands/idea.md +81 -8
- package/templates/commands/now.md +41 -17
- package/templates/commands/ship.md +64 -25
- package/templates/commands/sync.md +28 -3
- package/core/agentic/agent-router.js +0 -140
- package/core/agentic/chain-of-thought.js +0 -578
- package/core/agentic/command-executor.js +0 -417
- package/core/agentic/context-filter.js +0 -354
- package/core/agentic/ground-truth.js +0 -591
- package/core/agentic/loop-detector.js +0 -406
- package/core/agentic/memory-system.js +0 -845
- package/core/agentic/parallel-tools.js +0 -366
- package/core/agentic/plan-mode.js +0 -572
- package/core/agentic/prompt-builder.js +0 -352
- package/core/agentic/response-templates.js +0 -290
- package/core/agentic/semantic-compression.js +0 -517
- package/core/agentic/think-blocks.js +0 -657
- package/core/agentic/tool-registry.js +0 -184
- package/core/agentic/validation-rules.js +0 -380
- package/core/command-registry.js +0 -698
- package/core/commands.js +0 -2237
- package/core/domain/task-stack.js +0 -497
- package/core/infrastructure/legacy-installer-detector.js +0 -546
- package/core/infrastructure/migrator.js +0 -796
- package/core/infrastructure/session-manager.js +0 -390
- package/core/utils/file-helper.js +0 -329
- package/packages/web/app/api/projects/[id]/delete/route.ts +0 -21
- package/packages/web/app/api/stats/route.ts +0 -38
- package/packages/web/components/AppSidebar.tsx +0 -113
- package/packages/web/components/stats/ActivityTimeline.tsx +0 -201
- package/packages/web/components/stats/AgentsCard.tsx +0 -56
- package/packages/web/components/stats/BentoCard.tsx +0 -88
- package/packages/web/components/stats/HeroSection.tsx +0 -172
- package/packages/web/components/stats/NowCard.tsx +0 -71
- package/packages/web/components/stats/QueueCard.tsx +0 -58
- package/packages/web/components/stats/VelocityCard.tsx +0 -60
- package/packages/web/components/stats/index.ts +0 -17
- package/packages/web/hooks/useStats.ts +0 -28
- /package/packages/web/components/{CommandButton.tsx → CommandButton/CommandButton.tsx} +0 -0
- /package/packages/web/components/{ConnectionStatus.tsx → ConnectionStatus/ConnectionStatus.tsx} +0 -0
- /package/packages/web/components/{Logo.tsx → Logo/Logo.tsx} +0 -0
- /package/packages/web/components/{MarkdownContent.tsx → MarkdownContent/MarkdownContent.tsx} +0 -0
- /package/packages/web/components/{ProjectAvatar.tsx → ProjectAvatar/ProjectAvatar.tsx} +0 -0
- /package/packages/web/components/{providers.tsx → Providers/Providers.tsx} +0 -0
- /package/packages/web/components/{TechStackBadges.tsx → TechStackBadges/TechStackBadges.tsx} +0 -0
|
@@ -1,497 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Task Stack Manager - Handles multiple concurrent tasks with pause/resume capability
|
|
3
|
-
* Enables natural workflow with interruptions and context switching
|
|
4
|
-
*/
|
|
5
|
-
|
|
6
|
-
const path = require('path');
|
|
7
|
-
const fs = require('fs').promises;
|
|
8
|
-
const log = require('../utils/logger');
|
|
9
|
-
|
|
10
|
-
class TaskStack {
|
|
11
|
-
constructor(projectPath) {
|
|
12
|
-
this.projectPath = projectPath;
|
|
13
|
-
this.stackPath = path.join(projectPath, 'core', 'stack.jsonl');
|
|
14
|
-
this.nowPath = path.join(projectPath, 'core', 'now.md');
|
|
15
|
-
}
|
|
16
|
-
|
|
17
|
-
/**
|
|
18
|
-
* Initialize stack system - migrate from legacy now.md if needed
|
|
19
|
-
*/
|
|
20
|
-
async initialize() {
|
|
21
|
-
try {
|
|
22
|
-
// Check if stack already exists
|
|
23
|
-
await fs.access(this.stackPath);
|
|
24
|
-
return { migrated: false };
|
|
25
|
-
} catch {
|
|
26
|
-
// Stack doesn't exist, check for legacy now.md
|
|
27
|
-
return await this.migrateFromLegacy();
|
|
28
|
-
}
|
|
29
|
-
}
|
|
30
|
-
|
|
31
|
-
/**
|
|
32
|
-
* Migrate from legacy now.md to stack system
|
|
33
|
-
*/
|
|
34
|
-
async migrateFromLegacy() {
|
|
35
|
-
try {
|
|
36
|
-
const nowContent = await fs.readFile(this.nowPath, 'utf8');
|
|
37
|
-
|
|
38
|
-
if (!nowContent.trim() || nowContent.includes('No active task')) {
|
|
39
|
-
// Empty or no task, just create empty stack
|
|
40
|
-
await this.ensureStackFile();
|
|
41
|
-
return { migrated: true, hadTask: false };
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
// Parse task from now.md
|
|
45
|
-
const task = this.parseNowFile(nowContent);
|
|
46
|
-
|
|
47
|
-
// Create initial stack entry
|
|
48
|
-
const entry = {
|
|
49
|
-
id: `task-${Date.now()}`,
|
|
50
|
-
task: task.description || 'Migrated task',
|
|
51
|
-
agent: task.agent || 'unknown',
|
|
52
|
-
status: 'active',
|
|
53
|
-
started: task.started || new Date().toISOString(),
|
|
54
|
-
paused: null,
|
|
55
|
-
resumed: null,
|
|
56
|
-
completed: null,
|
|
57
|
-
duration: null,
|
|
58
|
-
complexity: task.complexity || 'moderate',
|
|
59
|
-
dev: task.dev || 'unknown'
|
|
60
|
-
};
|
|
61
|
-
|
|
62
|
-
// Write to stack
|
|
63
|
-
await this.appendToStack(entry);
|
|
64
|
-
|
|
65
|
-
return { migrated: true, hadTask: true, task: entry };
|
|
66
|
-
} catch (error) {
|
|
67
|
-
// No now.md or error reading, just create empty stack
|
|
68
|
-
await this.ensureStackFile();
|
|
69
|
-
return { migrated: true, hadTask: false, error: error.message };
|
|
70
|
-
}
|
|
71
|
-
}
|
|
72
|
-
|
|
73
|
-
/**
|
|
74
|
-
* Parse legacy now.md format
|
|
75
|
-
*/
|
|
76
|
-
parseNowFile(content) {
|
|
77
|
-
const result = {
|
|
78
|
-
description: '',
|
|
79
|
-
started: null,
|
|
80
|
-
agent: null,
|
|
81
|
-
complexity: null,
|
|
82
|
-
dev: null
|
|
83
|
-
};
|
|
84
|
-
|
|
85
|
-
// Check for frontmatter
|
|
86
|
-
if (content.startsWith('---')) {
|
|
87
|
-
const frontmatterEnd = content.indexOf('---', 3);
|
|
88
|
-
if (frontmatterEnd > 0) {
|
|
89
|
-
const frontmatter = content.substring(3, frontmatterEnd);
|
|
90
|
-
const lines = frontmatter.split('\n');
|
|
91
|
-
|
|
92
|
-
for (const line of lines) {
|
|
93
|
-
if (line.includes('task:')) {
|
|
94
|
-
result.description = line.split('task:')[1].trim().replace(/['"]/g, '');
|
|
95
|
-
}
|
|
96
|
-
if (line.includes('started:')) {
|
|
97
|
-
result.started = line.split('started:')[1].trim();
|
|
98
|
-
}
|
|
99
|
-
if (line.includes('agent:')) {
|
|
100
|
-
result.agent = line.split('agent:')[1].trim();
|
|
101
|
-
}
|
|
102
|
-
if (line.includes('complexity:')) {
|
|
103
|
-
result.complexity = line.split('complexity:')[1].trim();
|
|
104
|
-
}
|
|
105
|
-
if (line.includes('dev:')) {
|
|
106
|
-
result.dev = line.split('dev:')[1].trim();
|
|
107
|
-
}
|
|
108
|
-
}
|
|
109
|
-
|
|
110
|
-
// Get description from content if not in frontmatter
|
|
111
|
-
if (!result.description) {
|
|
112
|
-
const contentBody = content.substring(frontmatterEnd + 3).trim();
|
|
113
|
-
const firstLine = contentBody.split('\n')[0];
|
|
114
|
-
if (firstLine && !firstLine.startsWith('#')) {
|
|
115
|
-
result.description = firstLine.replace(/^[*-]\s*/, '').trim();
|
|
116
|
-
}
|
|
117
|
-
}
|
|
118
|
-
}
|
|
119
|
-
} else {
|
|
120
|
-
// No frontmatter, try to extract task from content
|
|
121
|
-
const lines = content.split('\n');
|
|
122
|
-
for (const line of lines) {
|
|
123
|
-
if (line.trim() && !line.startsWith('#') && !line.startsWith('---')) {
|
|
124
|
-
result.description = line.replace(/^[*-]\s*/, '').trim();
|
|
125
|
-
break;
|
|
126
|
-
}
|
|
127
|
-
}
|
|
128
|
-
}
|
|
129
|
-
|
|
130
|
-
return result;
|
|
131
|
-
}
|
|
132
|
-
|
|
133
|
-
/**
|
|
134
|
-
* Ensure stack file exists
|
|
135
|
-
*/
|
|
136
|
-
async ensureStackFile() {
|
|
137
|
-
try {
|
|
138
|
-
await fs.access(this.stackPath);
|
|
139
|
-
} catch {
|
|
140
|
-
// Create empty file
|
|
141
|
-
await fs.writeFile(this.stackPath, '');
|
|
142
|
-
}
|
|
143
|
-
}
|
|
144
|
-
|
|
145
|
-
/**
|
|
146
|
-
* Append entry to stack
|
|
147
|
-
*/
|
|
148
|
-
async appendToStack(entry) {
|
|
149
|
-
await this.ensureStackFile();
|
|
150
|
-
const line = JSON.stringify(entry) + '\n';
|
|
151
|
-
await fs.appendFile(this.stackPath, line);
|
|
152
|
-
}
|
|
153
|
-
|
|
154
|
-
/**
|
|
155
|
-
* Read all stack entries
|
|
156
|
-
*/
|
|
157
|
-
async readStack() {
|
|
158
|
-
await this.ensureStackFile();
|
|
159
|
-
const content = await fs.readFile(this.stackPath, 'utf8');
|
|
160
|
-
|
|
161
|
-
if (!content.trim()) {
|
|
162
|
-
return [];
|
|
163
|
-
}
|
|
164
|
-
|
|
165
|
-
const entries = [];
|
|
166
|
-
const lines = content.split('\n').filter(line => line.trim());
|
|
167
|
-
|
|
168
|
-
for (const line of lines) {
|
|
169
|
-
try {
|
|
170
|
-
entries.push(JSON.parse(line));
|
|
171
|
-
} catch (error) {
|
|
172
|
-
log.error('Error parsing stack line:', error.message);
|
|
173
|
-
}
|
|
174
|
-
}
|
|
175
|
-
|
|
176
|
-
return entries;
|
|
177
|
-
}
|
|
178
|
-
|
|
179
|
-
/**
|
|
180
|
-
* Get active task
|
|
181
|
-
*/
|
|
182
|
-
async getActiveTask() {
|
|
183
|
-
const stack = await this.readStack();
|
|
184
|
-
return stack.find(task => task.status === 'active') || null;
|
|
185
|
-
}
|
|
186
|
-
|
|
187
|
-
/**
|
|
188
|
-
* Get paused tasks
|
|
189
|
-
*/
|
|
190
|
-
async getPausedTasks() {
|
|
191
|
-
const stack = await this.readStack();
|
|
192
|
-
return stack.filter(task => task.status === 'paused')
|
|
193
|
-
.sort((a, b) => new Date(b.paused) - new Date(a.paused)); // Most recently paused first
|
|
194
|
-
}
|
|
195
|
-
|
|
196
|
-
/**
|
|
197
|
-
* Get all incomplete tasks
|
|
198
|
-
*/
|
|
199
|
-
async getIncompleteTasks() {
|
|
200
|
-
const stack = await this.readStack();
|
|
201
|
-
return stack.filter(task => task.status !== 'completed');
|
|
202
|
-
}
|
|
203
|
-
|
|
204
|
-
/**
|
|
205
|
-
* Start a new task
|
|
206
|
-
*/
|
|
207
|
-
async startTask(description, agent = 'general', complexity = 'moderate') {
|
|
208
|
-
// Check if there's already an active task
|
|
209
|
-
const active = await this.getActiveTask();
|
|
210
|
-
if (active) {
|
|
211
|
-
throw new Error(`Already working on: ${active.task}. Use /p:pause to pause it first.`);
|
|
212
|
-
}
|
|
213
|
-
|
|
214
|
-
const entry = {
|
|
215
|
-
id: `task-${Date.now()}`,
|
|
216
|
-
task: description,
|
|
217
|
-
agent,
|
|
218
|
-
status: 'active',
|
|
219
|
-
started: new Date().toISOString(),
|
|
220
|
-
paused: null,
|
|
221
|
-
resumed: null,
|
|
222
|
-
completed: null,
|
|
223
|
-
duration: null,
|
|
224
|
-
complexity,
|
|
225
|
-
dev: await this.getCurrentDev()
|
|
226
|
-
};
|
|
227
|
-
|
|
228
|
-
await this.appendToStack(entry);
|
|
229
|
-
await this.updateNowFile(entry);
|
|
230
|
-
|
|
231
|
-
return entry;
|
|
232
|
-
}
|
|
233
|
-
|
|
234
|
-
/**
|
|
235
|
-
* Pause the active task
|
|
236
|
-
*/
|
|
237
|
-
async pauseTask(reason = '') {
|
|
238
|
-
const active = await this.getActiveTask();
|
|
239
|
-
if (!active) {
|
|
240
|
-
throw new Error('No active task to pause');
|
|
241
|
-
}
|
|
242
|
-
|
|
243
|
-
// Update the task
|
|
244
|
-
active.status = 'paused';
|
|
245
|
-
active.paused = new Date().toISOString();
|
|
246
|
-
if (reason) {
|
|
247
|
-
active.pauseReason = reason;
|
|
248
|
-
}
|
|
249
|
-
|
|
250
|
-
// Rewrite stack with updated task
|
|
251
|
-
await this.updateTask(active);
|
|
252
|
-
|
|
253
|
-
// Update now.md to show paused state
|
|
254
|
-
await this.updateNowFile(null, `Paused: ${active.task}`);
|
|
255
|
-
|
|
256
|
-
return active;
|
|
257
|
-
}
|
|
258
|
-
|
|
259
|
-
/**
|
|
260
|
-
* Resume a paused task
|
|
261
|
-
*/
|
|
262
|
-
async resumeTask(taskId = null) {
|
|
263
|
-
// Check if there's an active task
|
|
264
|
-
const active = await this.getActiveTask();
|
|
265
|
-
if (active) {
|
|
266
|
-
throw new Error(`Already working on: ${active.task}. Complete or pause it first.`);
|
|
267
|
-
}
|
|
268
|
-
|
|
269
|
-
const paused = await this.getPausedTasks();
|
|
270
|
-
if (paused.length === 0) {
|
|
271
|
-
throw new Error('No paused tasks to resume');
|
|
272
|
-
}
|
|
273
|
-
|
|
274
|
-
let taskToResume;
|
|
275
|
-
if (taskId) {
|
|
276
|
-
taskToResume = paused.find(t => t.id === taskId);
|
|
277
|
-
if (!taskToResume) {
|
|
278
|
-
throw new Error(`Task ${taskId} not found or not paused`);
|
|
279
|
-
}
|
|
280
|
-
} else {
|
|
281
|
-
// Resume most recently paused
|
|
282
|
-
taskToResume = paused[0];
|
|
283
|
-
}
|
|
284
|
-
|
|
285
|
-
// Update the task
|
|
286
|
-
taskToResume.status = 'active';
|
|
287
|
-
taskToResume.resumed = new Date().toISOString();
|
|
288
|
-
|
|
289
|
-
// Calculate paused duration
|
|
290
|
-
if (taskToResume.paused) {
|
|
291
|
-
const pausedMs = new Date() - new Date(taskToResume.paused);
|
|
292
|
-
taskToResume.pausedDuration = (taskToResume.pausedDuration || 0) + pausedMs;
|
|
293
|
-
}
|
|
294
|
-
|
|
295
|
-
// Rewrite stack with updated task
|
|
296
|
-
await this.updateTask(taskToResume);
|
|
297
|
-
|
|
298
|
-
// Update now.md
|
|
299
|
-
await this.updateNowFile(taskToResume);
|
|
300
|
-
|
|
301
|
-
return taskToResume;
|
|
302
|
-
}
|
|
303
|
-
|
|
304
|
-
/**
|
|
305
|
-
* Complete the active task
|
|
306
|
-
*/
|
|
307
|
-
async completeTask() {
|
|
308
|
-
const active = await this.getActiveTask();
|
|
309
|
-
if (!active) {
|
|
310
|
-
throw new Error('No active task to complete');
|
|
311
|
-
}
|
|
312
|
-
|
|
313
|
-
// Update the task
|
|
314
|
-
active.status = 'completed';
|
|
315
|
-
active.completed = new Date().toISOString();
|
|
316
|
-
|
|
317
|
-
// Calculate duration (excluding paused time)
|
|
318
|
-
const totalMs = new Date() - new Date(active.started);
|
|
319
|
-
const pausedMs = active.pausedDuration || 0;
|
|
320
|
-
active.duration = totalMs - pausedMs;
|
|
321
|
-
active.durationFormatted = this.formatDuration(active.duration);
|
|
322
|
-
|
|
323
|
-
// Rewrite stack with updated task
|
|
324
|
-
await this.updateTask(active);
|
|
325
|
-
|
|
326
|
-
// Clear now.md
|
|
327
|
-
await this.updateNowFile(null, '');
|
|
328
|
-
|
|
329
|
-
return active;
|
|
330
|
-
}
|
|
331
|
-
|
|
332
|
-
/**
|
|
333
|
-
* Switch tasks (atomic pause + resume/start)
|
|
334
|
-
*/
|
|
335
|
-
async switchTask(targetTaskOrDescription) {
|
|
336
|
-
const active = await this.getActiveTask();
|
|
337
|
-
let pausedTask = null;
|
|
338
|
-
|
|
339
|
-
// Pause current if exists
|
|
340
|
-
if (active) {
|
|
341
|
-
pausedTask = await this.pauseTask('Switched to another task');
|
|
342
|
-
}
|
|
343
|
-
|
|
344
|
-
try {
|
|
345
|
-
// Check if target is a task ID or description
|
|
346
|
-
const paused = await this.getPausedTasks();
|
|
347
|
-
const existingTask = paused.find(t => t.id === targetTaskOrDescription);
|
|
348
|
-
|
|
349
|
-
if (existingTask) {
|
|
350
|
-
// Resume existing task
|
|
351
|
-
return {
|
|
352
|
-
paused: pausedTask,
|
|
353
|
-
resumed: await this.resumeTask(targetTaskOrDescription),
|
|
354
|
-
type: 'resumed'
|
|
355
|
-
};
|
|
356
|
-
} else {
|
|
357
|
-
// Start new task
|
|
358
|
-
return {
|
|
359
|
-
paused: pausedTask,
|
|
360
|
-
started: await this.startTask(targetTaskOrDescription),
|
|
361
|
-
type: 'started'
|
|
362
|
-
};
|
|
363
|
-
}
|
|
364
|
-
} catch (error) {
|
|
365
|
-
// If switch fails, resume the original task
|
|
366
|
-
if (pausedTask) {
|
|
367
|
-
await this.resumeTask(pausedTask.id);
|
|
368
|
-
}
|
|
369
|
-
throw error;
|
|
370
|
-
}
|
|
371
|
-
}
|
|
372
|
-
|
|
373
|
-
/**
|
|
374
|
-
* Update a task in the stack
|
|
375
|
-
*/
|
|
376
|
-
async updateTask(updatedTask) {
|
|
377
|
-
const stack = await this.readStack();
|
|
378
|
-
const index = stack.findIndex(t => t.id === updatedTask.id);
|
|
379
|
-
|
|
380
|
-
if (index === -1) {
|
|
381
|
-
throw new Error(`Task ${updatedTask.id} not found`);
|
|
382
|
-
}
|
|
383
|
-
|
|
384
|
-
stack[index] = updatedTask;
|
|
385
|
-
|
|
386
|
-
// Rewrite entire file (JSONL format)
|
|
387
|
-
const content = stack.map(task => JSON.stringify(task)).join('\n') + '\n';
|
|
388
|
-
await fs.writeFile(this.stackPath, content);
|
|
389
|
-
}
|
|
390
|
-
|
|
391
|
-
/**
|
|
392
|
-
* Update now.md to reflect current state
|
|
393
|
-
*/
|
|
394
|
-
async updateNowFile(task, customContent = null) {
|
|
395
|
-
let content;
|
|
396
|
-
|
|
397
|
-
if (customContent !== undefined && customContent !== null) {
|
|
398
|
-
content = customContent;
|
|
399
|
-
} else if (!task) {
|
|
400
|
-
content = `# Current Task
|
|
401
|
-
|
|
402
|
-
**No active task**
|
|
403
|
-
|
|
404
|
-
Use \`/p:work\` or \`/p:resume\` to start working.
|
|
405
|
-
|
|
406
|
-
---
|
|
407
|
-
|
|
408
|
-
_Track your focus with \`/p:work [task]\`_
|
|
409
|
-
`;
|
|
410
|
-
} else {
|
|
411
|
-
const started = new Date(task.started);
|
|
412
|
-
const now = new Date();
|
|
413
|
-
const elapsed = this.formatDuration(now - started - (task.pausedDuration || 0));
|
|
414
|
-
|
|
415
|
-
content = `---
|
|
416
|
-
task: "${task.task}"
|
|
417
|
-
started: ${task.started}
|
|
418
|
-
agent: ${task.agent}
|
|
419
|
-
complexity: ${task.complexity}
|
|
420
|
-
dev: ${task.dev}
|
|
421
|
-
---
|
|
422
|
-
|
|
423
|
-
# Current Task
|
|
424
|
-
|
|
425
|
-
**${task.task}**
|
|
426
|
-
|
|
427
|
-
- Started: ${started.toLocaleTimeString()} (${elapsed} ago)
|
|
428
|
-
- Agent: ${task.agent}
|
|
429
|
-
- Complexity: ${task.complexity}
|
|
430
|
-
|
|
431
|
-
---
|
|
432
|
-
|
|
433
|
-
When done: \`/p:done\`
|
|
434
|
-
Need to pause: \`/p:pause\`
|
|
435
|
-
`;
|
|
436
|
-
}
|
|
437
|
-
|
|
438
|
-
await fs.writeFile(this.nowPath, content);
|
|
439
|
-
}
|
|
440
|
-
|
|
441
|
-
/**
|
|
442
|
-
* Get current developer from git or system
|
|
443
|
-
*/
|
|
444
|
-
async getCurrentDev() {
|
|
445
|
-
try {
|
|
446
|
-
const { exec } = require('child_process');
|
|
447
|
-
const { promisify } = require('util');
|
|
448
|
-
const execAsync = promisify(exec);
|
|
449
|
-
|
|
450
|
-
const { stdout } = await execAsync('git config user.name');
|
|
451
|
-
return stdout.trim();
|
|
452
|
-
} catch {
|
|
453
|
-
return 'unknown';
|
|
454
|
-
}
|
|
455
|
-
}
|
|
456
|
-
|
|
457
|
-
/**
|
|
458
|
-
* Format duration in human-readable format
|
|
459
|
-
*/
|
|
460
|
-
formatDuration(ms) {
|
|
461
|
-
const seconds = Math.floor(ms / 1000);
|
|
462
|
-
const minutes = Math.floor(seconds / 60);
|
|
463
|
-
const hours = Math.floor(minutes / 60);
|
|
464
|
-
const days = Math.floor(hours / 24);
|
|
465
|
-
|
|
466
|
-
if (days > 0) {
|
|
467
|
-
return `${days}d ${hours % 24}h`;
|
|
468
|
-
} else if (hours > 0) {
|
|
469
|
-
return `${hours}h ${minutes % 60}m`;
|
|
470
|
-
} else if (minutes > 0) {
|
|
471
|
-
return `${minutes}m`;
|
|
472
|
-
} else {
|
|
473
|
-
return `${seconds}s`;
|
|
474
|
-
}
|
|
475
|
-
}
|
|
476
|
-
|
|
477
|
-
/**
|
|
478
|
-
* Get stack summary for display
|
|
479
|
-
*/
|
|
480
|
-
async getStackSummary() {
|
|
481
|
-
const active = await this.getActiveTask();
|
|
482
|
-
const paused = await this.getPausedTasks();
|
|
483
|
-
const stack = await this.readStack();
|
|
484
|
-
const completed = stack.filter(t => t.status === 'completed');
|
|
485
|
-
|
|
486
|
-
return {
|
|
487
|
-
active,
|
|
488
|
-
paused,
|
|
489
|
-
pausedCount: paused.length,
|
|
490
|
-
completed,
|
|
491
|
-
completedCount: completed.length,
|
|
492
|
-
totalTasks: stack.length
|
|
493
|
-
};
|
|
494
|
-
}
|
|
495
|
-
}
|
|
496
|
-
|
|
497
|
-
module.exports = TaskStack;
|