@k2works/claude-code-booster 3.0.0 → 3.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (524) hide show
  1. package/lib/assets/docs/article/functional-desgin-ppp/all/01-immutability-and-data-transformation.md +475 -0
  2. package/lib/assets/docs/article/functional-desgin-ppp/all/02-function-composition.md +519 -0
  3. package/lib/assets/docs/article/functional-desgin-ppp/all/03-polymorphism.md +537 -0
  4. package/lib/assets/docs/article/functional-desgin-ppp/all/04-data-validation.md +300 -0
  5. package/lib/assets/docs/article/functional-desgin-ppp/all/05-property-based-testing.md +320 -0
  6. package/lib/assets/docs/article/functional-desgin-ppp/all/06-tdd-and-functional.md +498 -0
  7. package/lib/assets/docs/article/functional-desgin-ppp/all/07-composite-pattern.md +298 -0
  8. package/lib/assets/docs/article/functional-desgin-ppp/all/08-decorator-pattern.md +291 -0
  9. package/lib/assets/docs/article/functional-desgin-ppp/all/09-adapter-pattern.md +336 -0
  10. package/lib/assets/docs/article/functional-desgin-ppp/all/10-strategy-pattern.md +303 -0
  11. package/lib/assets/docs/article/functional-desgin-ppp/all/11-command-pattern.md +286 -0
  12. package/lib/assets/docs/article/functional-desgin-ppp/all/12-visitor-pattern.md +322 -0
  13. package/lib/assets/docs/article/functional-desgin-ppp/all/13-abstract-factory-pattern.md +319 -0
  14. package/lib/assets/docs/article/functional-desgin-ppp/all/14-abstract-server-pattern.md +365 -0
  15. package/lib/assets/docs/article/functional-desgin-ppp/all/15-gossiping-bus-drivers.md +156 -0
  16. package/lib/assets/docs/article/functional-desgin-ppp/all/16-payroll-system.md +178 -0
  17. package/lib/assets/docs/article/functional-desgin-ppp/all/17-video-rental-system.md +312 -0
  18. package/lib/assets/docs/article/functional-desgin-ppp/all/18-concurrency-system.md +287 -0
  19. package/lib/assets/docs/article/functional-desgin-ppp/all/19-wa-tor-simulation.md +286 -0
  20. package/lib/assets/docs/article/functional-desgin-ppp/all/20-pattern-interactions.md +274 -0
  21. package/lib/assets/docs/article/functional-desgin-ppp/all/21-best-practices.md +294 -0
  22. package/lib/assets/docs/article/functional-desgin-ppp/all/22-oo-to-fp-migration.md +337 -0
  23. package/lib/assets/docs/article/functional-desgin-ppp/all/index.md +388 -0
  24. package/lib/assets/docs/article/functional-desgin-ppp/clojure/01-immutability-and-data-transformation.md +271 -0
  25. package/lib/assets/docs/article/functional-desgin-ppp/clojure/02-function-composition.md +380 -0
  26. package/lib/assets/docs/article/functional-desgin-ppp/clojure/03-polymorphism.md +384 -0
  27. package/lib/assets/docs/article/functional-desgin-ppp/clojure/04-clojure-spec.md +350 -0
  28. package/lib/assets/docs/article/functional-desgin-ppp/clojure/05-property-based-testing.md +352 -0
  29. package/lib/assets/docs/article/functional-desgin-ppp/clojure/06-tdd-in-functional.md +383 -0
  30. package/lib/assets/docs/article/functional-desgin-ppp/clojure/07-composite-pattern.md +529 -0
  31. package/lib/assets/docs/article/functional-desgin-ppp/clojure/08-decorator-pattern.md +395 -0
  32. package/lib/assets/docs/article/functional-desgin-ppp/clojure/09-adapter-pattern.md +399 -0
  33. package/lib/assets/docs/article/functional-desgin-ppp/clojure/10-strategy-pattern.md +485 -0
  34. package/lib/assets/docs/article/functional-desgin-ppp/clojure/11-command-pattern.md +566 -0
  35. package/lib/assets/docs/article/functional-desgin-ppp/clojure/12-visitor-pattern.md +567 -0
  36. package/lib/assets/docs/article/functional-desgin-ppp/clojure/13-abstract-factory-pattern.md +475 -0
  37. package/lib/assets/docs/article/functional-desgin-ppp/clojure/14-abstract-server-pattern.md +462 -0
  38. package/lib/assets/docs/article/functional-desgin-ppp/clojure/15-gossiping-bus-drivers.md +323 -0
  39. package/lib/assets/docs/article/functional-desgin-ppp/clojure/16-payroll-system.md +401 -0
  40. package/lib/assets/docs/article/functional-desgin-ppp/clojure/17-video-rental-system.md +450 -0
  41. package/lib/assets/docs/article/functional-desgin-ppp/clojure/18-concurrency-system.md +475 -0
  42. package/lib/assets/docs/article/functional-desgin-ppp/clojure/19-wator-simulation.md +739 -0
  43. package/lib/assets/docs/article/functional-desgin-ppp/clojure/20-pattern-interactions.md +562 -0
  44. package/lib/assets/docs/article/functional-desgin-ppp/clojure/21-best-practices.md +506 -0
  45. package/lib/assets/docs/article/functional-desgin-ppp/clojure/22-oo-to-fp-migration.md +526 -0
  46. package/lib/assets/docs/article/functional-desgin-ppp/clojure/index.md +197 -0
  47. package/lib/assets/docs/article/functional-desgin-ppp/elixir/01-immutability-and-data-transformation.md +381 -0
  48. package/lib/assets/docs/article/functional-desgin-ppp/elixir/02-function-composition.md +374 -0
  49. package/lib/assets/docs/article/functional-desgin-ppp/elixir/03-polymorphism.md +375 -0
  50. package/lib/assets/docs/article/functional-desgin-ppp/elixir/04-data-validation.md +195 -0
  51. package/lib/assets/docs/article/functional-desgin-ppp/elixir/05-property-based-testing.md +268 -0
  52. package/lib/assets/docs/article/functional-desgin-ppp/elixir/06-tdd-and-fp.md +294 -0
  53. package/lib/assets/docs/article/functional-desgin-ppp/elixir/07-effects-and-pure-functions.md +164 -0
  54. package/lib/assets/docs/article/functional-desgin-ppp/elixir/08-error-handling-strategies.md +168 -0
  55. package/lib/assets/docs/article/functional-desgin-ppp/elixir/09-io-and-external-systems.md +254 -0
  56. package/lib/assets/docs/article/functional-desgin-ppp/elixir/10-concurrency-patterns.md +269 -0
  57. package/lib/assets/docs/article/functional-desgin-ppp/elixir/11-command-pattern.md +148 -0
  58. package/lib/assets/docs/article/functional-desgin-ppp/elixir/12-visitor-pattern.md +176 -0
  59. package/lib/assets/docs/article/functional-desgin-ppp/elixir/13-abstract-factory-pattern.md +604 -0
  60. package/lib/assets/docs/article/functional-desgin-ppp/elixir/14-abstract-server-pattern.md +729 -0
  61. package/lib/assets/docs/article/functional-desgin-ppp/elixir/15-gossiping-bus-drivers.md +291 -0
  62. package/lib/assets/docs/article/functional-desgin-ppp/elixir/16-payroll-system.md +420 -0
  63. package/lib/assets/docs/article/functional-desgin-ppp/elixir/17-video-rental-system.md +319 -0
  64. package/lib/assets/docs/article/functional-desgin-ppp/elixir/18-concurrency-system.md +466 -0
  65. package/lib/assets/docs/article/functional-desgin-ppp/elixir/19-wator-simulation.md +523 -0
  66. package/lib/assets/docs/article/functional-desgin-ppp/elixir/20-pattern-interactions.md +287 -0
  67. package/lib/assets/docs/article/functional-desgin-ppp/elixir/21-best-practices.md +340 -0
  68. package/lib/assets/docs/article/functional-desgin-ppp/elixir/22-oo-to-fp-migration.md +395 -0
  69. package/lib/assets/docs/article/functional-desgin-ppp/elixir/index.md +204 -0
  70. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/01-immutability-and-data-transformation.md +382 -0
  71. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/02-function-composition.md +452 -0
  72. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/03-polymorphism.md +495 -0
  73. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/04-data-validation.md +416 -0
  74. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/05-property-based-testing.md +382 -0
  75. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/06-tdd-functional.md +687 -0
  76. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/07-composite-pattern.md +442 -0
  77. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/08-decorator-pattern.md +479 -0
  78. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/09-adapter-pattern.md +479 -0
  79. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/10-strategy-pattern.md +427 -0
  80. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/11-command-pattern.md +428 -0
  81. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/12-visitor-pattern.md +339 -0
  82. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/13-abstract-factory-pattern.md +309 -0
  83. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/14-abstract-server-pattern.md +596 -0
  84. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/15-gossiping-bus-drivers.md +353 -0
  85. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/16-payroll-system.md +350 -0
  86. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/17-video-rental-system.md +412 -0
  87. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/18-concurrency-system.md +367 -0
  88. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/19-wator-simulation.md +401 -0
  89. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/20-pattern-interactions.md +291 -0
  90. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/21-best-practices.md +320 -0
  91. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/22-oo-to-fp-migration.md +322 -0
  92. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/index.md +230 -0
  93. package/lib/assets/docs/article/functional-desgin-ppp/haskell/01-immutability-and-data-transformation.md +298 -0
  94. package/lib/assets/docs/article/functional-desgin-ppp/haskell/02-function-composition.md +304 -0
  95. package/lib/assets/docs/article/functional-desgin-ppp/haskell/03-polymorphism.md +362 -0
  96. package/lib/assets/docs/article/functional-desgin-ppp/haskell/04-data-validation.md +257 -0
  97. package/lib/assets/docs/article/functional-desgin-ppp/haskell/05-property-based-testing.md +254 -0
  98. package/lib/assets/docs/article/functional-desgin-ppp/haskell/06-tdd-functional.md +283 -0
  99. package/lib/assets/docs/article/functional-desgin-ppp/haskell/07-composite-pattern.md +395 -0
  100. package/lib/assets/docs/article/functional-desgin-ppp/haskell/08-decorator-pattern.md +319 -0
  101. package/lib/assets/docs/article/functional-desgin-ppp/haskell/09-adapter-pattern.md +382 -0
  102. package/lib/assets/docs/article/functional-desgin-ppp/haskell/10-strategy-pattern.md +287 -0
  103. package/lib/assets/docs/article/functional-desgin-ppp/haskell/11-command-pattern.md +303 -0
  104. package/lib/assets/docs/article/functional-desgin-ppp/haskell/12-visitor-pattern.md +326 -0
  105. package/lib/assets/docs/article/functional-desgin-ppp/haskell/13-abstract-factory-pattern.md +332 -0
  106. package/lib/assets/docs/article/functional-desgin-ppp/haskell/14-abstract-server-pattern.md +379 -0
  107. package/lib/assets/docs/article/functional-desgin-ppp/haskell/15-gossiping-bus-drivers.md +175 -0
  108. package/lib/assets/docs/article/functional-desgin-ppp/haskell/16-payroll-system.md +219 -0
  109. package/lib/assets/docs/article/functional-desgin-ppp/haskell/17-video-rental-system.md +244 -0
  110. package/lib/assets/docs/article/functional-desgin-ppp/haskell/18-concurrency-system.md +363 -0
  111. package/lib/assets/docs/article/functional-desgin-ppp/haskell/19-wator-simulation.md +438 -0
  112. package/lib/assets/docs/article/functional-desgin-ppp/haskell/20-pattern-interactions.md +323 -0
  113. package/lib/assets/docs/article/functional-desgin-ppp/haskell/21-best-practices.md +403 -0
  114. package/lib/assets/docs/article/functional-desgin-ppp/haskell/22-oo-to-fp-migration.md +469 -0
  115. package/lib/assets/docs/article/functional-desgin-ppp/haskell/index.md +174 -0
  116. package/lib/assets/docs/article/functional-desgin-ppp/index.md +90 -0
  117. package/lib/assets/docs/article/functional-desgin-ppp/rust/01-immutability-and-data-transformation.md +448 -0
  118. package/lib/assets/docs/article/functional-desgin-ppp/rust/02-function-composition.md +463 -0
  119. package/lib/assets/docs/article/functional-desgin-ppp/rust/03-polymorphism.md +425 -0
  120. package/lib/assets/docs/article/functional-desgin-ppp/rust/04-data-validation.md +273 -0
  121. package/lib/assets/docs/article/functional-desgin-ppp/rust/05-property-based-testing.md +247 -0
  122. package/lib/assets/docs/article/functional-desgin-ppp/rust/06-tdd-and-functional.md +841 -0
  123. package/lib/assets/docs/article/functional-desgin-ppp/rust/07-composite-pattern.md +384 -0
  124. package/lib/assets/docs/article/functional-desgin-ppp/rust/08-decorator-pattern.md +383 -0
  125. package/lib/assets/docs/article/functional-desgin-ppp/rust/09-adapter-pattern.md +339 -0
  126. package/lib/assets/docs/article/functional-desgin-ppp/rust/10-strategy-pattern.md +331 -0
  127. package/lib/assets/docs/article/functional-desgin-ppp/rust/11-command-pattern.md +356 -0
  128. package/lib/assets/docs/article/functional-desgin-ppp/rust/12-visitor-pattern.md +379 -0
  129. package/lib/assets/docs/article/functional-desgin-ppp/rust/13-abstract-factory-pattern.md +361 -0
  130. package/lib/assets/docs/article/functional-desgin-ppp/rust/14-abstract-server-pattern.md +392 -0
  131. package/lib/assets/docs/article/functional-desgin-ppp/rust/15-gossiping-bus-drivers.md +300 -0
  132. package/lib/assets/docs/article/functional-desgin-ppp/rust/16-payroll-system.md +297 -0
  133. package/lib/assets/docs/article/functional-desgin-ppp/rust/17-video-rental-system.md +304 -0
  134. package/lib/assets/docs/article/functional-desgin-ppp/rust/18-concurrency-system.md +315 -0
  135. package/lib/assets/docs/article/functional-desgin-ppp/rust/19-wator-simulation.md +311 -0
  136. package/lib/assets/docs/article/functional-desgin-ppp/rust/20-pattern-interactions.md +304 -0
  137. package/lib/assets/docs/article/functional-desgin-ppp/rust/21-best-practices.md +336 -0
  138. package/lib/assets/docs/article/functional-desgin-ppp/rust/22-oo-to-fp-migration.md +349 -0
  139. package/lib/assets/docs/article/functional-desgin-ppp/rust/index.md +199 -0
  140. package/lib/assets/docs/article/functional-desgin-ppp/scala/01-immutability-and-data-transformation.md +326 -0
  141. package/lib/assets/docs/article/functional-desgin-ppp/scala/02-function-composition.md +348 -0
  142. package/lib/assets/docs/article/functional-desgin-ppp/scala/03-polymorphism.md +357 -0
  143. package/lib/assets/docs/article/functional-desgin-ppp/scala/04-data-validation.md +364 -0
  144. package/lib/assets/docs/article/functional-desgin-ppp/scala/05-property-based-testing.md +515 -0
  145. package/lib/assets/docs/article/functional-desgin-ppp/scala/06-tdd-functional.md +557 -0
  146. package/lib/assets/docs/article/functional-desgin-ppp/scala/07-composite-pattern.md +363 -0
  147. package/lib/assets/docs/article/functional-desgin-ppp/scala/08-decorator-pattern.md +327 -0
  148. package/lib/assets/docs/article/functional-desgin-ppp/scala/09-adapter-pattern.md +517 -0
  149. package/lib/assets/docs/article/functional-desgin-ppp/scala/10-strategy-pattern.md +441 -0
  150. package/lib/assets/docs/article/functional-desgin-ppp/scala/11-command-pattern.md +407 -0
  151. package/lib/assets/docs/article/functional-desgin-ppp/scala/12-visitor-pattern.md +379 -0
  152. package/lib/assets/docs/article/functional-desgin-ppp/scala/13-abstract-factory-pattern.md +398 -0
  153. package/lib/assets/docs/article/functional-desgin-ppp/scala/14-abstract-server-pattern.md +476 -0
  154. package/lib/assets/docs/article/functional-desgin-ppp/scala/15-gossiping-bus-drivers.md +389 -0
  155. package/lib/assets/docs/article/functional-desgin-ppp/scala/16-payroll-system.md +342 -0
  156. package/lib/assets/docs/article/functional-desgin-ppp/scala/17-video-rental-system.md +324 -0
  157. package/lib/assets/docs/article/functional-desgin-ppp/scala/18-concurrency-system.md +730 -0
  158. package/lib/assets/docs/article/functional-desgin-ppp/scala/19-wator-simulation.md +624 -0
  159. package/lib/assets/docs/article/functional-desgin-ppp/scala/20-pattern-interactions.md +512 -0
  160. package/lib/assets/docs/article/functional-desgin-ppp/scala/21-best-practices.md +427 -0
  161. package/lib/assets/docs/article/functional-desgin-ppp/scala/22-oo-to-fp-migration.md +682 -0
  162. package/lib/assets/docs/article/functional-desgin-ppp/scala/index.md +199 -0
  163. package/lib/assets/docs/article/getting-start-tdd/clojure/01-todo-list-and-first-test.md +166 -0
  164. package/lib/assets/docs/article/getting-start-tdd/clojure/02-fake-it-and-triangulation.md +162 -0
  165. package/lib/assets/docs/article/getting-start-tdd/clojure/03-obvious-implementation-and-refactoring.md +135 -0
  166. package/lib/assets/docs/article/getting-start-tdd/clojure/04-version-control-and-conventional-commits.md +88 -0
  167. package/lib/assets/docs/article/getting-start-tdd/clojure/05-package-management-and-static-analysis.md +299 -0
  168. package/lib/assets/docs/article/getting-start-tdd/clojure/06-task-runner-and-ci-cd.md +241 -0
  169. package/lib/assets/docs/article/getting-start-tdd/clojure/07-protocols-and-records.md +131 -0
  170. package/lib/assets/docs/article/getting-start-tdd/clojure/08-multimethods-and-design-patterns.md +130 -0
  171. package/lib/assets/docs/article/getting-start-tdd/clojure/09-namespaces-and-module-design.md +127 -0
  172. package/lib/assets/docs/article/getting-start-tdd/clojure/10-higher-order-functions-and-composition.md +114 -0
  173. package/lib/assets/docs/article/getting-start-tdd/clojure/11-persistent-data-and-pipeline.md +138 -0
  174. package/lib/assets/docs/article/getting-start-tdd/clojure/12-error-handling-and-spec.md +161 -0
  175. package/lib/assets/docs/article/getting-start-tdd/clojure/index.md +65 -0
  176. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter01.md +232 -0
  177. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter02.md +244 -0
  178. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter03.md +202 -0
  179. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter04.md +92 -0
  180. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter05.md +256 -0
  181. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter06.md +195 -0
  182. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter07.md +214 -0
  183. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter08.md +249 -0
  184. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter09.md +174 -0
  185. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter10.md +166 -0
  186. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter11.md +192 -0
  187. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter12.md +211 -0
  188. package/lib/assets/docs/article/getting-start-tdd/csharp/index.md +83 -0
  189. package/lib/assets/docs/article/getting-start-tdd/elixir/01-todo-list-and-first-test.md +87 -0
  190. package/lib/assets/docs/article/getting-start-tdd/elixir/02-fake-it-and-triangulation.md +95 -0
  191. package/lib/assets/docs/article/getting-start-tdd/elixir/03-obvious-implementation-and-refactoring.md +109 -0
  192. package/lib/assets/docs/article/getting-start-tdd/elixir/04-version-control-and-conventional-commits.md +96 -0
  193. package/lib/assets/docs/article/getting-start-tdd/elixir/05-package-management-and-static-analysis.md +88 -0
  194. package/lib/assets/docs/article/getting-start-tdd/elixir/06-task-runner-and-ci-cd.md +71 -0
  195. package/lib/assets/docs/article/getting-start-tdd/elixir/07-structs-and-protocols.md +110 -0
  196. package/lib/assets/docs/article/getting-start-tdd/elixir/08-pattern-matching-and-guards.md +108 -0
  197. package/lib/assets/docs/article/getting-start-tdd/elixir/09-module-design-and-behaviours.md +104 -0
  198. package/lib/assets/docs/article/getting-start-tdd/elixir/10-higher-order-functions-and-pipeline.md +178 -0
  199. package/lib/assets/docs/article/getting-start-tdd/elixir/11-stream-and-lazy-evaluation.md +142 -0
  200. package/lib/assets/docs/article/getting-start-tdd/elixir/12-error-handling-and-with.md +145 -0
  201. package/lib/assets/docs/article/getting-start-tdd/elixir/index.md +35 -0
  202. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter01.md +202 -0
  203. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter02.md +246 -0
  204. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter03.md +218 -0
  205. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter04.md +179 -0
  206. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter05.md +267 -0
  207. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter06.md +190 -0
  208. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter07.md +161 -0
  209. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter08.md +175 -0
  210. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter09.md +222 -0
  211. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter10.md +189 -0
  212. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter11.md +212 -0
  213. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter12.md +215 -0
  214. package/lib/assets/docs/article/getting-start-tdd/fsharp/index.md +71 -0
  215. package/lib/assets/docs/article/getting-start-tdd/go/01-todo-list-and-first-test.md +213 -0
  216. package/lib/assets/docs/article/getting-start-tdd/go/02-fake-it-and-triangulation.md +302 -0
  217. package/lib/assets/docs/article/getting-start-tdd/go/03-obvious-implementation-and-refactoring.md +339 -0
  218. package/lib/assets/docs/article/getting-start-tdd/go/04-version-control-and-conventional-commits.md +112 -0
  219. package/lib/assets/docs/article/getting-start-tdd/go/05-package-management-and-static-analysis.md +272 -0
  220. package/lib/assets/docs/article/getting-start-tdd/go/06-task-runner-and-ci-cd.md +233 -0
  221. package/lib/assets/docs/article/getting-start-tdd/go/07-encapsulation-and-polymorphism.md +394 -0
  222. package/lib/assets/docs/article/getting-start-tdd/go/08-design-patterns.md +422 -0
  223. package/lib/assets/docs/article/getting-start-tdd/go/09-solid-principles-and-module-design.md +400 -0
  224. package/lib/assets/docs/article/getting-start-tdd/go/10-higher-order-functions-and-composition.md +226 -0
  225. package/lib/assets/docs/article/getting-start-tdd/go/11-immutable-data-and-pipeline.md +296 -0
  226. package/lib/assets/docs/article/getting-start-tdd/go/12-error-handling-and-type-safety.md +411 -0
  227. package/lib/assets/docs/article/getting-start-tdd/go/index.md +83 -0
  228. package/lib/assets/docs/article/getting-start-tdd/haskell/01-todo-list-and-first-test.md +279 -0
  229. package/lib/assets/docs/article/getting-start-tdd/haskell/02-fake-it-and-triangulation.md +337 -0
  230. package/lib/assets/docs/article/getting-start-tdd/haskell/03-obvious-implementation-and-refactoring.md +257 -0
  231. package/lib/assets/docs/article/getting-start-tdd/haskell/04-version-control-and-conventional-commits.md +182 -0
  232. package/lib/assets/docs/article/getting-start-tdd/haskell/05-package-management-and-static-analysis.md +313 -0
  233. package/lib/assets/docs/article/getting-start-tdd/haskell/06-task-runner-and-ci-cd.md +309 -0
  234. package/lib/assets/docs/article/getting-start-tdd/haskell/07-algebraic-data-types-and-type-classes.md +412 -0
  235. package/lib/assets/docs/article/getting-start-tdd/haskell/08-pattern-matching-and-guards.md +390 -0
  236. package/lib/assets/docs/article/getting-start-tdd/haskell/09-module-design-and-smart-constructors.md +461 -0
  237. package/lib/assets/docs/article/getting-start-tdd/haskell/10-higher-order-functions-and-currying.md +434 -0
  238. package/lib/assets/docs/article/getting-start-tdd/haskell/11-function-composition-and-point-free.md +392 -0
  239. package/lib/assets/docs/article/getting-start-tdd/haskell/12-monad-and-error-handling.md +631 -0
  240. package/lib/assets/docs/article/getting-start-tdd/haskell/index.md +49 -0
  241. package/lib/assets/docs/article/getting-start-tdd/index.md +93 -0
  242. package/lib/assets/docs/article/getting-start-tdd/integration/01-language-overview.md +375 -0
  243. package/lib/assets/docs/article/getting-start-tdd/integration/02-test-framework-comparison.md +349 -0
  244. package/lib/assets/docs/article/getting-start-tdd/integration/03-tdd-pattern-comparison.md +445 -0
  245. package/lib/assets/docs/article/getting-start-tdd/integration/04-type-system-comparison.md +405 -0
  246. package/lib/assets/docs/article/getting-start-tdd/integration/05-dev-environment-comparison.md +330 -0
  247. package/lib/assets/docs/article/getting-start-tdd/integration/06-learning-roadmap.md +274 -0
  248. package/lib/assets/docs/article/getting-start-tdd/integration/index.md +69 -0
  249. package/lib/assets/docs/article/getting-start-tdd/java/01-todo-list-and-first-test.md +234 -0
  250. package/lib/assets/docs/article/getting-start-tdd/java/02-fake-it-and-triangulation.md +261 -0
  251. package/lib/assets/docs/article/getting-start-tdd/java/03-obvious-implementation-and-refactoring.md +185 -0
  252. package/lib/assets/docs/article/getting-start-tdd/java/04-version-control-and-conventional-commits.md +115 -0
  253. package/lib/assets/docs/article/getting-start-tdd/java/05-package-management-and-static-analysis.md +382 -0
  254. package/lib/assets/docs/article/getting-start-tdd/java/06-task-runner-and-ci-cd.md +272 -0
  255. package/lib/assets/docs/article/getting-start-tdd/java/07-encapsulation-and-polymorphism.md +626 -0
  256. package/lib/assets/docs/article/getting-start-tdd/java/08-design-patterns.md +393 -0
  257. package/lib/assets/docs/article/getting-start-tdd/java/09-solid-principles-and-module-design.md +310 -0
  258. package/lib/assets/docs/article/getting-start-tdd/java/10-higher-order-functions-and-composition.md +188 -0
  259. package/lib/assets/docs/article/getting-start-tdd/java/11-immutable-data-and-pipeline.md +167 -0
  260. package/lib/assets/docs/article/getting-start-tdd/java/12-error-handling-and-type-safety.md +205 -0
  261. package/lib/assets/docs/article/getting-start-tdd/java/index.md +61 -0
  262. package/lib/assets/docs/article/getting-start-tdd/node/01-todo-list-and-first-test.md +244 -0
  263. package/lib/assets/docs/article/getting-start-tdd/node/02-fake-it-and-triangulation.md +262 -0
  264. package/lib/assets/docs/article/getting-start-tdd/node/03-obvious-implementation-and-refactoring.md +169 -0
  265. package/lib/assets/docs/article/getting-start-tdd/node/04-version-control-and-conventional-commits.md +112 -0
  266. package/lib/assets/docs/article/getting-start-tdd/node/05-package-management-and-static-analysis.md +314 -0
  267. package/lib/assets/docs/article/getting-start-tdd/node/06-task-runner-and-ci-cd.md +235 -0
  268. package/lib/assets/docs/article/getting-start-tdd/node/07-encapsulation-and-polymorphism.md +327 -0
  269. package/lib/assets/docs/article/getting-start-tdd/node/08-design-patterns.md +322 -0
  270. package/lib/assets/docs/article/getting-start-tdd/node/09-solid-principles-and-module-design.md +285 -0
  271. package/lib/assets/docs/article/getting-start-tdd/node/10-higher-order-functions-and-composition.md +199 -0
  272. package/lib/assets/docs/article/getting-start-tdd/node/11-immutable-data-and-pipeline.md +207 -0
  273. package/lib/assets/docs/article/getting-start-tdd/node/12-error-handling-and-type-safety.md +295 -0
  274. package/lib/assets/docs/article/getting-start-tdd/node/index.md +56 -0
  275. package/lib/assets/docs/article/getting-start-tdd/php/01-todo-list-and-first-test.md +259 -0
  276. package/lib/assets/docs/article/getting-start-tdd/php/02-fake-it-and-triangulation.md +200 -0
  277. package/lib/assets/docs/article/getting-start-tdd/php/03-obvious-implementation-and-refactoring.md +248 -0
  278. package/lib/assets/docs/article/getting-start-tdd/php/04-version-control-and-conventional-commits.md +141 -0
  279. package/lib/assets/docs/article/getting-start-tdd/php/05-package-management-and-static-analysis.md +410 -0
  280. package/lib/assets/docs/article/getting-start-tdd/php/06-task-runner-and-ci-cd.md +321 -0
  281. package/lib/assets/docs/article/getting-start-tdd/php/07-encapsulation-and-polymorphism.md +372 -0
  282. package/lib/assets/docs/article/getting-start-tdd/php/08-design-patterns.md +453 -0
  283. package/lib/assets/docs/article/getting-start-tdd/php/09-solid-principles-and-module-design.md +460 -0
  284. package/lib/assets/docs/article/getting-start-tdd/php/10-higher-order-functions-and-composition.md +182 -0
  285. package/lib/assets/docs/article/getting-start-tdd/php/11-immutable-data-and-pipeline.md +266 -0
  286. package/lib/assets/docs/article/getting-start-tdd/php/12-error-handling-and-type-safety.md +308 -0
  287. package/lib/assets/docs/article/getting-start-tdd/php/index.md +84 -0
  288. package/lib/assets/docs/article/getting-start-tdd/python/01-todo-list-and-first-test.md +201 -0
  289. package/lib/assets/docs/article/getting-start-tdd/python/02-fake-it-and-triangulation.md +247 -0
  290. package/lib/assets/docs/article/getting-start-tdd/python/03-obvious-implementation-and-refactoring.md +199 -0
  291. package/lib/assets/docs/article/getting-start-tdd/python/04-version-control-and-conventional-commits.md +87 -0
  292. package/lib/assets/docs/article/getting-start-tdd/python/05-package-management-and-static-analysis.md +274 -0
  293. package/lib/assets/docs/article/getting-start-tdd/python/06-task-runner-and-ci-cd.md +190 -0
  294. package/lib/assets/docs/article/getting-start-tdd/python/07-encapsulation-and-polymorphism.md +208 -0
  295. package/lib/assets/docs/article/getting-start-tdd/python/08-design-patterns.md +172 -0
  296. package/lib/assets/docs/article/getting-start-tdd/python/09-solid-principles-and-module-design.md +130 -0
  297. package/lib/assets/docs/article/getting-start-tdd/python/10-higher-order-functions-and-composition.md +122 -0
  298. package/lib/assets/docs/article/getting-start-tdd/python/11-immutable-data-and-pipeline.md +116 -0
  299. package/lib/assets/docs/article/getting-start-tdd/python/12-error-handling-and-type-safety.md +126 -0
  300. package/lib/assets/docs/article/getting-start-tdd/python/index.md +55 -0
  301. package/lib/assets/docs/article/getting-start-tdd/ruby/01-todo-list-and-first-test.md +231 -0
  302. package/lib/assets/docs/article/getting-start-tdd/ruby/02-fake-it-and-triangulation.md +238 -0
  303. package/lib/assets/docs/article/getting-start-tdd/ruby/03-obvious-implementation-and-refactoring.md +228 -0
  304. package/lib/assets/docs/article/getting-start-tdd/ruby/04-version-control-and-conventional-commits.md +112 -0
  305. package/lib/assets/docs/article/getting-start-tdd/ruby/05-package-management-and-static-analysis.md +287 -0
  306. package/lib/assets/docs/article/getting-start-tdd/ruby/06-task-runner-and-ci-cd.md +248 -0
  307. package/lib/assets/docs/article/getting-start-tdd/ruby/07-encapsulation-and-polymorphism.md +279 -0
  308. package/lib/assets/docs/article/getting-start-tdd/ruby/08-design-patterns.md +329 -0
  309. package/lib/assets/docs/article/getting-start-tdd/ruby/09-solid-principles-and-module-design.md +196 -0
  310. package/lib/assets/docs/article/getting-start-tdd/ruby/10-higher-order-functions-and-composition.md +175 -0
  311. package/lib/assets/docs/article/getting-start-tdd/ruby/11-immutable-data-and-pipeline.md +233 -0
  312. package/lib/assets/docs/article/getting-start-tdd/ruby/12-error-handling-and-type-safety.md +398 -0
  313. package/lib/assets/docs/article/getting-start-tdd/ruby/index.md +83 -0
  314. package/lib/assets/docs/article/getting-start-tdd/rust/01-todo-list-and-first-test.md +211 -0
  315. package/lib/assets/docs/article/getting-start-tdd/rust/02-fake-it-and-triangulation.md +264 -0
  316. package/lib/assets/docs/article/getting-start-tdd/rust/03-obvious-implementation-and-refactoring.md +233 -0
  317. package/lib/assets/docs/article/getting-start-tdd/rust/04-version-control-and-conventional-commits.md +92 -0
  318. package/lib/assets/docs/article/getting-start-tdd/rust/05-package-management-and-static-analysis.md +212 -0
  319. package/lib/assets/docs/article/getting-start-tdd/rust/06-task-runner-and-ci-cd.md +164 -0
  320. package/lib/assets/docs/article/getting-start-tdd/rust/07-encapsulation-and-polymorphism.md +142 -0
  321. package/lib/assets/docs/article/getting-start-tdd/rust/08-design-patterns.md +145 -0
  322. package/lib/assets/docs/article/getting-start-tdd/rust/09-solid-principles-and-module-design.md +110 -0
  323. package/lib/assets/docs/article/getting-start-tdd/rust/10-higher-order-functions-and-composition.md +94 -0
  324. package/lib/assets/docs/article/getting-start-tdd/rust/11-immutable-data-and-pipeline.md +105 -0
  325. package/lib/assets/docs/article/getting-start-tdd/rust/12-error-handling-and-type-safety.md +112 -0
  326. package/lib/assets/docs/article/getting-start-tdd/rust/index.md +83 -0
  327. package/lib/assets/docs/article/getting-start-tdd/scala/01-todo-list-and-first-test.md +111 -0
  328. package/lib/assets/docs/article/getting-start-tdd/scala/02-fake-it-and-triangulation.md +107 -0
  329. package/lib/assets/docs/article/getting-start-tdd/scala/03-obvious-implementation-and-refactoring.md +99 -0
  330. package/lib/assets/docs/article/getting-start-tdd/scala/04-version-control-and-conventional-commits.md +123 -0
  331. package/lib/assets/docs/article/getting-start-tdd/scala/05-package-management-and-static-analysis.md +196 -0
  332. package/lib/assets/docs/article/getting-start-tdd/scala/06-task-runner-and-ci-cd.md +186 -0
  333. package/lib/assets/docs/article/getting-start-tdd/scala/07-case-classes-and-traits.md +139 -0
  334. package/lib/assets/docs/article/getting-start-tdd/scala/08-pattern-matching-and-sealed-traits.md +106 -0
  335. package/lib/assets/docs/article/getting-start-tdd/scala/09-packages-and-module-design.md +75 -0
  336. package/lib/assets/docs/article/getting-start-tdd/scala/10-higher-order-functions-and-composition.md +104 -0
  337. package/lib/assets/docs/article/getting-start-tdd/scala/11-collections-and-lazy-evaluation.md +94 -0
  338. package/lib/assets/docs/article/getting-start-tdd/scala/12-error-handling-and-type-safety.md +92 -0
  339. package/lib/assets/docs/article/getting-start-tdd/scala/index.md +65 -0
  340. package/lib/assets/docs/article/grokking-concurrency/all/index.md +404 -0
  341. package/lib/assets/docs/article/grokking-concurrency/all/part-1-ch02-sequential.md +554 -0
  342. package/lib/assets/docs/article/grokking-concurrency/all/part-2-ch04-05-threads.md +469 -0
  343. package/lib/assets/docs/article/grokking-concurrency/all/part-3-ch06-multitasking.md +520 -0
  344. package/lib/assets/docs/article/grokking-concurrency/all/part-4-ch07-parallel-patterns.md +420 -0
  345. package/lib/assets/docs/article/grokking-concurrency/all/part-5-ch08-09-synchronization.md +510 -0
  346. package/lib/assets/docs/article/grokking-concurrency/all/part-6-ch10-11-nonblocking-io.md +435 -0
  347. package/lib/assets/docs/article/grokking-concurrency/all/part-7-ch12-async.md +465 -0
  348. package/lib/assets/docs/article/grokking-concurrency/all/part-8-ch13-mapreduce.md +377 -0
  349. package/lib/assets/docs/article/grokking-concurrency/clojure/index.md +116 -0
  350. package/lib/assets/docs/article/grokking-concurrency/clojure/part-1.md +108 -0
  351. package/lib/assets/docs/article/grokking-concurrency/clojure/part-2.md +101 -0
  352. package/lib/assets/docs/article/grokking-concurrency/clojure/part-3.md +122 -0
  353. package/lib/assets/docs/article/grokking-concurrency/clojure/part-4.md +123 -0
  354. package/lib/assets/docs/article/grokking-concurrency/clojure/part-5.md +118 -0
  355. package/lib/assets/docs/article/grokking-concurrency/clojure/part-6.md +89 -0
  356. package/lib/assets/docs/article/grokking-concurrency/clojure/part-7.md +100 -0
  357. package/lib/assets/docs/article/grokking-concurrency/clojure/part-8.md +120 -0
  358. package/lib/assets/docs/article/grokking-concurrency/csharp/index.md +101 -0
  359. package/lib/assets/docs/article/grokking-concurrency/csharp/part-1.md +97 -0
  360. package/lib/assets/docs/article/grokking-concurrency/csharp/part-2.md +123 -0
  361. package/lib/assets/docs/article/grokking-concurrency/csharp/part-3.md +101 -0
  362. package/lib/assets/docs/article/grokking-concurrency/csharp/part-4.md +112 -0
  363. package/lib/assets/docs/article/grokking-concurrency/csharp/part-5.md +99 -0
  364. package/lib/assets/docs/article/grokking-concurrency/csharp/part-6.md +61 -0
  365. package/lib/assets/docs/article/grokking-concurrency/csharp/part-7.md +84 -0
  366. package/lib/assets/docs/article/grokking-concurrency/csharp/part-8.md +92 -0
  367. package/lib/assets/docs/article/grokking-concurrency/fsharp/index.md +65 -0
  368. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-1.md +80 -0
  369. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-2.md +103 -0
  370. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-3.md +94 -0
  371. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-4.md +110 -0
  372. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-5.md +104 -0
  373. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-6.md +93 -0
  374. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-7.md +121 -0
  375. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-8.md +107 -0
  376. package/lib/assets/docs/article/grokking-concurrency/haskell/index.md +248 -0
  377. package/lib/assets/docs/article/grokking-concurrency/haskell/part-1.md +96 -0
  378. package/lib/assets/docs/article/grokking-concurrency/haskell/part-2.md +96 -0
  379. package/lib/assets/docs/article/grokking-concurrency/haskell/part-3.md +91 -0
  380. package/lib/assets/docs/article/grokking-concurrency/haskell/part-4.md +106 -0
  381. package/lib/assets/docs/article/grokking-concurrency/haskell/part-5.md +99 -0
  382. package/lib/assets/docs/article/grokking-concurrency/haskell/part-6.md +95 -0
  383. package/lib/assets/docs/article/grokking-concurrency/haskell/part-7.md +111 -0
  384. package/lib/assets/docs/article/grokking-concurrency/haskell/part-8.md +118 -0
  385. package/lib/assets/docs/article/grokking-concurrency/index.md +66 -0
  386. package/lib/assets/docs/article/grokking-concurrency/java/index.md +102 -0
  387. package/lib/assets/docs/article/grokking-concurrency/java/part-1.md +308 -0
  388. package/lib/assets/docs/article/grokking-concurrency/java/part-2.md +334 -0
  389. package/lib/assets/docs/article/grokking-concurrency/java/part-3.md +221 -0
  390. package/lib/assets/docs/article/grokking-concurrency/java/part-4.md +213 -0
  391. package/lib/assets/docs/article/grokking-concurrency/java/part-5.md +112 -0
  392. package/lib/assets/docs/article/grokking-concurrency/java/part-6.md +69 -0
  393. package/lib/assets/docs/article/grokking-concurrency/java/part-7.md +101 -0
  394. package/lib/assets/docs/article/grokking-concurrency/java/part-8.md +101 -0
  395. package/lib/assets/docs/article/grokking-concurrency/python/index.md +313 -0
  396. package/lib/assets/docs/article/grokking-concurrency/python/part-1.md +239 -0
  397. package/lib/assets/docs/article/grokking-concurrency/python/part-2.md +418 -0
  398. package/lib/assets/docs/article/grokking-concurrency/python/part-3.md +227 -0
  399. package/lib/assets/docs/article/grokking-concurrency/python/part-4.md +299 -0
  400. package/lib/assets/docs/article/grokking-concurrency/python/part-5.md +315 -0
  401. package/lib/assets/docs/article/grokking-concurrency/python/part-6.md +297 -0
  402. package/lib/assets/docs/article/grokking-concurrency/python/part-7.md +314 -0
  403. package/lib/assets/docs/article/grokking-concurrency/python/part-8.md +360 -0
  404. package/lib/assets/docs/article/grokking-concurrency/rust/index.md +270 -0
  405. package/lib/assets/docs/article/grokking-concurrency/rust/part-1.md +108 -0
  406. package/lib/assets/docs/article/grokking-concurrency/rust/part-2.md +120 -0
  407. package/lib/assets/docs/article/grokking-concurrency/rust/part-3.md +126 -0
  408. package/lib/assets/docs/article/grokking-concurrency/rust/part-4.md +175 -0
  409. package/lib/assets/docs/article/grokking-concurrency/rust/part-5.md +158 -0
  410. package/lib/assets/docs/article/grokking-concurrency/rust/part-6.md +94 -0
  411. package/lib/assets/docs/article/grokking-concurrency/rust/part-7.md +133 -0
  412. package/lib/assets/docs/article/grokking-concurrency/rust/part-8.md +155 -0
  413. package/lib/assets/docs/article/grokking-concurrency/scala/index.md +69 -0
  414. package/lib/assets/docs/article/grokking-concurrency/scala/part-1.md +78 -0
  415. package/lib/assets/docs/article/grokking-concurrency/scala/part-2.md +112 -0
  416. package/lib/assets/docs/article/grokking-concurrency/scala/part-3.md +93 -0
  417. package/lib/assets/docs/article/grokking-concurrency/scala/part-4.md +110 -0
  418. package/lib/assets/docs/article/grokking-concurrency/scala/part-5.md +119 -0
  419. package/lib/assets/docs/article/grokking-concurrency/scala/part-6.md +83 -0
  420. package/lib/assets/docs/article/grokking-concurrency/scala/part-7.md +131 -0
  421. package/lib/assets/docs/article/grokking-concurrency/scala/part-8.md +129 -0
  422. package/lib/assets/docs/article/grokkingfp/all/index.md +368 -0
  423. package/lib/assets/docs/article/grokkingfp/all/part-1-ch01-fp-introduction.md +530 -0
  424. package/lib/assets/docs/article/grokkingfp/all/part-1-ch02-pure-functions.md +923 -0
  425. package/lib/assets/docs/article/grokkingfp/all/part-2-ch03-immutable-data.md +1122 -0
  426. package/lib/assets/docs/article/grokkingfp/all/part-2-ch04-higher-order-functions.md +1104 -0
  427. package/lib/assets/docs/article/grokkingfp/all/part-2-ch05-flatmap.md +1026 -0
  428. package/lib/assets/docs/article/grokkingfp/all/part-3-ch06-option.md +777 -0
  429. package/lib/assets/docs/article/grokkingfp/all/part-3-ch07-either-adt.md +871 -0
  430. package/lib/assets/docs/article/grokkingfp/all/part-4-ch08-io-monad.md +972 -0
  431. package/lib/assets/docs/article/grokkingfp/all/part-4-ch09-streams.md +926 -0
  432. package/lib/assets/docs/article/grokkingfp/all/part-5-ch10-concurrency.md +870 -0
  433. package/lib/assets/docs/article/grokkingfp/all/part-6-ch11-application.md +715 -0
  434. package/lib/assets/docs/article/grokkingfp/all/part-6-ch12-testing.md +626 -0
  435. package/lib/assets/docs/article/grokkingfp/all/writing-plan.md +696 -0
  436. package/lib/assets/docs/article/grokkingfp/clojure/index.md +276 -0
  437. package/lib/assets/docs/article/grokkingfp/clojure/part-1.md +667 -0
  438. package/lib/assets/docs/article/grokkingfp/clojure/part-2.md +643 -0
  439. package/lib/assets/docs/article/grokkingfp/clojure/part-3.md +620 -0
  440. package/lib/assets/docs/article/grokkingfp/clojure/part-4.md +697 -0
  441. package/lib/assets/docs/article/grokkingfp/clojure/part-5.md +751 -0
  442. package/lib/assets/docs/article/grokkingfp/clojure/part-6.md +721 -0
  443. package/lib/assets/docs/article/grokkingfp/csharp/index.md +246 -0
  444. package/lib/assets/docs/article/grokkingfp/csharp/part-1.md +811 -0
  445. package/lib/assets/docs/article/grokkingfp/csharp/part-2.md +971 -0
  446. package/lib/assets/docs/article/grokkingfp/csharp/part-3.md +981 -0
  447. package/lib/assets/docs/article/grokkingfp/csharp/part-4.md +949 -0
  448. package/lib/assets/docs/article/grokkingfp/csharp/part-5.md +947 -0
  449. package/lib/assets/docs/article/grokkingfp/csharp/part-6.md +739 -0
  450. package/lib/assets/docs/article/grokkingfp/elixir/index.md +203 -0
  451. package/lib/assets/docs/article/grokkingfp/elixir/part-1.md +710 -0
  452. package/lib/assets/docs/article/grokkingfp/elixir/part-2.md +838 -0
  453. package/lib/assets/docs/article/grokkingfp/elixir/part-3.md +985 -0
  454. package/lib/assets/docs/article/grokkingfp/elixir/part-4.md +974 -0
  455. package/lib/assets/docs/article/grokkingfp/elixir/part-5.md +1284 -0
  456. package/lib/assets/docs/article/grokkingfp/elixir/part-6.md +1047 -0
  457. package/lib/assets/docs/article/grokkingfp/fsharp/index.md +210 -0
  458. package/lib/assets/docs/article/grokkingfp/fsharp/part-1.md +714 -0
  459. package/lib/assets/docs/article/grokkingfp/fsharp/part-2.md +961 -0
  460. package/lib/assets/docs/article/grokkingfp/fsharp/part-3.md +972 -0
  461. package/lib/assets/docs/article/grokkingfp/fsharp/part-4.md +832 -0
  462. package/lib/assets/docs/article/grokkingfp/fsharp/part-5.md +911 -0
  463. package/lib/assets/docs/article/grokkingfp/fsharp/part-6.md +920 -0
  464. package/lib/assets/docs/article/grokkingfp/haskell/index.md +234 -0
  465. package/lib/assets/docs/article/grokkingfp/haskell/part-1.md +591 -0
  466. package/lib/assets/docs/article/grokkingfp/haskell/part-2.md +866 -0
  467. package/lib/assets/docs/article/grokkingfp/haskell/part-3.md +915 -0
  468. package/lib/assets/docs/article/grokkingfp/haskell/part-4.md +876 -0
  469. package/lib/assets/docs/article/grokkingfp/haskell/part-5.md +845 -0
  470. package/lib/assets/docs/article/grokkingfp/haskell/part-6.md +842 -0
  471. package/lib/assets/docs/article/grokkingfp/index.md +143 -0
  472. package/lib/assets/docs/article/grokkingfp/java/index.md +211 -0
  473. package/lib/assets/docs/article/grokkingfp/java/part-1.md +646 -0
  474. package/lib/assets/docs/article/grokkingfp/java/part-2.md +667 -0
  475. package/lib/assets/docs/article/grokkingfp/java/part-3.md +672 -0
  476. package/lib/assets/docs/article/grokkingfp/java/part-4.md +771 -0
  477. package/lib/assets/docs/article/grokkingfp/java/part-5.md +959 -0
  478. package/lib/assets/docs/article/grokkingfp/java/part-6.md +1324 -0
  479. package/lib/assets/docs/article/grokkingfp/python/index.md +258 -0
  480. package/lib/assets/docs/article/grokkingfp/python/part-1.md +437 -0
  481. package/lib/assets/docs/article/grokkingfp/python/part-2.md +958 -0
  482. package/lib/assets/docs/article/grokkingfp/python/part-3.md +1004 -0
  483. package/lib/assets/docs/article/grokkingfp/python/part-4.md +765 -0
  484. package/lib/assets/docs/article/grokkingfp/python/part-5.md +747 -0
  485. package/lib/assets/docs/article/grokkingfp/python/part-6.md +861 -0
  486. package/lib/assets/docs/article/grokkingfp/ruby/index.md +330 -0
  487. package/lib/assets/docs/article/grokkingfp/ruby/part-1.md +753 -0
  488. package/lib/assets/docs/article/grokkingfp/ruby/part-2.md +938 -0
  489. package/lib/assets/docs/article/grokkingfp/ruby/part-3.md +946 -0
  490. package/lib/assets/docs/article/grokkingfp/ruby/part-4.md +921 -0
  491. package/lib/assets/docs/article/grokkingfp/ruby/part-5.md +908 -0
  492. package/lib/assets/docs/article/grokkingfp/ruby/part-6.md +1410 -0
  493. package/lib/assets/docs/article/grokkingfp/rust/index.md +242 -0
  494. package/lib/assets/docs/article/grokkingfp/rust/part-1.md +634 -0
  495. package/lib/assets/docs/article/grokkingfp/rust/part-2.md +1060 -0
  496. package/lib/assets/docs/article/grokkingfp/rust/part-3.md +994 -0
  497. package/lib/assets/docs/article/grokkingfp/rust/part-4.md +571 -0
  498. package/lib/assets/docs/article/grokkingfp/rust/part-5.md +705 -0
  499. package/lib/assets/docs/article/grokkingfp/rust/part-6.md +508 -0
  500. package/lib/assets/docs/article/grokkingfp/scala/index.md +171 -0
  501. package/lib/assets/docs/article/grokkingfp/scala/part-1.md +541 -0
  502. package/lib/assets/docs/article/grokkingfp/scala/part-2.md +946 -0
  503. package/lib/assets/docs/article/grokkingfp/scala/part-3.md +917 -0
  504. package/lib/assets/docs/article/grokkingfp/scala/part-4.md +742 -0
  505. package/lib/assets/docs/article/grokkingfp/scala/part-5.md +722 -0
  506. package/lib/assets/docs/article/grokkingfp/scala/part-6.md +865 -0
  507. package/lib/assets/docs/article/grokkingfp/typescript/index.md +273 -0
  508. package/lib/assets/docs/article/grokkingfp/typescript/part-1.md +559 -0
  509. package/lib/assets/docs/article/grokkingfp/typescript/part-2.md +1129 -0
  510. package/lib/assets/docs/article/grokkingfp/typescript/part-3.md +842 -0
  511. package/lib/assets/docs/article/grokkingfp/typescript/part-4.md +1085 -0
  512. package/lib/assets/docs/article/grokkingfp/typescript/part-5.md +717 -0
  513. package/lib/assets/docs/article/grokkingfp/typescript/part-6.md +980 -0
  514. package/lib/assets/docs/article/index.md +36 -0
  515. package/lib/assets/docs/design/index.md +39 -27
  516. package/lib/assets/docs/development/index.md +11 -1
  517. package/lib/assets/docs/index.md +33 -106
  518. package/lib/assets/docs/operation/index.md +16 -6
  519. package/lib/assets/docs/reference/index.md +5 -4
  520. package/lib/assets/docs/requirements/index.md +13 -6
  521. package/lib/assets/docs/strategy/index.md +11 -4
  522. package/lib/assets/docs/template/index.md +9 -5
  523. package/lib/assets/mkdocs.yml +29 -17
  524. package/package.json +1 -1
