claude-code-pilot 2.0.0 → 3.1.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/README.md +76 -97
- 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/{ecc → src}/agents/doc-updater.md +1 -1
- package/src/agents/docs-lookup.md +68 -0
- package/src/agents/flutter-reviewer.md +243 -0
- package/src/agents/gan-evaluator.md +209 -0
- package/src/agents/gan-generator.md +131 -0
- package/src/agents/gan-planner.md +99 -0
- package/src/agents/go-build-resolver.md +94 -0
- package/src/agents/go-reviewer.md +76 -0
- package/src/agents/harness-optimizer.md +35 -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/loop-operator.md +36 -0
- package/src/agents/opensource-forker.md +198 -0
- package/src/agents/opensource-packager.md +249 -0
- package/src/agents/opensource-sanitizer.md +188 -0
- package/src/agents/performance-optimizer.md +446 -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/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/src/commands/ccp/aside.md +165 -0
- 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/src/commands/ccp/build-fix.md +67 -0
- package/{gsd/commands-gsd → src/commands/ccp}/check-todos.md +3 -3
- package/{ecc/commands → src/commands/ccp}/checkpoint.md +12 -7
- package/{gsd/commands-gsd → src/commands/ccp}/cleanup.md +3 -3
- package/src/commands/ccp/code-review.md +45 -0
- package/{gsd/commands-gsd → src/commands/ccp}/complete-milestone.md +9 -9
- package/src/commands/ccp/context-budget.md +30 -0
- package/src/commands/ccp/cpp-build.md +174 -0
- package/src/commands/ccp/cpp-review.md +133 -0
- package/src/commands/ccp/cpp-test.md +252 -0
- 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/docs-update.md +48 -0
- package/src/commands/ccp/docs.md +32 -0
- package/src/commands/ccp/e2e.md +365 -0
- package/src/commands/ccp/eval.md +125 -0
- package/{ecc/commands → src/commands/ccp}/evolve.md +5 -5
- 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/src/commands/ccp/go-build.md +184 -0
- package/src/commands/ccp/go-review.md +149 -0
- package/src/commands/ccp/go-test.md +269 -0
- package/src/commands/ccp/gradle-build.md +71 -0
- package/src/commands/ccp/harness-audit.md +76 -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/src/commands/ccp/kotlin-build.md +175 -0
- package/src/commands/ccp/kotlin-review.md +141 -0
- package/src/commands/ccp/kotlin-test.md +313 -0
- package/{ecc/commands → src/commands/ccp}/learn.md +7 -2
- 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/{ecc/commands → src/commands/ccp}/model-route.md +6 -1
- 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/src/commands/ccp/orchestrate.md +232 -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/plan.md +115 -0
- package/src/commands/ccp/plant-seed.md +28 -0
- package/src/commands/ccp/pr-branch.md +25 -0
- package/src/commands/ccp/profile-user.md +46 -0
- package/{gsd/commands-gsd → src/commands/ccp}/progress.md +3 -3
- package/src/commands/ccp/prompt-optimize.md +39 -0
- package/src/commands/ccp/prune.md +25 -0
- package/src/commands/ccp/python-review.md +298 -0
- package/{ecc/commands → src/commands/ccp}/quality-gate.md +7 -2
- package/{gsd/commands-gsd → src/commands/ccp}/quick.md +10 -8
- package/src/commands/ccp/refactor-clean.md +85 -0
- 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/{ecc/commands → src/commands/ccp}/resume-session.md +9 -8
- 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/review.md +37 -0
- package/src/commands/ccp/rules-distill.md +12 -0
- package/src/commands/ccp/rust-build.md +188 -0
- package/src/commands/ccp/rust-review.md +143 -0
- package/src/commands/ccp/rust-test.md +309 -0
- package/{ecc/commands → src/commands/ccp}/save-session.md +2 -1
- package/src/commands/ccp/secure-phase.md +35 -0
- package/src/commands/ccp/session-report.md +19 -0
- package/{ecc/commands → src/commands/ccp}/sessions.md +39 -34
- package/src/commands/ccp/set-profile.md +12 -0
- package/{gsd/commands-gsd → src/commands/ccp}/settings.md +5 -5
- package/src/commands/ccp/setup-pm.md +81 -0
- package/{kit/commands → src/commands/ccp}/setup-refresh.md +4 -3
- package/{kit/commands → src/commands/ccp}/setup.md +67 -40
- package/src/commands/ccp/ship.md +23 -0
- package/src/commands/ccp/skill-create.md +172 -0
- package/src/commands/ccp/skill-health.md +51 -0
- package/src/commands/ccp/stats.md +18 -0
- package/src/commands/ccp/tdd.md +329 -0
- package/src/commands/ccp/test-coverage.md +74 -0
- package/src/commands/ccp/thread.md +127 -0
- package/{kit/commands → src/commands/ccp}/tool-guide.md +2 -1
- package/src/commands/ccp/ui-phase.md +34 -0
- package/src/commands/ccp/ui-review.md +32 -0
- package/src/commands/ccp/update-codemaps.md +77 -0
- package/src/commands/ccp/update-docs.md +89 -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/{ecc/commands → src/commands/ccp}/verify.md +5 -0
- package/src/commands/ccp/workstreams.md +68 -0
- package/{ecc → src}/examples/CLAUDE.md +4 -4
- package/{ecc → src}/examples/django-api-CLAUDE.md +5 -5
- package/{ecc → src}/examples/go-microservice-CLAUDE.md +6 -6
- package/{ecc → src}/examples/rust-api-CLAUDE.md +4 -4
- package/{ecc → src}/examples/saas-nextjs-CLAUDE.md +8 -8
- 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 +76 -10
- 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/docs-update.md +1165 -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 +767 -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/src/pilot/workflows/profile-user.md +452 -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/review.md +244 -0
- package/src/pilot/workflows/secure-phase.md +164 -0
- 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/src/rules/common/code-review.md +124 -0
- package/{ecc → src}/rules/common/coding-style.md +21 -0
- package/src/rules/zh/README.md +108 -0
- package/src/rules/zh/agents.md +50 -0
- package/src/rules/zh/code-review.md +124 -0
- package/src/rules/zh/coding-style.md +48 -0
- package/src/rules/zh/development-workflow.md +44 -0
- package/src/rules/zh/git-workflow.md +24 -0
- package/src/rules/zh/hooks.md +30 -0
- package/src/rules/zh/patterns.md +31 -0
- package/src/rules/zh/performance.md +55 -0
- package/src/rules/zh/security.md +29 -0
- package/src/rules/zh/testing.md +29 -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/autonomous-agent-harness/SKILL.md +267 -0
- package/src/skills/autonomous-loops/SKILL.md +610 -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/bun-runtime/SKILL.md +84 -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/content-hash-cache-pattern/SKILL.md +161 -0
- package/src/skills/context-budget/SKILL.md +135 -0
- package/{ecc → src}/skills/continuous-learning-v2/SKILL.md +6 -6
- package/{ecc → src}/skills/continuous-learning-v2/agents/observer-loop.sh +1 -1
- package/{ecc → src}/skills/continuous-learning-v2/agents/observer.md +1 -1
- package/src/skills/cost-aware-llm-pipeline/SKILL.md +183 -0
- 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/design-system/SKILL.md +82 -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/eval-harness/SKILL.md +270 -0
- package/src/skills/exa-search/SKILL.md +103 -0
- package/src/skills/flutter-dart-code-review/SKILL.md +435 -0
- package/src/skills/frontend-patterns/SKILL.md +642 -0
- package/src/skills/gan-style-harness/SKILL.md +278 -0
- package/src/skills/git-workflow/SKILL.md +715 -0
- package/src/skills/golang-patterns/SKILL.md +674 -0
- package/src/skills/golang-testing/SKILL.md +720 -0
- package/src/skills/hexagonal-architecture/SKILL.md +276 -0
- package/src/skills/iterative-retrieval/SKILL.md +211 -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-plugin-discovery/SKILL.md +229 -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/nextjs-turbopack/SKILL.md +44 -0
- package/src/skills/nuxt4-patterns/SKILL.md +100 -0
- package/src/skills/opensource-pipeline/SKILL.md +255 -0
- package/src/skills/perl-patterns/SKILL.md +504 -0
- package/src/skills/perl-security/SKILL.md +503 -0
- package/src/skills/perl-testing/SKILL.md +475 -0
- package/src/skills/postgres-patterns/SKILL.md +147 -0
- package/src/skills/project-flow-ops/SKILL.md +111 -0
- package/src/skills/project-guidelines-example/SKILL.md +349 -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/pytorch-patterns/SKILL.md +396 -0
- package/src/skills/regex-vs-llm-structured-text/SKILL.md +220 -0
- package/src/skills/repo-scan/SKILL.md +78 -0
- package/src/skills/rules-distill/SKILL.md +264 -0
- package/src/skills/rules-distill/scripts/scan-rules.sh +58 -0
- package/src/skills/rules-distill/scripts/scan-skills.sh +129 -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/swift-concurrency-6-2/SKILL.md +216 -0
- package/src/skills/tdd-workflow/SKILL.md +410 -0
- package/src/skills/token-budget-advisor/SKILL.md +133 -0
- package/{ecc/skills/verification-loop-SKILL.md → src/skills/verification-loop/SKILL.md} +1 -1
- package/src/skills/workspace-surface-audit/SKILL.md +125 -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/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}/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/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/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
|
@@ -0,0 +1,530 @@
|
|
|
1
|
+
---
|
|
2
|
+
name: coding-standards
|
|
3
|
+
description: Universal coding standards, best practices, and patterns for TypeScript, JavaScript, React, and Node.js development.
|
|
4
|
+
origin: ECC
|
|
5
|
+
---
|
|
6
|
+
|
|
7
|
+
# Coding Standards & Best Practices
|
|
8
|
+
|
|
9
|
+
Universal coding standards applicable across all projects.
|
|
10
|
+
|
|
11
|
+
## When to Activate
|
|
12
|
+
|
|
13
|
+
- Starting a new project or module
|
|
14
|
+
- Reviewing code for quality and maintainability
|
|
15
|
+
- Refactoring existing code to follow conventions
|
|
16
|
+
- Enforcing naming, formatting, or structural consistency
|
|
17
|
+
- Setting up linting, formatting, or type-checking rules
|
|
18
|
+
- Onboarding new contributors to coding conventions
|
|
19
|
+
|
|
20
|
+
## Code Quality Principles
|
|
21
|
+
|
|
22
|
+
### 1. Readability First
|
|
23
|
+
- Code is read more than written
|
|
24
|
+
- Clear variable and function names
|
|
25
|
+
- Self-documenting code preferred over comments
|
|
26
|
+
- Consistent formatting
|
|
27
|
+
|
|
28
|
+
### 2. KISS (Keep It Simple, Stupid)
|
|
29
|
+
- Simplest solution that works
|
|
30
|
+
- Avoid over-engineering
|
|
31
|
+
- No premature optimization
|
|
32
|
+
- Easy to understand > clever code
|
|
33
|
+
|
|
34
|
+
### 3. DRY (Don't Repeat Yourself)
|
|
35
|
+
- Extract common logic into functions
|
|
36
|
+
- Create reusable components
|
|
37
|
+
- Share utilities across modules
|
|
38
|
+
- Avoid copy-paste programming
|
|
39
|
+
|
|
40
|
+
### 4. YAGNI (You Aren't Gonna Need It)
|
|
41
|
+
- Don't build features before they're needed
|
|
42
|
+
- Avoid speculative generality
|
|
43
|
+
- Add complexity only when required
|
|
44
|
+
- Start simple, refactor when needed
|
|
45
|
+
|
|
46
|
+
## TypeScript/JavaScript Standards
|
|
47
|
+
|
|
48
|
+
### Variable Naming
|
|
49
|
+
|
|
50
|
+
```typescript
|
|
51
|
+
// ✅ GOOD: Descriptive names
|
|
52
|
+
const marketSearchQuery = 'election'
|
|
53
|
+
const isUserAuthenticated = true
|
|
54
|
+
const totalRevenue = 1000
|
|
55
|
+
|
|
56
|
+
// ❌ BAD: Unclear names
|
|
57
|
+
const q = 'election'
|
|
58
|
+
const flag = true
|
|
59
|
+
const x = 1000
|
|
60
|
+
```
|
|
61
|
+
|
|
62
|
+
### Function Naming
|
|
63
|
+
|
|
64
|
+
```typescript
|
|
65
|
+
// ✅ GOOD: Verb-noun pattern
|
|
66
|
+
async function fetchMarketData(marketId: string) { }
|
|
67
|
+
function calculateSimilarity(a: number[], b: number[]) { }
|
|
68
|
+
function isValidEmail(email: string): boolean { }
|
|
69
|
+
|
|
70
|
+
// ❌ BAD: Unclear or noun-only
|
|
71
|
+
async function market(id: string) { }
|
|
72
|
+
function similarity(a, b) { }
|
|
73
|
+
function email(e) { }
|
|
74
|
+
```
|
|
75
|
+
|
|
76
|
+
### Immutability Pattern (CRITICAL)
|
|
77
|
+
|
|
78
|
+
```typescript
|
|
79
|
+
// ✅ ALWAYS use spread operator
|
|
80
|
+
const updatedUser = {
|
|
81
|
+
...user,
|
|
82
|
+
name: 'New Name'
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
const updatedArray = [...items, newItem]
|
|
86
|
+
|
|
87
|
+
// ❌ NEVER mutate directly
|
|
88
|
+
user.name = 'New Name' // BAD
|
|
89
|
+
items.push(newItem) // BAD
|
|
90
|
+
```
|
|
91
|
+
|
|
92
|
+
### Error Handling
|
|
93
|
+
|
|
94
|
+
```typescript
|
|
95
|
+
// ✅ GOOD: Comprehensive error handling
|
|
96
|
+
async function fetchData(url: string) {
|
|
97
|
+
try {
|
|
98
|
+
const response = await fetch(url)
|
|
99
|
+
|
|
100
|
+
if (!response.ok) {
|
|
101
|
+
throw new Error(`HTTP ${response.status}: ${response.statusText}`)
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
return await response.json()
|
|
105
|
+
} catch (error) {
|
|
106
|
+
console.error('Fetch failed:', error)
|
|
107
|
+
throw new Error('Failed to fetch data')
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
// ❌ BAD: No error handling
|
|
112
|
+
async function fetchData(url) {
|
|
113
|
+
const response = await fetch(url)
|
|
114
|
+
return response.json()
|
|
115
|
+
}
|
|
116
|
+
```
|
|
117
|
+
|
|
118
|
+
### Async/Await Best Practices
|
|
119
|
+
|
|
120
|
+
```typescript
|
|
121
|
+
// ✅ GOOD: Parallel execution when possible
|
|
122
|
+
const [users, markets, stats] = await Promise.all([
|
|
123
|
+
fetchUsers(),
|
|
124
|
+
fetchMarkets(),
|
|
125
|
+
fetchStats()
|
|
126
|
+
])
|
|
127
|
+
|
|
128
|
+
// ❌ BAD: Sequential when unnecessary
|
|
129
|
+
const users = await fetchUsers()
|
|
130
|
+
const markets = await fetchMarkets()
|
|
131
|
+
const stats = await fetchStats()
|
|
132
|
+
```
|
|
133
|
+
|
|
134
|
+
### Type Safety
|
|
135
|
+
|
|
136
|
+
```typescript
|
|
137
|
+
// ✅ GOOD: Proper types
|
|
138
|
+
interface Market {
|
|
139
|
+
id: string
|
|
140
|
+
name: string
|
|
141
|
+
status: 'active' | 'resolved' | 'closed'
|
|
142
|
+
created_at: Date
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
function getMarket(id: string): Promise<Market> {
|
|
146
|
+
// Implementation
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
// ❌ BAD: Using 'any'
|
|
150
|
+
function getMarket(id: any): Promise<any> {
|
|
151
|
+
// Implementation
|
|
152
|
+
}
|
|
153
|
+
```
|
|
154
|
+
|
|
155
|
+
## React Best Practices
|
|
156
|
+
|
|
157
|
+
### Component Structure
|
|
158
|
+
|
|
159
|
+
```typescript
|
|
160
|
+
// ✅ GOOD: Functional component with types
|
|
161
|
+
interface ButtonProps {
|
|
162
|
+
children: React.ReactNode
|
|
163
|
+
onClick: () => void
|
|
164
|
+
disabled?: boolean
|
|
165
|
+
variant?: 'primary' | 'secondary'
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
export function Button({
|
|
169
|
+
children,
|
|
170
|
+
onClick,
|
|
171
|
+
disabled = false,
|
|
172
|
+
variant = 'primary'
|
|
173
|
+
}: ButtonProps) {
|
|
174
|
+
return (
|
|
175
|
+
<button
|
|
176
|
+
onClick={onClick}
|
|
177
|
+
disabled={disabled}
|
|
178
|
+
className={`btn btn-${variant}`}
|
|
179
|
+
>
|
|
180
|
+
{children}
|
|
181
|
+
</button>
|
|
182
|
+
)
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
// ❌ BAD: No types, unclear structure
|
|
186
|
+
export function Button(props) {
|
|
187
|
+
return <button onClick={props.onClick}>{props.children}</button>
|
|
188
|
+
}
|
|
189
|
+
```
|
|
190
|
+
|
|
191
|
+
### Custom Hooks
|
|
192
|
+
|
|
193
|
+
```typescript
|
|
194
|
+
// ✅ GOOD: Reusable custom hook
|
|
195
|
+
export function useDebounce<T>(value: T, delay: number): T {
|
|
196
|
+
const [debouncedValue, setDebouncedValue] = useState<T>(value)
|
|
197
|
+
|
|
198
|
+
useEffect(() => {
|
|
199
|
+
const handler = setTimeout(() => {
|
|
200
|
+
setDebouncedValue(value)
|
|
201
|
+
}, delay)
|
|
202
|
+
|
|
203
|
+
return () => clearTimeout(handler)
|
|
204
|
+
}, [value, delay])
|
|
205
|
+
|
|
206
|
+
return debouncedValue
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
// Usage
|
|
210
|
+
const debouncedQuery = useDebounce(searchQuery, 500)
|
|
211
|
+
```
|
|
212
|
+
|
|
213
|
+
### State Management
|
|
214
|
+
|
|
215
|
+
```typescript
|
|
216
|
+
// ✅ GOOD: Proper state updates
|
|
217
|
+
const [count, setCount] = useState(0)
|
|
218
|
+
|
|
219
|
+
// Functional update for state based on previous state
|
|
220
|
+
setCount(prev => prev + 1)
|
|
221
|
+
|
|
222
|
+
// ❌ BAD: Direct state reference
|
|
223
|
+
setCount(count + 1) // Can be stale in async scenarios
|
|
224
|
+
```
|
|
225
|
+
|
|
226
|
+
### Conditional Rendering
|
|
227
|
+
|
|
228
|
+
```typescript
|
|
229
|
+
// ✅ GOOD: Clear conditional rendering
|
|
230
|
+
{isLoading && <Spinner />}
|
|
231
|
+
{error && <ErrorMessage error={error} />}
|
|
232
|
+
{data && <DataDisplay data={data} />}
|
|
233
|
+
|
|
234
|
+
// ❌ BAD: Ternary hell
|
|
235
|
+
{isLoading ? <Spinner /> : error ? <ErrorMessage error={error} /> : data ? <DataDisplay data={data} /> : null}
|
|
236
|
+
```
|
|
237
|
+
|
|
238
|
+
## API Design Standards
|
|
239
|
+
|
|
240
|
+
### REST API Conventions
|
|
241
|
+
|
|
242
|
+
```
|
|
243
|
+
GET /api/markets # List all markets
|
|
244
|
+
GET /api/markets/:id # Get specific market
|
|
245
|
+
POST /api/markets # Create new market
|
|
246
|
+
PUT /api/markets/:id # Update market (full)
|
|
247
|
+
PATCH /api/markets/:id # Update market (partial)
|
|
248
|
+
DELETE /api/markets/:id # Delete market
|
|
249
|
+
|
|
250
|
+
# Query parameters for filtering
|
|
251
|
+
GET /api/markets?status=active&limit=10&offset=0
|
|
252
|
+
```
|
|
253
|
+
|
|
254
|
+
### Response Format
|
|
255
|
+
|
|
256
|
+
```typescript
|
|
257
|
+
// ✅ GOOD: Consistent response structure
|
|
258
|
+
interface ApiResponse<T> {
|
|
259
|
+
success: boolean
|
|
260
|
+
data?: T
|
|
261
|
+
error?: string
|
|
262
|
+
meta?: {
|
|
263
|
+
total: number
|
|
264
|
+
page: number
|
|
265
|
+
limit: number
|
|
266
|
+
}
|
|
267
|
+
}
|
|
268
|
+
|
|
269
|
+
// Success response
|
|
270
|
+
return NextResponse.json({
|
|
271
|
+
success: true,
|
|
272
|
+
data: markets,
|
|
273
|
+
meta: { total: 100, page: 1, limit: 10 }
|
|
274
|
+
})
|
|
275
|
+
|
|
276
|
+
// Error response
|
|
277
|
+
return NextResponse.json({
|
|
278
|
+
success: false,
|
|
279
|
+
error: 'Invalid request'
|
|
280
|
+
}, { status: 400 })
|
|
281
|
+
```
|
|
282
|
+
|
|
283
|
+
### Input Validation
|
|
284
|
+
|
|
285
|
+
```typescript
|
|
286
|
+
import { z } from 'zod'
|
|
287
|
+
|
|
288
|
+
// ✅ GOOD: Schema validation
|
|
289
|
+
const CreateMarketSchema = z.object({
|
|
290
|
+
name: z.string().min(1).max(200),
|
|
291
|
+
description: z.string().min(1).max(2000),
|
|
292
|
+
endDate: z.string().datetime(),
|
|
293
|
+
categories: z.array(z.string()).min(1)
|
|
294
|
+
})
|
|
295
|
+
|
|
296
|
+
export async function POST(request: Request) {
|
|
297
|
+
const body = await request.json()
|
|
298
|
+
|
|
299
|
+
try {
|
|
300
|
+
const validated = CreateMarketSchema.parse(body)
|
|
301
|
+
// Proceed with validated data
|
|
302
|
+
} catch (error) {
|
|
303
|
+
if (error instanceof z.ZodError) {
|
|
304
|
+
return NextResponse.json({
|
|
305
|
+
success: false,
|
|
306
|
+
error: 'Validation failed',
|
|
307
|
+
details: error.errors
|
|
308
|
+
}, { status: 400 })
|
|
309
|
+
}
|
|
310
|
+
}
|
|
311
|
+
}
|
|
312
|
+
```
|
|
313
|
+
|
|
314
|
+
## File Organization
|
|
315
|
+
|
|
316
|
+
### Project Structure
|
|
317
|
+
|
|
318
|
+
```
|
|
319
|
+
src/
|
|
320
|
+
├── app/ # Next.js App Router
|
|
321
|
+
│ ├── api/ # API routes
|
|
322
|
+
│ ├── markets/ # Market pages
|
|
323
|
+
│ └── (auth)/ # Auth pages (route groups)
|
|
324
|
+
├── components/ # React components
|
|
325
|
+
│ ├── ui/ # Generic UI components
|
|
326
|
+
│ ├── forms/ # Form components
|
|
327
|
+
│ └── layouts/ # Layout components
|
|
328
|
+
├── hooks/ # Custom React hooks
|
|
329
|
+
├── lib/ # Utilities and configs
|
|
330
|
+
│ ├── api/ # API clients
|
|
331
|
+
│ ├── utils/ # Helper functions
|
|
332
|
+
│ └── constants/ # Constants
|
|
333
|
+
├── types/ # TypeScript types
|
|
334
|
+
└── styles/ # Global styles
|
|
335
|
+
```
|
|
336
|
+
|
|
337
|
+
### File Naming
|
|
338
|
+
|
|
339
|
+
```
|
|
340
|
+
components/Button.tsx # PascalCase for components
|
|
341
|
+
hooks/useAuth.ts # camelCase with 'use' prefix
|
|
342
|
+
lib/formatDate.ts # camelCase for utilities
|
|
343
|
+
types/market.types.ts # camelCase with .types suffix
|
|
344
|
+
```
|
|
345
|
+
|
|
346
|
+
## Comments & Documentation
|
|
347
|
+
|
|
348
|
+
### When to Comment
|
|
349
|
+
|
|
350
|
+
```typescript
|
|
351
|
+
// ✅ GOOD: Explain WHY, not WHAT
|
|
352
|
+
// Use exponential backoff to avoid overwhelming the API during outages
|
|
353
|
+
const delay = Math.min(1000 * Math.pow(2, retryCount), 30000)
|
|
354
|
+
|
|
355
|
+
// Deliberately using mutation here for performance with large arrays
|
|
356
|
+
items.push(newItem)
|
|
357
|
+
|
|
358
|
+
// ❌ BAD: Stating the obvious
|
|
359
|
+
// Increment counter by 1
|
|
360
|
+
count++
|
|
361
|
+
|
|
362
|
+
// Set name to user's name
|
|
363
|
+
name = user.name
|
|
364
|
+
```
|
|
365
|
+
|
|
366
|
+
### JSDoc for Public APIs
|
|
367
|
+
|
|
368
|
+
```typescript
|
|
369
|
+
/**
|
|
370
|
+
* Searches markets using semantic similarity.
|
|
371
|
+
*
|
|
372
|
+
* @param query - Natural language search query
|
|
373
|
+
* @param limit - Maximum number of results (default: 10)
|
|
374
|
+
* @returns Array of markets sorted by similarity score
|
|
375
|
+
* @throws {Error} If OpenAI API fails or Redis unavailable
|
|
376
|
+
*
|
|
377
|
+
* @example
|
|
378
|
+
* ```typescript
|
|
379
|
+
* const results = await searchMarkets('election', 5)
|
|
380
|
+
* console.log(results[0].name) // "Trump vs Biden"
|
|
381
|
+
* ```
|
|
382
|
+
*/
|
|
383
|
+
export async function searchMarkets(
|
|
384
|
+
query: string,
|
|
385
|
+
limit: number = 10
|
|
386
|
+
): Promise<Market[]> {
|
|
387
|
+
// Implementation
|
|
388
|
+
}
|
|
389
|
+
```
|
|
390
|
+
|
|
391
|
+
## Performance Best Practices
|
|
392
|
+
|
|
393
|
+
### Memoization
|
|
394
|
+
|
|
395
|
+
```typescript
|
|
396
|
+
import { useMemo, useCallback } from 'react'
|
|
397
|
+
|
|
398
|
+
// ✅ GOOD: Memoize expensive computations
|
|
399
|
+
const sortedMarkets = useMemo(() => {
|
|
400
|
+
return markets.sort((a, b) => b.volume - a.volume)
|
|
401
|
+
}, [markets])
|
|
402
|
+
|
|
403
|
+
// ✅ GOOD: Memoize callbacks
|
|
404
|
+
const handleSearch = useCallback((query: string) => {
|
|
405
|
+
setSearchQuery(query)
|
|
406
|
+
}, [])
|
|
407
|
+
```
|
|
408
|
+
|
|
409
|
+
### Lazy Loading
|
|
410
|
+
|
|
411
|
+
```typescript
|
|
412
|
+
import { lazy, Suspense } from 'react'
|
|
413
|
+
|
|
414
|
+
// ✅ GOOD: Lazy load heavy components
|
|
415
|
+
const HeavyChart = lazy(() => import('./HeavyChart'))
|
|
416
|
+
|
|
417
|
+
export function Dashboard() {
|
|
418
|
+
return (
|
|
419
|
+
<Suspense fallback={<Spinner />}>
|
|
420
|
+
<HeavyChart />
|
|
421
|
+
</Suspense>
|
|
422
|
+
)
|
|
423
|
+
}
|
|
424
|
+
```
|
|
425
|
+
|
|
426
|
+
### Database Queries
|
|
427
|
+
|
|
428
|
+
```typescript
|
|
429
|
+
// ✅ GOOD: Select only needed columns
|
|
430
|
+
const { data } = await supabase
|
|
431
|
+
.from('markets')
|
|
432
|
+
.select('id, name, status')
|
|
433
|
+
.limit(10)
|
|
434
|
+
|
|
435
|
+
// ❌ BAD: Select everything
|
|
436
|
+
const { data } = await supabase
|
|
437
|
+
.from('markets')
|
|
438
|
+
.select('*')
|
|
439
|
+
```
|
|
440
|
+
|
|
441
|
+
## Testing Standards
|
|
442
|
+
|
|
443
|
+
### Test Structure (AAA Pattern)
|
|
444
|
+
|
|
445
|
+
```typescript
|
|
446
|
+
test('calculates similarity correctly', () => {
|
|
447
|
+
// Arrange
|
|
448
|
+
const vector1 = [1, 0, 0]
|
|
449
|
+
const vector2 = [0, 1, 0]
|
|
450
|
+
|
|
451
|
+
// Act
|
|
452
|
+
const similarity = calculateCosineSimilarity(vector1, vector2)
|
|
453
|
+
|
|
454
|
+
// Assert
|
|
455
|
+
expect(similarity).toBe(0)
|
|
456
|
+
})
|
|
457
|
+
```
|
|
458
|
+
|
|
459
|
+
### Test Naming
|
|
460
|
+
|
|
461
|
+
```typescript
|
|
462
|
+
// ✅ GOOD: Descriptive test names
|
|
463
|
+
test('returns empty array when no markets match query', () => { })
|
|
464
|
+
test('throws error when OpenAI API key is missing', () => { })
|
|
465
|
+
test('falls back to substring search when Redis unavailable', () => { })
|
|
466
|
+
|
|
467
|
+
// ❌ BAD: Vague test names
|
|
468
|
+
test('works', () => { })
|
|
469
|
+
test('test search', () => { })
|
|
470
|
+
```
|
|
471
|
+
|
|
472
|
+
## Code Smell Detection
|
|
473
|
+
|
|
474
|
+
Watch for these anti-patterns:
|
|
475
|
+
|
|
476
|
+
### 1. Long Functions
|
|
477
|
+
```typescript
|
|
478
|
+
// ❌ BAD: Function > 50 lines
|
|
479
|
+
function processMarketData() {
|
|
480
|
+
// 100 lines of code
|
|
481
|
+
}
|
|
482
|
+
|
|
483
|
+
// ✅ GOOD: Split into smaller functions
|
|
484
|
+
function processMarketData() {
|
|
485
|
+
const validated = validateData()
|
|
486
|
+
const transformed = transformData(validated)
|
|
487
|
+
return saveData(transformed)
|
|
488
|
+
}
|
|
489
|
+
```
|
|
490
|
+
|
|
491
|
+
### 2. Deep Nesting
|
|
492
|
+
```typescript
|
|
493
|
+
// ❌ BAD: 5+ levels of nesting
|
|
494
|
+
if (user) {
|
|
495
|
+
if (user.isAdmin) {
|
|
496
|
+
if (market) {
|
|
497
|
+
if (market.isActive) {
|
|
498
|
+
if (hasPermission) {
|
|
499
|
+
// Do something
|
|
500
|
+
}
|
|
501
|
+
}
|
|
502
|
+
}
|
|
503
|
+
}
|
|
504
|
+
}
|
|
505
|
+
|
|
506
|
+
// ✅ GOOD: Early returns
|
|
507
|
+
if (!user) return
|
|
508
|
+
if (!user.isAdmin) return
|
|
509
|
+
if (!market) return
|
|
510
|
+
if (!market.isActive) return
|
|
511
|
+
if (!hasPermission) return
|
|
512
|
+
|
|
513
|
+
// Do something
|
|
514
|
+
```
|
|
515
|
+
|
|
516
|
+
### 3. Magic Numbers
|
|
517
|
+
```typescript
|
|
518
|
+
// ❌ BAD: Unexplained numbers
|
|
519
|
+
if (retryCount > 3) { }
|
|
520
|
+
setTimeout(callback, 500)
|
|
521
|
+
|
|
522
|
+
// ✅ GOOD: Named constants
|
|
523
|
+
const MAX_RETRIES = 3
|
|
524
|
+
const DEBOUNCE_DELAY_MS = 500
|
|
525
|
+
|
|
526
|
+
if (retryCount > MAX_RETRIES) { }
|
|
527
|
+
setTimeout(callback, DEBOUNCE_DELAY_MS)
|
|
528
|
+
```
|
|
529
|
+
|
|
530
|
+
**Remember**: Code quality is not negotiable. Clear, maintainable code enables rapid development and confident refactoring.
|
|
@@ -0,0 +1,161 @@
|
|
|
1
|
+
---
|
|
2
|
+
name: content-hash-cache-pattern
|
|
3
|
+
description: Cache expensive file processing results using SHA-256 content hashes — path-independent, auto-invalidating, with service layer separation.
|
|
4
|
+
origin: ECC
|
|
5
|
+
---
|
|
6
|
+
|
|
7
|
+
# Content-Hash File Cache Pattern
|
|
8
|
+
|
|
9
|
+
Cache expensive file processing results (PDF parsing, text extraction, image analysis) using SHA-256 content hashes as cache keys. Unlike path-based caching, this approach survives file moves/renames and auto-invalidates when content changes.
|
|
10
|
+
|
|
11
|
+
## When to Activate
|
|
12
|
+
|
|
13
|
+
- Building file processing pipelines (PDF, images, text extraction)
|
|
14
|
+
- Processing cost is high and same files are processed repeatedly
|
|
15
|
+
- Need a `--cache/--no-cache` CLI option
|
|
16
|
+
- Want to add caching to existing pure functions without modifying them
|
|
17
|
+
|
|
18
|
+
## Core Pattern
|
|
19
|
+
|
|
20
|
+
### 1. Content-Hash Based Cache Key
|
|
21
|
+
|
|
22
|
+
Use file content (not path) as the cache key:
|
|
23
|
+
|
|
24
|
+
```python
|
|
25
|
+
import hashlib
|
|
26
|
+
from pathlib import Path
|
|
27
|
+
|
|
28
|
+
_HASH_CHUNK_SIZE = 65536 # 64KB chunks for large files
|
|
29
|
+
|
|
30
|
+
def compute_file_hash(path: Path) -> str:
|
|
31
|
+
"""SHA-256 of file contents (chunked for large files)."""
|
|
32
|
+
if not path.is_file():
|
|
33
|
+
raise FileNotFoundError(f"File not found: {path}")
|
|
34
|
+
sha256 = hashlib.sha256()
|
|
35
|
+
with open(path, "rb") as f:
|
|
36
|
+
while True:
|
|
37
|
+
chunk = f.read(_HASH_CHUNK_SIZE)
|
|
38
|
+
if not chunk:
|
|
39
|
+
break
|
|
40
|
+
sha256.update(chunk)
|
|
41
|
+
return sha256.hexdigest()
|
|
42
|
+
```
|
|
43
|
+
|
|
44
|
+
**Why content hash?** File rename/move = cache hit. Content change = automatic invalidation. No index file needed.
|
|
45
|
+
|
|
46
|
+
### 2. Frozen Dataclass for Cache Entry
|
|
47
|
+
|
|
48
|
+
```python
|
|
49
|
+
from dataclasses import dataclass
|
|
50
|
+
|
|
51
|
+
@dataclass(frozen=True, slots=True)
|
|
52
|
+
class CacheEntry:
|
|
53
|
+
file_hash: str
|
|
54
|
+
source_path: str
|
|
55
|
+
document: ExtractedDocument # The cached result
|
|
56
|
+
```
|
|
57
|
+
|
|
58
|
+
### 3. File-Based Cache Storage
|
|
59
|
+
|
|
60
|
+
Each cache entry is stored as `{hash}.json` — O(1) lookup by hash, no index file required.
|
|
61
|
+
|
|
62
|
+
```python
|
|
63
|
+
import json
|
|
64
|
+
from typing import Any
|
|
65
|
+
|
|
66
|
+
def write_cache(cache_dir: Path, entry: CacheEntry) -> None:
|
|
67
|
+
cache_dir.mkdir(parents=True, exist_ok=True)
|
|
68
|
+
cache_file = cache_dir / f"{entry.file_hash}.json"
|
|
69
|
+
data = serialize_entry(entry)
|
|
70
|
+
cache_file.write_text(json.dumps(data, ensure_ascii=False), encoding="utf-8")
|
|
71
|
+
|
|
72
|
+
def read_cache(cache_dir: Path, file_hash: str) -> CacheEntry | None:
|
|
73
|
+
cache_file = cache_dir / f"{file_hash}.json"
|
|
74
|
+
if not cache_file.is_file():
|
|
75
|
+
return None
|
|
76
|
+
try:
|
|
77
|
+
raw = cache_file.read_text(encoding="utf-8")
|
|
78
|
+
data = json.loads(raw)
|
|
79
|
+
return deserialize_entry(data)
|
|
80
|
+
except (json.JSONDecodeError, ValueError, KeyError):
|
|
81
|
+
return None # Treat corruption as cache miss
|
|
82
|
+
```
|
|
83
|
+
|
|
84
|
+
### 4. Service Layer Wrapper (SRP)
|
|
85
|
+
|
|
86
|
+
Keep the processing function pure. Add caching as a separate service layer.
|
|
87
|
+
|
|
88
|
+
```python
|
|
89
|
+
def extract_with_cache(
|
|
90
|
+
file_path: Path,
|
|
91
|
+
*,
|
|
92
|
+
cache_enabled: bool = True,
|
|
93
|
+
cache_dir: Path = Path(".cache"),
|
|
94
|
+
) -> ExtractedDocument:
|
|
95
|
+
"""Service layer: cache check -> extraction -> cache write."""
|
|
96
|
+
if not cache_enabled:
|
|
97
|
+
return extract_text(file_path) # Pure function, no cache knowledge
|
|
98
|
+
|
|
99
|
+
file_hash = compute_file_hash(file_path)
|
|
100
|
+
|
|
101
|
+
# Check cache
|
|
102
|
+
cached = read_cache(cache_dir, file_hash)
|
|
103
|
+
if cached is not None:
|
|
104
|
+
logger.info("Cache hit: %s (hash=%s)", file_path.name, file_hash[:12])
|
|
105
|
+
return cached.document
|
|
106
|
+
|
|
107
|
+
# Cache miss -> extract -> store
|
|
108
|
+
logger.info("Cache miss: %s (hash=%s)", file_path.name, file_hash[:12])
|
|
109
|
+
doc = extract_text(file_path)
|
|
110
|
+
entry = CacheEntry(file_hash=file_hash, source_path=str(file_path), document=doc)
|
|
111
|
+
write_cache(cache_dir, entry)
|
|
112
|
+
return doc
|
|
113
|
+
```
|
|
114
|
+
|
|
115
|
+
## Key Design Decisions
|
|
116
|
+
|
|
117
|
+
| Decision | Rationale |
|
|
118
|
+
|----------|-----------|
|
|
119
|
+
| SHA-256 content hash | Path-independent, auto-invalidates on content change |
|
|
120
|
+
| `{hash}.json` file naming | O(1) lookup, no index file needed |
|
|
121
|
+
| Service layer wrapper | SRP: extraction stays pure, cache is a separate concern |
|
|
122
|
+
| Manual JSON serialization | Full control over frozen dataclass serialization |
|
|
123
|
+
| Corruption returns `None` | Graceful degradation, re-processes on next run |
|
|
124
|
+
| `cache_dir.mkdir(parents=True)` | Lazy directory creation on first write |
|
|
125
|
+
|
|
126
|
+
## Best Practices
|
|
127
|
+
|
|
128
|
+
- **Hash content, not paths** — paths change, content identity doesn't
|
|
129
|
+
- **Chunk large files** when hashing — avoid loading entire files into memory
|
|
130
|
+
- **Keep processing functions pure** — they should know nothing about caching
|
|
131
|
+
- **Log cache hit/miss** with truncated hashes for debugging
|
|
132
|
+
- **Handle corruption gracefully** — treat invalid cache entries as misses, never crash
|
|
133
|
+
|
|
134
|
+
## Anti-Patterns to Avoid
|
|
135
|
+
|
|
136
|
+
```python
|
|
137
|
+
# BAD: Path-based caching (breaks on file move/rename)
|
|
138
|
+
cache = {"/path/to/file.pdf": result}
|
|
139
|
+
|
|
140
|
+
# BAD: Adding cache logic inside the processing function (SRP violation)
|
|
141
|
+
def extract_text(path, *, cache_enabled=False, cache_dir=None):
|
|
142
|
+
if cache_enabled: # Now this function has two responsibilities
|
|
143
|
+
...
|
|
144
|
+
|
|
145
|
+
# BAD: Using dataclasses.asdict() with nested frozen dataclasses
|
|
146
|
+
# (can cause issues with complex nested types)
|
|
147
|
+
data = dataclasses.asdict(entry) # Use manual serialization instead
|
|
148
|
+
```
|
|
149
|
+
|
|
150
|
+
## When to Use
|
|
151
|
+
|
|
152
|
+
- File processing pipelines (PDF parsing, OCR, text extraction, image analysis)
|
|
153
|
+
- CLI tools that benefit from `--cache/--no-cache` options
|
|
154
|
+
- Batch processing where the same files appear across runs
|
|
155
|
+
- Adding caching to existing pure functions without modifying them
|
|
156
|
+
|
|
157
|
+
## When NOT to Use
|
|
158
|
+
|
|
159
|
+
- Data that must always be fresh (real-time feeds)
|
|
160
|
+
- Cache entries that would be extremely large (consider streaming instead)
|
|
161
|
+
- Results that depend on parameters beyond file content (e.g., different extraction configs)
|