claude-code-pilot 2.0.0 → 3.0.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/bin/install.js +267 -250
- package/manifest.json +5 -18
- package/package.json +5 -7
- package/src/agents/build-error-resolver.md +114 -0
- package/src/agents/ccp-advisor-researcher.md +104 -0
- package/src/agents/ccp-assumptions-analyzer.md +105 -0
- package/{gsd/agents/gsd-codebase-mapper.md → src/agents/ccp-codebase-mapper.md} +7 -7
- package/{gsd/agents/gsd-debugger.md → src/agents/ccp-debugger.md} +125 -8
- package/{gsd/agents/gsd-executor.md → src/agents/ccp-executor.md} +31 -20
- package/{gsd/agents/gsd-integration-checker.md → src/agents/ccp-integration-checker.md} +2 -2
- package/{gsd/agents/gsd-nyquist-auditor.md → src/agents/ccp-nyquist-auditor.md} +3 -3
- package/{gsd/agents/gsd-phase-researcher.md → src/agents/ccp-phase-researcher.md} +127 -13
- package/{gsd/agents/gsd-plan-checker.md → src/agents/ccp-plan-checker.md} +57 -21
- package/{gsd/agents/gsd-planner.md → src/agents/ccp-planner.md} +61 -23
- package/{gsd/agents/gsd-project-researcher.md → src/agents/ccp-project-researcher.md} +33 -6
- package/{gsd/agents/gsd-research-synthesizer.md → src/agents/ccp-research-synthesizer.md} +11 -11
- package/{gsd/agents/gsd-roadmapper.md → src/agents/ccp-roadmapper.md} +39 -10
- package/src/agents/ccp-ui-auditor.md +439 -0
- package/src/agents/ccp-ui-checker.md +300 -0
- package/src/agents/ccp-ui-researcher.md +357 -0
- package/{gsd/agents/gsd-verifier.md → src/agents/ccp-verifier.md} +81 -15
- package/src/agents/cpp-build-resolver.md +90 -0
- package/src/agents/cpp-reviewer.md +72 -0
- package/src/agents/database-reviewer.md +91 -0
- package/src/agents/docs-lookup.md +68 -0
- package/src/agents/flutter-reviewer.md +243 -0
- package/src/agents/go-build-resolver.md +94 -0
- package/src/agents/go-reviewer.md +76 -0
- package/src/agents/java-build-resolver.md +153 -0
- package/src/agents/java-reviewer.md +92 -0
- package/src/agents/kotlin-build-resolver.md +118 -0
- package/src/agents/kotlin-reviewer.md +159 -0
- package/src/agents/planner.md +212 -0
- package/src/agents/python-reviewer.md +98 -0
- package/src/agents/pytorch-build-resolver.md +120 -0
- package/src/agents/refactor-cleaner.md +85 -0
- package/src/agents/rust-build-resolver.md +148 -0
- package/src/agents/rust-reviewer.md +94 -0
- package/src/agents/typescript-reviewer.md +112 -0
- package/src/available-rules/README.md +80 -0
- package/src/available-rules/cpp/coding-style.md +44 -0
- package/src/available-rules/cpp/hooks.md +39 -0
- package/src/available-rules/cpp/patterns.md +51 -0
- package/src/available-rules/cpp/security.md +51 -0
- package/src/available-rules/cpp/testing.md +44 -0
- package/src/available-rules/csharp/coding-style.md +72 -0
- package/src/available-rules/csharp/hooks.md +25 -0
- package/src/available-rules/csharp/patterns.md +50 -0
- package/src/available-rules/csharp/security.md +58 -0
- package/src/available-rules/csharp/testing.md +46 -0
- package/src/available-rules/java/coding-style.md +114 -0
- package/src/available-rules/java/hooks.md +18 -0
- package/src/available-rules/java/patterns.md +146 -0
- package/src/available-rules/java/security.md +100 -0
- package/src/available-rules/java/testing.md +131 -0
- package/src/available-rules/kotlin/hooks.md +17 -0
- package/src/available-rules/rust/coding-style.md +151 -0
- package/src/available-rules/rust/hooks.md +16 -0
- package/src/available-rules/rust/patterns.md +168 -0
- package/src/available-rules/rust/security.md +141 -0
- package/src/available-rules/rust/testing.md +154 -0
- package/src/commands/aside.md +164 -0
- package/src/commands/build-fix.md +62 -0
- package/src/commands/ccp/add-backlog.md +76 -0
- package/{gsd/commands-gsd → src/commands/ccp}/add-phase.md +3 -3
- package/{gsd/commands-gsd → src/commands/ccp}/add-tests.md +5 -5
- package/{gsd/commands-gsd → src/commands/ccp}/add-todo.md +4 -4
- package/{gsd/commands-gsd → src/commands/ccp}/audit-milestone.md +3 -3
- package/src/commands/ccp/audit-uat.md +24 -0
- package/src/commands/ccp/autonomous.md +41 -0
- package/{gsd/commands-gsd → src/commands/ccp}/check-todos.md +3 -3
- package/{gsd/commands-gsd → src/commands/ccp}/cleanup.md +3 -3
- package/{gsd/commands-gsd → src/commands/ccp}/complete-milestone.md +9 -9
- package/{gsd/commands-gsd → src/commands/ccp}/debug.md +14 -9
- package/src/commands/ccp/discuss-phase.md +64 -0
- package/src/commands/ccp/do.md +30 -0
- package/src/commands/ccp/execute-phase.md +59 -0
- package/src/commands/ccp/fast.md +30 -0
- package/src/commands/ccp/forensics.md +56 -0
- package/{gsd/commands-gsd → src/commands/ccp}/health.md +3 -3
- package/{gsd/commands-gsd → src/commands/ccp}/help.md +5 -5
- package/{gsd/commands-gsd → src/commands/ccp}/insert-phase.md +3 -3
- package/{gsd/commands-gsd → src/commands/ccp}/list-phase-assumptions.md +2 -2
- package/src/commands/ccp/manager.md +39 -0
- package/{gsd/commands-gsd → src/commands/ccp}/map-codebase.md +7 -7
- package/src/commands/ccp/milestone-summary.md +51 -0
- package/{gsd/commands-gsd → src/commands/ccp}/new-milestone.md +8 -8
- package/{gsd/commands-gsd → src/commands/ccp}/new-project.md +8 -8
- package/src/commands/ccp/next.md +24 -0
- package/src/commands/ccp/note.md +34 -0
- package/{gsd/commands-gsd → src/commands/ccp}/pause-work.md +3 -3
- package/{gsd/commands-gsd → src/commands/ccp}/plan-milestone-gaps.md +5 -5
- package/{gsd/commands-gsd → src/commands/ccp}/plan-phase.md +9 -7
- package/src/commands/ccp/plant-seed.md +28 -0
- package/src/commands/ccp/pr-branch.md +25 -0
- package/{gsd/commands-gsd → src/commands/ccp}/progress.md +3 -3
- package/{gsd/commands-gsd → src/commands/ccp}/quick.md +10 -8
- package/{gsd/commands-gsd → src/commands/ccp}/remove-phase.md +3 -3
- package/{gsd/commands-gsd → src/commands/ccp}/research-phase.md +17 -12
- package/{gsd/commands-gsd → src/commands/ccp}/resume-work.md +3 -3
- package/src/commands/ccp/review-backlog.md +61 -0
- package/src/commands/ccp/session-report.md +19 -0
- package/src/commands/ccp/set-profile.md +12 -0
- package/{gsd/commands-gsd → src/commands/ccp}/settings.md +5 -5
- package/src/commands/ccp/ship.md +23 -0
- package/src/commands/ccp/stats.md +18 -0
- package/src/commands/ccp/thread.md +127 -0
- package/src/commands/ccp/ui-phase.md +34 -0
- package/src/commands/ccp/ui-review.md +32 -0
- package/{gsd/commands-gsd → src/commands/ccp}/update.md +5 -5
- package/{gsd/commands-gsd → src/commands/ccp}/validate-phase.md +3 -3
- package/{gsd/commands-gsd → src/commands/ccp}/verify-work.md +5 -5
- package/src/commands/code-review.md +40 -0
- package/src/commands/context-budget.md +29 -0
- package/src/commands/cpp-build.md +173 -0
- package/src/commands/cpp-review.md +132 -0
- package/src/commands/cpp-test.md +251 -0
- package/src/commands/docs.md +31 -0
- package/src/commands/e2e.md +364 -0
- package/src/commands/eval.md +120 -0
- package/{ecc → src}/commands/evolve.md +2 -2
- package/src/commands/go-build.md +183 -0
- package/src/commands/go-review.md +148 -0
- package/src/commands/go-test.md +268 -0
- package/src/commands/gradle-build.md +70 -0
- package/src/commands/harness-audit.md +71 -0
- package/src/commands/kotlin-build.md +174 -0
- package/src/commands/kotlin-review.md +140 -0
- package/src/commands/kotlin-test.md +312 -0
- package/src/commands/orchestrate.md +231 -0
- package/src/commands/plan.md +114 -0
- package/src/commands/prompt-optimize.md +38 -0
- package/src/commands/prune.md +25 -0
- package/src/commands/python-review.md +297 -0
- package/{ecc → src}/commands/quality-gate.md +1 -1
- package/src/commands/refactor-clean.md +80 -0
- package/src/commands/rules-distill.md +11 -0
- package/src/commands/rust-build.md +187 -0
- package/src/commands/rust-review.md +142 -0
- package/src/commands/rust-test.md +308 -0
- package/{ecc → src}/commands/sessions.md +10 -10
- package/src/commands/setup-pm.md +80 -0
- package/{kit → src}/commands/setup.md +45 -19
- package/src/commands/skill-create.md +172 -0
- package/src/commands/skill-health.md +51 -0
- package/src/commands/tdd.md +328 -0
- package/src/commands/test-coverage.md +69 -0
- package/src/commands/update-codemaps.md +72 -0
- package/src/commands/update-docs.md +84 -0
- package/{gsd/hooks/gsd-context-monitor.js → src/hooks/ccp-context-monitor.js} +3 -3
- package/src/hooks/ccp-prompt-guard.js +96 -0
- package/{gsd/hooks/gsd-statusline.js → src/hooks/ccp-statusline.js} +7 -7
- package/src/hooks/ccp-workflow-guard.js +94 -0
- package/src/hooks/config-protection.js +141 -0
- package/{kit → src}/hooks/kit-check-update.js +7 -4
- package/src/hooks/mcp-health-check.js +620 -0
- package/{ecc/scripts → src}/hooks/run-with-flags-shell.sh +1 -1
- package/{ecc/scripts → src}/hooks/run-with-flags.js +74 -13
- package/src/hooks/session-end-marker.js +29 -0
- package/{ecc/scripts → src}/hooks/session-end.js +83 -40
- package/{ecc/scripts → src}/hooks/session-start.js +75 -9
- package/{ecc/scripts → src}/lib/hook-flags.js +8 -4
- package/{ecc/scripts → src}/lib/project-detect.js +2 -1
- package/{ecc/scripts → src}/lib/session-manager.d.ts +5 -1
- package/{ecc/scripts → src}/lib/session-manager.js +202 -92
- package/{ecc/scripts → src}/lib/utils.d.ts +23 -1
- package/{ecc/scripts → src}/lib/utils.js +91 -3
- package/{gsd/get-shit-done/bin/gsd-tools.cjs → src/pilot/bin/ccp-tools.cjs} +257 -86
- package/{gsd/get-shit-done → src/pilot}/bin/lib/commands.cjs +1 -1
- package/src/pilot/bin/lib/config.cjs +444 -0
- package/src/pilot/bin/lib/core.cjs +1190 -0
- package/src/pilot/bin/lib/init.cjs +1281 -0
- package/src/pilot/bin/lib/model-profiles.cjs +67 -0
- package/{gsd/get-shit-done → src/pilot}/bin/lib/phase.cjs +2 -2
- package/src/pilot/bin/lib/security.cjs +382 -0
- package/{gsd/get-shit-done → src/pilot}/bin/lib/state.cjs +1 -1
- package/src/pilot/bin/lib/uat.cjs +282 -0
- package/{gsd/get-shit-done → src/pilot}/bin/lib/verify.cjs +10 -10
- package/{gsd/get-shit-done → src/pilot}/references/continuation-format.md +16 -16
- package/{gsd/get-shit-done → src/pilot}/references/decimal-phase-calculation.md +5 -5
- package/{gsd/get-shit-done → src/pilot}/references/git-integration.md +5 -5
- package/{gsd/get-shit-done → src/pilot}/references/git-planning-commit.md +4 -4
- package/src/pilot/references/mcp-servers.json +153 -0
- package/{gsd/get-shit-done → src/pilot}/references/model-profile-resolution.md +2 -2
- package/{gsd/get-shit-done → src/pilot}/references/model-profiles.md +20 -20
- package/{gsd/get-shit-done → src/pilot}/references/phase-argument-parsing.md +4 -4
- package/{gsd/get-shit-done → src/pilot}/references/planning-config.md +15 -15
- package/{gsd/get-shit-done → src/pilot}/references/ui-brand.md +5 -5
- package/{gsd/get-shit-done → src/pilot}/references/verification-patterns.md +1 -1
- package/{gsd/get-shit-done → src/pilot}/templates/DEBUG.md +1 -1
- package/{gsd/get-shit-done → src/pilot}/templates/UAT.md +3 -3
- package/src/pilot/templates/UI-SPEC.md +100 -0
- package/{gsd/get-shit-done → src/pilot}/templates/VALIDATION.md +1 -1
- package/src/pilot/templates/claude-md.md +122 -0
- package/{gsd/get-shit-done → src/pilot}/templates/codebase/architecture.md +2 -2
- package/{gsd/get-shit-done → src/pilot}/templates/codebase/structure.md +13 -13
- package/{gsd/get-shit-done → src/pilot}/templates/context.md +4 -4
- package/src/pilot/templates/copilot-instructions.md +7 -0
- package/{gsd/get-shit-done → src/pilot}/templates/debug-subagent-prompt.md +4 -4
- package/src/pilot/templates/dev-preferences.md +21 -0
- package/{gsd/get-shit-done → src/pilot}/templates/discovery.md +2 -2
- package/src/pilot/templates/discussion-log.md +63 -0
- package/{gsd/get-shit-done → src/pilot}/templates/phase-prompt.md +12 -12
- package/{gsd/get-shit-done → src/pilot}/templates/planner-subagent-prompt.md +7 -7
- package/{gsd/get-shit-done → src/pilot}/templates/project.md +1 -1
- package/{gsd/get-shit-done → src/pilot}/templates/research.md +2 -2
- package/{gsd/get-shit-done → src/pilot}/templates/state.md +2 -2
- package/{gsd/get-shit-done → src/pilot}/templates/summary-complex.md +1 -1
- package/{gsd/get-shit-done → src/pilot}/workflows/add-phase.md +11 -11
- package/{gsd/get-shit-done → src/pilot}/workflows/add-tests.md +15 -15
- package/{gsd/get-shit-done → src/pilot}/workflows/add-todo.md +7 -7
- package/{gsd/get-shit-done → src/pilot}/workflows/audit-milestone.md +24 -16
- package/src/pilot/workflows/audit-uat.md +109 -0
- package/src/pilot/workflows/autonomous.md +891 -0
- package/{gsd/get-shit-done → src/pilot}/workflows/check-todos.md +10 -10
- package/{gsd/get-shit-done → src/pilot}/workflows/cleanup.md +3 -3
- package/{gsd/get-shit-done → src/pilot}/workflows/complete-milestone.md +19 -16
- package/{gsd/get-shit-done → src/pilot}/workflows/diagnose-issues.md +9 -4
- package/{gsd/get-shit-done → src/pilot}/workflows/discovery-phase.md +8 -8
- package/src/pilot/workflows/discuss-phase-assumptions.md +653 -0
- package/{gsd/get-shit-done → src/pilot}/workflows/discuss-phase.md +407 -49
- package/src/pilot/workflows/do.md +104 -0
- package/src/pilot/workflows/execute-phase.md +821 -0
- package/{gsd/get-shit-done → src/pilot}/workflows/execute-plan.md +79 -28
- package/src/pilot/workflows/fast.md +105 -0
- package/src/pilot/workflows/forensics.md +265 -0
- package/{gsd/get-shit-done → src/pilot}/workflows/health.md +34 -11
- package/src/pilot/workflows/help.md +775 -0
- package/{gsd/get-shit-done → src/pilot}/workflows/insert-phase.md +10 -10
- package/{gsd/get-shit-done → src/pilot}/workflows/list-phase-assumptions.md +4 -4
- package/src/pilot/workflows/manager.md +362 -0
- package/{gsd/get-shit-done → src/pilot}/workflows/map-codebase.md +27 -17
- package/src/pilot/workflows/milestone-summary.md +223 -0
- package/{gsd/get-shit-done → src/pilot}/workflows/new-milestone.md +135 -33
- package/{gsd/get-shit-done → src/pilot}/workflows/new-project.md +152 -79
- package/src/pilot/workflows/next.md +97 -0
- package/src/pilot/workflows/node-repair.md +92 -0
- package/src/pilot/workflows/note.md +156 -0
- package/src/pilot/workflows/pause-work.md +177 -0
- package/{gsd/get-shit-done → src/pilot}/workflows/plan-milestone-gaps.md +10 -11
- package/src/pilot/workflows/plan-phase.md +859 -0
- package/src/pilot/workflows/plant-seed.md +169 -0
- package/src/pilot/workflows/pr-branch.md +129 -0
- package/{gsd/get-shit-done → src/pilot}/workflows/progress.md +95 -34
- package/{gsd/get-shit-done → src/pilot}/workflows/quick.md +33 -21
- package/{gsd/get-shit-done → src/pilot}/workflows/remove-phase.md +14 -14
- package/{gsd/get-shit-done → src/pilot}/workflows/research-phase.md +18 -10
- package/{gsd/get-shit-done → src/pilot}/workflows/resume-project.md +37 -18
- package/src/pilot/workflows/session-report.md +146 -0
- package/{gsd/get-shit-done → src/pilot}/workflows/set-profile.md +7 -7
- package/{gsd/get-shit-done → src/pilot}/workflows/settings.md +75 -22
- package/src/pilot/workflows/ship.md +228 -0
- package/src/pilot/workflows/stats.md +60 -0
- package/{gsd/get-shit-done → src/pilot}/workflows/transition.md +57 -17
- package/src/pilot/workflows/ui-phase.md +302 -0
- package/src/pilot/workflows/ui-review.md +165 -0
- package/{gsd/get-shit-done → src/pilot}/workflows/update.md +88 -58
- package/{gsd/get-shit-done → src/pilot}/workflows/validate-phase.md +24 -17
- package/{gsd/get-shit-done → src/pilot}/workflows/verify-phase.md +26 -15
- package/{gsd/get-shit-done → src/pilot}/workflows/verify-work.md +89 -37
- package/{ecc → src}/rules/common/agents.md +1 -0
- package/{ecc → src}/rules/common/coding-style.md +21 -0
- package/src/skills/agentic-engineering/SKILL.md +63 -0
- package/src/skills/ai-first-engineering/SKILL.md +51 -0
- package/src/skills/ai-regression-testing/SKILL.md +385 -0
- package/src/skills/api-design/SKILL.md +523 -0
- package/src/skills/architecture-decision-records/SKILL.md +179 -0
- package/src/skills/backend-patterns/SKILL.md +598 -0
- package/src/skills/benchmark/SKILL.md +87 -0
- package/src/skills/blueprint/SKILL.md +90 -0
- package/src/skills/browser-qa/SKILL.md +81 -0
- package/src/skills/claude-api/SKILL.md +337 -0
- package/src/skills/codebase-onboarding/SKILL.md +233 -0
- package/src/skills/coding-standards/SKILL.md +530 -0
- package/src/skills/context-budget/SKILL.md +135 -0
- package/{ecc → src}/skills/continuous-learning-v2/SKILL.md +2 -2
- package/{ecc → src}/skills/continuous-learning-v2/agents/observer-loop.sh +1 -1
- package/src/skills/cpp-coding-standards/SKILL.md +723 -0
- package/src/skills/cpp-testing/SKILL.md +324 -0
- package/src/skills/database-migrations/SKILL.md +429 -0
- package/src/skills/deep-research/SKILL.md +155 -0
- package/src/skills/deployment-patterns/SKILL.md +427 -0
- package/src/skills/django-patterns/SKILL.md +734 -0
- package/src/skills/django-security/SKILL.md +593 -0
- package/src/skills/django-tdd/SKILL.md +729 -0
- package/src/skills/django-verification/SKILL.md +469 -0
- package/src/skills/docker-patterns/SKILL.md +364 -0
- package/src/skills/documentation-lookup/SKILL.md +90 -0
- package/src/skills/e2e-testing/SKILL.md +326 -0
- package/src/skills/exa-search/SKILL.md +103 -0
- package/src/skills/frontend-patterns/SKILL.md +642 -0
- package/src/skills/golang-patterns/SKILL.md +674 -0
- package/src/skills/golang-testing/SKILL.md +720 -0
- package/src/skills/java-coding-standards/SKILL.md +147 -0
- package/src/skills/jpa-patterns/SKILL.md +151 -0
- package/src/skills/kotlin-coroutines-flows/SKILL.md +284 -0
- package/src/skills/kotlin-exposed-patterns/SKILL.md +719 -0
- package/src/skills/kotlin-ktor-patterns/SKILL.md +689 -0
- package/src/skills/kotlin-patterns/SKILL.md +711 -0
- package/src/skills/kotlin-testing/SKILL.md +824 -0
- package/src/skills/laravel-patterns/SKILL.md +415 -0
- package/src/skills/laravel-security/SKILL.md +285 -0
- package/src/skills/laravel-tdd/SKILL.md +283 -0
- package/src/skills/laravel-verification/SKILL.md +179 -0
- package/src/skills/mcp-server-patterns/SKILL.md +67 -0
- package/src/skills/perl-patterns/SKILL.md +504 -0
- package/src/skills/perl-testing/SKILL.md +475 -0
- package/src/skills/postgres-patterns/SKILL.md +147 -0
- package/src/skills/prompt-optimizer/SKILL.md +397 -0
- package/src/skills/python-patterns/SKILL.md +750 -0
- package/src/skills/python-testing/SKILL.md +816 -0
- package/src/skills/rust-patterns/SKILL.md +499 -0
- package/src/skills/rust-testing/SKILL.md +500 -0
- package/src/skills/safety-guard/SKILL.md +69 -0
- package/src/skills/search-first/SKILL.md +161 -0
- package/src/skills/security-review/SKILL.md +495 -0
- package/src/skills/security-review/cloud-infrastructure-security.md +361 -0
- package/src/skills/security-scan/SKILL.md +165 -0
- package/src/skills/springboot-patterns/SKILL.md +314 -0
- package/src/skills/springboot-security/SKILL.md +272 -0
- package/src/skills/springboot-tdd/SKILL.md +158 -0
- package/src/skills/springboot-verification/SKILL.md +231 -0
- package/src/skills/tdd-workflow/SKILL.md +410 -0
- package/ecc/scripts/hooks/session-end-marker.js +0 -15
- package/gsd/LICENSE +0 -21
- package/gsd/commands-gsd/discuss-phase.md +0 -90
- package/gsd/commands-gsd/execute-phase.md +0 -41
- package/gsd/commands-gsd/join-discord.md +0 -18
- package/gsd/commands-gsd/reapply-patches.md +0 -123
- package/gsd/commands-gsd/set-profile.md +0 -34
- package/gsd/get-shit-done/bin/lib/config.cjs +0 -169
- package/gsd/get-shit-done/bin/lib/core.cjs +0 -492
- package/gsd/get-shit-done/bin/lib/init.cjs +0 -710
- package/gsd/get-shit-done/workflows/execute-phase.md +0 -459
- package/gsd/get-shit-done/workflows/help.md +0 -489
- package/gsd/get-shit-done/workflows/pause-work.md +0 -122
- package/gsd/get-shit-done/workflows/plan-phase.md +0 -560
- package/gsd/hooks/gsd-check-update.js +0 -81
- package/kit/CLAUDE.md +0 -43
- package/kit/commands/kit/update.md +0 -46
- package/kit/mcp.json +0 -10
- package/kit/rules/code-style.md +0 -24
- /package/{ecc → src}/agents/architect.md +0 -0
- /package/{ecc → src}/agents/code-reviewer.md +0 -0
- /package/{ecc → src}/agents/doc-updater.md +0 -0
- /package/{ecc → src}/agents/e2e-runner.md +0 -0
- /package/{ecc → src}/agents/security-reviewer.md +0 -0
- /package/{ecc → src}/agents/tdd-guide.md +0 -0
- /package/{ecc/rules → src/available-rules}/golang/coding-style.md +0 -0
- /package/{ecc/rules → src/available-rules}/golang/hooks.md +0 -0
- /package/{ecc/rules → src/available-rules}/golang/patterns.md +0 -0
- /package/{ecc/rules → src/available-rules}/golang/security.md +0 -0
- /package/{ecc/rules → src/available-rules}/golang/testing.md +0 -0
- /package/{ecc/rules → src/available-rules}/kotlin/coding-style.md +0 -0
- /package/{ecc/rules → src/available-rules}/kotlin/patterns.md +0 -0
- /package/{ecc/rules → src/available-rules}/kotlin/security.md +0 -0
- /package/{ecc/rules → src/available-rules}/kotlin/testing.md +0 -0
- /package/{ecc/rules → src/available-rules}/perl/coding-style.md +0 -0
- /package/{ecc/rules → src/available-rules}/perl/hooks.md +0 -0
- /package/{ecc/rules → src/available-rules}/perl/patterns.md +0 -0
- /package/{ecc/rules → src/available-rules}/perl/security.md +0 -0
- /package/{ecc/rules → src/available-rules}/perl/testing.md +0 -0
- /package/{ecc/rules → src/available-rules}/php/coding-style.md +0 -0
- /package/{ecc/rules → src/available-rules}/php/hooks.md +0 -0
- /package/{ecc/rules → src/available-rules}/php/patterns.md +0 -0
- /package/{ecc/rules → src/available-rules}/php/security.md +0 -0
- /package/{ecc/rules → src/available-rules}/php/testing.md +0 -0
- /package/{ecc/rules → src/available-rules}/python/coding-style.md +0 -0
- /package/{ecc/rules → src/available-rules}/python/hooks.md +0 -0
- /package/{ecc/rules → src/available-rules}/python/patterns.md +0 -0
- /package/{ecc/rules → src/available-rules}/python/security.md +0 -0
- /package/{ecc/rules → src/available-rules}/python/testing.md +0 -0
- /package/{ecc/rules → src/available-rules}/swift/coding-style.md +0 -0
- /package/{ecc/rules → src/available-rules}/swift/hooks.md +0 -0
- /package/{ecc/rules → src/available-rules}/swift/patterns.md +0 -0
- /package/{ecc/rules → src/available-rules}/swift/security.md +0 -0
- /package/{ecc/rules → src/available-rules}/swift/testing.md +0 -0
- /package/{ecc/rules → src/available-rules}/typescript/coding-style.md +0 -0
- /package/{ecc/rules → src/available-rules}/typescript/hooks.md +0 -0
- /package/{ecc/rules → src/available-rules}/typescript/patterns.md +0 -0
- /package/{ecc/rules → src/available-rules}/typescript/security.md +0 -0
- /package/{ecc/rules → src/available-rules}/typescript/testing.md +0 -0
- /package/{ecc → src}/commands/checkpoint.md +0 -0
- /package/{ecc → src}/commands/learn.md +0 -0
- /package/{ecc → src}/commands/model-route.md +0 -0
- /package/{ecc → src}/commands/resume-session.md +0 -0
- /package/{ecc → src}/commands/save-session.md +0 -0
- /package/{kit → src}/commands/setup-refresh.md +0 -0
- /package/{kit → src}/commands/tool-guide.md +0 -0
- /package/{ecc → src}/commands/verify.md +0 -0
- /package/{ecc → src}/contexts/dev.md +0 -0
- /package/{ecc → src}/contexts/research.md +0 -0
- /package/{ecc → src}/contexts/review.md +0 -0
- /package/{ecc → src}/examples/CLAUDE.md +0 -0
- /package/{ecc → src}/examples/django-api-CLAUDE.md +0 -0
- /package/{ecc → src}/examples/go-microservice-CLAUDE.md +0 -0
- /package/{ecc → src}/examples/rust-api-CLAUDE.md +0 -0
- /package/{ecc → src}/examples/saas-nextjs-CLAUDE.md +0 -0
- /package/{ecc → src}/examples/user-CLAUDE.md +0 -0
- /package/{ecc/scripts → src}/hooks/check-hook-enabled.js +0 -0
- /package/{ecc/scripts → src}/hooks/evaluate-session.js +0 -0
- /package/{ecc/scripts → src}/hooks/pre-compact.js +0 -0
- /package/{ecc/scripts → src}/hooks/suggest-compact.js +0 -0
- /package/{ecc/scripts → src}/lib/package-manager.d.ts +0 -0
- /package/{ecc/scripts → src}/lib/package-manager.js +0 -0
- /package/{ecc/scripts → src}/lib/resolve-formatter.js +0 -0
- /package/{ecc/scripts → src}/lib/session-aliases.d.ts +0 -0
- /package/{ecc/scripts → src}/lib/session-aliases.js +0 -0
- /package/{ecc/scripts → src}/lib/shell-split.js +0 -0
- /package/{gsd/get-shit-done → src/pilot}/bin/lib/frontmatter.cjs +0 -0
- /package/{gsd/get-shit-done → src/pilot}/bin/lib/milestone.cjs +0 -0
- /package/{gsd/get-shit-done → src/pilot}/bin/lib/roadmap.cjs +0 -0
- /package/{gsd/get-shit-done → src/pilot}/bin/lib/template.cjs +0 -0
- /package/{gsd/get-shit-done → src/pilot}/references/checkpoints.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/references/questioning.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/references/tdd.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/templates/codebase/concerns.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/templates/codebase/conventions.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/templates/codebase/integrations.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/templates/codebase/stack.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/templates/codebase/testing.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/templates/config.json +0 -0
- /package/{gsd/get-shit-done → src/pilot}/templates/continue-here.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/templates/milestone-archive.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/templates/milestone.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/templates/requirements.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/templates/research-project/ARCHITECTURE.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/templates/research-project/FEATURES.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/templates/research-project/PITFALLS.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/templates/research-project/STACK.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/templates/research-project/SUMMARY.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/templates/retrospective.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/templates/roadmap.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/templates/summary-minimal.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/templates/summary-standard.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/templates/summary.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/templates/user-setup.md +0 -0
- /package/{gsd/get-shit-done → src/pilot}/templates/verification-report.md +0 -0
- /package/{ecc → src}/rules/common/development-workflow.md +0 -0
- /package/{ecc → src}/rules/common/git-workflow.md +0 -0
- /package/{ecc → src}/rules/common/hooks.md +0 -0
- /package/{ecc → src}/rules/common/patterns.md +0 -0
- /package/{ecc → src}/rules/common/performance.md +0 -0
- /package/{ecc → src}/rules/common/security.md +0 -0
- /package/{ecc → src}/rules/common/testing.md +0 -0
- /package/{ecc → src}/skills/continuous-learning-v2/agents/observer.md +0 -0
- /package/{ecc → src}/skills/continuous-learning-v2/agents/start-observer.sh +0 -0
- /package/{ecc → src}/skills/continuous-learning-v2/config.json +0 -0
- /package/{ecc → src}/skills/continuous-learning-v2/hooks/observe.sh +0 -0
- /package/{ecc → src}/skills/continuous-learning-v2/scripts/detect-project.sh +0 -0
- /package/{ecc → src}/skills/continuous-learning-v2/scripts/instinct-cli.py +0 -0
- /package/{ecc → src}/skills/continuous-learning-v2/scripts/test_parse_instinct.py +0 -0
- /package/{ecc → src}/skills/strategic-compact/SKILL.md +0 -0
- /package/{ecc → src}/skills/strategic-compact/suggest-compact.sh +0 -0
- /package/{ecc/skills/verification-loop-SKILL.md → src/skills/verification-loop/SKILL.md} +0 -0
|
@@ -0,0 +1,168 @@
|
|
|
1
|
+
---
|
|
2
|
+
paths:
|
|
3
|
+
- "**/*.rs"
|
|
4
|
+
---
|
|
5
|
+
# Rust Patterns
|
|
6
|
+
|
|
7
|
+
> This file extends [common/patterns.md](../common/patterns.md) with Rust-specific content.
|
|
8
|
+
|
|
9
|
+
## Repository Pattern with Traits
|
|
10
|
+
|
|
11
|
+
Encapsulate data access behind a trait:
|
|
12
|
+
|
|
13
|
+
```rust
|
|
14
|
+
pub trait OrderRepository: Send + Sync {
|
|
15
|
+
fn find_by_id(&self, id: u64) -> Result<Option<Order>, StorageError>;
|
|
16
|
+
fn find_all(&self) -> Result<Vec<Order>, StorageError>;
|
|
17
|
+
fn save(&self, order: &Order) -> Result<Order, StorageError>;
|
|
18
|
+
fn delete(&self, id: u64) -> Result<(), StorageError>;
|
|
19
|
+
}
|
|
20
|
+
```
|
|
21
|
+
|
|
22
|
+
Concrete implementations handle storage details (Postgres, SQLite, in-memory for tests).
|
|
23
|
+
|
|
24
|
+
## Service Layer
|
|
25
|
+
|
|
26
|
+
Business logic in service structs; inject dependencies via constructor:
|
|
27
|
+
|
|
28
|
+
```rust
|
|
29
|
+
pub struct OrderService {
|
|
30
|
+
repo: Box<dyn OrderRepository>,
|
|
31
|
+
payment: Box<dyn PaymentGateway>,
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
impl OrderService {
|
|
35
|
+
pub fn new(repo: Box<dyn OrderRepository>, payment: Box<dyn PaymentGateway>) -> Self {
|
|
36
|
+
Self { repo, payment }
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
pub fn place_order(&self, request: CreateOrderRequest) -> anyhow::Result<OrderSummary> {
|
|
40
|
+
let order = Order::from(request);
|
|
41
|
+
self.payment.charge(order.total())?;
|
|
42
|
+
let saved = self.repo.save(&order)?;
|
|
43
|
+
Ok(OrderSummary::from(saved))
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
```
|
|
47
|
+
|
|
48
|
+
## Newtype Pattern for Type Safety
|
|
49
|
+
|
|
50
|
+
Prevent argument mix-ups with distinct wrapper types:
|
|
51
|
+
|
|
52
|
+
```rust
|
|
53
|
+
struct UserId(u64);
|
|
54
|
+
struct OrderId(u64);
|
|
55
|
+
|
|
56
|
+
fn get_order(user: UserId, order: OrderId) -> anyhow::Result<Order> {
|
|
57
|
+
// Can't accidentally swap user and order IDs at call sites
|
|
58
|
+
todo!()
|
|
59
|
+
}
|
|
60
|
+
```
|
|
61
|
+
|
|
62
|
+
## Enum State Machines
|
|
63
|
+
|
|
64
|
+
Model states as enums — make illegal states unrepresentable:
|
|
65
|
+
|
|
66
|
+
```rust
|
|
67
|
+
enum ConnectionState {
|
|
68
|
+
Disconnected,
|
|
69
|
+
Connecting { attempt: u32 },
|
|
70
|
+
Connected { session_id: String },
|
|
71
|
+
Failed { reason: String, retries: u32 },
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
fn handle(state: &ConnectionState) {
|
|
75
|
+
match state {
|
|
76
|
+
ConnectionState::Disconnected => connect(),
|
|
77
|
+
ConnectionState::Connecting { attempt } if *attempt > 3 => abort(),
|
|
78
|
+
ConnectionState::Connecting { .. } => wait(),
|
|
79
|
+
ConnectionState::Connected { session_id } => use_session(session_id),
|
|
80
|
+
ConnectionState::Failed { retries, .. } if *retries < 5 => retry(),
|
|
81
|
+
ConnectionState::Failed { reason, .. } => log_failure(reason),
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
```
|
|
85
|
+
|
|
86
|
+
Always match exhaustively — no wildcard `_` for business-critical enums.
|
|
87
|
+
|
|
88
|
+
## Builder Pattern
|
|
89
|
+
|
|
90
|
+
Use for structs with many optional parameters:
|
|
91
|
+
|
|
92
|
+
```rust
|
|
93
|
+
pub struct ServerConfig {
|
|
94
|
+
host: String,
|
|
95
|
+
port: u16,
|
|
96
|
+
max_connections: usize,
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
impl ServerConfig {
|
|
100
|
+
pub fn builder(host: impl Into<String>, port: u16) -> ServerConfigBuilder {
|
|
101
|
+
ServerConfigBuilder {
|
|
102
|
+
host: host.into(),
|
|
103
|
+
port,
|
|
104
|
+
max_connections: 100,
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
pub struct ServerConfigBuilder {
|
|
110
|
+
host: String,
|
|
111
|
+
port: u16,
|
|
112
|
+
max_connections: usize,
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
impl ServerConfigBuilder {
|
|
116
|
+
pub fn max_connections(mut self, n: usize) -> Self {
|
|
117
|
+
self.max_connections = n;
|
|
118
|
+
self
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
pub fn build(self) -> ServerConfig {
|
|
122
|
+
ServerConfig {
|
|
123
|
+
host: self.host,
|
|
124
|
+
port: self.port,
|
|
125
|
+
max_connections: self.max_connections,
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
```
|
|
130
|
+
|
|
131
|
+
## Sealed Traits for Extensibility Control
|
|
132
|
+
|
|
133
|
+
Use a private module to seal a trait, preventing external implementations:
|
|
134
|
+
|
|
135
|
+
```rust
|
|
136
|
+
mod private {
|
|
137
|
+
pub trait Sealed {}
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
pub trait Format: private::Sealed {
|
|
141
|
+
fn encode(&self, data: &[u8]) -> Vec<u8>;
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
pub struct Json;
|
|
145
|
+
impl private::Sealed for Json {}
|
|
146
|
+
impl Format for Json {
|
|
147
|
+
fn encode(&self, data: &[u8]) -> Vec<u8> { todo!() }
|
|
148
|
+
}
|
|
149
|
+
```
|
|
150
|
+
|
|
151
|
+
## API Response Envelope
|
|
152
|
+
|
|
153
|
+
Consistent API responses using a generic enum:
|
|
154
|
+
|
|
155
|
+
```rust
|
|
156
|
+
#[derive(Debug, serde::Serialize)]
|
|
157
|
+
#[serde(tag = "status")]
|
|
158
|
+
pub enum ApiResponse<T: serde::Serialize> {
|
|
159
|
+
#[serde(rename = "ok")]
|
|
160
|
+
Ok { data: T },
|
|
161
|
+
#[serde(rename = "error")]
|
|
162
|
+
Error { message: String },
|
|
163
|
+
}
|
|
164
|
+
```
|
|
165
|
+
|
|
166
|
+
## References
|
|
167
|
+
|
|
168
|
+
See skill: `rust-patterns` for comprehensive patterns including ownership, traits, generics, concurrency, and async.
|
|
@@ -0,0 +1,141 @@
|
|
|
1
|
+
---
|
|
2
|
+
paths:
|
|
3
|
+
- "**/*.rs"
|
|
4
|
+
---
|
|
5
|
+
# Rust Security
|
|
6
|
+
|
|
7
|
+
> This file extends [common/security.md](../common/security.md) with Rust-specific content.
|
|
8
|
+
|
|
9
|
+
## Secrets Management
|
|
10
|
+
|
|
11
|
+
- Never hardcode API keys, tokens, or credentials in source code
|
|
12
|
+
- Use environment variables: `std::env::var("API_KEY")`
|
|
13
|
+
- Fail fast if required secrets are missing at startup
|
|
14
|
+
- Keep `.env` files in `.gitignore`
|
|
15
|
+
|
|
16
|
+
```rust
|
|
17
|
+
// BAD
|
|
18
|
+
const API_KEY: &str = "sk-abc123...";
|
|
19
|
+
|
|
20
|
+
// GOOD — environment variable with early validation
|
|
21
|
+
fn load_api_key() -> anyhow::Result<String> {
|
|
22
|
+
std::env::var("PAYMENT_API_KEY")
|
|
23
|
+
.context("PAYMENT_API_KEY must be set")
|
|
24
|
+
}
|
|
25
|
+
```
|
|
26
|
+
|
|
27
|
+
## SQL Injection Prevention
|
|
28
|
+
|
|
29
|
+
- Always use parameterized queries — never format user input into SQL strings
|
|
30
|
+
- Use query builder or ORM (sqlx, diesel, sea-orm) with bind parameters
|
|
31
|
+
|
|
32
|
+
```rust
|
|
33
|
+
// BAD — SQL injection via format string
|
|
34
|
+
let query = format!("SELECT * FROM users WHERE name = '{name}'");
|
|
35
|
+
sqlx::query(&query).fetch_one(&pool).await?;
|
|
36
|
+
|
|
37
|
+
// GOOD — parameterized query with sqlx
|
|
38
|
+
// Placeholder syntax varies by backend: Postgres: $1 | MySQL: ? | SQLite: $1
|
|
39
|
+
sqlx::query("SELECT * FROM users WHERE name = $1")
|
|
40
|
+
.bind(&name)
|
|
41
|
+
.fetch_one(&pool)
|
|
42
|
+
.await?;
|
|
43
|
+
```
|
|
44
|
+
|
|
45
|
+
## Input Validation
|
|
46
|
+
|
|
47
|
+
- Validate all user input at system boundaries before processing
|
|
48
|
+
- Use the type system to enforce invariants (newtype pattern)
|
|
49
|
+
- Parse, don't validate — convert unstructured data to typed structs at the boundary
|
|
50
|
+
- Reject invalid input with clear error messages
|
|
51
|
+
|
|
52
|
+
```rust
|
|
53
|
+
// Parse, don't validate — invalid states are unrepresentable
|
|
54
|
+
pub struct Email(String);
|
|
55
|
+
|
|
56
|
+
impl Email {
|
|
57
|
+
pub fn parse(input: &str) -> Result<Self, ValidationError> {
|
|
58
|
+
let trimmed = input.trim();
|
|
59
|
+
let at_pos = trimmed.find('@')
|
|
60
|
+
.filter(|&p| p > 0 && p < trimmed.len() - 1)
|
|
61
|
+
.ok_or_else(|| ValidationError::InvalidEmail(input.to_string()))?;
|
|
62
|
+
let domain = &trimmed[at_pos + 1..];
|
|
63
|
+
if trimmed.len() > 254 || !domain.contains('.') {
|
|
64
|
+
return Err(ValidationError::InvalidEmail(input.to_string()));
|
|
65
|
+
}
|
|
66
|
+
// For production use, prefer a validated email crate (e.g., `email_address`)
|
|
67
|
+
Ok(Self(trimmed.to_string()))
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
pub fn as_str(&self) -> &str {
|
|
71
|
+
&self.0
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
```
|
|
75
|
+
|
|
76
|
+
## Unsafe Code
|
|
77
|
+
|
|
78
|
+
- Minimize `unsafe` blocks — prefer safe abstractions
|
|
79
|
+
- Every `unsafe` block must have a `// SAFETY:` comment explaining the invariant
|
|
80
|
+
- Never use `unsafe` to bypass the borrow checker for convenience
|
|
81
|
+
- Audit all `unsafe` code during review — it is a red flag without justification
|
|
82
|
+
- Prefer `safe` FFI wrappers around C libraries
|
|
83
|
+
|
|
84
|
+
```rust
|
|
85
|
+
// GOOD — safety comment documents ALL required invariants
|
|
86
|
+
let widget: &Widget = {
|
|
87
|
+
// SAFETY: `ptr` is non-null, aligned, points to an initialized Widget,
|
|
88
|
+
// and no mutable references or mutations exist for its lifetime.
|
|
89
|
+
unsafe { &*ptr }
|
|
90
|
+
};
|
|
91
|
+
|
|
92
|
+
// BAD — no safety justification
|
|
93
|
+
unsafe { &*ptr }
|
|
94
|
+
```
|
|
95
|
+
|
|
96
|
+
## Dependency Security
|
|
97
|
+
|
|
98
|
+
- Run `cargo audit` to scan for known CVEs in dependencies
|
|
99
|
+
- Run `cargo deny check` for license and advisory compliance
|
|
100
|
+
- Use `cargo tree` to audit transitive dependencies
|
|
101
|
+
- Keep dependencies updated — set up Dependabot or Renovate
|
|
102
|
+
- Minimize dependency count — evaluate before adding new crates
|
|
103
|
+
|
|
104
|
+
```bash
|
|
105
|
+
# Security audit
|
|
106
|
+
cargo audit
|
|
107
|
+
|
|
108
|
+
# Deny advisories, duplicate versions, and restricted licenses
|
|
109
|
+
cargo deny check
|
|
110
|
+
|
|
111
|
+
# Inspect dependency tree
|
|
112
|
+
cargo tree
|
|
113
|
+
cargo tree -d # Show duplicates only
|
|
114
|
+
```
|
|
115
|
+
|
|
116
|
+
## Error Messages
|
|
117
|
+
|
|
118
|
+
- Never expose internal paths, stack traces, or database errors in API responses
|
|
119
|
+
- Log detailed errors server-side; return generic messages to clients
|
|
120
|
+
- Use `tracing` or `log` for structured server-side logging
|
|
121
|
+
|
|
122
|
+
```rust
|
|
123
|
+
// Map errors to appropriate status codes and generic messages
|
|
124
|
+
// (Example uses axum; adapt the response type to your framework)
|
|
125
|
+
match order_service.find_by_id(id) {
|
|
126
|
+
Ok(order) => Ok((StatusCode::OK, Json(order))),
|
|
127
|
+
Err(ServiceError::NotFound(_)) => {
|
|
128
|
+
tracing::info!(order_id = id, "order not found");
|
|
129
|
+
Err((StatusCode::NOT_FOUND, "Resource not found"))
|
|
130
|
+
}
|
|
131
|
+
Err(e) => {
|
|
132
|
+
tracing::error!(order_id = id, error = %e, "unexpected error");
|
|
133
|
+
Err((StatusCode::INTERNAL_SERVER_ERROR, "Internal server error"))
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
```
|
|
137
|
+
|
|
138
|
+
## References
|
|
139
|
+
|
|
140
|
+
See skill: `rust-patterns` for unsafe code guidelines and ownership patterns.
|
|
141
|
+
See skill: `security-review` for general security checklists.
|
|
@@ -0,0 +1,154 @@
|
|
|
1
|
+
---
|
|
2
|
+
paths:
|
|
3
|
+
- "**/*.rs"
|
|
4
|
+
---
|
|
5
|
+
# Rust Testing
|
|
6
|
+
|
|
7
|
+
> This file extends [common/testing.md](../common/testing.md) with Rust-specific content.
|
|
8
|
+
|
|
9
|
+
## Test Framework
|
|
10
|
+
|
|
11
|
+
- **`#[test]`** with `#[cfg(test)]` modules for unit tests
|
|
12
|
+
- **rstest** for parameterized tests and fixtures
|
|
13
|
+
- **proptest** for property-based testing
|
|
14
|
+
- **mockall** for trait-based mocking
|
|
15
|
+
- **`#[tokio::test]`** for async tests
|
|
16
|
+
|
|
17
|
+
## Test Organization
|
|
18
|
+
|
|
19
|
+
```text
|
|
20
|
+
my_crate/
|
|
21
|
+
├── src/
|
|
22
|
+
│ ├── lib.rs # Unit tests in #[cfg(test)] modules
|
|
23
|
+
│ ├── auth/
|
|
24
|
+
│ │ └── mod.rs # #[cfg(test)] mod tests { ... }
|
|
25
|
+
│ └── orders/
|
|
26
|
+
│ └── service.rs # #[cfg(test)] mod tests { ... }
|
|
27
|
+
├── tests/ # Integration tests (each file = separate binary)
|
|
28
|
+
│ ├── api_test.rs
|
|
29
|
+
│ ├── db_test.rs
|
|
30
|
+
│ └── common/ # Shared test utilities
|
|
31
|
+
│ └── mod.rs
|
|
32
|
+
└── benches/ # Criterion benchmarks
|
|
33
|
+
└── benchmark.rs
|
|
34
|
+
```
|
|
35
|
+
|
|
36
|
+
Unit tests go inside `#[cfg(test)]` modules in the same file. Integration tests go in `tests/`.
|
|
37
|
+
|
|
38
|
+
## Unit Test Pattern
|
|
39
|
+
|
|
40
|
+
```rust
|
|
41
|
+
#[cfg(test)]
|
|
42
|
+
mod tests {
|
|
43
|
+
use super::*;
|
|
44
|
+
|
|
45
|
+
#[test]
|
|
46
|
+
fn creates_user_with_valid_email() {
|
|
47
|
+
let user = User::new("Alice", "alice@example.com").unwrap();
|
|
48
|
+
assert_eq!(user.name, "Alice");
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
#[test]
|
|
52
|
+
fn rejects_invalid_email() {
|
|
53
|
+
let result = User::new("Bob", "not-an-email");
|
|
54
|
+
assert!(result.is_err());
|
|
55
|
+
assert!(result.unwrap_err().to_string().contains("invalid email"));
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
```
|
|
59
|
+
|
|
60
|
+
## Parameterized Tests
|
|
61
|
+
|
|
62
|
+
```rust
|
|
63
|
+
use rstest::rstest;
|
|
64
|
+
|
|
65
|
+
#[rstest]
|
|
66
|
+
#[case("hello", 5)]
|
|
67
|
+
#[case("", 0)]
|
|
68
|
+
#[case("rust", 4)]
|
|
69
|
+
fn test_string_length(#[case] input: &str, #[case] expected: usize) {
|
|
70
|
+
assert_eq!(input.len(), expected);
|
|
71
|
+
}
|
|
72
|
+
```
|
|
73
|
+
|
|
74
|
+
## Async Tests
|
|
75
|
+
|
|
76
|
+
```rust
|
|
77
|
+
#[tokio::test]
|
|
78
|
+
async fn fetches_data_successfully() {
|
|
79
|
+
let client = TestClient::new().await;
|
|
80
|
+
let result = client.get("/data").await;
|
|
81
|
+
assert!(result.is_ok());
|
|
82
|
+
}
|
|
83
|
+
```
|
|
84
|
+
|
|
85
|
+
## Mocking with mockall
|
|
86
|
+
|
|
87
|
+
Define traits in production code; generate mocks in test modules:
|
|
88
|
+
|
|
89
|
+
```rust
|
|
90
|
+
// Production trait — pub so integration tests can import it
|
|
91
|
+
pub trait UserRepository {
|
|
92
|
+
fn find_by_id(&self, id: u64) -> Option<User>;
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
#[cfg(test)]
|
|
96
|
+
mod tests {
|
|
97
|
+
use super::*;
|
|
98
|
+
use mockall::predicate::eq;
|
|
99
|
+
|
|
100
|
+
mockall::mock! {
|
|
101
|
+
pub Repo {}
|
|
102
|
+
impl UserRepository for Repo {
|
|
103
|
+
fn find_by_id(&self, id: u64) -> Option<User>;
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
#[test]
|
|
108
|
+
fn service_returns_user_when_found() {
|
|
109
|
+
let mut mock = MockRepo::new();
|
|
110
|
+
mock.expect_find_by_id()
|
|
111
|
+
.with(eq(42))
|
|
112
|
+
.times(1)
|
|
113
|
+
.returning(|_| Some(User { id: 42, name: "Alice".into() }));
|
|
114
|
+
|
|
115
|
+
let service = UserService::new(Box::new(mock));
|
|
116
|
+
let user = service.get_user(42).unwrap();
|
|
117
|
+
assert_eq!(user.name, "Alice");
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
```
|
|
121
|
+
|
|
122
|
+
## Test Naming
|
|
123
|
+
|
|
124
|
+
Use descriptive names that explain the scenario:
|
|
125
|
+
- `creates_user_with_valid_email()`
|
|
126
|
+
- `rejects_order_when_insufficient_stock()`
|
|
127
|
+
- `returns_none_when_not_found()`
|
|
128
|
+
|
|
129
|
+
## Coverage
|
|
130
|
+
|
|
131
|
+
- Target 80%+ line coverage
|
|
132
|
+
- Use **cargo-llvm-cov** for coverage reporting
|
|
133
|
+
- Focus on business logic — exclude generated code and FFI bindings
|
|
134
|
+
|
|
135
|
+
```bash
|
|
136
|
+
cargo llvm-cov # Summary
|
|
137
|
+
cargo llvm-cov --html # HTML report
|
|
138
|
+
cargo llvm-cov --fail-under-lines 80 # Fail if below threshold
|
|
139
|
+
```
|
|
140
|
+
|
|
141
|
+
## Testing Commands
|
|
142
|
+
|
|
143
|
+
```bash
|
|
144
|
+
cargo test # Run all tests
|
|
145
|
+
cargo test -- --nocapture # Show println output
|
|
146
|
+
cargo test test_name # Run tests matching pattern
|
|
147
|
+
cargo test --lib # Unit tests only
|
|
148
|
+
cargo test --test api_test # Specific integration test (tests/api_test.rs)
|
|
149
|
+
cargo test --doc # Doc tests only
|
|
150
|
+
```
|
|
151
|
+
|
|
152
|
+
## References
|
|
153
|
+
|
|
154
|
+
See skill: `rust-testing` for comprehensive testing patterns including property-based testing, fixtures, and benchmarking with Criterion.
|
|
@@ -0,0 +1,164 @@
|
|
|
1
|
+
---
|
|
2
|
+
description: Answer a quick side question without interrupting or losing context from the current task. Resume work automatically after answering.
|
|
3
|
+
---
|
|
4
|
+
|
|
5
|
+
# Aside Command
|
|
6
|
+
|
|
7
|
+
Ask a question mid-task and get an immediate, focused answer — then continue right where you left off. The current task, files, and context are never modified.
|
|
8
|
+
|
|
9
|
+
## When to Use
|
|
10
|
+
|
|
11
|
+
- You're curious about something while Claude is working and don't want to lose momentum
|
|
12
|
+
- You need a quick explanation of code Claude is currently editing
|
|
13
|
+
- You want a second opinion or clarification on a decision without derailing the task
|
|
14
|
+
- You need to understand an error, concept, or pattern before Claude proceeds
|
|
15
|
+
- You want to ask something unrelated to the current task without starting a new session
|
|
16
|
+
|
|
17
|
+
## Usage
|
|
18
|
+
|
|
19
|
+
```
|
|
20
|
+
/aside <your question>
|
|
21
|
+
/aside what does this function actually return?
|
|
22
|
+
/aside is this pattern thread-safe?
|
|
23
|
+
/aside why are we using X instead of Y here?
|
|
24
|
+
/aside what's the difference between foo() and bar()?
|
|
25
|
+
/aside should we be worried about the N+1 query we just added?
|
|
26
|
+
```
|
|
27
|
+
|
|
28
|
+
## Process
|
|
29
|
+
|
|
30
|
+
### Step 1: Freeze the current task state
|
|
31
|
+
|
|
32
|
+
Before answering anything, mentally note:
|
|
33
|
+
- What is the active task? (what file, feature, or problem was being worked on)
|
|
34
|
+
- What step was in progress at the moment `/aside` was invoked?
|
|
35
|
+
- What was about to happen next?
|
|
36
|
+
|
|
37
|
+
Do NOT touch, edit, create, or delete any files during the aside.
|
|
38
|
+
|
|
39
|
+
### Step 2: Answer the question directly
|
|
40
|
+
|
|
41
|
+
Answer the question in the most concise form that is still complete and useful.
|
|
42
|
+
|
|
43
|
+
- Lead with the answer, not the reasoning
|
|
44
|
+
- Keep it short — if a full explanation is needed, offer to go deeper after the task
|
|
45
|
+
- If the question is about the current file or code being worked on, reference it precisely (file path and line number if relevant)
|
|
46
|
+
- If answering requires reading a file, read it — but read only, never write
|
|
47
|
+
|
|
48
|
+
Format the response as:
|
|
49
|
+
|
|
50
|
+
```
|
|
51
|
+
ASIDE: [restate the question briefly]
|
|
52
|
+
|
|
53
|
+
[Your answer here]
|
|
54
|
+
|
|
55
|
+
— Back to task: [one-line description of what was being done]
|
|
56
|
+
```
|
|
57
|
+
|
|
58
|
+
### Step 3: Resume the main task
|
|
59
|
+
|
|
60
|
+
After delivering the answer, immediately continue the active task from the exact point it was paused. Do not ask for permission to resume unless the aside answer revealed a blocker or a reason to reconsider the current approach (see Edge Cases).
|
|
61
|
+
|
|
62
|
+
---
|
|
63
|
+
|
|
64
|
+
## Edge Cases
|
|
65
|
+
|
|
66
|
+
**No question provided (`/aside` with nothing after it):**
|
|
67
|
+
Respond:
|
|
68
|
+
```
|
|
69
|
+
ASIDE: no question provided
|
|
70
|
+
|
|
71
|
+
What would you like to know? (ask your question and I'll answer without losing the current task context)
|
|
72
|
+
|
|
73
|
+
— Back to task: [one-line description of what was being done]
|
|
74
|
+
```
|
|
75
|
+
|
|
76
|
+
**Question reveals a potential problem with the current task:**
|
|
77
|
+
Flag it clearly before resuming:
|
|
78
|
+
```
|
|
79
|
+
ASIDE: [answer]
|
|
80
|
+
|
|
81
|
+
⚠️ Note: This answer suggests [issue] with the current approach. Want to address this before continuing, or proceed as planned?
|
|
82
|
+
```
|
|
83
|
+
Wait for the user's decision before resuming.
|
|
84
|
+
|
|
85
|
+
**Question is actually a task redirect (not a side question):**
|
|
86
|
+
If the question implies changing what is being built (e.g., `/aside actually, let's use Redis instead`), clarify:
|
|
87
|
+
```
|
|
88
|
+
ASIDE: That sounds like a direction change, not just a side question.
|
|
89
|
+
Do you want to:
|
|
90
|
+
(a) Answer this as information only and keep the current plan
|
|
91
|
+
(b) Pause the current task and change approach
|
|
92
|
+
```
|
|
93
|
+
Wait for the user's answer — do not make assumptions.
|
|
94
|
+
|
|
95
|
+
**Question is about the currently open file or code:**
|
|
96
|
+
Answer from the live context. If the file was read earlier in the session, reference it directly. If not, read it now (read-only) and answer with a file:line reference.
|
|
97
|
+
|
|
98
|
+
**No active task (nothing in progress when `/aside` is invoked):**
|
|
99
|
+
Still use the standard wrapper so the response shape stays consistent:
|
|
100
|
+
```
|
|
101
|
+
ASIDE: [restate the question briefly]
|
|
102
|
+
|
|
103
|
+
[Your answer here]
|
|
104
|
+
|
|
105
|
+
— Back to task: no active task to resume
|
|
106
|
+
```
|
|
107
|
+
|
|
108
|
+
**Question requires a long answer:**
|
|
109
|
+
Give the essential answer concisely, then offer:
|
|
110
|
+
```
|
|
111
|
+
That's the short version. Want a deeper explanation after we finish [current task]?
|
|
112
|
+
```
|
|
113
|
+
|
|
114
|
+
**Multiple `/aside` questions in a row:**
|
|
115
|
+
Answer each one in sequence. After the last answer, resume the main task. Do not lose task state across a chain of asides.
|
|
116
|
+
|
|
117
|
+
**Aside answer implies a code change is needed:**
|
|
118
|
+
Note the change needed but do not make it during the aside:
|
|
119
|
+
```
|
|
120
|
+
ASIDE: [answer]
|
|
121
|
+
|
|
122
|
+
📝 Worth fixing: [what should be changed]. I'll flag this after the current task unless you want to address it now.
|
|
123
|
+
```
|
|
124
|
+
|
|
125
|
+
**Question is ambiguous or too vague:**
|
|
126
|
+
Ask one clarifying question — the shortest question that gets the information needed to answer. Do not ask multiple questions.
|
|
127
|
+
|
|
128
|
+
---
|
|
129
|
+
|
|
130
|
+
## Example Output
|
|
131
|
+
|
|
132
|
+
```
|
|
133
|
+
User: /aside what does fetchWithRetry() actually do?
|
|
134
|
+
|
|
135
|
+
ASIDE: what does fetchWithRetry() do?
|
|
136
|
+
|
|
137
|
+
fetchWithRetry() (src/api/retry.ts:12) attempts the request up to 3 times with
|
|
138
|
+
exponential backoff (250ms → 500ms → 1s). It only retries on 5xx responses and
|
|
139
|
+
network errors — 4xx errors are treated as final and not retried.
|
|
140
|
+
|
|
141
|
+
— Back to task: refactoring the auth middleware in src/middleware/auth.ts
|
|
142
|
+
```
|
|
143
|
+
|
|
144
|
+
```
|
|
145
|
+
User: /aside is the approach we're taking thread-safe?
|
|
146
|
+
|
|
147
|
+
ASIDE: is the current approach thread-safe?
|
|
148
|
+
|
|
149
|
+
No — the shared cache object in src/cache/store.ts:34 is mutated without locking.
|
|
150
|
+
Under concurrent requests this is a race condition. It's low risk in a single-process
|
|
151
|
+
Node.js server but would be a real problem with worker threads or clustering.
|
|
152
|
+
|
|
153
|
+
⚠️ Note: This could affect the feature we're building. Want to address this now or continue and fix it in a follow-up?
|
|
154
|
+
```
|
|
155
|
+
|
|
156
|
+
---
|
|
157
|
+
|
|
158
|
+
## Notes
|
|
159
|
+
|
|
160
|
+
- Never modify files during an aside — read-only access only
|
|
161
|
+
- The aside is a conversation pause, not a new task — the original task must always resume
|
|
162
|
+
- Keep answers focused: the goal is to unblock the user quickly, not to deliver a lecture
|
|
163
|
+
- If an aside sparks a larger discussion, finish the current task first unless the aside reveals a blocker
|
|
164
|
+
- Asides are not saved to session files unless explicitly relevant to the task outcome
|
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
# Build and Fix
|
|
2
|
+
|
|
3
|
+
Incrementally fix build and type errors with minimal, safe changes.
|
|
4
|
+
|
|
5
|
+
## Step 1: Detect Build System
|
|
6
|
+
|
|
7
|
+
Identify the project's build tool and run the build:
|
|
8
|
+
|
|
9
|
+
| Indicator | Build Command |
|
|
10
|
+
|-----------|---------------|
|
|
11
|
+
| `package.json` with `build` script | `npm run build` or `pnpm build` |
|
|
12
|
+
| `tsconfig.json` (TypeScript only) | `npx tsc --noEmit` |
|
|
13
|
+
| `Cargo.toml` | `cargo build 2>&1` |
|
|
14
|
+
| `pom.xml` | `mvn compile` |
|
|
15
|
+
| `build.gradle` | `./gradlew compileJava` |
|
|
16
|
+
| `go.mod` | `go build ./...` |
|
|
17
|
+
| `pyproject.toml` | `python -m compileall -q .` or `mypy .` |
|
|
18
|
+
|
|
19
|
+
## Step 2: Parse and Group Errors
|
|
20
|
+
|
|
21
|
+
1. Run the build command and capture stderr
|
|
22
|
+
2. Group errors by file path
|
|
23
|
+
3. Sort by dependency order (fix imports/types before logic errors)
|
|
24
|
+
4. Count total errors for progress tracking
|
|
25
|
+
|
|
26
|
+
## Step 3: Fix Loop (One Error at a Time)
|
|
27
|
+
|
|
28
|
+
For each error:
|
|
29
|
+
|
|
30
|
+
1. **Read the file** — Use Read tool to see error context (10 lines around the error)
|
|
31
|
+
2. **Diagnose** — Identify root cause (missing import, wrong type, syntax error)
|
|
32
|
+
3. **Fix minimally** — Use Edit tool for the smallest change that resolves the error
|
|
33
|
+
4. **Re-run build** — Verify the error is gone and no new errors introduced
|
|
34
|
+
5. **Move to next** — Continue with remaining errors
|
|
35
|
+
|
|
36
|
+
## Step 4: Guardrails
|
|
37
|
+
|
|
38
|
+
Stop and ask the user if:
|
|
39
|
+
- A fix introduces **more errors than it resolves**
|
|
40
|
+
- The **same error persists after 3 attempts** (likely a deeper issue)
|
|
41
|
+
- The fix requires **architectural changes** (not just a build fix)
|
|
42
|
+
- Build errors stem from **missing dependencies** (need `npm install`, `cargo add`, etc.)
|
|
43
|
+
|
|
44
|
+
## Step 5: Summary
|
|
45
|
+
|
|
46
|
+
Show results:
|
|
47
|
+
- Errors fixed (with file paths)
|
|
48
|
+
- Errors remaining (if any)
|
|
49
|
+
- New errors introduced (should be zero)
|
|
50
|
+
- Suggested next steps for unresolved issues
|
|
51
|
+
|
|
52
|
+
## Recovery Strategies
|
|
53
|
+
|
|
54
|
+
| Situation | Action |
|
|
55
|
+
|-----------|--------|
|
|
56
|
+
| Missing module/import | Check if package is installed; suggest install command |
|
|
57
|
+
| Type mismatch | Read both type definitions; fix the narrower type |
|
|
58
|
+
| Circular dependency | Identify cycle with import graph; suggest extraction |
|
|
59
|
+
| Version conflict | Check `package.json` / `Cargo.toml` for version constraints |
|
|
60
|
+
| Build tool misconfiguration | Read config file; compare with working defaults |
|
|
61
|
+
|
|
62
|
+
Fix one error at a time for safety. Prefer minimal diffs over refactoring.
|