@arimakouyou/spec-workflow-mcp 2.2.7
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.claude-plugin/.mcp.json +8 -0
- package/.claude-plugin/agents/code-simplifier.md +80 -0
- package/.claude-plugin/agents/integ-test-auditor.md +91 -0
- package/.claude-plugin/agents/integ-test-worker.md +73 -0
- package/.claude-plugin/agents/parallel-worker.md +136 -0
- package/.claude-plugin/agents/review-worker.md +279 -0
- package/.claude-plugin/agents/unit-test-engineer.md +148 -0
- package/.claude-plugin/agents/wave-harness-worker.md +158 -0
- package/.claude-plugin/hooks/hooks.json +16 -0
- package/.claude-plugin/hooks/tasks-read-guard.sh +17 -0
- package/.claude-plugin/marketplace.json +33 -0
- package/.claude-plugin/plugin.json +11 -0
- package/.claude-plugin/rules/axum.md +154 -0
- package/.claude-plugin/rules/cargo-toml.md +63 -0
- package/.claude-plugin/rules/context7.md +17 -0
- package/.claude-plugin/rules/design-conformance.md +82 -0
- package/.claude-plugin/rules/design-principles.md +53 -0
- package/.claude-plugin/rules/diesel.md +176 -0
- package/.claude-plugin/rules/feedback-loop.md +33 -0
- package/.claude-plugin/rules/leptos.md +319 -0
- package/.claude-plugin/rules/project-architecture.md +134 -0
- package/.claude-plugin/rules/quality-checks.md +265 -0
- package/.claude-plugin/rules/rust-style.md +242 -0
- package/.claude-plugin/rules/security.md +67 -0
- package/.claude-plugin/rules/spec-workflow-enforcement.md +47 -0
- package/.claude-plugin/rules/valkey.md +167 -0
- package/.claude-plugin/skills/integration-test/SKILL.md +230 -0
- package/.claude-plugin/skills/integration-test/references/auditor-prompt.md +78 -0
- package/.claude-plugin/skills/integration-test/references/external-api-mock.md +98 -0
- package/.claude-plugin/skills/integration-test/references/fixture-catalog.md +155 -0
- package/.claude-plugin/skills/integration-test/references/parallel-execution.md +124 -0
- package/.claude-plugin/skills/integration-test/references/quality-gate.md +80 -0
- package/.claude-plugin/skills/integration-test/references/test-case-design.md +88 -0
- package/.claude-plugin/skills/integration-test/references/test-patterns.md +215 -0
- package/.claude-plugin/skills/integration-test/references/whiteboard-template.md +81 -0
- package/.claude-plugin/skills/integration-test/references/worker-prompt.md +70 -0
- package/.claude-plugin/skills/knowhow-capture/SKILL.md +143 -0
- package/.claude-plugin/skills/phase-review-team/SKILL.md +380 -0
- package/.claude-plugin/skills/spec-design/SKILL.md +282 -0
- package/.claude-plugin/skills/spec-e2e-implement/SKILL.md +259 -0
- package/.claude-plugin/skills/spec-impl-code/SKILL.md +101 -0
- package/.claude-plugin/skills/spec-impl-review/SKILL.md +115 -0
- package/.claude-plugin/skills/spec-impl-test-run/SKILL.md +98 -0
- package/.claude-plugin/skills/spec-impl-test-write/SKILL.md +121 -0
- package/.claude-plugin/skills/spec-implement/SKILL.md +822 -0
- package/.claude-plugin/skills/spec-requirements/SKILL.md +130 -0
- package/.claude-plugin/skills/spec-review/SKILL.md +274 -0
- package/.claude-plugin/skills/spec-tasks/SKILL.md +372 -0
- package/.claude-plugin/skills/spec-test-design/SKILL.md +233 -0
- package/.claude-plugin/skills/tdd-skills/SKILL.md +95 -0
- package/.claude-plugin/skills/tdd-skills/references/advanced-techniques.md +49 -0
- package/.claude-plugin/skills/tdd-skills/references/green-strategies.md +70 -0
- package/.claude-plugin/skills/tdd-skills/references/tdd-and-design.md +48 -0
- package/.claude-plugin/skills/tdd-skills/references/test-design.md +43 -0
- package/.claude-plugin/skills/tdd-skills/references/test-doubles.md +53 -0
- package/.claude-plugin/skills/tdd-skills/references/test-patterns.md +40 -0
- package/.claude-plugin/skills/tdd-skills-rust/SKILL.md +128 -0
- package/.claude-plugin/skills/tdd-skills-rust/references/advanced-techniques.md +205 -0
- package/.claude-plugin/skills/tdd-skills-rust/references/green-strategies.md +166 -0
- package/.claude-plugin/skills/tdd-skills-rust/references/tdd-and-design.md +215 -0
- package/.claude-plugin/skills/tdd-skills-rust/references/test-design.md +128 -0
- package/.claude-plugin/skills/tdd-skills-rust/references/test-doubles.md +208 -0
- package/.claude-plugin/skills/tdd-skills-rust/references/test-patterns.md +223 -0
- package/.claude-plugin/with-dashboard/.mcp.json +8 -0
- package/.claude-plugin/with-dashboard/plugin.json +10 -0
- package/CHANGELOG.md +1007 -0
- package/LICENSE +674 -0
- package/README.ja.md +380 -0
- package/README.md +437 -0
- package/dist/__tests__/config.test.d.ts +2 -0
- package/dist/__tests__/config.test.d.ts.map +1 -0
- package/dist/__tests__/config.test.js +264 -0
- package/dist/__tests__/config.test.js.map +1 -0
- package/dist/__tests__/index-args.test.d.ts +2 -0
- package/dist/__tests__/index-args.test.d.ts.map +1 -0
- package/dist/__tests__/index-args.test.js +43 -0
- package/dist/__tests__/index-args.test.js.map +1 -0
- package/dist/__tests__/index-entrypoint.test.d.ts +2 -0
- package/dist/__tests__/index-entrypoint.test.d.ts.map +1 -0
- package/dist/__tests__/index-entrypoint.test.js +23 -0
- package/dist/__tests__/index-entrypoint.test.js.map +1 -0
- package/dist/config.d.ts +26 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +188 -0
- package/dist/config.js.map +1 -0
- package/dist/core/__tests__/git-utils.test.d.ts +2 -0
- package/dist/core/__tests__/git-utils.test.d.ts.map +1 -0
- package/dist/core/__tests__/git-utils.test.js +179 -0
- package/dist/core/__tests__/git-utils.test.js.map +1 -0
- package/dist/core/__tests__/mdx-validator.test.d.ts +2 -0
- package/dist/core/__tests__/mdx-validator.test.d.ts.map +1 -0
- package/dist/core/__tests__/mdx-validator.test.js +42 -0
- package/dist/core/__tests__/mdx-validator.test.js.map +1 -0
- package/dist/core/__tests__/path-utils.test.d.ts +2 -0
- package/dist/core/__tests__/path-utils.test.d.ts.map +1 -0
- package/dist/core/__tests__/path-utils.test.js +342 -0
- package/dist/core/__tests__/path-utils.test.js.map +1 -0
- package/dist/core/__tests__/project-registry.test.d.ts +2 -0
- package/dist/core/__tests__/project-registry.test.d.ts.map +1 -0
- package/dist/core/__tests__/project-registry.test.js +62 -0
- package/dist/core/__tests__/project-registry.test.js.map +1 -0
- package/dist/core/__tests__/security-utils.test.d.ts +2 -0
- package/dist/core/__tests__/security-utils.test.d.ts.map +1 -0
- package/dist/core/__tests__/security-utils.test.js +657 -0
- package/dist/core/__tests__/security-utils.test.js.map +1 -0
- package/dist/core/__tests__/task-parser.test.d.ts +2 -0
- package/dist/core/__tests__/task-parser.test.d.ts.map +1 -0
- package/dist/core/__tests__/task-parser.test.js +222 -0
- package/dist/core/__tests__/task-parser.test.js.map +1 -0
- package/dist/core/__tests__/task-validator.test.d.ts +2 -0
- package/dist/core/__tests__/task-validator.test.d.ts.map +1 -0
- package/dist/core/__tests__/task-validator.test.js +308 -0
- package/dist/core/__tests__/task-validator.test.js.map +1 -0
- package/dist/core/archive-service.d.ts +10 -0
- package/dist/core/archive-service.d.ts.map +1 -0
- package/dist/core/archive-service.js +99 -0
- package/dist/core/archive-service.js.map +1 -0
- package/dist/core/dashboard-session.d.ts +49 -0
- package/dist/core/dashboard-session.d.ts.map +1 -0
- package/dist/core/dashboard-session.js +132 -0
- package/dist/core/dashboard-session.js.map +1 -0
- package/dist/core/git-utils.d.ts +25 -0
- package/dist/core/git-utils.d.ts.map +1 -0
- package/dist/core/git-utils.js +87 -0
- package/dist/core/git-utils.js.map +1 -0
- package/dist/core/global-dir.d.ts +44 -0
- package/dist/core/global-dir.d.ts.map +1 -0
- package/dist/core/global-dir.js +74 -0
- package/dist/core/global-dir.js.map +1 -0
- package/dist/core/implementation-log-migrator.d.ts +41 -0
- package/dist/core/implementation-log-migrator.d.ts.map +1 -0
- package/dist/core/implementation-log-migrator.js +258 -0
- package/dist/core/implementation-log-migrator.js.map +1 -0
- package/dist/core/mdx-validator.d.ts +14 -0
- package/dist/core/mdx-validator.d.ts.map +1 -0
- package/dist/core/mdx-validator.js +34 -0
- package/dist/core/mdx-validator.js.map +1 -0
- package/dist/core/parser.d.ts +11 -0
- package/dist/core/parser.d.ts.map +1 -0
- package/dist/core/parser.js +128 -0
- package/dist/core/parser.js.map +1 -0
- package/dist/core/path-utils.d.ts +68 -0
- package/dist/core/path-utils.d.ts.map +1 -0
- package/dist/core/path-utils.js +302 -0
- package/dist/core/path-utils.js.map +1 -0
- package/dist/core/project-registry.d.ts +94 -0
- package/dist/core/project-registry.d.ts.map +1 -0
- package/dist/core/project-registry.js +297 -0
- package/dist/core/project-registry.js.map +1 -0
- package/dist/core/security-utils.d.ts +99 -0
- package/dist/core/security-utils.d.ts.map +1 -0
- package/dist/core/security-utils.js +275 -0
- package/dist/core/security-utils.js.map +1 -0
- package/dist/core/task-parser.d.ts +90 -0
- package/dist/core/task-parser.d.ts.map +1 -0
- package/dist/core/task-parser.js +477 -0
- package/dist/core/task-parser.js.map +1 -0
- package/dist/core/task-validator.d.ts +37 -0
- package/dist/core/task-validator.d.ts.map +1 -0
- package/dist/core/task-validator.js +499 -0
- package/dist/core/task-validator.js.map +1 -0
- package/dist/core/workspace-initializer.d.ts +16 -0
- package/dist/core/workspace-initializer.d.ts.map +1 -0
- package/dist/core/workspace-initializer.js +168 -0
- package/dist/core/workspace-initializer.js.map +1 -0
- package/dist/dashboard/__tests__/approval-storage-path-resolution.test.d.ts +2 -0
- package/dist/dashboard/__tests__/approval-storage-path-resolution.test.d.ts.map +1 -0
- package/dist/dashboard/__tests__/approval-storage-path-resolution.test.js +78 -0
- package/dist/dashboard/__tests__/approval-storage-path-resolution.test.js.map +1 -0
- package/dist/dashboard/__tests__/multi-server-approvals-content.test.d.ts +2 -0
- package/dist/dashboard/__tests__/multi-server-approvals-content.test.d.ts.map +1 -0
- package/dist/dashboard/__tests__/multi-server-approvals-content.test.js +115 -0
- package/dist/dashboard/__tests__/multi-server-approvals-content.test.js.map +1 -0
- package/dist/dashboard/__tests__/watcher-error-handling.test.d.ts +2 -0
- package/dist/dashboard/__tests__/watcher-error-handling.test.d.ts.map +1 -0
- package/dist/dashboard/__tests__/watcher-error-handling.test.js +118 -0
- package/dist/dashboard/__tests__/watcher-error-handling.test.js.map +1 -0
- package/dist/dashboard/approval-storage.d.ts +139 -0
- package/dist/dashboard/approval-storage.d.ts.map +1 -0
- package/dist/dashboard/approval-storage.js +608 -0
- package/dist/dashboard/approval-storage.js.map +1 -0
- package/dist/dashboard/execution-history-manager.d.ts +52 -0
- package/dist/dashboard/execution-history-manager.d.ts.map +1 -0
- package/dist/dashboard/execution-history-manager.js +161 -0
- package/dist/dashboard/execution-history-manager.js.map +1 -0
- package/dist/dashboard/implementation-log-manager.d.ts +97 -0
- package/dist/dashboard/implementation-log-manager.d.ts.map +1 -0
- package/dist/dashboard/implementation-log-manager.js +617 -0
- package/dist/dashboard/implementation-log-manager.js.map +1 -0
- package/dist/dashboard/job-scheduler.d.ts +91 -0
- package/dist/dashboard/job-scheduler.d.ts.map +1 -0
- package/dist/dashboard/job-scheduler.js +321 -0
- package/dist/dashboard/job-scheduler.js.map +1 -0
- package/dist/dashboard/multi-server.d.ts +42 -0
- package/dist/dashboard/multi-server.d.ts.map +1 -0
- package/dist/dashboard/multi-server.js +1460 -0
- package/dist/dashboard/multi-server.js.map +1 -0
- package/dist/dashboard/parser.d.ts +18 -0
- package/dist/dashboard/parser.d.ts.map +1 -0
- package/dist/dashboard/parser.js +269 -0
- package/dist/dashboard/parser.js.map +1 -0
- package/dist/dashboard/project-manager.d.ts +82 -0
- package/dist/dashboard/project-manager.d.ts.map +1 -0
- package/dist/dashboard/project-manager.js +257 -0
- package/dist/dashboard/project-manager.js.map +1 -0
- package/dist/dashboard/public/assets/Inter-Bold-CD3Pr7BX.woff2 +0 -0
- package/dist/dashboard/public/assets/Inter-Medium-B_8v_WHh.woff2 +0 -0
- package/dist/dashboard/public/assets/Inter-Regular-DRVdRqcI.woff2 +0 -0
- package/dist/dashboard/public/assets/Inter-SemiBold-CtskMddL.woff2 +0 -0
- package/dist/dashboard/public/assets/JetBrainsMono-Bold-D4WEaHbo.woff2 +0 -0
- package/dist/dashboard/public/assets/JetBrainsMono-Medium-3S3k2nMz.woff2 +0 -0
- package/dist/dashboard/public/assets/JetBrainsMono-Regular-BQaDgvhP.woff2 +0 -0
- package/dist/dashboard/public/assets/Tableau10-B-NsZVaP.js +1 -0
- package/dist/dashboard/public/assets/apl-B4CMkyY2.js +1 -0
- package/dist/dashboard/public/assets/arc-a5wW942W.js +1 -0
- package/dist/dashboard/public/assets/array-BKyUJesY.js +1 -0
- package/dist/dashboard/public/assets/asciiarmor-Df11BRmG.js +1 -0
- package/dist/dashboard/public/assets/asn1-EdZsLKOL.js +1 -0
- package/dist/dashboard/public/assets/asterisk-B-8jnY81.js +1 -0
- package/dist/dashboard/public/assets/blockDiagram-c4efeb88-CvjTuK-w.js +118 -0
- package/dist/dashboard/public/assets/brainfuck-C4LP7Hcl.js +1 -0
- package/dist/dashboard/public/assets/c4Diagram-c83219d4-NwVQo5kf.js +10 -0
- package/dist/dashboard/public/assets/channel-Bi16YZhk.js +1 -0
- package/dist/dashboard/public/assets/classDiagram-beda092f-BmSeXDdU.js +2 -0
- package/dist/dashboard/public/assets/classDiagram-v2-2358418a-D7GvvuPr.js +2 -0
- package/dist/dashboard/public/assets/clike-B9uivgTg.js +1 -0
- package/dist/dashboard/public/assets/clojure-BMjYHr_A.js +1 -0
- package/dist/dashboard/public/assets/clone-BpKTiq7P.js +1 -0
- package/dist/dashboard/public/assets/cmake-BQqOBYOt.js +1 -0
- package/dist/dashboard/public/assets/cobol-CWcv1MsR.js +1 -0
- package/dist/dashboard/public/assets/coffeescript-S37ZYGWr.js +1 -0
- package/dist/dashboard/public/assets/commonlisp-DBKNyK5s.js +1 -0
- package/dist/dashboard/public/assets/createText-1719965b-qASbqHUP.js +7 -0
- package/dist/dashboard/public/assets/crystal-SjHAIU92.js +1 -0
- package/dist/dashboard/public/assets/css-BnMrqG3P.js +1 -0
- package/dist/dashboard/public/assets/cypher-C_CwsFkJ.js +1 -0
- package/dist/dashboard/public/assets/d-pRatUO7H.js +1 -0
- package/dist/dashboard/public/assets/diff-DbItnlRl.js +1 -0
- package/dist/dashboard/public/assets/dockerfile-BKs6k2Af.js +1 -0
- package/dist/dashboard/public/assets/dtd-DF_7sFjM.js +1 -0
- package/dist/dashboard/public/assets/dylan-DwRh75JA.js +1 -0
- package/dist/dashboard/public/assets/ebnf-CDyGwa7X.js +1 -0
- package/dist/dashboard/public/assets/ecl-Cabwm37j.js +1 -0
- package/dist/dashboard/public/assets/edges-96097737-BItTSnH7.js +4 -0
- package/dist/dashboard/public/assets/eiffel-CnydiIhH.js +1 -0
- package/dist/dashboard/public/assets/elm-vLlmbW-K.js +1 -0
- package/dist/dashboard/public/assets/erDiagram-0228fc6a-DT224olg.js +51 -0
- package/dist/dashboard/public/assets/erlang-BNw1qcRV.js +1 -0
- package/dist/dashboard/public/assets/factor-kuTfRLto.js +1 -0
- package/dist/dashboard/public/assets/fcl-Kvtd6kyn.js +1 -0
- package/dist/dashboard/public/assets/flowDb-c6c81e3f-D9_ukKtv.js +10 -0
- package/dist/dashboard/public/assets/flowDiagram-50d868cf-CylE8siG.js +4 -0
- package/dist/dashboard/public/assets/flowDiagram-v2-4f6560a1-B2O3JN7Y.js +1 -0
- package/dist/dashboard/public/assets/flowchart-elk-definition-6af322e1-BCaqFKf3.js +139 -0
- package/dist/dashboard/public/assets/forth-Ffai-XNe.js +1 -0
- package/dist/dashboard/public/assets/fortran-DYz_wnZ1.js +1 -0
- package/dist/dashboard/public/assets/ganttDiagram-a2739b55-WQUL1QW_.js +257 -0
- package/dist/dashboard/public/assets/gas-Bneqetm1.js +1 -0
- package/dist/dashboard/public/assets/gherkin-heZmZLOM.js +1 -0
- package/dist/dashboard/public/assets/gitGraphDiagram-82fe8481-CttZrdmr.js +70 -0
- package/dist/dashboard/public/assets/graph-Ch-rVueN.js +1 -0
- package/dist/dashboard/public/assets/groovy-D9Dt4D0W.js +1 -0
- package/dist/dashboard/public/assets/haskell-Cw1EW3IL.js +1 -0
- package/dist/dashboard/public/assets/haxe-H-WmDvRZ.js +1 -0
- package/dist/dashboard/public/assets/http-DBlCnlav.js +1 -0
- package/dist/dashboard/public/assets/idl-BEugSyMb.js +1 -0
- package/dist/dashboard/public/assets/index--kbPpDKv.js +1 -0
- package/dist/dashboard/public/assets/index-3scDwWm6.js +1 -0
- package/dist/dashboard/public/assets/index-5325376f-BL2zVOJU.js +1 -0
- package/dist/dashboard/public/assets/index-BZdjbO25.js +1 -0
- package/dist/dashboard/public/assets/index-BmA_batZ.js +1 -0
- package/dist/dashboard/public/assets/index-Bu0u99kF.js +2 -0
- package/dist/dashboard/public/assets/index-Ch-lr7F4.js +1 -0
- package/dist/dashboard/public/assets/index-ClgWbdoq.js +1 -0
- package/dist/dashboard/public/assets/index-CzLwOMQ_.js +3 -0
- package/dist/dashboard/public/assets/index-DAOEjGO7.js +1 -0
- package/dist/dashboard/public/assets/index-DXqf0B9c.js +1 -0
- package/dist/dashboard/public/assets/index-DegWdR16.js +1 -0
- package/dist/dashboard/public/assets/index-DiHyYGim.js +1 -0
- package/dist/dashboard/public/assets/index-DlZtG7I5.js +1 -0
- package/dist/dashboard/public/assets/index-DmhGE2M8.js +1 -0
- package/dist/dashboard/public/assets/index-QEGvld4x.js +1 -0
- package/dist/dashboard/public/assets/index-RfZPGAJu.js +1 -0
- package/dist/dashboard/public/assets/index-UybBj_7u.js +319 -0
- package/dist/dashboard/public/assets/index-bVekzPnl.js +7 -0
- package/dist/dashboard/public/assets/index-f5bysQzW.css +1 -0
- package/dist/dashboard/public/assets/infoDiagram-8eee0895-DjzkkE3o.js +7 -0
- package/dist/dashboard/public/assets/init-Gi6I4Gst.js +1 -0
- package/dist/dashboard/public/assets/javascript-iXu5QeM3.js +1 -0
- package/dist/dashboard/public/assets/journeyDiagram-c64418c1-CxPZkNdB.js +139 -0
- package/dist/dashboard/public/assets/julia-DuME0IfC.js +1 -0
- package/dist/dashboard/public/assets/katex-XbL3y5x-.js +261 -0
- package/dist/dashboard/public/assets/layout-DX7DNTRm.js +1 -0
- package/dist/dashboard/public/assets/line-DfvpmKOn.js +1 -0
- package/dist/dashboard/public/assets/linear-gQbBPHO5.js +1 -0
- package/dist/dashboard/public/assets/livescript-BwQOo05w.js +1 -0
- package/dist/dashboard/public/assets/lua-BgMRiT3U.js +1 -0
- package/dist/dashboard/public/assets/mathematica-DTrFuWx2.js +1 -0
- package/dist/dashboard/public/assets/mbox-CNhZ1qSd.js +1 -0
- package/dist/dashboard/public/assets/mindmap-definition-8da855dc-CNxmpyG6.js +415 -0
- package/dist/dashboard/public/assets/mirc-CjQqDB4T.js +1 -0
- package/dist/dashboard/public/assets/mllike-CXdrOF99.js +1 -0
- package/dist/dashboard/public/assets/modelica-Dc1JOy9r.js +1 -0
- package/dist/dashboard/public/assets/mscgen-BA5vi2Kp.js +1 -0
- package/dist/dashboard/public/assets/mumps-BT43cFF4.js +1 -0
- package/dist/dashboard/public/assets/nginx-DdIZxoE0.js +1 -0
- package/dist/dashboard/public/assets/nsis-LdVXkNf5.js +1 -0
- package/dist/dashboard/public/assets/ntriples-BfvgReVJ.js +1 -0
- package/dist/dashboard/public/assets/octave-Ck1zUtKM.js +1 -0
- package/dist/dashboard/public/assets/ordinal-Cboi1Yqb.js +1 -0
- package/dist/dashboard/public/assets/oz-BzwKVEFT.js +1 -0
- package/dist/dashboard/public/assets/pascal--L3eBynH.js +1 -0
- package/dist/dashboard/public/assets/path-CbwjOpE9.js +1 -0
- package/dist/dashboard/public/assets/perl-CdXCOZ3F.js +1 -0
- package/dist/dashboard/public/assets/pieDiagram-a8764435-D-xy_NSA.js +35 -0
- package/dist/dashboard/public/assets/pig-CevX1Tat.js +1 -0
- package/dist/dashboard/public/assets/powershell-CFHJl5sT.js +1 -0
- package/dist/dashboard/public/assets/properties-C78fOPTZ.js +1 -0
- package/dist/dashboard/public/assets/protobuf-ChK-085T.js +1 -0
- package/dist/dashboard/public/assets/pug-DeIclll2.js +1 -0
- package/dist/dashboard/public/assets/puppet-DMA9R1ak.js +1 -0
- package/dist/dashboard/public/assets/python-BuPzkPfP.js +1 -0
- package/dist/dashboard/public/assets/q-pXgVlZs6.js +1 -0
- package/dist/dashboard/public/assets/quadrantDiagram-1e28029f-BoL2wzz0.js +7 -0
- package/dist/dashboard/public/assets/r-B6wPVr8A.js +1 -0
- package/dist/dashboard/public/assets/requirementDiagram-08caed73-BujFz0q1.js +52 -0
- package/dist/dashboard/public/assets/rpm-CTu-6PCP.js +1 -0
- package/dist/dashboard/public/assets/ruby-B2Rjki9n.js +1 -0
- package/dist/dashboard/public/assets/sankeyDiagram-a04cb91d-D03_NARm.js +8 -0
- package/dist/dashboard/public/assets/sas-B4kiWyti.js +1 -0
- package/dist/dashboard/public/assets/scheme-C41bIUwD.js +1 -0
- package/dist/dashboard/public/assets/sequenceDiagram-c5b8d532-B65eFcaT.js +122 -0
- package/dist/dashboard/public/assets/shell-CjFT_Tl9.js +1 -0
- package/dist/dashboard/public/assets/sieve-C3Gn_uJK.js +1 -0
- package/dist/dashboard/public/assets/simple-mode-GW_nhZxv.js +1 -0
- package/dist/dashboard/public/assets/smalltalk-CnHTOXQT.js +1 -0
- package/dist/dashboard/public/assets/solr-DehyRSwq.js +1 -0
- package/dist/dashboard/public/assets/sparql-DkYu6x3z.js +1 -0
- package/dist/dashboard/public/assets/spreadsheet-BCZA_wO0.js +1 -0
- package/dist/dashboard/public/assets/sql-D0XecflT.js +1 -0
- package/dist/dashboard/public/assets/stateDiagram-1ecb1508-BDbqu0Vl.js +1 -0
- package/dist/dashboard/public/assets/stateDiagram-v2-c2b004d7-CBHvk4b8.js +1 -0
- package/dist/dashboard/public/assets/stex-C3f8Ysf7.js +1 -0
- package/dist/dashboard/public/assets/styles-b4e223ce-CELsPqaO.js +160 -0
- package/dist/dashboard/public/assets/styles-ca3715f6-BRqMqT6F.js +207 -0
- package/dist/dashboard/public/assets/styles-d45a18b0-e8N-oLPy.js +116 -0
- package/dist/dashboard/public/assets/stylus-B533Al4x.js +1 -0
- package/dist/dashboard/public/assets/svgDrawCommon-b86b1483-vNDtmQc-.js +1 -0
- package/dist/dashboard/public/assets/swift-BzpIVaGY.js +1 -0
- package/dist/dashboard/public/assets/tcl-DVfN8rqt.js +1 -0
- package/dist/dashboard/public/assets/textile-CnDTJFAw.js +1 -0
- package/dist/dashboard/public/assets/tiddlywiki-DO-Gjzrf.js +1 -0
- package/dist/dashboard/public/assets/tiki-DGYXhP31.js +1 -0
- package/dist/dashboard/public/assets/timeline-definition-faaaa080-Dh2_A5VU.js +61 -0
- package/dist/dashboard/public/assets/toml-Bm5Em-hy.js +1 -0
- package/dist/dashboard/public/assets/troff-wAsdV37c.js +1 -0
- package/dist/dashboard/public/assets/ttcn-CfJYG6tj.js +1 -0
- package/dist/dashboard/public/assets/ttcn-cfg-B9xdYoR4.js +1 -0
- package/dist/dashboard/public/assets/turtle-B1tBg_DP.js +1 -0
- package/dist/dashboard/public/assets/vb-CmGdzxic.js +1 -0
- package/dist/dashboard/public/assets/vbscript-BuJXcnF6.js +1 -0
- package/dist/dashboard/public/assets/velocity-D8B20fx6.js +1 -0
- package/dist/dashboard/public/assets/verilog-C6RDOZhf.js +1 -0
- package/dist/dashboard/public/assets/vhdl-lSbBsy5d.js +1 -0
- package/dist/dashboard/public/assets/webidl-ZXfAyPTL.js +1 -0
- package/dist/dashboard/public/assets/xquery-DzFWVndE.js +1 -0
- package/dist/dashboard/public/assets/xychartDiagram-f5964ef8-B76v1AVF.js +7 -0
- package/dist/dashboard/public/assets/yacas-BJ4BC0dw.js +1 -0
- package/dist/dashboard/public/assets/z80-Hz9HOZM7.js +1 -0
- package/dist/dashboard/public/claude-icon-dark.svg +1 -0
- package/dist/dashboard/public/claude-icon.svg +1 -0
- package/dist/dashboard/public/index.html +16 -0
- package/dist/dashboard/settings-manager.d.ts +47 -0
- package/dist/dashboard/settings-manager.d.ts.map +1 -0
- package/dist/dashboard/settings-manager.js +180 -0
- package/dist/dashboard/settings-manager.js.map +1 -0
- package/dist/dashboard/utils.d.ts +31 -0
- package/dist/dashboard/utils.d.ts.map +1 -0
- package/dist/dashboard/utils.js +102 -0
- package/dist/dashboard/utils.js.map +1 -0
- package/dist/dashboard/watcher.d.ts +32 -0
- package/dist/dashboard/watcher.d.ts.map +1 -0
- package/dist/dashboard/watcher.js +173 -0
- package/dist/dashboard/watcher.js.map +1 -0
- package/dist/index.d.ts +13 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +380 -0
- package/dist/index.js.map +1 -0
- package/dist/markdown/templates/design-template.md +126 -0
- package/dist/markdown/templates/product-template.md +51 -0
- package/dist/markdown/templates/requirements-template.md +50 -0
- package/dist/markdown/templates/structure-template.md +145 -0
- package/dist/markdown/templates/tasks-template.md +100 -0
- package/dist/markdown/templates/tech-template.md +99 -0
- package/dist/markdown/templates/test-design-template.md +221 -0
- package/dist/prompts/create-spec.d.ts +3 -0
- package/dist/prompts/create-spec.d.ts.map +1 -0
- package/dist/prompts/create-spec.js +97 -0
- package/dist/prompts/create-spec.js.map +1 -0
- package/dist/prompts/create-steering-doc.d.ts +3 -0
- package/dist/prompts/create-steering-doc.d.ts.map +1 -0
- package/dist/prompts/create-steering-doc.js +75 -0
- package/dist/prompts/create-steering-doc.js.map +1 -0
- package/dist/prompts/implement-task.d.ts +3 -0
- package/dist/prompts/implement-task.d.ts.map +1 -0
- package/dist/prompts/implement-task.js +174 -0
- package/dist/prompts/implement-task.js.map +1 -0
- package/dist/prompts/index.d.ts +20 -0
- package/dist/prompts/index.d.ts.map +1 -0
- package/dist/prompts/index.js +103 -0
- package/dist/prompts/index.js.map +1 -0
- package/dist/prompts/inject-spec-workflow-guide.d.ts +3 -0
- package/dist/prompts/inject-spec-workflow-guide.d.ts.map +1 -0
- package/dist/prompts/inject-spec-workflow-guide.js +60 -0
- package/dist/prompts/inject-spec-workflow-guide.js.map +1 -0
- package/dist/prompts/inject-steering-guide.d.ts +3 -0
- package/dist/prompts/inject-steering-guide.d.ts.map +1 -0
- package/dist/prompts/inject-steering-guide.js +64 -0
- package/dist/prompts/inject-steering-guide.js.map +1 -0
- package/dist/prompts/refresh-tasks.d.ts +3 -0
- package/dist/prompts/refresh-tasks.d.ts.map +1 -0
- package/dist/prompts/refresh-tasks.js +237 -0
- package/dist/prompts/refresh-tasks.js.map +1 -0
- package/dist/prompts/spec-status.d.ts +3 -0
- package/dist/prompts/spec-status.d.ts.map +1 -0
- package/dist/prompts/spec-status.js +77 -0
- package/dist/prompts/spec-status.js.map +1 -0
- package/dist/prompts/types.d.ts +13 -0
- package/dist/prompts/types.d.ts.map +1 -0
- package/dist/prompts/types.js +2 -0
- package/dist/prompts/types.js.map +1 -0
- package/dist/server.d.ts +17 -0
- package/dist/server.d.ts.map +1 -0
- package/dist/server.js +175 -0
- package/dist/server.js.map +1 -0
- package/dist/tools/__tests__/log-implementation-review-process.test.d.ts +2 -0
- package/dist/tools/__tests__/log-implementation-review-process.test.d.ts.map +1 -0
- package/dist/tools/__tests__/log-implementation-review-process.test.js +190 -0
- package/dist/tools/__tests__/log-implementation-review-process.test.js.map +1 -0
- package/dist/tools/__tests__/projectPath.test.d.ts +2 -0
- package/dist/tools/__tests__/projectPath.test.d.ts.map +1 -0
- package/dist/tools/__tests__/projectPath.test.js +187 -0
- package/dist/tools/__tests__/projectPath.test.js.map +1 -0
- package/dist/tools/approvals.d.ts +14 -0
- package/dist/tools/approvals.d.ts.map +1 -0
- package/dist/tools/approvals.js +505 -0
- package/dist/tools/approvals.js.map +1 -0
- package/dist/tools/index.d.ts +5 -0
- package/dist/tools/index.d.ts.map +1 -0
- package/dist/tools/index.js +52 -0
- package/dist/tools/index.js.map +1 -0
- package/dist/tools/log-implementation.d.ts +5 -0
- package/dist/tools/log-implementation.d.ts.map +1 -0
- package/dist/tools/log-implementation.js +498 -0
- package/dist/tools/log-implementation.js.map +1 -0
- package/dist/tools/spec-status.d.ts +5 -0
- package/dist/tools/spec-status.d.ts.map +1 -0
- package/dist/tools/spec-status.js +192 -0
- package/dist/tools/spec-status.js.map +1 -0
- package/dist/tools/spec-workflow-guide.d.ts +5 -0
- package/dist/tools/spec-workflow-guide.d.ts.map +1 -0
- package/dist/tools/spec-workflow-guide.js +116 -0
- package/dist/tools/spec-workflow-guide.js.map +1 -0
- package/dist/tools/steering-guide.d.ts +5 -0
- package/dist/tools/steering-guide.d.ts.map +1 -0
- package/dist/tools/steering-guide.js +192 -0
- package/dist/tools/steering-guide.js.map +1 -0
- package/dist/types.d.ts +183 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +13 -0
- package/dist/types.js.map +1 -0
- package/package.json +106 -0
|
@@ -0,0 +1,205 @@
|
|
|
1
|
+
# TDD 実践テクニックとアンチパターン
|
|
2
|
+
|
|
3
|
+
## 実践テクニック
|
|
4
|
+
|
|
5
|
+
### 仕様の不確実性への対処
|
|
6
|
+
|
|
7
|
+
問題: 仕様が曖昧で、どうテストを書けばいいか分からない。
|
|
8
|
+
|
|
9
|
+
対策: 具体例から始める。
|
|
10
|
+
|
|
11
|
+
```rust
|
|
12
|
+
// 悪い: 抽象的すぎる
|
|
13
|
+
#[test]
|
|
14
|
+
fn calculate_price() {
|
|
15
|
+
// 何をテストする?
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
// 良い: 具体的なユースケース
|
|
19
|
+
#[test]
|
|
20
|
+
fn calculate_price_for_single_item_without_discount() {
|
|
21
|
+
let calculator = PriceCalculator::new();
|
|
22
|
+
let items = vec![Item { price: 1000 }];
|
|
23
|
+
|
|
24
|
+
let total = calculator.calculate(&items);
|
|
25
|
+
|
|
26
|
+
assert_eq!(total, 1000);
|
|
27
|
+
}
|
|
28
|
+
```
|
|
29
|
+
|
|
30
|
+
### レガシーコードへの TDD 適用
|
|
31
|
+
|
|
32
|
+
1. 既存の振る舞いを保護するテストを書く(特性テスト)
|
|
33
|
+
2. 小さくリファクタリング
|
|
34
|
+
3. 徐々にテストカバレッジを上げる
|
|
35
|
+
|
|
36
|
+
```rust
|
|
37
|
+
// Step 1: 既存の振る舞いを記録
|
|
38
|
+
#[test]
|
|
39
|
+
fn existing_behavior() {
|
|
40
|
+
let result = legacy_function(&input_data);
|
|
41
|
+
assert_eq!(result, expected_output);
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
// Step 2: trait で依存を抽象化してテスト可能にする
|
|
45
|
+
// Step 3: 新機能は TDD で
|
|
46
|
+
```
|
|
47
|
+
|
|
48
|
+
### テストが複雑になってきたら
|
|
49
|
+
|
|
50
|
+
対策:
|
|
51
|
+
1. テストヘルパー関数を作る
|
|
52
|
+
2. Builder パターンを使う
|
|
53
|
+
3. テストを分割する
|
|
54
|
+
|
|
55
|
+
```rust
|
|
56
|
+
#[cfg(test)]
|
|
57
|
+
mod tests {
|
|
58
|
+
use super::*;
|
|
59
|
+
|
|
60
|
+
fn create_premium_user() -> User {
|
|
61
|
+
User { id: 1, name: "Alice".into(), is_premium: true }
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
fn create_cart_with_items(user: &User) -> ShoppingCart {
|
|
65
|
+
let mut cart = ShoppingCart::new(user.id);
|
|
66
|
+
cart.add_item(ItemBuilder::new().name("Book").price(1000).build());
|
|
67
|
+
cart.add_item(ItemBuilder::new().name("Pen").price(500).build());
|
|
68
|
+
cart
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
#[test]
|
|
72
|
+
fn premium_user_gets_discount() {
|
|
73
|
+
let user = create_premium_user();
|
|
74
|
+
let cart = create_cart_with_items(&user);
|
|
75
|
+
|
|
76
|
+
let total = cart.calculate_total(&user);
|
|
77
|
+
|
|
78
|
+
assert_eq!(total, 1350); // 10% discount
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
```
|
|
82
|
+
|
|
83
|
+
## TDD のアンチパターン
|
|
84
|
+
|
|
85
|
+
### 1. テストを書かずに実装
|
|
86
|
+
|
|
87
|
+
```rust
|
|
88
|
+
// 悪い: いきなり実装
|
|
89
|
+
fn calculate_total(items: &[Item]) -> u64 {
|
|
90
|
+
items.iter().map(|i| i.price).sum()
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
// 良い: まずテスト
|
|
94
|
+
#[test]
|
|
95
|
+
fn calculate_total_for_empty_list() {
|
|
96
|
+
assert_eq!(calculate_total(&[]), 0);
|
|
97
|
+
}
|
|
98
|
+
```
|
|
99
|
+
|
|
100
|
+
### 2. 大きすぎるステップ
|
|
101
|
+
|
|
102
|
+
```rust
|
|
103
|
+
// 悪い: いきなり完璧を目指す
|
|
104
|
+
#[test]
|
|
105
|
+
fn complete_order_system() {
|
|
106
|
+
// カート、決済、在庫管理、メール送信...全部
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
// 良い: 小さく分割
|
|
110
|
+
#[test]
|
|
111
|
+
fn create_empty_cart() {
|
|
112
|
+
let cart = ShoppingCart::new();
|
|
113
|
+
assert!(cart.is_empty());
|
|
114
|
+
}
|
|
115
|
+
```
|
|
116
|
+
|
|
117
|
+
### 3. テストのためのテスト
|
|
118
|
+
|
|
119
|
+
```rust
|
|
120
|
+
// 悪い: 自明すぎる(getter のテスト)
|
|
121
|
+
#[test]
|
|
122
|
+
fn getter() {
|
|
123
|
+
let user = User { name: "Alice".into() };
|
|
124
|
+
assert_eq!(user.name, "Alice");
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
// 良い: 振る舞いをテスト
|
|
128
|
+
#[test]
|
|
129
|
+
fn user_full_name() {
|
|
130
|
+
let user = User { first_name: "Alice".into(), last_name: "Smith".into() };
|
|
131
|
+
assert_eq!(user.full_name(), "Alice Smith");
|
|
132
|
+
}
|
|
133
|
+
```
|
|
134
|
+
|
|
135
|
+
### 4. プライベート関数のテスト
|
|
136
|
+
|
|
137
|
+
```rust
|
|
138
|
+
// 悪い: 内部関数を直接テスト
|
|
139
|
+
#[test]
|
|
140
|
+
fn internal_calculation() {
|
|
141
|
+
assert_eq!(internal_helper(5), 10); // pub(crate) にして無理やりテスト
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
// 良い: 公開 API を通してテスト
|
|
145
|
+
#[test]
|
|
146
|
+
fn public_method_that_uses_internal() {
|
|
147
|
+
let obj = MyStruct::new();
|
|
148
|
+
assert_eq!(obj.calculate(5), expected_result);
|
|
149
|
+
}
|
|
150
|
+
```
|
|
151
|
+
|
|
152
|
+
Rust ではモジュール内テスト (`#[cfg(test)] mod tests`) からプライベート関数にアクセスできるが、
|
|
153
|
+
公開 API を通してテストする方が設計上望ましい。
|
|
154
|
+
|
|
155
|
+
### 5. テストの相互依存
|
|
156
|
+
|
|
157
|
+
```rust
|
|
158
|
+
// 悪い: テストが順序に依存(static mut 等)
|
|
159
|
+
// 良い: 各テストが独立
|
|
160
|
+
#[test]
|
|
161
|
+
fn create_cart() {
|
|
162
|
+
let cart = ShoppingCart::new();
|
|
163
|
+
assert!(cart.is_empty());
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
#[test]
|
|
167
|
+
fn add_item() {
|
|
168
|
+
let mut cart = ShoppingCart::new(); // 毎回新規作成
|
|
169
|
+
cart.add_item(Item { price: 100 });
|
|
170
|
+
assert_eq!(cart.item_count(), 1);
|
|
171
|
+
}
|
|
172
|
+
```
|
|
173
|
+
|
|
174
|
+
## TODO リストの活用
|
|
175
|
+
|
|
176
|
+
実装中に思いついたアイデアを記録:
|
|
177
|
+
|
|
178
|
+
```markdown
|
|
179
|
+
## TODO
|
|
180
|
+
- [x] 空のカートの合計は0
|
|
181
|
+
- [x] 1つの商品を追加した場合の合計
|
|
182
|
+
- [ ] 複数の商品を追加した場合の合計
|
|
183
|
+
- [ ] 割引適用時の合計
|
|
184
|
+
- [ ] 負の価格の商品は追加できない(エラー)
|
|
185
|
+
- [ ] 在庫がない商品は追加できない
|
|
186
|
+
```
|
|
187
|
+
|
|
188
|
+
メリット:
|
|
189
|
+
- 今やるべきことに集中できる
|
|
190
|
+
- 進捗が可視化される
|
|
191
|
+
- 実装漏れを防げる
|
|
192
|
+
|
|
193
|
+
## まとめ
|
|
194
|
+
|
|
195
|
+
やるべきこと:
|
|
196
|
+
- 小さいステップで進める
|
|
197
|
+
- 具体例から始める
|
|
198
|
+
- TODO リストを活用
|
|
199
|
+
- テストヘルパーで整理
|
|
200
|
+
|
|
201
|
+
避けるべきこと:
|
|
202
|
+
- テストを飛ばす
|
|
203
|
+
- 大きすぎるステップ
|
|
204
|
+
- プライベート関数のテスト
|
|
205
|
+
- テストの相互依存
|
|
@@ -0,0 +1,166 @@
|
|
|
1
|
+
# Green の3つの戦略
|
|
2
|
+
|
|
3
|
+
テストを最速で通すための3つの戦略と使い分け。
|
|
4
|
+
|
|
5
|
+
## 1. 仮実装(Fake It)
|
|
6
|
+
|
|
7
|
+
最も安全な方法。まず定数を返してテストを通す。
|
|
8
|
+
|
|
9
|
+
```rust
|
|
10
|
+
#[test]
|
|
11
|
+
fn empty_cart_total_should_be_zero() {
|
|
12
|
+
let cart = ShoppingCart::new();
|
|
13
|
+
assert_eq!(cart.total(), 0);
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
// 仮実装
|
|
17
|
+
impl ShoppingCart {
|
|
18
|
+
fn total(&self) -> u64 {
|
|
19
|
+
0 // まずは定数で仮実装
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
```
|
|
23
|
+
|
|
24
|
+
### 使うべき場面
|
|
25
|
+
- 実装方法がまだ明確でない時
|
|
26
|
+
- 複雑な処理が必要な時
|
|
27
|
+
- TDD に慣れていない時
|
|
28
|
+
|
|
29
|
+
### メリット
|
|
30
|
+
- 最も安全で失敗しにくい
|
|
31
|
+
- 小さいステップで確実に進められる
|
|
32
|
+
- 思考を整理する時間ができる
|
|
33
|
+
|
|
34
|
+
## 2. 三角測量(Triangulation)
|
|
35
|
+
|
|
36
|
+
複数のテストケースから一般化を導く。
|
|
37
|
+
|
|
38
|
+
```rust
|
|
39
|
+
// Test 1: 空のカート
|
|
40
|
+
#[test]
|
|
41
|
+
fn empty_cart_total_should_be_zero() {
|
|
42
|
+
let cart = ShoppingCart::new();
|
|
43
|
+
assert_eq!(cart.total(), 0);
|
|
44
|
+
// 実装: return 0
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
// Test 2: 商品1つ追加(ここで仮実装から一般化)
|
|
48
|
+
#[test]
|
|
49
|
+
fn one_item_cart_total() {
|
|
50
|
+
let mut cart = ShoppingCart::new();
|
|
51
|
+
cart.add_item(Item { price: 100 });
|
|
52
|
+
assert_eq!(cart.total(), 100);
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
// 一般化した実装
|
|
56
|
+
impl ShoppingCart {
|
|
57
|
+
fn total(&self) -> u64 {
|
|
58
|
+
self.items.iter().map(|item| item.price).sum()
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
```
|
|
62
|
+
|
|
63
|
+
### 使うべき場面
|
|
64
|
+
- どう一般化すべきか不明確な時
|
|
65
|
+
- 複数のテストケースから共通パターンを見つけたい時
|
|
66
|
+
|
|
67
|
+
### プロセス
|
|
68
|
+
1. 最初のテストで仮実装(定数を返す)
|
|
69
|
+
2. 2つ目のテストを追加
|
|
70
|
+
3. 両方のテストが通るように一般化
|
|
71
|
+
4. 必要に応じて3つ目、4つ目...
|
|
72
|
+
|
|
73
|
+
## 3. 明白な実装(Obvious Implementation)
|
|
74
|
+
|
|
75
|
+
最も高速だが、慣れが必要。いきなり正解を実装。
|
|
76
|
+
|
|
77
|
+
```rust
|
|
78
|
+
#[test]
|
|
79
|
+
fn total_calculates_sum_of_item_prices() {
|
|
80
|
+
let mut cart = ShoppingCart::new();
|
|
81
|
+
cart.add_item(Item { price: 100 });
|
|
82
|
+
cart.add_item(Item { price: 200 });
|
|
83
|
+
assert_eq!(cart.total(), 300);
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
// 明白な実装(仮実装を経ずに直接実装)
|
|
87
|
+
impl ShoppingCart {
|
|
88
|
+
fn total(&self) -> u64 {
|
|
89
|
+
self.items.iter().map(|item| item.price).sum()
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
```
|
|
93
|
+
|
|
94
|
+
### 使うべき場面
|
|
95
|
+
- 実装方法が明白な時
|
|
96
|
+
- シンプルな処理の時
|
|
97
|
+
- TDD に慣れている時
|
|
98
|
+
|
|
99
|
+
### 重要な注意
|
|
100
|
+
明白だと思って実装したがテストが通らない場合は、躊躇なく仮実装に戻る。
|
|
101
|
+
|
|
102
|
+
## 戦略の使い分けフローチャート
|
|
103
|
+
|
|
104
|
+
```
|
|
105
|
+
実装は明白か?
|
|
106
|
+
├─ Yes → 明白な実装を試す
|
|
107
|
+
│ ├─ 成功 → 完了
|
|
108
|
+
│ └─ 失敗 → 仮実装に戻る
|
|
109
|
+
│
|
|
110
|
+
└─ No → 仮実装から開始
|
|
111
|
+
├─ テストが1つ → 仮実装のまま
|
|
112
|
+
└─ テストが複数 → 三角測量で一般化
|
|
113
|
+
```
|
|
114
|
+
|
|
115
|
+
## 実践例: フィボナッチ数列
|
|
116
|
+
|
|
117
|
+
### ステップ1: 仮実装
|
|
118
|
+
|
|
119
|
+
```rust
|
|
120
|
+
#[test]
|
|
121
|
+
fn fib_0() {
|
|
122
|
+
assert_eq!(fib(0), 0);
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
fn fib(_n: u64) -> u64 {
|
|
126
|
+
0 // 仮実装
|
|
127
|
+
}
|
|
128
|
+
```
|
|
129
|
+
|
|
130
|
+
### ステップ2: 三角測量
|
|
131
|
+
|
|
132
|
+
```rust
|
|
133
|
+
#[test]
|
|
134
|
+
fn fib_1() {
|
|
135
|
+
assert_eq!(fib(1), 1);
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
fn fib(n: u64) -> u64 {
|
|
139
|
+
if n == 0 { return 0; }
|
|
140
|
+
1 // まだ仮実装
|
|
141
|
+
}
|
|
142
|
+
```
|
|
143
|
+
|
|
144
|
+
### ステップ3: さらに三角測量
|
|
145
|
+
|
|
146
|
+
```rust
|
|
147
|
+
#[test]
|
|
148
|
+
fn fib_2() {
|
|
149
|
+
assert_eq!(fib(2), 1);
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
fn fib(n: u64) -> u64 {
|
|
153
|
+
if n <= 1 { return n; }
|
|
154
|
+
fib(n - 1) + fib(n - 2) // ここで一般化
|
|
155
|
+
}
|
|
156
|
+
```
|
|
157
|
+
|
|
158
|
+
## まとめ
|
|
159
|
+
|
|
160
|
+
| 戦略 | 速度 | 安全性 | 推奨レベル |
|
|
161
|
+
|------|------|--------|-----------|
|
|
162
|
+
| 仮実装 | 遅い | 高い | 初心者〜上級者 |
|
|
163
|
+
| 三角測量 | 中間 | 高い | 中級者〜上級者 |
|
|
164
|
+
| 明白な実装 | 速い | 低い | 上級者 |
|
|
165
|
+
|
|
166
|
+
原則: 迷ったら仮実装。慣れたら明白な実装。複雑なら三角測量。
|
|
@@ -0,0 +1,215 @@
|
|
|
1
|
+
# TDD と設計
|
|
2
|
+
|
|
3
|
+
## TDD は設計手法でもある
|
|
4
|
+
|
|
5
|
+
TDD を実践すると、自然に以下の設計原則が守られる:
|
|
6
|
+
|
|
7
|
+
1. **YAGNI(You Aren't Gonna Need It)**: 必要最小限の実装
|
|
8
|
+
2. **単一責任の原則**: テストしやすい構造体は責任が明確
|
|
9
|
+
3. **依存性逆転の原則**: テストダブルを使うと自然に trait 設計
|
|
10
|
+
4. **疎結合**: テストしやすいコードは結合度が低い
|
|
11
|
+
|
|
12
|
+
## テストしやすい設計
|
|
13
|
+
|
|
14
|
+
### テストしにくい設計
|
|
15
|
+
|
|
16
|
+
```rust
|
|
17
|
+
struct OrderService;
|
|
18
|
+
|
|
19
|
+
impl OrderService {
|
|
20
|
+
fn process_order(&self, order_id: i64) -> Result<(), AppError> {
|
|
21
|
+
// DB に直接アクセス
|
|
22
|
+
let mut conn = PgConnection::establish("postgres://...")?;
|
|
23
|
+
let order = orders::table.find(order_id).first(&mut conn)?;
|
|
24
|
+
|
|
25
|
+
// 外部 API を直接呼び出し
|
|
26
|
+
let client = reqwest::blocking::Client::new();
|
|
27
|
+
client.post("https://payment.api/charge").json(&order).send()?;
|
|
28
|
+
Ok(())
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
```
|
|
32
|
+
|
|
33
|
+
問題点:
|
|
34
|
+
- DB が必要(遅い)
|
|
35
|
+
- 外部 API が必要(不安定)
|
|
36
|
+
- テストが環境に依存
|
|
37
|
+
|
|
38
|
+
### テストしやすい設計
|
|
39
|
+
|
|
40
|
+
```rust
|
|
41
|
+
// trait で依存を抽象化
|
|
42
|
+
trait OrderRepository: Send + Sync {
|
|
43
|
+
fn find_by_id(&self, id: i64) -> Result<Order, DbError>;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
trait PaymentGateway: Send + Sync {
|
|
47
|
+
fn charge(&self, amount: u64) -> Result<PaymentResult, PaymentError>;
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
// 依存性注入
|
|
51
|
+
struct OrderService {
|
|
52
|
+
order_repo: Box<dyn OrderRepository>,
|
|
53
|
+
payment: Box<dyn PaymentGateway>,
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
impl OrderService {
|
|
57
|
+
fn new(
|
|
58
|
+
order_repo: Box<dyn OrderRepository>,
|
|
59
|
+
payment: Box<dyn PaymentGateway>,
|
|
60
|
+
) -> Self {
|
|
61
|
+
Self { order_repo, payment }
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
fn process_order(&self, order_id: i64) -> Result<(), AppError> {
|
|
65
|
+
let order = self.order_repo.find_by_id(order_id)?;
|
|
66
|
+
self.payment.charge(order.amount)?;
|
|
67
|
+
Ok(())
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
```
|
|
71
|
+
|
|
72
|
+
テスト:
|
|
73
|
+
|
|
74
|
+
```rust
|
|
75
|
+
#[test]
|
|
76
|
+
fn process_order_charges_payment() {
|
|
77
|
+
let mut mock_repo = MockOrderRepository::new();
|
|
78
|
+
mock_repo.expect_find_by_id()
|
|
79
|
+
.returning(|_| Ok(Order { id: 1, amount: 5000 }));
|
|
80
|
+
|
|
81
|
+
let mut mock_payment = MockPaymentGateway::new();
|
|
82
|
+
mock_payment.expect_charge()
|
|
83
|
+
.with(mockall::predicate::eq(5000))
|
|
84
|
+
.returning(|_| Ok(PaymentResult::Success));
|
|
85
|
+
|
|
86
|
+
let service = OrderService::new(
|
|
87
|
+
Box::new(mock_repo),
|
|
88
|
+
Box::new(mock_payment),
|
|
89
|
+
);
|
|
90
|
+
|
|
91
|
+
assert!(service.process_order(1).is_ok());
|
|
92
|
+
}
|
|
93
|
+
```
|
|
94
|
+
|
|
95
|
+
## TDD がもたらす設計上の利点
|
|
96
|
+
|
|
97
|
+
### 1. インターフェース(trait)の明確化
|
|
98
|
+
|
|
99
|
+
テストを先に書くことで、使いやすい API が設計される。
|
|
100
|
+
|
|
101
|
+
```rust
|
|
102
|
+
// テストから始めるので、シンプルで直感的な API になる
|
|
103
|
+
#[test]
|
|
104
|
+
fn cart_add_item() {
|
|
105
|
+
let mut cart = ShoppingCart::new();
|
|
106
|
+
cart.add(Item::new("Book", 1000));
|
|
107
|
+
assert_eq!(cart.total(), 1000);
|
|
108
|
+
}
|
|
109
|
+
```
|
|
110
|
+
|
|
111
|
+
### 2. 責任の分離
|
|
112
|
+
|
|
113
|
+
テストが複雑になる = 構造体が複雑すぎるサイン。
|
|
114
|
+
|
|
115
|
+
```rust
|
|
116
|
+
// 責任が多すぎる → テストが複雑
|
|
117
|
+
struct OrderProcessor { /* 在庫確認 + 決済 + メール + 配送 */ }
|
|
118
|
+
|
|
119
|
+
// 責任を分離 → テストが簡単
|
|
120
|
+
struct OrderProcessor {
|
|
121
|
+
inventory: Box<dyn InventoryService>,
|
|
122
|
+
payment: Box<dyn PaymentService>,
|
|
123
|
+
notification: Box<dyn NotificationService>,
|
|
124
|
+
shipping: Box<dyn ShippingService>,
|
|
125
|
+
}
|
|
126
|
+
```
|
|
127
|
+
|
|
128
|
+
### 3. 疎結合
|
|
129
|
+
|
|
130
|
+
trait を使うことで、自然に疎結合になる。
|
|
131
|
+
|
|
132
|
+
```rust
|
|
133
|
+
// 密結合(テストしにくい)
|
|
134
|
+
impl UserService {
|
|
135
|
+
fn create_user(&self, email: &str) -> Result<User, AppError> {
|
|
136
|
+
let mut conn = PgConnection::establish("...")?; // 直接生成
|
|
137
|
+
// ...
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
// 疎結合(テストしやすい)
|
|
142
|
+
impl UserService {
|
|
143
|
+
fn new(repository: Box<dyn UserRepository>) -> Self {
|
|
144
|
+
Self { repository } // 注入
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
```
|
|
148
|
+
|
|
149
|
+
## テスタビリティの原則
|
|
150
|
+
|
|
151
|
+
### 1. 外部依存を注入する
|
|
152
|
+
|
|
153
|
+
```rust
|
|
154
|
+
// テストしにくい: 時刻を直接取得
|
|
155
|
+
fn generate_report(&self) -> Report {
|
|
156
|
+
let now = chrono::Utc::now();
|
|
157
|
+
// ...
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
// テストしやすい: trait で抽象化
|
|
161
|
+
trait Clock: Send + Sync {
|
|
162
|
+
fn now(&self) -> DateTime<Utc>;
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
fn generate_report(&self, clock: &dyn Clock) -> Report {
|
|
166
|
+
let now = clock.now();
|
|
167
|
+
// ...
|
|
168
|
+
}
|
|
169
|
+
```
|
|
170
|
+
|
|
171
|
+
### 2. 副作用を分離する
|
|
172
|
+
|
|
173
|
+
```rust
|
|
174
|
+
// 副作用が混在
|
|
175
|
+
fn process_and_save(data: &Data, conn: &mut PgConnection) -> Result<Report, AppError> {
|
|
176
|
+
let result = expensive_calculation(data); // 純粋な計算
|
|
177
|
+
diesel::insert_into(reports::table).values(&result).execute(conn)?; // 副作用
|
|
178
|
+
Ok(result)
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
// 副作用を分離
|
|
182
|
+
fn process(data: &Data) -> Report {
|
|
183
|
+
expensive_calculation(data) // 純粋
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
fn save(report: &Report, conn: &mut PgConnection) -> Result<(), DbError> {
|
|
187
|
+
diesel::insert_into(reports::table).values(report).execute(conn)?;
|
|
188
|
+
Ok(())
|
|
189
|
+
}
|
|
190
|
+
```
|
|
191
|
+
|
|
192
|
+
### 3. 決定論的にする
|
|
193
|
+
|
|
194
|
+
```rust
|
|
195
|
+
// ランダム(再現不可)
|
|
196
|
+
fn generate_token() -> String {
|
|
197
|
+
use rand::Rng;
|
|
198
|
+
rand::thread_rng().gen::<[u8; 32]>().encode_hex()
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
// 決定論的(trait で抽象化)
|
|
202
|
+
trait TokenGenerator: Send + Sync {
|
|
203
|
+
fn generate(&self) -> String;
|
|
204
|
+
}
|
|
205
|
+
```
|
|
206
|
+
|
|
207
|
+
## まとめ
|
|
208
|
+
|
|
209
|
+
TDD を実践すると:
|
|
210
|
+
- trait による抽象化が自然に使われる
|
|
211
|
+
- 責任が適切に分離される
|
|
212
|
+
- 疎結合なコードになる
|
|
213
|
+
- 依存性注入が自然に使われる
|
|
214
|
+
|
|
215
|
+
TDD = 設計駆動開発
|
|
@@ -0,0 +1,128 @@
|
|
|
1
|
+
# テスト設計
|
|
2
|
+
|
|
3
|
+
## 境界値分析(Boundary Value Analysis)
|
|
4
|
+
|
|
5
|
+
境界付近の値は特にバグが発生しやすい。
|
|
6
|
+
|
|
7
|
+
### 例: 年齢区分
|
|
8
|
+
|
|
9
|
+
```rust
|
|
10
|
+
// 0-17: 未成年, 18-64: 成人, 65以上: 高齢者
|
|
11
|
+
|
|
12
|
+
#[test]
|
|
13
|
+
fn age_17_is_minor() {
|
|
14
|
+
assert!(is_minor(17));
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
#[test]
|
|
18
|
+
fn age_18_is_not_minor() {
|
|
19
|
+
assert!(!is_minor(18));
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
#[test]
|
|
23
|
+
fn age_64_is_adult() {
|
|
24
|
+
assert!(is_adult(64));
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
#[test]
|
|
28
|
+
fn age_65_is_senior() {
|
|
29
|
+
assert!(is_senior(65));
|
|
30
|
+
}
|
|
31
|
+
```
|
|
32
|
+
|
|
33
|
+
## 同値分割(Equivalence Partitioning)
|
|
34
|
+
|
|
35
|
+
同じ振る舞いをするグループに分割してテストケースを削減。
|
|
36
|
+
|
|
37
|
+
### 例: 割引計算
|
|
38
|
+
|
|
39
|
+
```rust
|
|
40
|
+
// 0-999: 割引なし
|
|
41
|
+
// 1000-4999: 5%割引
|
|
42
|
+
// 5000以上: 10%割引
|
|
43
|
+
|
|
44
|
+
use rstest::rstest;
|
|
45
|
+
|
|
46
|
+
// 各クラスから代表値を選んでテスト
|
|
47
|
+
#[rstest]
|
|
48
|
+
#[case(500, 0)]
|
|
49
|
+
#[case(3000, 150)]
|
|
50
|
+
#[case(10000, 1000)]
|
|
51
|
+
fn discount_representative_values(#[case] amount: u64, #[case] expected: u64) {
|
|
52
|
+
assert_eq!(calculate_discount(amount), expected);
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
// 境界値も必ずテスト
|
|
56
|
+
#[rstest]
|
|
57
|
+
#[case(999, 0)]
|
|
58
|
+
#[case(1000, 50)]
|
|
59
|
+
#[case(4999, 249)]
|
|
60
|
+
#[case(5000, 500)]
|
|
61
|
+
fn discount_boundary_values(#[case] amount: u64, #[case] expected: u64) {
|
|
62
|
+
assert_eq!(calculate_discount(amount), expected);
|
|
63
|
+
}
|
|
64
|
+
```
|
|
65
|
+
|
|
66
|
+
## テスト命名規則
|
|
67
|
+
|
|
68
|
+
### パターン1: 英語構造的命名(推奨)
|
|
69
|
+
|
|
70
|
+
```rust
|
|
71
|
+
#[test]
|
|
72
|
+
fn total_should_be_zero_when_cart_is_empty() { /* ... */ }
|
|
73
|
+
|
|
74
|
+
#[test]
|
|
75
|
+
fn total_should_increase_when_item_added() { /* ... */ }
|
|
76
|
+
|
|
77
|
+
#[test]
|
|
78
|
+
fn should_return_error_when_negative_price() { /* ... */ }
|
|
79
|
+
```
|
|
80
|
+
|
|
81
|
+
### パターン2: 日本語(可読性重視の場合)
|
|
82
|
+
|
|
83
|
+
```rust
|
|
84
|
+
#[test]
|
|
85
|
+
fn 空のカートの合計金額は0円() { /* ... */ }
|
|
86
|
+
|
|
87
|
+
#[test]
|
|
88
|
+
fn 商品追加でカートの合計金額が増える() { /* ... */ }
|
|
89
|
+
|
|
90
|
+
#[test]
|
|
91
|
+
fn 負の価格の商品追加で例外発生() { /* ... */ }
|
|
92
|
+
```
|
|
93
|
+
|
|
94
|
+
### 命名のポイント
|
|
95
|
+
- テスト名から何をテストしているか分かる
|
|
96
|
+
- 失敗時に原因が推測できる
|
|
97
|
+
- 「対象_条件_期待結果」パターン
|
|
98
|
+
|
|
99
|
+
## エラーケースのテスト
|
|
100
|
+
|
|
101
|
+
```rust
|
|
102
|
+
// #[should_panic] を使う方法
|
|
103
|
+
#[test]
|
|
104
|
+
#[should_panic(expected = "division by zero")]
|
|
105
|
+
fn divide_by_zero_panics() {
|
|
106
|
+
let calc = Calculator::new();
|
|
107
|
+
calc.divide(10, 0);
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
// Result を返す方法(推奨)
|
|
111
|
+
#[test]
|
|
112
|
+
fn invalid_email_returns_validation_error() {
|
|
113
|
+
let result = User::create("invalid-email");
|
|
114
|
+
|
|
115
|
+
assert!(result.is_err());
|
|
116
|
+
let err = result.unwrap_err();
|
|
117
|
+
assert!(matches!(err, AppError::Validation(msg) if msg.contains("email")));
|
|
118
|
+
}
|
|
119
|
+
```
|
|
120
|
+
|
|
121
|
+
Result を返すパターンの方が Rust では一般的。`#[should_panic]` は panic が意図された場合のみ使用する。
|
|
122
|
+
|
|
123
|
+
## まとめ
|
|
124
|
+
|
|
125
|
+
- 境界値は必ずテスト
|
|
126
|
+
- 同値分割で効率化
|
|
127
|
+
- 明確な命名
|
|
128
|
+
- エラーケースは Result ベースで検証
|