@@ -0,0 +1,350 @@
1
+ # 第4章: Clojure Spec による仕様定義
2
+
3
+ ## はじめに
4
+
5
+ Clojure Spec は、データ構造と関数の仕様を定義するための強力なライブラリです。型システムとは異なり、Spec は実行時にデータを検証し、自動テストを生成し、ドキュメントとしても機能します。
6
+
7
+ 本章では、Spec を使ってデータの仕様を定義し、関数の契約を表現し、自動テスト生成を活用する方法を学びます。
8
+
9
+ ## 1. 基本的なスペック定義
10
+
11
+ ### スペックとは
12
+
13
+ スペックは、データが満たすべき条件を宣言的に定義します。
14
+
15
+ ```clojure
16
+ (require '[clojure.spec.alpha :as s])
17
+
18
+ ;; シンプルなスペック
19
+ (s/def ::name (s/and string? #(< 0 (count %) 100)))
20
+ (s/def ::age (s/and int? #(<= 0 % 150)))
21
+ (s/def ::email (s/and string? #(re-matches #".+@.+\..+" %)))
22
+
23
+ ;; 列挙型のスペック
24
+ (s/def ::membership #{:bronze :silver :gold :platinum})
25
+ (s/def ::status #{:active :inactive :suspended})
26
+ ```
27
+
28
+ ### スペックの検証
29
+
30
+ ```clojure
31
+ ;; valid? で検証
32
+ (s/valid? ::name "田中太郎") ;; => true
33
+ (s/valid? ::name "") ;; => false(空文字列)
34
+ (s/valid? ::age 25) ;; => true
35
+ (s/valid? ::age -1) ;; => false(負の数)
36
+
37
+ ;; explain でエラー詳細を表示
38
+ (s/explain ::age -1)
39
+ ;; val: -1 fails spec: :user/age predicate: (<= 0 % 150)
40
+ ```
41
+
42
+ ## 2. コレクションのスペック
43
+
44
+ ### ベクターとリスト
45
+
46
+ ```clojure
47
+ ;; ベクターのスペック
48
+ (s/def ::tags (s/coll-of string? :kind vector? :min-count 0 :max-count 10))
49
+
50
+ (s/valid? ::tags ["tag1" "tag2"]) ;; => true
51
+ (s/valid? ::tags '("tag1")) ;; => false(リストは不可)
52
+ ```
53
+
54
+ ### マップのスペック
55
+
56
+ ```clojure
57
+ ;; 必須キーとオプションキー
58
+ (s/def ::person
59
+ (s/keys :req-un [::name ::age]
60
+ :opt-un [::email ::membership]))
61
+
62
+ (s/valid? ::person {:name "田中" :age 30}) ;; => true
63
+ (s/valid? ::person {:name "田中" :age 30 :email "tanaka@example.com"}) ;; => true
64
+ (s/valid? ::person {:name "田中"}) ;; => false(age が欠落)
65
+ ```
66
+
67
+ ### 入れ子のマップ
68
+
69
+ ```clojure
70
+ (s/def ::street string?)
71
+ (s/def ::city string?)
72
+ (s/def ::postal-code (s/and string? #(re-matches #"\d{3}-\d{4}" %)))
73
+
74
+ (s/def ::address
75
+ (s/keys :req-un [::street ::city ::postal-code]))
76
+
77
+ (s/def ::person-with-address
78
+ (s/keys :req-un [::name ::age ::address]))
79
+
80
+ (s/valid? ::person-with-address
81
+ {:name "田中"
82
+ :age 30
83
+ :address {:street "東京都渋谷区1-1-1"
84
+ :city "渋谷区"
85
+ :postal-code "150-0001"}})
86
+ ;; => true
87
+ ```
88
+
89
+ ## 3. ドメインモデルの定義
90
+
91
+ ### 商品と注文
92
+
93
+ ```clojure
94
+ ;; 商品
95
+ (s/def ::product-id (s/and string? #(re-matches #"PROD-\d{5}" %)))
96
+ (s/def ::product-name (s/and string? #(< 0 (count %) 200)))
97
+ (s/def ::price (s/and number? pos?))
98
+ (s/def ::quantity (s/and int? pos?))
99
+
100
+ (s/def ::product
101
+ (s/keys :req-un [::product-id ::product-name ::price]
102
+ :opt-un [::description ::category]))
103
+
104
+ ;; 注文アイテム
105
+ (s/def ::order-item
106
+ (s/keys :req-un [::product-id ::quantity ::price]))
107
+
108
+ ;; 注文
109
+ (s/def ::order-id (s/and string? #(re-matches #"ORD-\d{8}" %)))
110
+ (s/def ::customer-id (s/and string? #(re-matches #"CUST-\d{6}" %)))
111
+ (s/def ::items (s/coll-of ::order-item :min-count 1))
112
+
113
+ (s/def ::order
114
+ (s/keys :req-un [::order-id ::customer-id ::items ::order-date]
115
+ :opt-un [::total ::status]))
116
+ ```
117
+
118
+ ## 4. 関数仕様の定義(fdef)
119
+
120
+ ### 基本的な fdef
121
+
122
+ `s/fdef` を使って関数の引数、戻り値、および引数と戻り値の関係を定義します。
123
+
124
+ ```clojure
125
+ (defn calculate-item-total
126
+ "注文アイテムの合計を計算する"
127
+ [{:keys [price quantity]}]
128
+ (* price quantity))
129
+
130
+ (s/fdef calculate-item-total
131
+ :args (s/cat :item ::order-item)
132
+ :ret number?
133
+ :fn (s/and #(pos? (:ret %))
134
+ #(= (:ret %)
135
+ (* (get-in % [:args :item :price])
136
+ (get-in % [:args :item :quantity])))))
137
+ ```
138
+
139
+ ### fdef の構成要素
140
+
141
+ - **:args**: 引数のスペック(`s/cat` で名前付き引数を定義)
142
+ - **:ret**: 戻り値のスペック
143
+ - **:fn**: 引数と戻り値の関係を定義する述語
144
+
145
+ ```clojure
146
+ (defn apply-discount
147
+ "割引を適用する"
148
+ [total discount-rate]
149
+ (* total (- 1 discount-rate)))
150
+
151
+ (s/fdef apply-discount
152
+ :args (s/cat :total ::total :discount-rate (s/and number? #(<= 0 % 1)))
153
+ :ret number?
154
+ :fn #(<= (:ret %) (get-in % [:args :total]))) ;; 戻り値は元の値以下
155
+ ```
156
+
157
+ ## 5. 多引数と可変長引数
158
+
159
+ ### オーバーロードされた関数
160
+
161
+ ```clojure
162
+ (defn create-person
163
+ "人物を作成する"
164
+ ([name age]
165
+ {:name name :age age})
166
+ ([name age email]
167
+ {:name name :age age :email email}))
168
+
169
+ (s/fdef create-person
170
+ :args (s/alt :two-args (s/cat :name ::name :age ::age)
171
+ :three-args (s/cat :name ::name :age ::age :email ::email))
172
+ :ret ::person)
173
+ ```
174
+
175
+ ### 可変長引数
176
+
177
+ ```clojure
178
+ (defn sum-prices
179
+ "複数の価格を合計する"
180
+ [& prices]
181
+ (reduce + 0 prices))
182
+
183
+ (s/fdef sum-prices
184
+ :args (s/* ::price)
185
+ :ret number?)
186
+ ```
187
+
188
+ ## 6. 条件付きスペック(マルチスペック)
189
+
190
+ データの内容に応じて異なるスペックを適用する場合は、マルチスペックを使用します。
191
+
192
+ ```clojure
193
+ (s/def ::notification-type #{:email :sms :push})
194
+
195
+ (defmulti notification-spec :type)
196
+
197
+ (defmethod notification-spec :email [_]
198
+ (s/keys :req-un [::type ::to ::subject ::body]))
199
+
200
+ (defmethod notification-spec :sms [_]
201
+ (s/keys :req-un [::type ::phone-number ::body]))
202
+
203
+ (defmethod notification-spec :push [_]
204
+ (s/keys :req-un [::type ::device-token ::body]))
205
+
206
+ (s/def ::notification (s/multi-spec notification-spec :type))
207
+
208
+ ;; 検証
209
+ (s/valid? ::notification
210
+ {:type :email
211
+ :to "test@example.com"
212
+ :subject "テスト"
213
+ :body "本文"})
214
+ ;; => true
215
+
216
+ (s/valid? ::notification
217
+ {:type :sms
218
+ :phone-number "090-1234-5678"
219
+ :body "本文"})
220
+ ;; => true
221
+ ```
222
+
223
+ ## 7. カスタムジェネレータ
224
+
225
+ スペックに対応するテストデータを自動生成するジェネレータを定義できます。
226
+
227
+ ```clojure
228
+ (require '[clojure.spec.gen.alpha :as gen])
229
+
230
+ (defn product-id-gen
231
+ "商品IDのジェネレータ"
232
+ []
233
+ (gen/fmap #(str "PROD-" (format "%05d" %))
234
+ (gen/choose 0 99999)))
235
+
236
+ (s/def ::product-id-with-gen
237
+ (s/with-gen
238
+ (s/and string? #(re-matches #"PROD-\d{5}" %))
239
+ product-id-gen))
240
+
241
+ ;; サンプル生成
242
+ (gen/sample (s/gen ::product-id-with-gen) 5)
243
+ ;; => ("PROD-00042" "PROD-00001" "PROD-12345" "PROD-00000" "PROD-99999")
244
+ ```
245
+
246
+ ### メールアドレスのジェネレータ
247
+
248
+ ```clojure
249
+ (defn email-gen
250
+ "メールアドレスのジェネレータ"
251
+ []
252
+ (gen/fmap (fn [[user domain]]
253
+ (str user "@" domain ".com"))
254
+ (gen/tuple (gen/such-that #(not (empty? %))
255
+ (gen/string-alphanumeric))
256
+ (gen/such-that #(not (empty? %))
257
+ (gen/string-alphanumeric)))))
258
+ ```
259
+
260
+ ## 8. バリデーションとエラーハンドリング
261
+
262
+ ### バリデーション関数
263
+
264
+ ```clojure
265
+ (defn validate-person
266
+ "人物データを検証し、結果を返す"
267
+ [person]
268
+ (if (s/valid? ::person person)
269
+ {:valid true :data person}
270
+ {:valid false
271
+ :errors (s/explain-data ::person person)}))
272
+
273
+ (validate-person {:name "田中" :age 30})
274
+ ;; => {:valid true, :data {:name "田中", :age 30}}
275
+
276
+ (validate-person {:name "" :age 30})
277
+ ;; => {:valid false, :errors {...}}
278
+ ```
279
+
280
+ ### conform による値の変換
281
+
282
+ ```clojure
283
+ (defn conform-or-throw
284
+ "スペックに適合しない場合は例外をスロー"
285
+ [spec data]
286
+ (let [conformed (s/conform spec data)]
287
+ (if (= conformed ::s/invalid)
288
+ (throw (ex-info "Validation failed"
289
+ {:spec spec
290
+ :data data
291
+ :problems (s/explain-data spec data)}))
292
+ conformed)))
293
+ ```
294
+
295
+ ## 9. インストルメンテーション
296
+
297
+ 開発時に関数の引数を自動検証できます。
298
+
299
+ ```clojure
300
+ (require '[clojure.spec.test.alpha :as stest])
301
+
302
+ ;; インストルメンテーションを有効化
303
+ (stest/instrument)
304
+
305
+ ;; これ以降、fdef が定義された関数は引数が自動検証される
306
+ (calculate-item-total {:product-id "INVALID" :quantity -1 :price 0})
307
+ ;; => ExceptionInfo: Call to #'user/calculate-item-total did not conform to spec
308
+
309
+ ;; 無効化
310
+ (stest/unstrument)
311
+ ```
312
+
313
+ ## 10. テスト生成
314
+
315
+ `check` 関数で自動的にテストを生成・実行できます。
316
+
317
+ ```clojure
318
+ (stest/check `calculate-item-total)
319
+ ;; 100個のランダムな入力でテストを実行
320
+
321
+ (stest/check `apply-discount {:clojure.spec.test.check/opts {:num-tests 1000}})
322
+ ;; 1000個のテストケースで実行
323
+ ```
324
+
325
+ ## まとめ
326
+
327
+ 本章では、Clojure Spec について学びました:
328
+
329
+ 1. **基本的なスペック**: 述語と組み合わせたデータ検証
330
+ 2. **コレクションスペック**: ベクター、マップの構造定義
331
+ 3. **ドメインモデル**: ビジネスデータの仕様定義
332
+ 4. **fdef**: 関数の契約(引数、戻り値、関係)
333
+ 5. **マルチスペック**: 条件付きの仕様定義
334
+ 6. **カスタムジェネレータ**: テストデータの自動生成
335
+ 7. **バリデーション**: 実行時検証とエラーハンドリング
336
+ 8. **インストルメンテーション**: 開発時の自動検証
337
+ 9. **テスト生成**: 自動プロパティベーステスト
338
+
339
+ Spec を活用することで、動的型付け言語の柔軟性を保ちながら、堅牢なデータ検証とドキュメント化を実現できます。
340
+
341
+ ## 参考コード
342
+
343
+ 本章のコード例は以下のファイルで確認できます:
344
+
345
+ - ソースコード: `apps/clojure/part2/src/clojure_spec.clj`
346
+ - テストコード: `apps/clojure/part2/spec/clojure_spec_spec.clj`
347
+
348
+ ## 次章予告
349
+
350
+ 次章では、**プロパティベーステスト**について学びます。test.check を使った生成的テストの手法を探ります。
@@ -0,0 +1,352 @@
1
+ # 第5章: プロパティベーステスト
2
+
3
+ ## はじめに
4
+
5
+ 従来の単体テストでは、特定の入力に対する期待される出力を検証します。一方、**プロパティベーステスト**では、すべての入力に対して成り立つべき「性質(プロパティ)」を定義し、ランダムに生成された多数のテストケースで検証します。
6
+
7
+ 本章では、Clojure の `test.check` ライブラリを使ったプロパティベーステストの手法を学びます。
8
+
9
+ ## 1. プロパティベーステストとは
10
+
11
+ ### 従来のテストとの違い
12
+
13
+ ```clojure
14
+ ;; 従来のテスト:特定の入力に対する出力を検証
15
+ (deftest test-reverse
16
+ (is (= "olleh" (reverse-string "hello")))
17
+ (is (= "" (reverse-string "")))
18
+ (is (= "a" (reverse-string "a"))))
19
+
20
+ ;; プロパティベーステスト:性質を検証
21
+ (def prop-reverse-involutory
22
+ "文字列を2回反転すると元に戻る"
23
+ (prop/for-all [s gen/string]
24
+ (= s (reverse-string (reverse-string s)))))
25
+ ```
26
+
27
+ ### プロパティベーステストの利点
28
+
29
+ 1. **網羅性**: 手動では思いつかないエッジケースを発見
30
+ 2. **ドキュメント性**: コードの性質を明確に表現
31
+ 3. **回帰防止**: リファクタリング時の安全網
32
+ 4. **シュリンキング**: 失敗時に最小の反例を提示
33
+
34
+ ## 2. 基本的なジェネレータ
35
+
36
+ ### プリミティブジェネレータ
37
+
38
+ ```clojure
39
+ (require '[clojure.test.check.generators :as gen])
40
+
41
+ ;; 整数
42
+ gen/small-integer ;; -200〜200程度の整数
43
+ gen/nat ;; 自然数(0以上)
44
+ gen/pos-int ;; 正の整数
45
+ gen/neg-int ;; 負の整数
46
+
47
+ ;; 文字列
48
+ gen/string ;; 任意の文字列
49
+ gen/string-alphanumeric ;; 英数字のみ
50
+
51
+ ;; その他
52
+ gen/boolean ;; true/false
53
+ gen/char ;; 文字
54
+ gen/keyword ;; キーワード
55
+
56
+ ;; サンプル生成
57
+ (gen/sample gen/small-integer 5)
58
+ ;; => (0 1 -2 1 4)
59
+ ```
60
+
61
+ ### 範囲と列挙
62
+
63
+ ```clojure
64
+ ;; 範囲指定
65
+ (def age-gen (gen/choose 0 150))
66
+
67
+ ;; 浮動小数点数
68
+ (def percentage-gen (gen/double* {:min 0.0 :max 1.0}))
69
+
70
+ ;; 列挙型
71
+ (def membership-gen (gen/elements [:bronze :silver :gold :platinum]))
72
+ ```
73
+
74
+ ## 3. コレクションジェネレータ
75
+
76
+ ### ベクター・リスト
77
+
78
+ ```clojure
79
+ ;; ベクター
80
+ (gen/vector gen/small-integer) ;; 任意長
81
+ (gen/vector gen/small-integer 1 10) ;; 1〜10要素
82
+
83
+ ;; リスト
84
+ (gen/list gen/small-integer)
85
+
86
+ ;; 非空コレクション
87
+ (gen/not-empty (gen/vector gen/small-integer))
88
+ ```
89
+
90
+ ### マップとセット
91
+
92
+ ```clojure
93
+ ;; マップ
94
+ (gen/map gen/string-alphanumeric gen/small-integer)
95
+
96
+ ;; セット
97
+ (gen/set gen/small-integer)
98
+
99
+ ;; 構造化されたマップ
100
+ (gen/hash-map
101
+ :name gen/string-alphanumeric
102
+ :age (gen/choose 0 150)
103
+ :active gen/boolean)
104
+ ```
105
+
106
+ ## 4. 複合ジェネレータ
107
+
108
+ ### ドメインモデルのジェネレータ
109
+
110
+ ```clojure
111
+ (def person-gen
112
+ "人物データのジェネレータ"
113
+ (gen/hash-map
114
+ :name (gen/such-that #(not (empty? %)) gen/string-alphanumeric)
115
+ :age (gen/choose 0 150)
116
+ :membership (gen/elements [:bronze :silver :gold :platinum])))
117
+
118
+ (def product-gen
119
+ "商品データのジェネレータ"
120
+ (gen/hash-map
121
+ :product-id (gen/fmap #(str "PROD-" (format "%05d" (mod % 100000))) gen/nat)
122
+ :name (gen/such-that #(not (empty? %)) gen/string-alphanumeric)
123
+ :price (gen/fmap #(+ 1 (mod % 10000)) gen/nat)
124
+ :quantity (gen/fmap #(+ 1 (mod % 100)) gen/nat)))
125
+
126
+ (gen/sample product-gen 3)
127
+ ;; => ({:product-id "PROD-00000", :name "a", :price 1, :quantity 1}
128
+ ;; {:product-id "PROD-00001", :name "Ab", :price 2, :quantity 2}
129
+ ;; {:product-id "PROD-00003", :name "x2Y", :price 4, :quantity 3})
130
+ ```
131
+
132
+ ## 5. ジェネレータの変換
133
+
134
+ ### fmap: 値の変換
135
+
136
+ ```clojure
137
+ ;; 生成された値を変換
138
+ (def uppercase-string-gen
139
+ (gen/fmap clojure.string/upper-case gen/string-alphanumeric))
140
+
141
+ ;; フォーマットされたID
142
+ (def order-id-gen
143
+ (gen/fmap #(str "ORD-" (format "%08d" %))
144
+ (gen/choose 0 99999999)))
145
+ ```
146
+
147
+ ### bind: 依存関係のある生成
148
+
149
+ ```clojure
150
+ ;; 生成された値に基づいて別のジェネレータを選択
151
+ (def non-empty-subset-gen
152
+ (gen/bind (gen/not-empty (gen/vector gen/small-integer))
153
+ (fn [v]
154
+ (gen/fmap #(take % v)
155
+ (gen/choose 1 (count v))))))
156
+ ```
157
+
158
+ ### such-that: フィルタリング
159
+
160
+ ```clojure
161
+ ;; 条件を満たす値のみを生成
162
+ (def positive-even-gen
163
+ (gen/such-that #(and (pos? %) (even? %)) gen/small-integer 100))
164
+ ```
165
+
166
+ ## 6. プロパティの定義
167
+
168
+ ### 基本的なプロパティ
169
+
170
+ ```clojure
171
+ (require '[clojure.test.check.properties :as prop])
172
+
173
+ ;; 文字列反転の性質
174
+ (def prop-reverse-involutory
175
+ "文字列を2回反転すると元に戻る"
176
+ (prop/for-all [s gen/string]
177
+ (= s (reverse-string (reverse-string s)))))
178
+
179
+ (def prop-reverse-length-preserved
180
+ "反転しても長さは変わらない"
181
+ (prop/for-all [s gen/string]
182
+ (= (count s) (count (reverse-string s)))))
183
+ ```
184
+
185
+ ### ソートの性質
186
+
187
+ ```clojure
188
+ (def prop-sort-idempotent
189
+ "ソートは冪等(2回ソートしても結果は同じ)"
190
+ (prop/for-all [nums (gen/vector gen/small-integer)]
191
+ (= (sort-numbers nums) (sort-numbers (sort-numbers nums)))))
192
+
193
+ (def prop-sort-preserves-elements
194
+ "ソートは要素を保存する(追加も削除もしない)"
195
+ (prop/for-all [nums (gen/vector gen/small-integer)]
196
+ (= (frequencies nums) (frequencies (sort-numbers nums)))))
197
+
198
+ (def prop-sort-ordered
199
+ "ソート結果は昇順に並ぶ"
200
+ (prop/for-all [nums (gen/vector gen/small-integer)]
201
+ (let [sorted (sort-numbers nums)]
202
+ (or (empty? sorted)
203
+ (apply <= sorted)))))
204
+ ```
205
+
206
+ ### ビジネスロジックの性質
207
+
208
+ ```clojure
209
+ (def prop-discount-bounds
210
+ "割引後の価格は0以上、元の価格以下"
211
+ (prop/for-all [price (gen/fmap #(Math/abs %) gen/small-integer)
212
+ rate (gen/double* {:min 0.0 :max 1.0})]
213
+ (let [discounted (calculate-discount price rate)]
214
+ (and (>= discounted 0)
215
+ (<= discounted price)))))
216
+ ```
217
+
218
+ ## 7. プロパティの実行
219
+
220
+ ### quick-check
221
+
222
+ ```clojure
223
+ (require '[clojure.test.check :as tc])
224
+
225
+ ;; 100回テスト
226
+ (tc/quick-check 100 prop-reverse-involutory)
227
+ ;; => {:result true, :pass? true, :num-tests 100, :time-elapsed-ms 15, ...}
228
+
229
+ ;; 1000回テスト
230
+ (tc/quick-check 1000 prop-sort-ordered)
231
+ ```
232
+
233
+ ### 失敗時の出力
234
+
235
+ ```clojure
236
+ ;; バグのある関数
237
+ (defn buggy-abs [n]
238
+ (if (neg? n) (- n) n))
239
+
240
+ (def prop-abs-positive
241
+ (prop/for-all [n gen/small-integer]
242
+ (>= (buggy-abs n) 0)))
243
+
244
+ (tc/quick-check 10000 prop-abs-positive)
245
+ ;; Integer/MIN_VALUE で失敗する場合、シュリンキングで最小反例を提示
246
+ ```
247
+
248
+ ## 8. シュリンキング
249
+
250
+ テストが失敗すると、test.check は自動的に**シュリンキング**を行い、同じ失敗を引き起こす最小の入力を探します。
251
+
252
+ ```clojure
253
+ ;; 例:リストの長さが5未満というバグのあるプロパティ
254
+ (def prop-buggy
255
+ (prop/for-all [v (gen/vector gen/small-integer)]
256
+ (< (count v) 5)))
257
+
258
+ (tc/quick-check 100 prop-buggy)
259
+ ;; => {:result false,
260
+ ;; :pass? false,
261
+ ;; :shrunk {:total-nodes-visited 7,
262
+ ;; :smallest [[0 0 0 0 0]]}, ;; 最小の反例
263
+ ;; ...}
264
+ ```
265
+
266
+ シュリンキングにより、「長さ100のリストで失敗」ではなく「長さ5のリストで失敗」という最小の反例が得られます。
267
+
268
+ ## 9. ラウンドトリッププロパティ
269
+
270
+ エンコード/デコードの可逆性は典型的なプロパティです。
271
+
272
+ ```clojure
273
+ (defn encode-run-length [s]
274
+ (->> s
275
+ (partition-by identity)
276
+ (map (fn [group] [(first group) (count group)]))
277
+ (into [])))
278
+
279
+ (defn decode-run-length [encoded]
280
+ (->> encoded
281
+ (mapcat (fn [[ch n]] (repeat n ch)))
282
+ (apply str)))
283
+
284
+ (def prop-run-length-roundtrip
285
+ "ランレングス符号化は可逆"
286
+ (prop/for-all [s (gen/fmap #(apply str %) (gen/vector gen/char-alpha))]
287
+ (= s (decode-run-length (encode-run-length s)))))
288
+ ```
289
+
290
+ ## 10. 実践的なパターン
291
+
292
+ ### 代数的性質
293
+
294
+ ```clojure
295
+ ;; 結合律
296
+ (def prop-addition-associative
297
+ (prop/for-all [a gen/small-integer
298
+ b gen/small-integer
299
+ c gen/small-integer]
300
+ (= (+ (+ a b) c) (+ a (+ b c)))))
301
+
302
+ ;; 交換律
303
+ (def prop-addition-commutative
304
+ (prop/for-all [a gen/small-integer
305
+ b gen/small-integer]
306
+ (= (+ a b) (+ b a))))
307
+ ```
308
+
309
+ ### オラクルテスト
310
+
311
+ 既知の正しい実装と比較します。
312
+
313
+ ```clojure
314
+ (def prop-my-sort-matches-builtin
315
+ "自作ソートは組み込みソートと同じ結果"
316
+ (prop/for-all [nums (gen/vector gen/small-integer)]
317
+ (= (my-sort nums) (sort nums))))
318
+ ```
319
+
320
+ ### 不変条件
321
+
322
+ ```clojure
323
+ (def prop-order-total-positive
324
+ "注文合計は常に正"
325
+ (prop/for-all [order order-gen]
326
+ (pos? (calculate-order-total order))))
327
+ ```
328
+
329
+ ## まとめ
330
+
331
+ 本章では、プロパティベーステストについて学びました:
332
+
333
+ 1. **ジェネレータ**: テストデータの自動生成
334
+ 2. **プリミティブ**: 整数、文字列、ブール値など
335
+ 3. **コレクション**: ベクター、マップ、セット
336
+ 4. **変換**: fmap, bind, such-that による加工
337
+ 5. **プロパティ**: すべての入力で成り立つべき性質
338
+ 6. **シュリンキング**: 失敗時の最小反例探索
339
+ 7. **パターン**: ラウンドトリップ、代数的性質、オラクルテスト
340
+
341
+ プロパティベーステストは、従来のテストを補完し、より堅牢なソフトウェアを実現します。
342
+
343
+ ## 参考コード
344
+
345
+ 本章のコード例は以下のファイルで確認できます:
346
+
347
+ - ソースコード: `apps/clojure/part2/src/property_based_testing.clj`
348
+ - テストコード: `apps/clojure/part2/spec/property_based_testing_spec.clj`
349
+
350
+ ## 次章予告
351
+
352
+ 次章では、**テスト駆動開発と関数型プログラミング**について学びます。Red-Green-Refactor サイクルを関数型スタイルで実践する方法を探ります。