@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.
Files changed (472) hide show
  1. package/.claude-plugin/.mcp.json +8 -0
  2. package/.claude-plugin/agents/code-simplifier.md +80 -0
  3. package/.claude-plugin/agents/integ-test-auditor.md +91 -0
  4. package/.claude-plugin/agents/integ-test-worker.md +73 -0
  5. package/.claude-plugin/agents/parallel-worker.md +136 -0
  6. package/.claude-plugin/agents/review-worker.md +279 -0
  7. package/.claude-plugin/agents/unit-test-engineer.md +148 -0
  8. package/.claude-plugin/agents/wave-harness-worker.md +158 -0
  9. package/.claude-plugin/hooks/hooks.json +16 -0
  10. package/.claude-plugin/hooks/tasks-read-guard.sh +17 -0
  11. package/.claude-plugin/marketplace.json +33 -0
  12. package/.claude-plugin/plugin.json +11 -0
  13. package/.claude-plugin/rules/axum.md +154 -0
  14. package/.claude-plugin/rules/cargo-toml.md +63 -0
  15. package/.claude-plugin/rules/context7.md +17 -0
  16. package/.claude-plugin/rules/design-conformance.md +82 -0
  17. package/.claude-plugin/rules/design-principles.md +53 -0
  18. package/.claude-plugin/rules/diesel.md +176 -0
  19. package/.claude-plugin/rules/feedback-loop.md +33 -0
  20. package/.claude-plugin/rules/leptos.md +319 -0
  21. package/.claude-plugin/rules/project-architecture.md +134 -0
  22. package/.claude-plugin/rules/quality-checks.md +265 -0
  23. package/.claude-plugin/rules/rust-style.md +242 -0
  24. package/.claude-plugin/rules/security.md +67 -0
  25. package/.claude-plugin/rules/spec-workflow-enforcement.md +47 -0
  26. package/.claude-plugin/rules/valkey.md +167 -0
  27. package/.claude-plugin/skills/integration-test/SKILL.md +230 -0
  28. package/.claude-plugin/skills/integration-test/references/auditor-prompt.md +78 -0
  29. package/.claude-plugin/skills/integration-test/references/external-api-mock.md +98 -0
  30. package/.claude-plugin/skills/integration-test/references/fixture-catalog.md +155 -0
  31. package/.claude-plugin/skills/integration-test/references/parallel-execution.md +124 -0
  32. package/.claude-plugin/skills/integration-test/references/quality-gate.md +80 -0
  33. package/.claude-plugin/skills/integration-test/references/test-case-design.md +88 -0
  34. package/.claude-plugin/skills/integration-test/references/test-patterns.md +215 -0
  35. package/.claude-plugin/skills/integration-test/references/whiteboard-template.md +81 -0
  36. package/.claude-plugin/skills/integration-test/references/worker-prompt.md +70 -0
  37. package/.claude-plugin/skills/knowhow-capture/SKILL.md +143 -0
  38. package/.claude-plugin/skills/phase-review-team/SKILL.md +380 -0
  39. package/.claude-plugin/skills/spec-design/SKILL.md +282 -0
  40. package/.claude-plugin/skills/spec-e2e-implement/SKILL.md +259 -0
  41. package/.claude-plugin/skills/spec-impl-code/SKILL.md +101 -0
  42. package/.claude-plugin/skills/spec-impl-review/SKILL.md +115 -0
  43. package/.claude-plugin/skills/spec-impl-test-run/SKILL.md +98 -0
  44. package/.claude-plugin/skills/spec-impl-test-write/SKILL.md +121 -0
  45. package/.claude-plugin/skills/spec-implement/SKILL.md +822 -0
  46. package/.claude-plugin/skills/spec-requirements/SKILL.md +130 -0
  47. package/.claude-plugin/skills/spec-review/SKILL.md +274 -0
  48. package/.claude-plugin/skills/spec-tasks/SKILL.md +372 -0
  49. package/.claude-plugin/skills/spec-test-design/SKILL.md +233 -0
  50. package/.claude-plugin/skills/tdd-skills/SKILL.md +95 -0
  51. package/.claude-plugin/skills/tdd-skills/references/advanced-techniques.md +49 -0
  52. package/.claude-plugin/skills/tdd-skills/references/green-strategies.md +70 -0
  53. package/.claude-plugin/skills/tdd-skills/references/tdd-and-design.md +48 -0
  54. package/.claude-plugin/skills/tdd-skills/references/test-design.md +43 -0
  55. package/.claude-plugin/skills/tdd-skills/references/test-doubles.md +53 -0
  56. package/.claude-plugin/skills/tdd-skills/references/test-patterns.md +40 -0
  57. package/.claude-plugin/skills/tdd-skills-rust/SKILL.md +128 -0
  58. package/.claude-plugin/skills/tdd-skills-rust/references/advanced-techniques.md +205 -0
  59. package/.claude-plugin/skills/tdd-skills-rust/references/green-strategies.md +166 -0
  60. package/.claude-plugin/skills/tdd-skills-rust/references/tdd-and-design.md +215 -0
  61. package/.claude-plugin/skills/tdd-skills-rust/references/test-design.md +128 -0
  62. package/.claude-plugin/skills/tdd-skills-rust/references/test-doubles.md +208 -0
  63. package/.claude-plugin/skills/tdd-skills-rust/references/test-patterns.md +223 -0
  64. package/.claude-plugin/with-dashboard/.mcp.json +8 -0
  65. package/.claude-plugin/with-dashboard/plugin.json +10 -0
  66. package/CHANGELOG.md +1007 -0
  67. package/LICENSE +674 -0
  68. package/README.ja.md +380 -0
  69. package/README.md +437 -0
  70. package/dist/__tests__/config.test.d.ts +2 -0
  71. package/dist/__tests__/config.test.d.ts.map +1 -0
  72. package/dist/__tests__/config.test.js +264 -0
  73. package/dist/__tests__/config.test.js.map +1 -0
  74. package/dist/__tests__/index-args.test.d.ts +2 -0
  75. package/dist/__tests__/index-args.test.d.ts.map +1 -0
  76. package/dist/__tests__/index-args.test.js +43 -0
  77. package/dist/__tests__/index-args.test.js.map +1 -0
  78. package/dist/__tests__/index-entrypoint.test.d.ts +2 -0
  79. package/dist/__tests__/index-entrypoint.test.d.ts.map +1 -0
  80. package/dist/__tests__/index-entrypoint.test.js +23 -0
  81. package/dist/__tests__/index-entrypoint.test.js.map +1 -0
  82. package/dist/config.d.ts +26 -0
  83. package/dist/config.d.ts.map +1 -0
  84. package/dist/config.js +188 -0
  85. package/dist/config.js.map +1 -0
  86. package/dist/core/__tests__/git-utils.test.d.ts +2 -0
  87. package/dist/core/__tests__/git-utils.test.d.ts.map +1 -0
  88. package/dist/core/__tests__/git-utils.test.js +179 -0
  89. package/dist/core/__tests__/git-utils.test.js.map +1 -0
  90. package/dist/core/__tests__/mdx-validator.test.d.ts +2 -0
  91. package/dist/core/__tests__/mdx-validator.test.d.ts.map +1 -0
  92. package/dist/core/__tests__/mdx-validator.test.js +42 -0
  93. package/dist/core/__tests__/mdx-validator.test.js.map +1 -0
  94. package/dist/core/__tests__/path-utils.test.d.ts +2 -0
  95. package/dist/core/__tests__/path-utils.test.d.ts.map +1 -0
  96. package/dist/core/__tests__/path-utils.test.js +342 -0
  97. package/dist/core/__tests__/path-utils.test.js.map +1 -0
  98. package/dist/core/__tests__/project-registry.test.d.ts +2 -0
  99. package/dist/core/__tests__/project-registry.test.d.ts.map +1 -0
  100. package/dist/core/__tests__/project-registry.test.js +62 -0
  101. package/dist/core/__tests__/project-registry.test.js.map +1 -0
  102. package/dist/core/__tests__/security-utils.test.d.ts +2 -0
  103. package/dist/core/__tests__/security-utils.test.d.ts.map +1 -0
  104. package/dist/core/__tests__/security-utils.test.js +657 -0
  105. package/dist/core/__tests__/security-utils.test.js.map +1 -0
  106. package/dist/core/__tests__/task-parser.test.d.ts +2 -0
  107. package/dist/core/__tests__/task-parser.test.d.ts.map +1 -0
  108. package/dist/core/__tests__/task-parser.test.js +222 -0
  109. package/dist/core/__tests__/task-parser.test.js.map +1 -0
  110. package/dist/core/__tests__/task-validator.test.d.ts +2 -0
  111. package/dist/core/__tests__/task-validator.test.d.ts.map +1 -0
  112. package/dist/core/__tests__/task-validator.test.js +308 -0
  113. package/dist/core/__tests__/task-validator.test.js.map +1 -0
  114. package/dist/core/archive-service.d.ts +10 -0
  115. package/dist/core/archive-service.d.ts.map +1 -0
  116. package/dist/core/archive-service.js +99 -0
  117. package/dist/core/archive-service.js.map +1 -0
  118. package/dist/core/dashboard-session.d.ts +49 -0
  119. package/dist/core/dashboard-session.d.ts.map +1 -0
  120. package/dist/core/dashboard-session.js +132 -0
  121. package/dist/core/dashboard-session.js.map +1 -0
  122. package/dist/core/git-utils.d.ts +25 -0
  123. package/dist/core/git-utils.d.ts.map +1 -0
  124. package/dist/core/git-utils.js +87 -0
  125. package/dist/core/git-utils.js.map +1 -0
  126. package/dist/core/global-dir.d.ts +44 -0
  127. package/dist/core/global-dir.d.ts.map +1 -0
  128. package/dist/core/global-dir.js +74 -0
  129. package/dist/core/global-dir.js.map +1 -0
  130. package/dist/core/implementation-log-migrator.d.ts +41 -0
  131. package/dist/core/implementation-log-migrator.d.ts.map +1 -0
  132. package/dist/core/implementation-log-migrator.js +258 -0
  133. package/dist/core/implementation-log-migrator.js.map +1 -0
  134. package/dist/core/mdx-validator.d.ts +14 -0
  135. package/dist/core/mdx-validator.d.ts.map +1 -0
  136. package/dist/core/mdx-validator.js +34 -0
  137. package/dist/core/mdx-validator.js.map +1 -0
  138. package/dist/core/parser.d.ts +11 -0
  139. package/dist/core/parser.d.ts.map +1 -0
  140. package/dist/core/parser.js +128 -0
  141. package/dist/core/parser.js.map +1 -0
  142. package/dist/core/path-utils.d.ts +68 -0
  143. package/dist/core/path-utils.d.ts.map +1 -0
  144. package/dist/core/path-utils.js +302 -0
  145. package/dist/core/path-utils.js.map +1 -0
  146. package/dist/core/project-registry.d.ts +94 -0
  147. package/dist/core/project-registry.d.ts.map +1 -0
  148. package/dist/core/project-registry.js +297 -0
  149. package/dist/core/project-registry.js.map +1 -0
  150. package/dist/core/security-utils.d.ts +99 -0
  151. package/dist/core/security-utils.d.ts.map +1 -0
  152. package/dist/core/security-utils.js +275 -0
  153. package/dist/core/security-utils.js.map +1 -0
  154. package/dist/core/task-parser.d.ts +90 -0
  155. package/dist/core/task-parser.d.ts.map +1 -0
  156. package/dist/core/task-parser.js +477 -0
  157. package/dist/core/task-parser.js.map +1 -0
  158. package/dist/core/task-validator.d.ts +37 -0
  159. package/dist/core/task-validator.d.ts.map +1 -0
  160. package/dist/core/task-validator.js +499 -0
  161. package/dist/core/task-validator.js.map +1 -0
  162. package/dist/core/workspace-initializer.d.ts +16 -0
  163. package/dist/core/workspace-initializer.d.ts.map +1 -0
  164. package/dist/core/workspace-initializer.js +168 -0
  165. package/dist/core/workspace-initializer.js.map +1 -0
  166. package/dist/dashboard/__tests__/approval-storage-path-resolution.test.d.ts +2 -0
  167. package/dist/dashboard/__tests__/approval-storage-path-resolution.test.d.ts.map +1 -0
  168. package/dist/dashboard/__tests__/approval-storage-path-resolution.test.js +78 -0
  169. package/dist/dashboard/__tests__/approval-storage-path-resolution.test.js.map +1 -0
  170. package/dist/dashboard/__tests__/multi-server-approvals-content.test.d.ts +2 -0
  171. package/dist/dashboard/__tests__/multi-server-approvals-content.test.d.ts.map +1 -0
  172. package/dist/dashboard/__tests__/multi-server-approvals-content.test.js +115 -0
  173. package/dist/dashboard/__tests__/multi-server-approvals-content.test.js.map +1 -0
  174. package/dist/dashboard/__tests__/watcher-error-handling.test.d.ts +2 -0
  175. package/dist/dashboard/__tests__/watcher-error-handling.test.d.ts.map +1 -0
  176. package/dist/dashboard/__tests__/watcher-error-handling.test.js +118 -0
  177. package/dist/dashboard/__tests__/watcher-error-handling.test.js.map +1 -0
  178. package/dist/dashboard/approval-storage.d.ts +139 -0
  179. package/dist/dashboard/approval-storage.d.ts.map +1 -0
  180. package/dist/dashboard/approval-storage.js +608 -0
  181. package/dist/dashboard/approval-storage.js.map +1 -0
  182. package/dist/dashboard/execution-history-manager.d.ts +52 -0
  183. package/dist/dashboard/execution-history-manager.d.ts.map +1 -0
  184. package/dist/dashboard/execution-history-manager.js +161 -0
  185. package/dist/dashboard/execution-history-manager.js.map +1 -0
  186. package/dist/dashboard/implementation-log-manager.d.ts +97 -0
  187. package/dist/dashboard/implementation-log-manager.d.ts.map +1 -0
  188. package/dist/dashboard/implementation-log-manager.js +617 -0
  189. package/dist/dashboard/implementation-log-manager.js.map +1 -0
  190. package/dist/dashboard/job-scheduler.d.ts +91 -0
  191. package/dist/dashboard/job-scheduler.d.ts.map +1 -0
  192. package/dist/dashboard/job-scheduler.js +321 -0
  193. package/dist/dashboard/job-scheduler.js.map +1 -0
  194. package/dist/dashboard/multi-server.d.ts +42 -0
  195. package/dist/dashboard/multi-server.d.ts.map +1 -0
  196. package/dist/dashboard/multi-server.js +1460 -0
  197. package/dist/dashboard/multi-server.js.map +1 -0
  198. package/dist/dashboard/parser.d.ts +18 -0
  199. package/dist/dashboard/parser.d.ts.map +1 -0
  200. package/dist/dashboard/parser.js +269 -0
  201. package/dist/dashboard/parser.js.map +1 -0
  202. package/dist/dashboard/project-manager.d.ts +82 -0
  203. package/dist/dashboard/project-manager.d.ts.map +1 -0
  204. package/dist/dashboard/project-manager.js +257 -0
  205. package/dist/dashboard/project-manager.js.map +1 -0
  206. package/dist/dashboard/public/assets/Inter-Bold-CD3Pr7BX.woff2 +0 -0
  207. package/dist/dashboard/public/assets/Inter-Medium-B_8v_WHh.woff2 +0 -0
  208. package/dist/dashboard/public/assets/Inter-Regular-DRVdRqcI.woff2 +0 -0
  209. package/dist/dashboard/public/assets/Inter-SemiBold-CtskMddL.woff2 +0 -0
  210. package/dist/dashboard/public/assets/JetBrainsMono-Bold-D4WEaHbo.woff2 +0 -0
  211. package/dist/dashboard/public/assets/JetBrainsMono-Medium-3S3k2nMz.woff2 +0 -0
  212. package/dist/dashboard/public/assets/JetBrainsMono-Regular-BQaDgvhP.woff2 +0 -0
  213. package/dist/dashboard/public/assets/Tableau10-B-NsZVaP.js +1 -0
  214. package/dist/dashboard/public/assets/apl-B4CMkyY2.js +1 -0
  215. package/dist/dashboard/public/assets/arc-a5wW942W.js +1 -0
  216. package/dist/dashboard/public/assets/array-BKyUJesY.js +1 -0
  217. package/dist/dashboard/public/assets/asciiarmor-Df11BRmG.js +1 -0
  218. package/dist/dashboard/public/assets/asn1-EdZsLKOL.js +1 -0
  219. package/dist/dashboard/public/assets/asterisk-B-8jnY81.js +1 -0
  220. package/dist/dashboard/public/assets/blockDiagram-c4efeb88-CvjTuK-w.js +118 -0
  221. package/dist/dashboard/public/assets/brainfuck-C4LP7Hcl.js +1 -0
  222. package/dist/dashboard/public/assets/c4Diagram-c83219d4-NwVQo5kf.js +10 -0
  223. package/dist/dashboard/public/assets/channel-Bi16YZhk.js +1 -0
  224. package/dist/dashboard/public/assets/classDiagram-beda092f-BmSeXDdU.js +2 -0
  225. package/dist/dashboard/public/assets/classDiagram-v2-2358418a-D7GvvuPr.js +2 -0
  226. package/dist/dashboard/public/assets/clike-B9uivgTg.js +1 -0
  227. package/dist/dashboard/public/assets/clojure-BMjYHr_A.js +1 -0
  228. package/dist/dashboard/public/assets/clone-BpKTiq7P.js +1 -0
  229. package/dist/dashboard/public/assets/cmake-BQqOBYOt.js +1 -0
  230. package/dist/dashboard/public/assets/cobol-CWcv1MsR.js +1 -0
  231. package/dist/dashboard/public/assets/coffeescript-S37ZYGWr.js +1 -0
  232. package/dist/dashboard/public/assets/commonlisp-DBKNyK5s.js +1 -0
  233. package/dist/dashboard/public/assets/createText-1719965b-qASbqHUP.js +7 -0
  234. package/dist/dashboard/public/assets/crystal-SjHAIU92.js +1 -0
  235. package/dist/dashboard/public/assets/css-BnMrqG3P.js +1 -0
  236. package/dist/dashboard/public/assets/cypher-C_CwsFkJ.js +1 -0
  237. package/dist/dashboard/public/assets/d-pRatUO7H.js +1 -0
  238. package/dist/dashboard/public/assets/diff-DbItnlRl.js +1 -0
  239. package/dist/dashboard/public/assets/dockerfile-BKs6k2Af.js +1 -0
  240. package/dist/dashboard/public/assets/dtd-DF_7sFjM.js +1 -0
  241. package/dist/dashboard/public/assets/dylan-DwRh75JA.js +1 -0
  242. package/dist/dashboard/public/assets/ebnf-CDyGwa7X.js +1 -0
  243. package/dist/dashboard/public/assets/ecl-Cabwm37j.js +1 -0
  244. package/dist/dashboard/public/assets/edges-96097737-BItTSnH7.js +4 -0
  245. package/dist/dashboard/public/assets/eiffel-CnydiIhH.js +1 -0
  246. package/dist/dashboard/public/assets/elm-vLlmbW-K.js +1 -0
  247. package/dist/dashboard/public/assets/erDiagram-0228fc6a-DT224olg.js +51 -0
  248. package/dist/dashboard/public/assets/erlang-BNw1qcRV.js +1 -0
  249. package/dist/dashboard/public/assets/factor-kuTfRLto.js +1 -0
  250. package/dist/dashboard/public/assets/fcl-Kvtd6kyn.js +1 -0
  251. package/dist/dashboard/public/assets/flowDb-c6c81e3f-D9_ukKtv.js +10 -0
  252. package/dist/dashboard/public/assets/flowDiagram-50d868cf-CylE8siG.js +4 -0
  253. package/dist/dashboard/public/assets/flowDiagram-v2-4f6560a1-B2O3JN7Y.js +1 -0
  254. package/dist/dashboard/public/assets/flowchart-elk-definition-6af322e1-BCaqFKf3.js +139 -0
  255. package/dist/dashboard/public/assets/forth-Ffai-XNe.js +1 -0
  256. package/dist/dashboard/public/assets/fortran-DYz_wnZ1.js +1 -0
  257. package/dist/dashboard/public/assets/ganttDiagram-a2739b55-WQUL1QW_.js +257 -0
  258. package/dist/dashboard/public/assets/gas-Bneqetm1.js +1 -0
  259. package/dist/dashboard/public/assets/gherkin-heZmZLOM.js +1 -0
  260. package/dist/dashboard/public/assets/gitGraphDiagram-82fe8481-CttZrdmr.js +70 -0
  261. package/dist/dashboard/public/assets/graph-Ch-rVueN.js +1 -0
  262. package/dist/dashboard/public/assets/groovy-D9Dt4D0W.js +1 -0
  263. package/dist/dashboard/public/assets/haskell-Cw1EW3IL.js +1 -0
  264. package/dist/dashboard/public/assets/haxe-H-WmDvRZ.js +1 -0
  265. package/dist/dashboard/public/assets/http-DBlCnlav.js +1 -0
  266. package/dist/dashboard/public/assets/idl-BEugSyMb.js +1 -0
  267. package/dist/dashboard/public/assets/index--kbPpDKv.js +1 -0
  268. package/dist/dashboard/public/assets/index-3scDwWm6.js +1 -0
  269. package/dist/dashboard/public/assets/index-5325376f-BL2zVOJU.js +1 -0
  270. package/dist/dashboard/public/assets/index-BZdjbO25.js +1 -0
  271. package/dist/dashboard/public/assets/index-BmA_batZ.js +1 -0
  272. package/dist/dashboard/public/assets/index-Bu0u99kF.js +2 -0
  273. package/dist/dashboard/public/assets/index-Ch-lr7F4.js +1 -0
  274. package/dist/dashboard/public/assets/index-ClgWbdoq.js +1 -0
  275. package/dist/dashboard/public/assets/index-CzLwOMQ_.js +3 -0
  276. package/dist/dashboard/public/assets/index-DAOEjGO7.js +1 -0
  277. package/dist/dashboard/public/assets/index-DXqf0B9c.js +1 -0
  278. package/dist/dashboard/public/assets/index-DegWdR16.js +1 -0
  279. package/dist/dashboard/public/assets/index-DiHyYGim.js +1 -0
  280. package/dist/dashboard/public/assets/index-DlZtG7I5.js +1 -0
  281. package/dist/dashboard/public/assets/index-DmhGE2M8.js +1 -0
  282. package/dist/dashboard/public/assets/index-QEGvld4x.js +1 -0
  283. package/dist/dashboard/public/assets/index-RfZPGAJu.js +1 -0
  284. package/dist/dashboard/public/assets/index-UybBj_7u.js +319 -0
  285. package/dist/dashboard/public/assets/index-bVekzPnl.js +7 -0
  286. package/dist/dashboard/public/assets/index-f5bysQzW.css +1 -0
  287. package/dist/dashboard/public/assets/infoDiagram-8eee0895-DjzkkE3o.js +7 -0
  288. package/dist/dashboard/public/assets/init-Gi6I4Gst.js +1 -0
  289. package/dist/dashboard/public/assets/javascript-iXu5QeM3.js +1 -0
  290. package/dist/dashboard/public/assets/journeyDiagram-c64418c1-CxPZkNdB.js +139 -0
  291. package/dist/dashboard/public/assets/julia-DuME0IfC.js +1 -0
  292. package/dist/dashboard/public/assets/katex-XbL3y5x-.js +261 -0
  293. package/dist/dashboard/public/assets/layout-DX7DNTRm.js +1 -0
  294. package/dist/dashboard/public/assets/line-DfvpmKOn.js +1 -0
  295. package/dist/dashboard/public/assets/linear-gQbBPHO5.js +1 -0
  296. package/dist/dashboard/public/assets/livescript-BwQOo05w.js +1 -0
  297. package/dist/dashboard/public/assets/lua-BgMRiT3U.js +1 -0
  298. package/dist/dashboard/public/assets/mathematica-DTrFuWx2.js +1 -0
  299. package/dist/dashboard/public/assets/mbox-CNhZ1qSd.js +1 -0
  300. package/dist/dashboard/public/assets/mindmap-definition-8da855dc-CNxmpyG6.js +415 -0
  301. package/dist/dashboard/public/assets/mirc-CjQqDB4T.js +1 -0
  302. package/dist/dashboard/public/assets/mllike-CXdrOF99.js +1 -0
  303. package/dist/dashboard/public/assets/modelica-Dc1JOy9r.js +1 -0
  304. package/dist/dashboard/public/assets/mscgen-BA5vi2Kp.js +1 -0
  305. package/dist/dashboard/public/assets/mumps-BT43cFF4.js +1 -0
  306. package/dist/dashboard/public/assets/nginx-DdIZxoE0.js +1 -0
  307. package/dist/dashboard/public/assets/nsis-LdVXkNf5.js +1 -0
  308. package/dist/dashboard/public/assets/ntriples-BfvgReVJ.js +1 -0
  309. package/dist/dashboard/public/assets/octave-Ck1zUtKM.js +1 -0
  310. package/dist/dashboard/public/assets/ordinal-Cboi1Yqb.js +1 -0
  311. package/dist/dashboard/public/assets/oz-BzwKVEFT.js +1 -0
  312. package/dist/dashboard/public/assets/pascal--L3eBynH.js +1 -0
  313. package/dist/dashboard/public/assets/path-CbwjOpE9.js +1 -0
  314. package/dist/dashboard/public/assets/perl-CdXCOZ3F.js +1 -0
  315. package/dist/dashboard/public/assets/pieDiagram-a8764435-D-xy_NSA.js +35 -0
  316. package/dist/dashboard/public/assets/pig-CevX1Tat.js +1 -0
  317. package/dist/dashboard/public/assets/powershell-CFHJl5sT.js +1 -0
  318. package/dist/dashboard/public/assets/properties-C78fOPTZ.js +1 -0
  319. package/dist/dashboard/public/assets/protobuf-ChK-085T.js +1 -0
  320. package/dist/dashboard/public/assets/pug-DeIclll2.js +1 -0
  321. package/dist/dashboard/public/assets/puppet-DMA9R1ak.js +1 -0
  322. package/dist/dashboard/public/assets/python-BuPzkPfP.js +1 -0
  323. package/dist/dashboard/public/assets/q-pXgVlZs6.js +1 -0
  324. package/dist/dashboard/public/assets/quadrantDiagram-1e28029f-BoL2wzz0.js +7 -0
  325. package/dist/dashboard/public/assets/r-B6wPVr8A.js +1 -0
  326. package/dist/dashboard/public/assets/requirementDiagram-08caed73-BujFz0q1.js +52 -0
  327. package/dist/dashboard/public/assets/rpm-CTu-6PCP.js +1 -0
  328. package/dist/dashboard/public/assets/ruby-B2Rjki9n.js +1 -0
  329. package/dist/dashboard/public/assets/sankeyDiagram-a04cb91d-D03_NARm.js +8 -0
  330. package/dist/dashboard/public/assets/sas-B4kiWyti.js +1 -0
  331. package/dist/dashboard/public/assets/scheme-C41bIUwD.js +1 -0
  332. package/dist/dashboard/public/assets/sequenceDiagram-c5b8d532-B65eFcaT.js +122 -0
  333. package/dist/dashboard/public/assets/shell-CjFT_Tl9.js +1 -0
  334. package/dist/dashboard/public/assets/sieve-C3Gn_uJK.js +1 -0
  335. package/dist/dashboard/public/assets/simple-mode-GW_nhZxv.js +1 -0
  336. package/dist/dashboard/public/assets/smalltalk-CnHTOXQT.js +1 -0
  337. package/dist/dashboard/public/assets/solr-DehyRSwq.js +1 -0
  338. package/dist/dashboard/public/assets/sparql-DkYu6x3z.js +1 -0
  339. package/dist/dashboard/public/assets/spreadsheet-BCZA_wO0.js +1 -0
  340. package/dist/dashboard/public/assets/sql-D0XecflT.js +1 -0
  341. package/dist/dashboard/public/assets/stateDiagram-1ecb1508-BDbqu0Vl.js +1 -0
  342. package/dist/dashboard/public/assets/stateDiagram-v2-c2b004d7-CBHvk4b8.js +1 -0
  343. package/dist/dashboard/public/assets/stex-C3f8Ysf7.js +1 -0
  344. package/dist/dashboard/public/assets/styles-b4e223ce-CELsPqaO.js +160 -0
  345. package/dist/dashboard/public/assets/styles-ca3715f6-BRqMqT6F.js +207 -0
  346. package/dist/dashboard/public/assets/styles-d45a18b0-e8N-oLPy.js +116 -0
  347. package/dist/dashboard/public/assets/stylus-B533Al4x.js +1 -0
  348. package/dist/dashboard/public/assets/svgDrawCommon-b86b1483-vNDtmQc-.js +1 -0
  349. package/dist/dashboard/public/assets/swift-BzpIVaGY.js +1 -0
  350. package/dist/dashboard/public/assets/tcl-DVfN8rqt.js +1 -0
  351. package/dist/dashboard/public/assets/textile-CnDTJFAw.js +1 -0
  352. package/dist/dashboard/public/assets/tiddlywiki-DO-Gjzrf.js +1 -0
  353. package/dist/dashboard/public/assets/tiki-DGYXhP31.js +1 -0
  354. package/dist/dashboard/public/assets/timeline-definition-faaaa080-Dh2_A5VU.js +61 -0
  355. package/dist/dashboard/public/assets/toml-Bm5Em-hy.js +1 -0
  356. package/dist/dashboard/public/assets/troff-wAsdV37c.js +1 -0
  357. package/dist/dashboard/public/assets/ttcn-CfJYG6tj.js +1 -0
  358. package/dist/dashboard/public/assets/ttcn-cfg-B9xdYoR4.js +1 -0
  359. package/dist/dashboard/public/assets/turtle-B1tBg_DP.js +1 -0
  360. package/dist/dashboard/public/assets/vb-CmGdzxic.js +1 -0
  361. package/dist/dashboard/public/assets/vbscript-BuJXcnF6.js +1 -0
  362. package/dist/dashboard/public/assets/velocity-D8B20fx6.js +1 -0
  363. package/dist/dashboard/public/assets/verilog-C6RDOZhf.js +1 -0
  364. package/dist/dashboard/public/assets/vhdl-lSbBsy5d.js +1 -0
  365. package/dist/dashboard/public/assets/webidl-ZXfAyPTL.js +1 -0
  366. package/dist/dashboard/public/assets/xquery-DzFWVndE.js +1 -0
  367. package/dist/dashboard/public/assets/xychartDiagram-f5964ef8-B76v1AVF.js +7 -0
  368. package/dist/dashboard/public/assets/yacas-BJ4BC0dw.js +1 -0
  369. package/dist/dashboard/public/assets/z80-Hz9HOZM7.js +1 -0
  370. package/dist/dashboard/public/claude-icon-dark.svg +1 -0
  371. package/dist/dashboard/public/claude-icon.svg +1 -0
  372. package/dist/dashboard/public/index.html +16 -0
  373. package/dist/dashboard/settings-manager.d.ts +47 -0
  374. package/dist/dashboard/settings-manager.d.ts.map +1 -0
  375. package/dist/dashboard/settings-manager.js +180 -0
  376. package/dist/dashboard/settings-manager.js.map +1 -0
  377. package/dist/dashboard/utils.d.ts +31 -0
  378. package/dist/dashboard/utils.d.ts.map +1 -0
  379. package/dist/dashboard/utils.js +102 -0
  380. package/dist/dashboard/utils.js.map +1 -0
  381. package/dist/dashboard/watcher.d.ts +32 -0
  382. package/dist/dashboard/watcher.d.ts.map +1 -0
  383. package/dist/dashboard/watcher.js +173 -0
  384. package/dist/dashboard/watcher.js.map +1 -0
  385. package/dist/index.d.ts +13 -0
  386. package/dist/index.d.ts.map +1 -0
  387. package/dist/index.js +380 -0
  388. package/dist/index.js.map +1 -0
  389. package/dist/markdown/templates/design-template.md +126 -0
  390. package/dist/markdown/templates/product-template.md +51 -0
  391. package/dist/markdown/templates/requirements-template.md +50 -0
  392. package/dist/markdown/templates/structure-template.md +145 -0
  393. package/dist/markdown/templates/tasks-template.md +100 -0
  394. package/dist/markdown/templates/tech-template.md +99 -0
  395. package/dist/markdown/templates/test-design-template.md +221 -0
  396. package/dist/prompts/create-spec.d.ts +3 -0
  397. package/dist/prompts/create-spec.d.ts.map +1 -0
  398. package/dist/prompts/create-spec.js +97 -0
  399. package/dist/prompts/create-spec.js.map +1 -0
  400. package/dist/prompts/create-steering-doc.d.ts +3 -0
  401. package/dist/prompts/create-steering-doc.d.ts.map +1 -0
  402. package/dist/prompts/create-steering-doc.js +75 -0
  403. package/dist/prompts/create-steering-doc.js.map +1 -0
  404. package/dist/prompts/implement-task.d.ts +3 -0
  405. package/dist/prompts/implement-task.d.ts.map +1 -0
  406. package/dist/prompts/implement-task.js +174 -0
  407. package/dist/prompts/implement-task.js.map +1 -0
  408. package/dist/prompts/index.d.ts +20 -0
  409. package/dist/prompts/index.d.ts.map +1 -0
  410. package/dist/prompts/index.js +103 -0
  411. package/dist/prompts/index.js.map +1 -0
  412. package/dist/prompts/inject-spec-workflow-guide.d.ts +3 -0
  413. package/dist/prompts/inject-spec-workflow-guide.d.ts.map +1 -0
  414. package/dist/prompts/inject-spec-workflow-guide.js +60 -0
  415. package/dist/prompts/inject-spec-workflow-guide.js.map +1 -0
  416. package/dist/prompts/inject-steering-guide.d.ts +3 -0
  417. package/dist/prompts/inject-steering-guide.d.ts.map +1 -0
  418. package/dist/prompts/inject-steering-guide.js +64 -0
  419. package/dist/prompts/inject-steering-guide.js.map +1 -0
  420. package/dist/prompts/refresh-tasks.d.ts +3 -0
  421. package/dist/prompts/refresh-tasks.d.ts.map +1 -0
  422. package/dist/prompts/refresh-tasks.js +237 -0
  423. package/dist/prompts/refresh-tasks.js.map +1 -0
  424. package/dist/prompts/spec-status.d.ts +3 -0
  425. package/dist/prompts/spec-status.d.ts.map +1 -0
  426. package/dist/prompts/spec-status.js +77 -0
  427. package/dist/prompts/spec-status.js.map +1 -0
  428. package/dist/prompts/types.d.ts +13 -0
  429. package/dist/prompts/types.d.ts.map +1 -0
  430. package/dist/prompts/types.js +2 -0
  431. package/dist/prompts/types.js.map +1 -0
  432. package/dist/server.d.ts +17 -0
  433. package/dist/server.d.ts.map +1 -0
  434. package/dist/server.js +175 -0
  435. package/dist/server.js.map +1 -0
  436. package/dist/tools/__tests__/log-implementation-review-process.test.d.ts +2 -0
  437. package/dist/tools/__tests__/log-implementation-review-process.test.d.ts.map +1 -0
  438. package/dist/tools/__tests__/log-implementation-review-process.test.js +190 -0
  439. package/dist/tools/__tests__/log-implementation-review-process.test.js.map +1 -0
  440. package/dist/tools/__tests__/projectPath.test.d.ts +2 -0
  441. package/dist/tools/__tests__/projectPath.test.d.ts.map +1 -0
  442. package/dist/tools/__tests__/projectPath.test.js +187 -0
  443. package/dist/tools/__tests__/projectPath.test.js.map +1 -0
  444. package/dist/tools/approvals.d.ts +14 -0
  445. package/dist/tools/approvals.d.ts.map +1 -0
  446. package/dist/tools/approvals.js +505 -0
  447. package/dist/tools/approvals.js.map +1 -0
  448. package/dist/tools/index.d.ts +5 -0
  449. package/dist/tools/index.d.ts.map +1 -0
  450. package/dist/tools/index.js +52 -0
  451. package/dist/tools/index.js.map +1 -0
  452. package/dist/tools/log-implementation.d.ts +5 -0
  453. package/dist/tools/log-implementation.d.ts.map +1 -0
  454. package/dist/tools/log-implementation.js +498 -0
  455. package/dist/tools/log-implementation.js.map +1 -0
  456. package/dist/tools/spec-status.d.ts +5 -0
  457. package/dist/tools/spec-status.d.ts.map +1 -0
  458. package/dist/tools/spec-status.js +192 -0
  459. package/dist/tools/spec-status.js.map +1 -0
  460. package/dist/tools/spec-workflow-guide.d.ts +5 -0
  461. package/dist/tools/spec-workflow-guide.d.ts.map +1 -0
  462. package/dist/tools/spec-workflow-guide.js +116 -0
  463. package/dist/tools/spec-workflow-guide.js.map +1 -0
  464. package/dist/tools/steering-guide.d.ts +5 -0
  465. package/dist/tools/steering-guide.d.ts.map +1 -0
  466. package/dist/tools/steering-guide.js +192 -0
  467. package/dist/tools/steering-guide.js.map +1 -0
  468. package/dist/types.d.ts +183 -0
  469. package/dist/types.d.ts.map +1 -0
  470. package/dist/types.js +13 -0
  471. package/dist/types.js.map +1 -0
  472. 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 ベースで検証