@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,842 @@
1
+ # Part VI: 実践的なアプリケーション構築とテスト
2
+
3
+ 本章では、これまで学んだ関数型プログラミングの概念を統合し、実践的なアプリケーションを構築します。また、Haskell におけるテスト戦略についても学びます。
4
+
5
+ ---
6
+
7
+ ## 第12章: テスト戦略と実践アプリケーション
8
+
9
+ ### 12.1 TravelGuide アプリケーション
10
+
11
+ **ソースファイル**: `app/haskell/src/Ch12/TestingStrategies.hs`
12
+
13
+ 旅行ガイドアプリケーションを例に、実践的な FP アプリケーションの構築方法を学びます。
14
+
15
+ ```plantuml
16
+ @startuml
17
+ !theme plain
18
+
19
+ package "TravelGuide Application" {
20
+ rectangle "Model" {
21
+ class Location {
22
+ locId: LocationId
23
+ locName: String
24
+ locPopulation: Int
25
+ }
26
+
27
+ class Attraction {
28
+ attrName: String
29
+ attrDescription: Maybe String
30
+ attrLocation: Location
31
+ }
32
+
33
+ class TravelGuide {
34
+ tgAttraction: Attraction
35
+ tgSubjects: [String]
36
+ tgSearchReport: SearchReport
37
+ }
38
+ }
39
+
40
+ rectangle "Data Access" {
41
+ class DataAccess {
42
+ +findAttractions()
43
+ +findArtistsFromLocation()
44
+ +findMoviesAboutLocation()
45
+ }
46
+ }
47
+ }
48
+
49
+ @enduml
50
+ ```
51
+
52
+ ### 12.2 ドメインモデルの定義
53
+
54
+ ```haskell
55
+ -- 位置ID(値オブジェクト)
56
+ newtype LocationId = LocationId { unLocationId :: String }
57
+ deriving (Show, Eq, Ord)
58
+
59
+ -- ロケーション
60
+ data Location = Location
61
+ { locId :: LocationId
62
+ , locName :: String
63
+ , locPopulation :: Int
64
+ } deriving (Show, Eq)
65
+
66
+ -- アトラクション(観光地)
67
+ data Attraction = Attraction
68
+ { attrName :: String
69
+ , attrDescription :: Maybe String
70
+ , attrLocation :: Location
71
+ } deriving (Show, Eq)
72
+
73
+ -- 旅行ガイド
74
+ data TravelGuide = TravelGuide
75
+ { tgAttraction :: Attraction
76
+ , tgSubjects :: [String]
77
+ , tgSearchReport :: SearchReport
78
+ } deriving (Show, Eq)
79
+ ```
80
+
81
+ Scala との対応:
82
+
83
+ - `opaque type LocationId` → `newtype LocationId`
84
+ - `case class` → `data` with record syntax
85
+ - `Option[String]` → `Maybe String`
86
+
87
+ ### 12.3 データアクセス層の抽象化
88
+
89
+ 外部データソースへのアクセスをレコード型で抽象化します。
90
+
91
+ ```haskell
92
+ -- データアクセスインターフェース
93
+ data DataAccess = DataAccess
94
+ { findAttractions :: String -> AttractionOrdering -> Int -> IO [Attraction]
95
+ , findArtistsFromLocation :: LocationId -> Int -> IO (Either String [MusicArtist])
96
+ , findMoviesAboutLocation :: LocationId -> Int -> IO (Either String [Movie])
97
+ }
98
+ ```
99
+
100
+ ```plantuml
101
+ @startuml
102
+ !theme plain
103
+
104
+ rectangle "DataAccess(レコード型)" {
105
+ card "findAttractions"
106
+ card "findArtistsFromLocation"
107
+ card "findMoviesAboutLocation"
108
+ }
109
+
110
+ note bottom
111
+ Haskell ではレコード型で
112
+ インターフェースを表現
113
+ (Scala の trait に相当)
114
+ end note
115
+
116
+ @enduml
117
+ ```
118
+
119
+ Scala との対応:
120
+
121
+ - `trait DataAccess` → `data DataAccess = DataAccess { ... }`
122
+ - メソッド → レコードのフィールド(関数型)
123
+
124
+ ### 12.4 テスト用スタブの作成
125
+
126
+ ```haskell
127
+ -- テスト用データアクセスの作成
128
+ mkTestDataAccess :: IO DataAccess
129
+ mkTestDataAccess = return DataAccess
130
+ { findAttractions = \name _ limit ->
131
+ return $ take limit
132
+ [ Attraction
133
+ { attrName = "Test Attraction"
134
+ , attrDescription = Just "A test attraction"
135
+ , attrLocation = Location (LocationId "Q123") "Test City" 100000
136
+ }
137
+ | name == "Test" || name == ""
138
+ ]
139
+ , findArtistsFromLocation = \_ limit ->
140
+ return $ Right $ take limit [MusicArtist "Test Artist"]
141
+ , findMoviesAboutLocation = \_ limit ->
142
+ return $ Right $ take limit [Movie "Test Movie"]
143
+ }
144
+
145
+ -- 失敗するデータアクセス(エラーテスト用)
146
+ mkFailingDataAccess :: IO DataAccess
147
+ mkFailingDataAccess = return DataAccess
148
+ { findAttractions = \_ _ limit ->
149
+ return $ take limit [testAttraction]
150
+ , findArtistsFromLocation = \_ _ ->
151
+ return $ Left "Network error"
152
+ , findMoviesAboutLocation = \_ _ ->
153
+ return $ Left "Timeout"
154
+ }
155
+ ```
156
+
157
+ ```plantuml
158
+ @startuml
159
+ !theme plain
160
+
161
+ rectangle "テスト構成" {
162
+ rectangle "本番" as prod {
163
+ class "WikidataDataAccess" as wda
164
+ }
165
+
166
+ rectangle "テスト" as test {
167
+ class "mkTestDataAccess" as tda
168
+ class "mkFailingDataAccess" as fda
169
+ }
170
+ }
171
+
172
+ note bottom of tda
173
+ テストデータを返す
174
+ return で即座に結果を返す
175
+ end note
176
+
177
+ note bottom of fda
178
+ エラーケースのテスト用
179
+ Left でエラーを返す
180
+ end note
181
+
182
+ @enduml
183
+ ```
184
+
185
+ ### 12.5 SearchReport の導入
186
+
187
+ テスト可能性を高めるため、`SearchReport` を導入します。
188
+
189
+ ```haskell
190
+ -- 検索レポート
191
+ data SearchReport = SearchReport
192
+ { srAttractionsSearched :: Int
193
+ , srErrors :: [String]
194
+ } deriving (Show, Eq)
195
+
196
+ -- 旅行ガイド(SearchReport 付き)
197
+ data TravelGuide = TravelGuide
198
+ { tgAttraction :: Attraction
199
+ , tgSubjects :: [String]
200
+ , tgSearchReport :: SearchReport
201
+ } deriving (Show, Eq)
202
+ ```
203
+
204
+ ```plantuml
205
+ @startuml
206
+ !theme plain
207
+
208
+ class TravelGuide {
209
+ tgAttraction: Attraction
210
+ tgSubjects: [String]
211
+ tgSearchReport: SearchReport
212
+ }
213
+
214
+ class SearchReport {
215
+ srAttractionsSearched: Int
216
+ srErrors: [String]
217
+ }
218
+
219
+ TravelGuide --> SearchReport
220
+
221
+ note right of SearchReport
222
+ 検索の統計情報と
223
+ エラー情報を保持
224
+ end note
225
+
226
+ @enduml
227
+ ```
228
+
229
+ ### 12.6 アプリケーションロジック
230
+
231
+ ```haskell
232
+ -- 旅行ガイドを取得(SearchReport 付き)
233
+ travelGuideWithReport :: DataAccess -> String -> IO (Maybe TravelGuide)
234
+ travelGuideWithReport da attractionName = do
235
+ attractions <- findAttractions da attractionName AttrByLocationPopulation 3
236
+ case attractions of
237
+ [] -> return Nothing
238
+ (attraction:_) -> do
239
+ let locId' = locId $ attrLocation attraction
240
+ artistsResult <- findArtistsFromLocation da locId' 2
241
+ moviesResult <- findMoviesAboutLocation da locId' 2
242
+
243
+ let errors = collectErrors [artistsResult, moviesResult]
244
+ let artists = either (const []) id artistsResult
245
+ let movies = either (const []) id moviesResult
246
+ let subjects = map artistName artists ++ map movieName movies
247
+
248
+ return $ Just TravelGuide
249
+ { tgAttraction = attraction
250
+ , tgSubjects = subjects
251
+ , tgSearchReport = SearchReport (length attractions) errors
252
+ }
253
+ where
254
+ collectErrors :: [Either String a] -> [String]
255
+ collectErrors = foldr (\r acc -> either (:acc) (const acc) r) []
256
+ ```
257
+
258
+ ```plantuml
259
+ @startuml
260
+ !theme plain
261
+
262
+ rectangle "travelGuideWithReport 関数" {
263
+ card "1. アトラクション検索" as step1
264
+ card "2. アーティスト検索" as step2
265
+ card "3. 映画検索" as step3
266
+ card "4. エラー収集" as step4
267
+ card "5. TravelGuide 組み立て" as step5
268
+
269
+ step1 --> step2 : attraction
270
+ step1 --> step3 : location.id
271
+ step2 --> step4 : Either
272
+ step3 --> step4 : Either
273
+ step4 --> step5 : errors
274
+ }
275
+
276
+ note bottom
277
+ do 記法で
278
+ 複数の IO を合成
279
+ end note
280
+
281
+ @enduml
282
+ ```
283
+
284
+ ### 12.7 キャッシュの実装
285
+
286
+ `IORef` を使用したスレッドセーフなキャッシュの実装:
287
+
288
+ ```haskell
289
+ import Data.IORef
290
+ import Data.Map.Strict (Map)
291
+ import qualified Data.Map.Strict as Map
292
+
293
+ -- キャッシュ付きデータアクセス
294
+ type CachedDataAccess = (DataAccess, IORef (Map String [Attraction]))
295
+
296
+ -- キャッシュ付きデータアクセスの作成
297
+ mkCachedDataAccess :: DataAccess -> IO CachedDataAccess
298
+ mkCachedDataAccess da = do
299
+ cache <- newIORef Map.empty
300
+ return (da, cache)
301
+
302
+ -- キャッシュ付きアトラクション検索
303
+ cachedFindAttractions :: CachedDataAccess -> String -> AttractionOrdering -> Int -> IO [Attraction]
304
+ cachedFindAttractions (da, cache) name ordering limit = do
305
+ let key = name ++ "-" ++ show ordering ++ "-" ++ show limit
306
+ cached <- Map.lookup key <$> readIORef cache
307
+ case cached of
308
+ Just attractions -> return attractions -- キャッシュヒット
309
+ Nothing -> do
310
+ attractions <- findAttractions da name ordering limit
311
+ atomicModifyIORef' cache (\m -> (Map.insert key attractions m, ()))
312
+ return attractions
313
+ ```
314
+
315
+ ```plantuml
316
+ @startuml
317
+ !theme plain
318
+ skinparam activity {
319
+ BackgroundColor White
320
+ BorderColor Black
321
+ }
322
+
323
+ start
324
+
325
+ :キャッシュキー生成;
326
+
327
+ :readIORef でキャッシュ確認;
328
+
329
+ if (キャッシュにデータあり?) then (yes)
330
+ :キャッシュからデータ返却;
331
+ else (no)
332
+ :外部 API を呼び出し;
333
+ :atomicModifyIORef' でキャッシュ保存;
334
+ :結果を返却;
335
+ endif
336
+
337
+ stop
338
+
339
+ note right
340
+ IORef を使用した
341
+ スレッドセーフなキャッシュ
342
+ end note
343
+
344
+ @enduml
345
+ ```
346
+
347
+ ### 12.8 リソース管理(bracket)
348
+
349
+ Haskell の `bracket` を使用して、安全なリソース管理を実現します。
350
+
351
+ ```haskell
352
+ import Control.Exception (bracket)
353
+ import System.IO (Handle, openFile, hClose, hGetContents, IOMode(..))
354
+
355
+ -- リソースを安全に使用
356
+ withResource :: IO a -> (a -> IO ()) -> (a -> IO b) -> IO b
357
+ withResource acquire release = bracket acquire release
358
+
359
+ -- ファイルを安全に開く
360
+ withFile' :: FilePath -> IOMode -> (Handle -> IO a) -> IO a
361
+ withFile' path mode = bracket (openFile path mode) hClose
362
+
363
+ -- 使用例
364
+ readFileContents :: FilePath -> IO (Either String String)
365
+ readFileContents path = do
366
+ result <- try $ withFile' path ReadMode hGetContents
367
+ case result of
368
+ Left e -> return $ Left (show e)
369
+ Right c -> return $ Right c
370
+ ```
371
+
372
+ ```plantuml
373
+ @startuml
374
+ !theme plain
375
+
376
+ participant "Application" as app
377
+ participant "bracket" as res
378
+ participant "Handle" as handle
379
+ participant "File" as file
380
+
381
+ app -> res: bracket acquire release use
382
+ res -> handle: acquire: openFile
383
+ handle -> file: open
384
+ file --> handle: handle
385
+
386
+ app -> res: use (read contents)
387
+ res -> handle: hGetContents
388
+ handle -> file: read
389
+ file --> handle: contents
390
+ handle --> res: contents
391
+ res --> app: contents
392
+
393
+ app -> res: release (automatic)
394
+ res -> handle: hClose
395
+ handle -> file: close
396
+
397
+ note over res
398
+ bracket は例外が発生しても
399
+ 必ず release を実行する
400
+ end note
401
+
402
+ @enduml
403
+ ```
404
+
405
+ Scala との対応:
406
+
407
+ - `Resource[IO, A]` → `bracket acquire release use`
408
+ - `Resource.make(acquire)(release)` → `bracket acquire release`
409
+
410
+ ### 12.9 純粋関数のテスト
411
+
412
+ 純粋関数は副作用がないため、テストが非常に簡単です。
413
+
414
+ ```haskell
415
+ -- 人口でロケーションをフィルタリング(純粋関数)
416
+ filterPopularLocations :: [Location] -> Int -> [Location]
417
+ filterPopularLocations locations minPopulation =
418
+ filter (\loc -> locPopulation loc >= minPopulation) locations
419
+
420
+ -- アトラクションを人口でソート(純粋関数)
421
+ sortAttractionsByPopulation :: [Attraction] -> [Attraction]
422
+ sortAttractionsByPopulation =
423
+ sortBy (\a b -> compare (Down $ locPopulation $ attrLocation a)
424
+ (Down $ locPopulation $ attrLocation b))
425
+
426
+ -- バリデーション(純粋関数)
427
+ validateLocation :: Location -> Either String Location
428
+ validateLocation loc
429
+ | null (unLocationId $ locId loc) = Left "Location ID cannot be empty"
430
+ | null (locName loc) = Left "Location name cannot be empty"
431
+ | locPopulation loc < 0 = Left "Population cannot be negative"
432
+ | otherwise = Right loc
433
+ ```
434
+
435
+ ### 12.10 HSpec によるユニットテスト
436
+
437
+ ```haskell
438
+ import Test.Hspec
439
+
440
+ spec :: Spec
441
+ spec = do
442
+ describe "filterPopularLocations" $ do
443
+ it "filters by minimum population" $ do
444
+ let locs = [ Location (LocationId "1") "A" 100
445
+ , Location (LocationId "2") "B" 200
446
+ , Location (LocationId "3") "C" 150
447
+ ]
448
+ let filtered = filterPopularLocations locs 150
449
+ length filtered `shouldBe` 2
450
+
451
+ it "returns empty for high minimum" $ do
452
+ let locs = [Location (LocationId "1") "A" 100]
453
+ filterPopularLocations locs 1000 `shouldBe` []
454
+
455
+ describe "validateLocation" $ do
456
+ it "validates correct location" $ do
457
+ let loc = Location (LocationId "Q1") "Tokyo" 14000000
458
+ validateLocation loc `shouldBe` Right loc
459
+
460
+ it "rejects empty ID" $ do
461
+ let loc = Location (LocationId "") "Tokyo" 14000000
462
+ validateLocation loc `shouldBe` Left "Location ID cannot be empty"
463
+ ```
464
+
465
+ ### 12.11 QuickCheck によるプロパティベーステスト
466
+
467
+ QuickCheck を使用したプロパティベーステスト:
468
+
469
+ ```haskell
470
+ import Test.QuickCheck
471
+
472
+ -- Arbitrary インスタンスの定義
473
+ instance Arbitrary LocationId where
474
+ arbitrary = LocationId <$> listOf1 (elements ['a'..'z'])
475
+
476
+ instance Arbitrary Location where
477
+ arbitrary = Location
478
+ <$> arbitrary
479
+ <*> listOf1 (elements ['A'..'Z'])
480
+ <*> (abs <$> arbitrary)
481
+
482
+ -- プロパティテスト
483
+ spec :: Spec
484
+ spec = do
485
+ describe "Property-based tests" $ do
486
+ it "filterPopularLocations result size <= input size" $
487
+ property $ \(locs :: [Location]) (minPop :: Int) ->
488
+ length (filterPopularLocations locs (abs minPop)) <= length locs
489
+
490
+ it "filterPopularLocations all results meet minimum" $
491
+ property $ \(locs :: [Location]) (minPop :: Int) ->
492
+ let filtered = filterPopularLocations locs (abs minPop)
493
+ in all (\loc -> locPopulation loc >= abs minPop) filtered
494
+
495
+ it "sortAttractionsByPopulation preserves length" $
496
+ property $ \(attrs :: [Attraction]) ->
497
+ length (sortAttractionsByPopulation attrs) == length attrs
498
+ ```
499
+
500
+ ```plantuml
501
+ @startuml
502
+ !theme plain
503
+
504
+ rectangle "プロパティベーステスト" {
505
+ card "Arbitrary インスタンス" as gen
506
+ card "ランダムデータ生成" as random
507
+ card "プロパティ検証" as verify
508
+ card "100回以上テスト" as repeat
509
+
510
+ gen --> random
511
+ random --> verify
512
+ verify --> repeat
513
+ repeat --> random : 繰り返し
514
+ }
515
+
516
+ note bottom
517
+ QuickCheck が自動で
518
+ ランダムな入力を生成
519
+ end note
520
+
521
+ @enduml
522
+ ```
523
+
524
+ Scala との対応:
525
+
526
+ - ScalaCheck `Gen[A]` → QuickCheck `Arbitrary a`
527
+ - `forAll` → `property`
528
+
529
+ ### 12.12 統合テスト
530
+
531
+ ```haskell
532
+ describe "Integration tests" $ do
533
+ it "travelGuide returns guide for valid attraction" $ do
534
+ da <- mkTestDataAccess
535
+ guide <- travelGuide da "Test"
536
+ guide `shouldSatisfy` \g -> case g of
537
+ Just g' -> attrName (tgAttraction g') == "Test Attraction"
538
+ Nothing -> False
539
+
540
+ it "travelGuideWithReport collects errors" $ do
541
+ da <- mkFailingDataAccess
542
+ guide <- travelGuideWithReport da "Test"
543
+ case guide of
544
+ Just g -> do
545
+ let errors = srErrors (tgSearchReport g)
546
+ length errors `shouldBe` 2
547
+ errors `shouldContain` ["Network error"]
548
+ Nothing -> expectationFailure "Expected Just"
549
+
550
+ it "still returns guide even with errors" $ do
551
+ da <- mkFailingDataAccess
552
+ guide <- travelGuideWithReport da "Test"
553
+ guide `shouldSatisfy` \g -> case g of
554
+ Just _ -> True
555
+ Nothing -> False
556
+ ```
557
+
558
+ ### 12.13 テストピラミッド
559
+
560
+ ```plantuml
561
+ @startuml
562
+ !theme plain
563
+
564
+ rectangle "テストピラミッド" {
565
+ rectangle "E2E テスト\n(少数)" as e2e
566
+ rectangle "統合テスト\n(中程度)" as integration
567
+ rectangle "単体テスト + プロパティテスト\n(多数)" as unit
568
+ }
569
+
570
+ e2e -[hidden]down- integration
571
+ integration -[hidden]down- unit
572
+
573
+ note right of unit
574
+ Haskell では純粋関数が多いため
575
+ 単体テストとプロパティテストが
576
+ 非常に効果的
577
+ end note
578
+
579
+ @enduml
580
+ ```
581
+
582
+ ---
583
+
584
+ ## まとめ
585
+
586
+ ### Part VI で学んだこと
587
+
588
+ ```plantuml
589
+ @startuml
590
+ !theme plain
591
+
592
+ rectangle "Part VI: 実践的なアプリケーション" {
593
+ rectangle "第12章" as ch12 {
594
+ card "ドメインモデル設計"
595
+ card "DataAccess 抽象化"
596
+ card "bracket リソース管理"
597
+ card "IORef キャッシュ"
598
+ card "SearchReport"
599
+ card "HSpec ユニットテスト"
600
+ card "QuickCheck プロパティテスト"
601
+ }
602
+ }
603
+
604
+ @enduml
605
+ ```
606
+
607
+ ### Scala と Haskell の比較
608
+
609
+ | 概念 | Scala | Haskell |
610
+ |------|-------|---------|
611
+ | インターフェース | `trait` | `data` (レコード型) |
612
+ | 値オブジェクト | `opaque type` | `newtype` |
613
+ | オプショナル | `Option[A]` | `Maybe a` |
614
+ | リソース管理 | `Resource[IO, A]` | `bracket` |
615
+ | キャッシュ | `Ref[IO, Map]` | `IORef (Map ...)` |
616
+ | ユニットテスト | ScalaTest | HSpec |
617
+ | プロパティテスト | ScalaCheck | QuickCheck |
618
+ | ジェネレータ | `Gen[A]` | `Arbitrary a` |
619
+
620
+ ### キーポイント
621
+
622
+ 1. **抽象化の重要性**: レコード型で外部依存を抽象化
623
+ 2. **bracket でリソース管理**: 安全なリソースの取得と解放
624
+ 3. **IORef でキャッシュ**: スレッドセーフな状態管理
625
+ 4. **Either でエラー処理**: 明示的なエラーハンドリング
626
+ 5. **SearchReport**: テスト可能性と可観測性の向上
627
+ 6. **スタブ**: 外部依存を差し替えてテスト
628
+ 7. **プロパティベーステスト**: QuickCheck でランダム入力による不変条件の検証
629
+
630
+ ### 学習の総括
631
+
632
+ ```plantuml
633
+ @startuml
634
+ !theme plain
635
+ left to right direction
636
+
637
+ rectangle "FP の学習パス" {
638
+ card "Part I\n基礎" as p1
639
+ card "Part II\nイミュータブル操作" as p2
640
+ card "Part III\n型による安全性" as p3
641
+ card "Part IV\nIO/Stream" as p4
642
+ card "Part V\n並行処理" as p5
643
+ card "Part VI\n実践" as p6
644
+ }
645
+
646
+ p1 --> p2
647
+ p2 --> p3
648
+ p3 --> p4
649
+ p4 --> p5
650
+ p5 --> p6
651
+
652
+ @enduml
653
+ ```
654
+
655
+ ---
656
+
657
+ ## 演習問題
658
+
659
+ ### 問題 1: DataAccess の拡張
660
+
661
+ 以下の要件で `DataAccess` を拡張してください:
662
+ - 新しいフィールド `findHotelsNearLocation` を追加
663
+ - 戻り値は `IO (Either String [Hotel])`
664
+
665
+ ```haskell
666
+ data Hotel = Hotel
667
+ { hotelName :: String
668
+ , hotelRating :: Double
669
+ , hotelLocation :: Location
670
+ } deriving (Show, Eq)
671
+ ```
672
+
673
+ <details>
674
+ <summary>解答</summary>
675
+
676
+ ```haskell
677
+ data DataAccess = DataAccess
678
+ { findAttractions :: String -> AttractionOrdering -> Int -> IO [Attraction]
679
+ , findArtistsFromLocation :: LocationId -> Int -> IO (Either String [MusicArtist])
680
+ , findMoviesAboutLocation :: LocationId -> Int -> IO (Either String [Movie])
681
+ , findHotelsNearLocation :: LocationId -> Int -> IO (Either String [Hotel])
682
+ }
683
+
684
+ -- テスト用スタブ
685
+ mkTestDataAccess :: IO DataAccess
686
+ mkTestDataAccess = return DataAccess
687
+ { -- 既存の実装...
688
+ , findHotelsNearLocation = \_ limit ->
689
+ return $ Right $ take limit
690
+ [Hotel "Test Hotel" 4.5 (Location (LocationId "Q123") "Test City" 100000)]
691
+ }
692
+ ```
693
+
694
+ </details>
695
+
696
+ ### 問題 2: プロパティベーステスト
697
+
698
+ 以下の関数に対するプロパティベーステストを書いてください:
699
+
700
+ ```haskell
701
+ combineSubjects :: [MusicArtist] -> [Movie] -> [String]
702
+ combineSubjects artists movies =
703
+ map artistName artists ++ map movieName movies
704
+ ```
705
+
706
+ <details>
707
+ <summary>解答</summary>
708
+
709
+ ```haskell
710
+ import Test.QuickCheck
711
+
712
+ instance Arbitrary MusicArtist where
713
+ arbitrary = MusicArtist <$> listOf1 (elements ['A'..'Z'])
714
+
715
+ instance Arbitrary Movie where
716
+ arbitrary = Movie <$> listOf1 (elements ['A'..'Z'])
717
+
718
+ spec :: Spec
719
+ spec = do
720
+ describe "combineSubjects properties" $ do
721
+ it "length is sum of inputs" $
722
+ property $ \(artists :: [MusicArtist]) (movies :: [Movie]) ->
723
+ length (combineSubjects artists movies) == length artists + length movies
724
+
725
+ it "artists come first" $
726
+ property $ \(artists :: [MusicArtist]) (movies :: [Movie]) ->
727
+ let result = combineSubjects artists movies
728
+ artistNames = map artistName artists
729
+ in take (length artists) result == artistNames
730
+
731
+ it "movies come after artists" $
732
+ property $ \(artists :: [MusicArtist]) (movies :: [Movie]) ->
733
+ let result = combineSubjects artists movies
734
+ movieNames = map movieName movies
735
+ in drop (length artists) result == movieNames
736
+ ```
737
+
738
+ </details>
739
+
740
+ ### 問題 3: bracket の実装
741
+
742
+ データベース接続を安全に管理する関数を実装してください。
743
+
744
+ ```haskell
745
+ data Connection = Connection { connId :: Int }
746
+
747
+ -- 実装してください
748
+ withConnection :: IO Connection -> (Connection -> IO a) -> IO a
749
+ ```
750
+
751
+ <details>
752
+ <summary>解答</summary>
753
+
754
+ ```haskell
755
+ import Control.Exception (bracket)
756
+
757
+ data Connection = Connection { connId :: Int }
758
+
759
+ -- 接続を開く
760
+ openConnection :: IO Connection
761
+ openConnection = do
762
+ putStrLn "Opening connection..."
763
+ return $ Connection 1
764
+
765
+ -- 接続を閉じる
766
+ closeConnection :: Connection -> IO ()
767
+ closeConnection conn = putStrLn $ "Closing connection " ++ show (connId conn)
768
+
769
+ -- 安全に接続を使用
770
+ withConnection :: (Connection -> IO a) -> IO a
771
+ withConnection = bracket openConnection closeConnection
772
+
773
+ -- 使用例
774
+ queryDatabase :: IO String
775
+ queryDatabase = withConnection $ \conn -> do
776
+ putStrLn $ "Querying with connection " ++ show (connId conn)
777
+ return "Query result"
778
+ ```
779
+
780
+ </details>
781
+
782
+ ### 問題 4: エラー収集
783
+
784
+ 複数の `Either` からエラーを収集する関数を実装してください。
785
+
786
+ ```haskell
787
+ collectAllErrors :: [Either String a] -> [String]
788
+ collectAllErrors = ???
789
+ ```
790
+
791
+ <details>
792
+ <summary>解答</summary>
793
+
794
+ ```haskell
795
+ collectAllErrors :: [Either String a] -> [String]
796
+ collectAllErrors = foldr collect []
797
+ where
798
+ collect (Left err) acc = err : acc
799
+ collect (Right _) acc = acc
800
+
801
+ -- または
802
+ collectAllErrors :: [Either String a] -> [String]
803
+ collectAllErrors results = [err | Left err <- results]
804
+
805
+ -- テスト
806
+ -- collectAllErrors [Right 1, Left "Error1", Right 2, Left "Error2"]
807
+ -- => ["Error1", "Error2"]
808
+ ```
809
+
810
+ </details>
811
+
812
+ ---
813
+
814
+ ## シリーズ全体の総括
815
+
816
+ 本シリーズでは、「Grokking Functional Programming」の内容に沿って、Haskell で関数型プログラミングの基礎から実践的なアプリケーション構築までを学びました。
817
+
818
+ ### 学んだ主な概念
819
+
820
+ | Part | 章 | 主な概念 |
821
+ |------|-----|----------|
822
+ | I | 1-2 | 純粋関数、参照透過性、型推論 |
823
+ | II | 3-5 | イミュータブルデータ、高階関数、do 記法 |
824
+ | III | 6-7 | Maybe、Either、パターンマッチング |
825
+ | IV | 8-9 | IO モナド、遅延リスト(無限ストリーム) |
826
+ | V | 10-11 | 並行処理、IORef、Async、STM |
827
+ | VI | 12 | 実践アプリケーション、テスト戦略 |
828
+
829
+ ### Haskell の関数型プログラミングの利点
830
+
831
+ 1. **純粋性の保証**: 型システムが純粋関数と IO を分離
832
+ 2. **遅延評価**: 無限リストが自然に扱える
833
+ 3. **強力な型システム**: newtype、ADT で安全なモデリング
834
+ 4. **STM**: デッドロックフリーな並行処理
835
+ 5. **QuickCheck**: プロパティベーステストの先駆者
836
+
837
+ ### 次のステップ
838
+
839
+ - Monad Transformers(モナドトランスフォーマー)を学ぶ
840
+ - lens ライブラリでデータ操作を効率化
841
+ - servant で型安全な Web API を構築
842
+ - 実際のプロジェクトで Haskell を適用する