@k2works/claude-code-booster 2.7.1 → 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 (527) 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 -103
  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/review/index.md +5 -0
  522. package/lib/assets/docs/strategy/index.md +15 -0
  523. package/lib/assets/docs/template/index.md +9 -5
  524. package/lib/assets/mkdocs.yml +33 -19
  525. package/package.json +1 -1
  526. package/lib/assets/docs/analysis/index.md +0 -8
  527. /package/lib/assets/docs/{analysis → strategy}/slide/.gitkeep +0 -0
@@ -0,0 +1,865 @@
1
+ # Part VI: 実践的なアプリケーション構築とテスト
2
+
3
+ 本章では、これまで学んだ関数型プログラミングの概念を統合し、実践的なアプリケーションを構築します。また、関数型プログラミングにおけるテスト戦略についても学びます。
4
+
5
+ ---
6
+
7
+ ## 第11章: 実践的なアプリケーション構築
8
+
9
+ ### 11.1 TravelGuide アプリケーション
10
+
11
+ 旅行ガイドアプリケーションを例に、実践的な FP アプリケーションの構築方法を学びます。
12
+
13
+ ```plantuml
14
+ @startuml
15
+ !theme plain
16
+
17
+ package "TravelGuide Application" {
18
+ rectangle "Model" {
19
+ class Location {
20
+ id: LocationId
21
+ name: String
22
+ population: Int
23
+ }
24
+
25
+ class Attraction {
26
+ name: String
27
+ description: Option[String]
28
+ location: Location
29
+ }
30
+
31
+ class TravelGuide {
32
+ attraction: Attraction
33
+ subjects: List[String]
34
+ }
35
+ }
36
+
37
+ rectangle "Data Access" {
38
+ interface DataAccess {
39
+ +findAttractions(): IO[List[Attraction]]
40
+ +findArtistsFromLocation(): IO[List[MusicArtist]]
41
+ +findMoviesAboutLocation(): IO[List[Movie]]
42
+ }
43
+
44
+ class WikidataDataAccess
45
+ class CachedDataAccess
46
+ }
47
+
48
+ DataAccess <|.. WikidataDataAccess
49
+ DataAccess <|.. CachedDataAccess
50
+ }
51
+
52
+ @enduml
53
+ ```
54
+
55
+ ### 11.2 ドメインモデルの定義
56
+
57
+ **ソースファイル**: `app/scala/src/main/scala/ch11_TravelGuide.scala`
58
+
59
+ ```scala
60
+ // 位置情報を表す値オブジェクト
61
+ opaque type LocationId = String
62
+ object LocationId {
63
+ def apply(value: String): LocationId = value
64
+ extension (a: LocationId) def value: String = a
65
+ }
66
+
67
+ // ロケーション
68
+ case class Location(id: LocationId, name: String, population: Int)
69
+
70
+ // アトラクション(観光地)
71
+ case class Attraction(
72
+ name: String,
73
+ description: Option[String],
74
+ location: Location
75
+ )
76
+
77
+ // 旅行ガイド
78
+ case class TravelGuide(attraction: Attraction, subjects: List[String])
79
+ ```
80
+
81
+ ### 11.3 データアクセス層の抽象化
82
+
83
+ 外部データソースへのアクセスをトレイトで抽象化します。
84
+
85
+ ```scala
86
+ trait DataAccess {
87
+ def findAttractions(
88
+ name: String,
89
+ ordering: AttractionOrdering,
90
+ limit: Int
91
+ ): IO[List[Attraction]]
92
+
93
+ def findArtistsFromLocation(
94
+ locationId: LocationId,
95
+ limit: Int
96
+ ): IO[List[MusicArtist]]
97
+
98
+ def findMoviesAboutLocation(
99
+ locationId: LocationId,
100
+ limit: Int
101
+ ): IO[List[Movie]]
102
+ }
103
+ ```
104
+
105
+ ```plantuml
106
+ @startuml
107
+ !theme plain
108
+
109
+ interface DataAccess {
110
+ +findAttractions(): IO[List[Attraction]]
111
+ +findArtistsFromLocation(): IO[List[MusicArtist]]
112
+ +findMoviesAboutLocation(): IO[List[Movie]]
113
+ }
114
+
115
+ note right of DataAccess
116
+ 純粋な関数型インターフェース
117
+ すべての操作は IO でラップ
118
+ end note
119
+
120
+ class WikidataDataAccess {
121
+ -sparqlEndpoint: String
122
+ +findAttractions(): IO[List[Attraction]]
123
+ +findArtistsFromLocation(): IO[List[MusicArtist]]
124
+ +findMoviesAboutLocation(): IO[List[Movie]]
125
+ }
126
+
127
+ class TestDataAccess {
128
+ -testData: Map[String, List[Attraction]]
129
+ +findAttractions(): IO[List[Attraction]]
130
+ +findArtistsFromLocation(): IO[List[MusicArtist]]
131
+ +findMoviesAboutLocation(): IO[List[Movie]]
132
+ }
133
+
134
+ DataAccess <|.. WikidataDataAccess
135
+ DataAccess <|.. TestDataAccess
136
+
137
+ @enduml
138
+ ```
139
+
140
+ ### 11.4 Resource によるリソース管理
141
+
142
+ cats-effect の `Resource` を使用して、安全なリソース管理を実現します。
143
+
144
+ **ソースファイル**: `app/scala/src/main/scala/ch11_TravelGuide.scala`
145
+
146
+ ```scala
147
+ def execQuery(
148
+ query: String
149
+ ): Resource[IO, List[QuerySolution]] = {
150
+ val connection: Resource[IO, RDFConnection] =
151
+ Resource.make(
152
+ IO.blocking(
153
+ RDFConnectionRemote
154
+ .create()
155
+ .destination(WikidataSparqlEndpoint)
156
+ .build()
157
+ )
158
+ )(connection => IO.blocking(connection.close()))
159
+
160
+ for {
161
+ conn <- connection
162
+ result <- Resource.eval(IO.blocking {
163
+ val resultSet = conn.query(query).execSelect()
164
+ resultSet.asScala.toList
165
+ })
166
+ } yield result
167
+ }
168
+ ```
169
+
170
+ ```plantuml
171
+ @startuml
172
+ !theme plain
173
+
174
+ participant "Application" as app
175
+ participant "Resource" as res
176
+ participant "RDFConnection" as conn
177
+ participant "SPARQL Endpoint" as sparql
178
+
179
+ app -> res: Resource.make(acquire)(release)
180
+ res -> conn: acquire: IO.blocking(create())
181
+ conn -> sparql: connect
182
+ sparql --> conn: connection
183
+
184
+ app -> res: use (query execution)
185
+ res -> conn: query.execSelect()
186
+ conn -> sparql: SPARQL query
187
+ sparql --> conn: results
188
+ conn --> res: List[QuerySolution]
189
+ res --> app: results
190
+
191
+ app -> res: release (automatic)
192
+ res -> conn: IO.blocking(close())
193
+ conn -> sparql: disconnect
194
+
195
+ note over res
196
+ Resource は例外が発生しても
197
+ 必ず release を実行する
198
+ end note
199
+
200
+ @enduml
201
+ ```
202
+
203
+ ### 11.5 キャッシュの実装
204
+
205
+ `Ref` を使用したスレッドセーフなキャッシュの実装:
206
+
207
+ ```scala
208
+ def cachedAttractions(
209
+ dataAccess: DataAccess
210
+ ): IO[DataAccess] = {
211
+ for {
212
+ cache <- Ref.of[IO, Map[String, List[Attraction]]](Map.empty)
213
+ } yield new DataAccess {
214
+ def findAttractions(
215
+ name: String,
216
+ ordering: AttractionOrdering,
217
+ limit: Int
218
+ ): IO[List[Attraction]] = {
219
+ val key = s"$name-$ordering-$limit"
220
+ for {
221
+ cached <- cache.get.map(_.get(key))
222
+ result <- cached match {
223
+ case Some(attractions) => IO.pure(attractions)
224
+ case None =>
225
+ for {
226
+ attractions <- dataAccess.findAttractions(name, ordering, limit)
227
+ _ <- cache.update(_ + (key -> attractions))
228
+ } yield attractions
229
+ }
230
+ } yield result
231
+ }
232
+
233
+ // 他のメソッドも同様に実装...
234
+ }
235
+ }
236
+ ```
237
+
238
+ ```plantuml
239
+ @startuml
240
+ !theme plain
241
+ skinparam activity {
242
+ BackgroundColor White
243
+ BorderColor Black
244
+ }
245
+
246
+ start
247
+
248
+ :キャッシュキー生成;
249
+
250
+ :cache.get でキャッシュ確認;
251
+
252
+ if (キャッシュにデータあり?) then (yes)
253
+ :キャッシュからデータ返却;
254
+ else (no)
255
+ :外部 API を呼び出し;
256
+ :結果をキャッシュに保存;
257
+ :結果を返却;
258
+ endif
259
+
260
+ stop
261
+
262
+ note right
263
+ Ref を使用した
264
+ スレッドセーフなキャッシュ
265
+ end note
266
+
267
+ @enduml
268
+ ```
269
+
270
+ ### 11.6 アプリケーションの組み立て
271
+
272
+ すべてのコンポーネントを組み合わせてアプリケーションを構築します。
273
+
274
+ ```scala
275
+ def travelGuide(data: DataAccess, attractionName: String): IO[Option[TravelGuide]] = {
276
+ for {
277
+ attractions <- data.findAttractions(attractionName, ByLocationPopulation, 1)
278
+ guide <- attractions.headOption match {
279
+ case None => IO.pure(None)
280
+ case Some(attraction) =>
281
+ for {
282
+ artists <- data.findArtistsFromLocation(attraction.location.id, 2)
283
+ movies <- data.findMoviesAboutLocation(attraction.location.id, 2)
284
+ } yield Some(
285
+ TravelGuide(
286
+ attraction,
287
+ artists.map(_.name) ++ movies.map(_.name)
288
+ )
289
+ )
290
+ }
291
+ } yield guide
292
+ }
293
+ ```
294
+
295
+ ```plantuml
296
+ @startuml
297
+ !theme plain
298
+
299
+ rectangle "travelGuide 関数" {
300
+ card "1. アトラクション検索" as step1
301
+ card "2. アーティスト検索" as step2
302
+ card "3. 映画検索" as step3
303
+ card "4. TravelGuide 組み立て" as step4
304
+
305
+ step1 --> step2 : attraction
306
+ step1 --> step3 : location.id
307
+ step2 --> step4 : artists
308
+ step3 --> step4 : movies
309
+ }
310
+
311
+ note bottom
312
+ for 内包表記で
313
+ 複数の IO を合成
314
+ end note
315
+
316
+ @enduml
317
+ ```
318
+
319
+ ---
320
+
321
+ ## 第12章: テスト戦略
322
+
323
+ ### 12.1 関数型プログラミングのテスト
324
+
325
+ 関数型プログラミングでは、純粋関数のおかげでテストが非常に簡単になります。
326
+
327
+ ```plantuml
328
+ @startuml
329
+ !theme plain
330
+
331
+ rectangle "テストの種類" {
332
+ rectangle "単体テスト" as unit {
333
+ card "純粋関数のテスト"
334
+ card "高速・独立"
335
+ }
336
+
337
+ rectangle "プロパティベーステスト" as property {
338
+ card "ランダム入力"
339
+ card "不変条件の検証"
340
+ }
341
+
342
+ rectangle "統合テスト" as integration {
343
+ card "コンポーネント連携"
344
+ card "スタブ/モック使用"
345
+ }
346
+ }
347
+
348
+ unit --> property : 補完
349
+ property --> integration : 補完
350
+
351
+ @enduml
352
+ ```
353
+
354
+ ### 12.2 SearchReport の導入
355
+
356
+ **ソースファイル**: `app/scala/src/main/scala/ch12_TravelGuide.scala`
357
+
358
+ テスト可能性を高めるため、`SearchReport` を導入します。
359
+
360
+ ```scala
361
+ case class SearchReport(
362
+ attractionsSearched: Int,
363
+ errors: List[String]
364
+ )
365
+
366
+ case class TravelGuide(
367
+ attraction: Attraction,
368
+ subjects: List[String],
369
+ searchReport: SearchReport
370
+ )
371
+ ```
372
+
373
+ ```plantuml
374
+ @startuml
375
+ !theme plain
376
+
377
+ class TravelGuide {
378
+ attraction: Attraction
379
+ subjects: List[String]
380
+ searchReport: SearchReport
381
+ }
382
+
383
+ class SearchReport {
384
+ attractionsSearched: Int
385
+ errors: List[String]
386
+ }
387
+
388
+ TravelGuide --> SearchReport
389
+
390
+ note right of SearchReport
391
+ 検索の統計情報と
392
+ エラー情報を保持
393
+ end note
394
+
395
+ @enduml
396
+ ```
397
+
398
+ ### 12.3 エラーハンドリングの改善
399
+
400
+ ```scala
401
+ def findArtistsFromLocation(
402
+ locationId: LocationId,
403
+ limit: Int
404
+ ): IO[Either[String, List[MusicArtist]]]
405
+
406
+ def findMoviesAboutLocation(
407
+ locationId: LocationId,
408
+ limit: Int
409
+ ): IO[Either[String, List[Movie]]]
410
+ ```
411
+
412
+ Either を使用してエラーを明示的に扱います:
413
+
414
+ ```scala
415
+ def travelGuide(
416
+ data: DataAccess,
417
+ attractionName: String
418
+ ): IO[Option[TravelGuide]] = {
419
+ for {
420
+ attractions <- data.findAttractions(attractionName, ByLocationPopulation, 3)
421
+ guides <- attractions.traverse { attraction =>
422
+ for {
423
+ artistsResult <- data.findArtistsFromLocation(attraction.location.id, 2)
424
+ moviesResult <- data.findMoviesAboutLocation(attraction.location.id, 2)
425
+ } yield {
426
+ val errors = List(artistsResult, moviesResult).collect {
427
+ case Left(error) => error
428
+ }
429
+ val artists = artistsResult.getOrElse(Nil)
430
+ val movies = moviesResult.getOrElse(Nil)
431
+ TravelGuide(
432
+ attraction,
433
+ artists.map(_.name) ++ movies.map(_.name),
434
+ SearchReport(attractions.size, errors)
435
+ )
436
+ }
437
+ }
438
+ } yield guides.headOption
439
+ }
440
+ ```
441
+
442
+ ### 12.4 スタブを使用したテスト
443
+
444
+ **ソースファイル**: `app/scala/src/main/scala/ch12_TravelGuideTest.scala`
445
+
446
+ ```scala
447
+ // テスト用のスタブ実装
448
+ val testDataAccess: DataAccess = new DataAccess {
449
+ def findAttractions(
450
+ name: String,
451
+ ordering: AttractionOrdering,
452
+ limit: Int
453
+ ): IO[List[Attraction]] = IO.pure(
454
+ List(
455
+ Attraction(
456
+ "Test Attraction",
457
+ Some("A test attraction"),
458
+ Location(LocationId("Q123"), "Test City", 100000)
459
+ )
460
+ )
461
+ )
462
+
463
+ def findArtistsFromLocation(
464
+ locationId: LocationId,
465
+ limit: Int
466
+ ): IO[Either[String, List[MusicArtist]]] =
467
+ IO.pure(Right(List(MusicArtist("Test Artist"))))
468
+
469
+ def findMoviesAboutLocation(
470
+ locationId: LocationId,
471
+ limit: Int
472
+ ): IO[Either[String, List[Movie]]] =
473
+ IO.pure(Right(List(Movie("Test Movie"))))
474
+ }
475
+ ```
476
+
477
+ ```plantuml
478
+ @startuml
479
+ !theme plain
480
+
481
+ rectangle "テスト構成" {
482
+ rectangle "本番" as prod {
483
+ interface "DataAccess" as da1
484
+ class "WikidataDataAccess" as wda
485
+ da1 <|.. wda
486
+ }
487
+
488
+ rectangle "テスト" as test {
489
+ interface "DataAccess" as da2
490
+ class "TestDataAccess" as tda
491
+ da2 <|.. tda
492
+ }
493
+ }
494
+
495
+ note bottom of prod
496
+ 実際の Wikidata に接続
497
+ end note
498
+
499
+ note bottom of test
500
+ テストデータを返す
501
+ IO.pure で即座に結果を返す
502
+ end note
503
+
504
+ @enduml
505
+ ```
506
+
507
+ ### 12.5 プロパティベーステスト
508
+
509
+ ScalaCheck を使用したプロパティベーステスト:
510
+
511
+ ```scala
512
+ import org.scalacheck.Gen
513
+ import org.scalacheck.Prop.forAll
514
+
515
+ // ジェネレータの定義
516
+ val locationIdGen: Gen[LocationId] =
517
+ Gen.alphaNumStr.map(LocationId.apply)
518
+
519
+ val locationGen: Gen[Location] = for {
520
+ id <- locationIdGen
521
+ name <- Gen.alphaStr
522
+ population <- Gen.posNum[Int]
523
+ } yield Location(id, name, population)
524
+
525
+ val attractionGen: Gen[Attraction] = for {
526
+ name <- Gen.alphaStr
527
+ description <- Gen.option(Gen.alphaStr)
528
+ location <- locationGen
529
+ } yield Attraction(name, description, location)
530
+ ```
531
+
532
+ ```plantuml
533
+ @startuml
534
+ !theme plain
535
+
536
+ rectangle "プロパティベーステスト" {
537
+ card "Gen[A]" as gen
538
+ card "ランダムデータ生成" as random
539
+ card "プロパティ検証" as verify
540
+ card "100回以上テスト" as repeat
541
+
542
+ gen --> random
543
+ random --> verify
544
+ verify --> repeat
545
+ repeat --> random : 繰り返し
546
+ }
547
+
548
+ note bottom
549
+ ランダムな入力で
550
+ 不変条件を検証
551
+ end note
552
+
553
+ @enduml
554
+ ```
555
+
556
+ ### 12.6 不変条件のテスト
557
+
558
+ ```scala
559
+ // プロパティ: アトラクションの数は limit 以下
560
+ property("findAttractions returns at most limit attractions") =
561
+ forAll(Gen.alphaStr, Gen.posNum[Int]) { (name: String, limit: Int) =>
562
+ val result = dataAccess
563
+ .findAttractions(name, ByLocationPopulation, limit)
564
+ .unsafeRunSync()
565
+ result.size <= limit
566
+ }
567
+
568
+ // プロパティ: SearchReport のエラー数は検索数以下
569
+ property("SearchReport errors are bounded by searches") =
570
+ forAll(attractionGen) { attraction =>
571
+ val guide = travelGuide(testDataAccess, attraction.name).unsafeRunSync()
572
+ guide.forall { g =>
573
+ g.searchReport.errors.size <= 2 // artists + movies
574
+ }
575
+ }
576
+ ```
577
+
578
+ ### 12.7 統合テスト
579
+
580
+ ```scala
581
+ // 統合テスト: エンドツーエンドのシナリオ
582
+ test("travelGuide returns guide for valid attraction") {
583
+ val result = travelGuide(testDataAccess, "Test").unsafeRunSync()
584
+
585
+ assert(result.isDefined)
586
+ result.foreach { guide =>
587
+ assert(guide.attraction.name == "Test Attraction")
588
+ assert(guide.subjects.nonEmpty)
589
+ assert(guide.searchReport.errors.isEmpty)
590
+ }
591
+ }
592
+
593
+ // エラーケースのテスト
594
+ test("travelGuide handles errors gracefully") {
595
+ val failingDataAccess = new DataAccess {
596
+ def findAttractions(name: String, ordering: AttractionOrdering, limit: Int) =
597
+ IO.pure(List(testAttraction))
598
+
599
+ def findArtistsFromLocation(locationId: LocationId, limit: Int) =
600
+ IO.pure(Left("Network error"))
601
+
602
+ def findMoviesAboutLocation(locationId: LocationId, limit: Int) =
603
+ IO.pure(Left("Timeout"))
604
+ }
605
+
606
+ val result = travelGuide(failingDataAccess, "Test").unsafeRunSync()
607
+
608
+ assert(result.isDefined)
609
+ result.foreach { guide =>
610
+ assert(guide.searchReport.errors.size == 2)
611
+ }
612
+ }
613
+ ```
614
+
615
+ ### 12.8 テストピラミッド
616
+
617
+ ```plantuml
618
+ @startuml
619
+ !theme plain
620
+
621
+ rectangle "テストピラミッド" {
622
+ rectangle "E2E テスト\n(少数)" as e2e
623
+ rectangle "統合テスト\n(中程度)" as integration
624
+ rectangle "単体テスト + プロパティテスト\n(多数)" as unit
625
+ }
626
+
627
+ e2e -[hidden]down- integration
628
+ integration -[hidden]down- unit
629
+
630
+ note right of unit
631
+ FP では純粋関数が多いため
632
+ 単体テストが非常に効果的
633
+ end note
634
+
635
+ @enduml
636
+ ```
637
+
638
+ ---
639
+
640
+ ## まとめ
641
+
642
+ ### Part VI で学んだこと
643
+
644
+ ```plantuml
645
+ @startuml
646
+ !theme plain
647
+
648
+ rectangle "Part VI: 実践的なアプリケーション" {
649
+ rectangle "第11章" as ch11 {
650
+ card "ドメインモデル設計"
651
+ card "DataAccess 抽象化"
652
+ card "Resource 管理"
653
+ card "キャッシュ実装"
654
+ }
655
+
656
+ rectangle "第12章" as ch12 {
657
+ card "SearchReport"
658
+ card "スタブ/モック"
659
+ card "プロパティベーステスト"
660
+ card "統合テスト"
661
+ }
662
+ }
663
+
664
+ ch11 --> ch12
665
+
666
+ @enduml
667
+ ```
668
+
669
+ ### キーポイント
670
+
671
+ 1. **抽象化の重要性**: DataAccess トレイトで外部依存を抽象化
672
+ 2. **Resource でリソース管理**: 安全なリソースの取得と解放
673
+ 3. **Ref でキャッシュ**: スレッドセーフな状態管理
674
+ 4. **Either でエラー処理**: 明示的なエラーハンドリング
675
+ 5. **SearchReport**: テスト可能性と可観測性の向上
676
+ 6. **スタブ**: 外部依存を差し替えてテスト
677
+ 7. **プロパティベーステスト**: ランダム入力で不変条件を検証
678
+
679
+ ### 学習の総括
680
+
681
+ ```plantuml
682
+ @startuml
683
+ !theme plain
684
+ left to right direction
685
+
686
+ rectangle "FP の学習パス" {
687
+ card "Part I\n基礎" as p1
688
+ card "Part II\nイミュータブル操作" as p2
689
+ card "Part III\n型による安全性" as p3
690
+ card "Part IV\nIO/Stream" as p4
691
+ card "Part V\n並行処理" as p5
692
+ card "Part VI\n実践" as p6
693
+ }
694
+
695
+ p1 --> p2
696
+ p2 --> p3
697
+ p3 --> p4
698
+ p4 --> p5
699
+ p5 --> p6
700
+
701
+ @enduml
702
+ ```
703
+
704
+ ---
705
+
706
+ ## 演習問題
707
+
708
+ ### 問題 1: DataAccess の拡張
709
+
710
+ 以下の要件で `DataAccess` を拡張してください:
711
+ - 新しいメソッド `findHotelsNearLocation` を追加
712
+ - 戻り値は `IO[Either[String, List[Hotel]]]`
713
+
714
+ <details>
715
+ <summary>解答</summary>
716
+
717
+ ```scala
718
+ case class Hotel(name: String, rating: Double, location: Location)
719
+
720
+ trait DataAccess {
721
+ // 既存のメソッド...
722
+
723
+ def findHotelsNearLocation(
724
+ locationId: LocationId,
725
+ limit: Int
726
+ ): IO[Either[String, List[Hotel]]]
727
+ }
728
+
729
+ // テスト用スタブ
730
+ val testDataAccess: DataAccess = new DataAccess {
731
+ // 既存の実装...
732
+
733
+ def findHotelsNearLocation(
734
+ locationId: LocationId,
735
+ limit: Int
736
+ ): IO[Either[String, List[Hotel]]] =
737
+ IO.pure(Right(List(
738
+ Hotel("Test Hotel", 4.5, testLocation)
739
+ )))
740
+ }
741
+ ```
742
+
743
+ </details>
744
+
745
+ ### 問題 2: プロパティベーステスト
746
+
747
+ 以下の関数に対するプロパティベーステストを書いてください:
748
+
749
+ ```scala
750
+ def filterPopularLocations(
751
+ locations: List[Location],
752
+ minPopulation: Int
753
+ ): List[Location] =
754
+ locations.filter(_.population >= minPopulation)
755
+ ```
756
+
757
+ <details>
758
+ <summary>解答</summary>
759
+
760
+ ```scala
761
+ import org.scalacheck.Gen
762
+ import org.scalacheck.Prop.forAll
763
+
764
+ val locationGen: Gen[Location] = for {
765
+ id <- Gen.alphaNumStr.map(LocationId.apply)
766
+ name <- Gen.alphaStr
767
+ population <- Gen.chooseNum(0, 10000000)
768
+ } yield Location(id, name, population)
769
+
770
+ val locationsGen: Gen[List[Location]] =
771
+ Gen.listOf(locationGen)
772
+
773
+ // プロパティ1: 結果は入力以下の要素数
774
+ property("result size <= input size") =
775
+ forAll(locationsGen, Gen.posNum[Int]) {
776
+ (locations: List[Location], minPop: Int) =>
777
+ filterPopularLocations(locations, minPop).size <= locations.size
778
+ }
779
+
780
+ // プロパティ2: 結果のすべての要素は条件を満たす
781
+ property("all results meet minimum population") =
782
+ forAll(locationsGen, Gen.posNum[Int]) {
783
+ (locations: List[Location], minPop: Int) =>
784
+ filterPopularLocations(locations, minPop)
785
+ .forall(_.population >= minPop)
786
+ }
787
+
788
+ // プロパティ3: 条件を満たす要素はすべて結果に含まれる
789
+ property("all qualifying locations are in result") =
790
+ forAll(locationsGen, Gen.posNum[Int]) {
791
+ (locations: List[Location], minPop: Int) =>
792
+ val result = filterPopularLocations(locations, minPop)
793
+ locations.filter(_.population >= minPop).forall(result.contains)
794
+ }
795
+ ```
796
+
797
+ </details>
798
+
799
+ ### 問題 3: Resource の実装
800
+
801
+ ファイルを安全に読み取る `Resource` を実装してください。
802
+
803
+ <details>
804
+ <summary>解答</summary>
805
+
806
+ ```scala
807
+ import cats.effect.{IO, Resource}
808
+ import java.io.{BufferedReader, FileReader}
809
+
810
+ def fileResource(path: String): Resource[IO, BufferedReader] =
811
+ Resource.make(
812
+ IO.blocking(new BufferedReader(new FileReader(path)))
813
+ )(reader => IO.blocking(reader.close()))
814
+
815
+ def readLines(path: String): IO[List[String]] = {
816
+ fileResource(path).use { reader =>
817
+ IO.blocking {
818
+ Iterator
819
+ .continually(reader.readLine())
820
+ .takeWhile(_ != null)
821
+ .toList
822
+ }
823
+ }
824
+ }
825
+
826
+ // 使用例
827
+ val program: IO[Unit] = for {
828
+ lines <- readLines("data.txt")
829
+ _ <- IO.println(s"Read ${lines.size} lines")
830
+ } yield ()
831
+ ```
832
+
833
+ </details>
834
+
835
+ ---
836
+
837
+ ## シリーズ全体の総括
838
+
839
+ 本シリーズでは、「Grokking Functional Programming」の内容に沿って、関数型プログラミングの基礎から実践的なアプリケーション構築までを学びました。
840
+
841
+ ### 学んだ主な概念
842
+
843
+ | Part | 章 | 主な概念 |
844
+ |------|-----|----------|
845
+ | I | 1-2 | 純粋関数、参照透過性 |
846
+ | II | 3-5 | イミュータブルデータ、高階関数、flatMap |
847
+ | III | 6-7 | Option、Either、ADT |
848
+ | IV | 8-9 | IO モナド、Stream |
849
+ | V | 10 | 並行処理、Ref、Fiber |
850
+ | VI | 11-12 | 実践アプリケーション、テスト |
851
+
852
+ ### 関数型プログラミングの利点
853
+
854
+ 1. **予測可能性**: 純粋関数は同じ入力に対して常に同じ出力
855
+ 2. **テスト容易性**: 副作用がないためテストが簡単
856
+ 3. **合成可能性**: 小さな関数を組み合わせて複雑な処理を構築
857
+ 4. **並行安全性**: イミュータブルデータは競合状態を防ぐ
858
+ 5. **型安全性**: Option、Either で null や例外を型で表現
859
+
860
+ ### 次のステップ
861
+
862
+ - Cats Effect のより高度な機能を学ぶ
863
+ - fs2 の実践的な使用方法を深める
864
+ - ZIO など他の FP ライブラリを探索
865
+ - 実際のプロジェクトで FP を適用する