@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,715 @@
1
+ # 第11章: 実践アプリケーション — 11言語比較
2
+
3
+ ## 11.1 はじめに
4
+
5
+ 第 10 章までで、関数型プログラミングの基礎から並行処理まで幅広い概念を学んできました。本章では、これまでのすべての概念を統合し、**実践的なアプリケーション**を構築します。
6
+
7
+ 題材は **TravelGuide**(旅行ガイド)アプリケーションです。外部データソースからアトラクション、アーティスト、映画の情報を取得し、旅行ガイドを生成します。この過程で、**DataAccess の抽象化(DI パターン)**、**Resource によるリソース管理**、**キャッシュ**、**SearchReport による可観測性**という、FP アプリケーション設計の 4 つの柱を 11 言語で比較します。
8
+
9
+ ---
10
+
11
+ ## 11.2 共通の本質 — FP アプリケーション設計の 4 つの柱
12
+
13
+ ### 柱 1: イミュータブルなドメインモデル
14
+
15
+ すべての言語で、ドメインモデルはイミュータブルなデータ構造として定義されます。`Location`、`Attraction`、`TravelGuide` といった型は一度作成されたら変更できず、関数で新しい値を返す設計です。
16
+
17
+ ### 柱 2: DataAccess の抽象化(関数型 DI)
18
+
19
+ 外部依存をインターフェースで抽象化し、本番実装とテスト用スタブを差し替え可能にします。これにより、ビジネスロジックが外部システムから独立し、テスト容易性が飛躍的に向上します。
20
+
21
+ ### 柱 3: Resource による安全なリソース管理
22
+
23
+ データベース接続やファイルハンドルなどのリソースを、例外が発生しても確実に解放する仕組みです。`acquire` → `use` → `release` のパターンを型で保証します。
24
+
25
+ ### 柱 4: SearchReport による可観測性
26
+
27
+ 検索結果だけでなく、「何件検索したか」「どのエラーが発生したか」といったメタデータを `SearchReport` として返します。部分的な失敗があっても処理を続行し、失敗情報をレポートに含める設計です。
28
+
29
+ ---
30
+
31
+ ## 11.3 ドメインモデル — 全 11 言語比較
32
+
33
+ ### 代表 3 言語の詳細比較
34
+
35
+ **Scala — case class:**
36
+
37
+ ```scala
38
+ opaque type LocationId = String
39
+ case class Location(id: LocationId, name: String, population: Int)
40
+ case class Attraction(name: String, description: Option[String], location: Location)
41
+ case class TravelGuide(attraction: Attraction, subjects: List[String])
42
+ ```
43
+
44
+ **Haskell — data + record syntax:**
45
+
46
+ ```haskell
47
+ newtype LocationId = LocationId { unLocationId :: String }
48
+
49
+ data Location = Location
50
+ { locId :: LocationId, locName :: String, locPopulation :: Int }
51
+
52
+ data Attraction = Attraction
53
+ { attrName :: String, attrDescription :: Maybe String, attrLocation :: Location }
54
+
55
+ data TravelGuide = TravelGuide
56
+ { tgAttraction :: Attraction, tgSubjects :: [String], tgSearchReport :: SearchReport }
57
+ ```
58
+
59
+ **Rust — struct + Option:**
60
+
61
+ ```rust
62
+ #[derive(Debug, Clone, PartialEq, Eq, Hash)]
63
+ pub struct LocationId(pub String);
64
+
65
+ #[derive(Debug, Clone)]
66
+ pub struct Location { pub id: LocationId, pub name: String, pub population: i32 }
67
+
68
+ #[derive(Debug, Clone)]
69
+ pub struct Attraction { pub name: String, pub description: Option<String>, pub location: Location }
70
+
71
+ #[derive(Debug, Clone)]
72
+ pub struct TravelGuide { pub attraction: Attraction, pub subjects: Vec<String>, pub search_report: SearchReport }
73
+ ```
74
+
75
+ ### 全 11 言語のドメインモデル定義方式
76
+
77
+ <details>
78
+ <summary>Java — record + sealed interface</summary>
79
+
80
+ ```java
81
+ public record LocationId(String value) {}
82
+ public record Location(LocationId id, String name, int population) {}
83
+ public record Attraction(String name, Option<String> description, Location location) {}
84
+ public sealed interface PopCultureSubject permits Artist, Movie { String name(); }
85
+ public record Guide(Attraction attraction, List<PopCultureSubject> subjects) {}
86
+ ```
87
+
88
+ </details>
89
+
90
+ <details>
91
+ <summary>Python — frozen dataclass</summary>
92
+
93
+ ```python
94
+ LocationId = NewType("LocationId", str)
95
+
96
+ @dataclass(frozen=True)
97
+ class Location:
98
+ id: LocationId
99
+ name: str
100
+ population: int
101
+
102
+ @dataclass(frozen=True)
103
+ class Attraction:
104
+ name: str
105
+ description: str | None
106
+ location: Location
107
+ ```
108
+
109
+ </details>
110
+
111
+ <details>
112
+ <summary>TypeScript — readonly interface + ブランド型</summary>
113
+
114
+ ```typescript
115
+ type LocationId = string & { readonly _brand: unique symbol }
116
+
117
+ interface Location {
118
+ readonly id: LocationId
119
+ readonly name: string
120
+ readonly population: number
121
+ }
122
+
123
+ interface Attraction {
124
+ readonly name: string
125
+ readonly description: Option<string>
126
+ readonly location: Location
127
+ }
128
+ ```
129
+
130
+ </details>
131
+
132
+ <details>
133
+ <summary>F# — レコード型 + 判別共用体</summary>
134
+
135
+ ```fsharp
136
+ type LocationId = LocationId of string
137
+
138
+ type Location = { Id: LocationId; Name: string; Population: int }
139
+ type Attraction = { Name: string; Description: string option; Location: Location }
140
+ type TravelGuide = { Attraction: Attraction; Subjects: string list }
141
+ ```
142
+
143
+ </details>
144
+
145
+ <details>
146
+ <summary>C# — record</summary>
147
+
148
+ ```csharp
149
+ public readonly record struct LocationId(string Value);
150
+ public record Location(LocationId Id, string Name, int Population);
151
+ public record Attraction(string Name, Option<string> Description, Location Location);
152
+ public record TravelGuide(Attraction Attraction, Seq<string> Subjects);
153
+ ```
154
+
155
+ </details>
156
+
157
+ <details>
158
+ <summary>Clojure — defrecord</summary>
159
+
160
+ ```clojure
161
+ (defrecord Location [id name population])
162
+ (defrecord Attraction [name description location])
163
+ (defrecord TravelGuide [attraction subjects search-report])
164
+ ```
165
+
166
+ </details>
167
+
168
+ <details>
169
+ <summary>Elixir — defstruct + @type</summary>
170
+
171
+ ```elixir
172
+ defmodule Location do
173
+ defstruct [:id, :name, :population]
174
+ @type t :: %__MODULE__{id: LocationId.t(), name: String.t(), population: non_neg_integer()}
175
+ end
176
+
177
+ defmodule Attraction do
178
+ defstruct [:name, :description, :location]
179
+ @type t :: %__MODULE__{name: String.t(), description: String.t() | nil, location: Location.t()}
180
+ end
181
+ ```
182
+
183
+ </details>
184
+
185
+ <details>
186
+ <summary>Ruby — Struct</summary>
187
+
188
+ ```ruby
189
+ class LocationId
190
+ attr_reader :value
191
+ def initialize(value) = @value = value
192
+ end
193
+
194
+ Location = Struct.new(:id, :name, :population, keyword_init: true)
195
+ Attraction = Struct.new(:name, :description, :location, keyword_init: true)
196
+ TravelGuide = Struct.new(:attraction, :subjects, :search_report, keyword_init: true)
197
+ ```
198
+
199
+ </details>
200
+
201
+ ### イミュータブル性の保証レベル
202
+
203
+ | 言語 | 手段 | 保証レベル |
204
+ |------|------|-----------|
205
+ | Scala | `case class` (デフォルトで `val`) | コンパイル時 |
206
+ | Haskell | すべての値がイミュータブル | 言語レベル |
207
+ | Rust | デフォルトがイミュータブル (`mut` 明示) | コンパイル時 |
208
+ | Java | `record` | コンパイル時 |
209
+ | Python | `@dataclass(frozen=True)` | ランタイム |
210
+ | TypeScript | `readonly` 修飾子 | コンパイル時(型レベル) |
211
+ | F# | レコード型(デフォルトでイミュータブル) | コンパイル時 |
212
+ | C# | `record` + `readonly` | コンパイル時 |
213
+ | Clojure | すべての値がイミュータブル | 言語レベル |
214
+ | Elixir | すべての値がイミュータブル | 言語レベル |
215
+ | Ruby | `freeze` / 慣習 | 規約ベース |
216
+
217
+ ---
218
+
219
+ ## 11.4 DataAccess の抽象化 — 関数型 DI の全 11 言語比較
220
+
221
+ ### 代表 3 言語の詳細比較
222
+
223
+ FP における DI は、OOP のコンストラクタインジェクションとは異なり、**インターフェースを引数として受け取る**シンプルなパターンです。
224
+
225
+ **Scala — trait:**
226
+
227
+ ```scala
228
+ trait DataAccess {
229
+ def findAttractions(name: String, ordering: AttractionOrdering, limit: Int): IO[List[Attraction]]
230
+ def findArtistsFromLocation(locationId: LocationId, limit: Int): IO[List[MusicArtist]]
231
+ def findMoviesAboutLocation(locationId: LocationId, limit: Int): IO[List[Movie]]
232
+ }
233
+ ```
234
+
235
+ **Haskell — レコード型(関数の集合):**
236
+
237
+ ```haskell
238
+ data DataAccess = DataAccess
239
+ { findAttractions :: String -> AttractionOrdering -> Int -> IO [Attraction]
240
+ , findArtistsFromLocation :: LocationId -> Int -> IO (Either String [MusicArtist])
241
+ , findMoviesAboutLocation :: LocationId -> Int -> IO (Either String [Movie])
242
+ }
243
+ ```
244
+
245
+ **Rust — async_trait:**
246
+
247
+ ```rust
248
+ #[async_trait]
249
+ pub trait DataAccess: Send + Sync {
250
+ async fn find_attractions(&self, name: &str, ordering: AttractionOrdering, limit: usize)
251
+ -> Vec<Attraction>;
252
+ async fn find_artists_from_location(&self, location_id: &LocationId, limit: usize)
253
+ -> Result<Vec<MusicArtist>, String>;
254
+ async fn find_movies_about_location(&self, location_id: &LocationId, limit: usize)
255
+ -> Result<Vec<Movie>, String>;
256
+ }
257
+ ```
258
+
259
+ ### 全 11 言語の DataAccess 抽象化
260
+
261
+ <details>
262
+ <summary>Java — interface</summary>
263
+
264
+ ```java
265
+ public interface DataAccess {
266
+ IO<List<Attraction>> findAttractions(String name, AttractionOrdering ordering, int limit);
267
+ IO<List<Artist>> findArtistsFromLocation(LocationId locationId, int limit);
268
+ IO<List<Movie>> findMoviesAboutLocation(LocationId locationId, int limit);
269
+ }
270
+ ```
271
+
272
+ </details>
273
+
274
+ <details>
275
+ <summary>Python — ABC(抽象基底クラス)</summary>
276
+
277
+ ```python
278
+ class DataAccess(ABC):
279
+ @abstractmethod
280
+ def find_attractions(self, name: str, ordering: AttractionOrdering, limit: int) -> list[Attraction]:
281
+ pass
282
+
283
+ @abstractmethod
284
+ def find_artists_from_location(self, location_id: LocationId, limit: int) -> Result[list[MusicArtist], str]:
285
+ pass
286
+ ```
287
+
288
+ </details>
289
+
290
+ <details>
291
+ <summary>TypeScript — interface(関数フィールド)</summary>
292
+
293
+ ```typescript
294
+ interface DataAccess {
295
+ readonly findAttractions: (name: string, ordering: AttractionOrdering, limit: number)
296
+ => Task<readonly Attraction[]>
297
+ readonly findArtistsFromLocation: (locationId: LocationId, limit: number)
298
+ => Task<Either<string, readonly MusicArtist[]>>
299
+ readonly findMoviesAboutLocation: (locationId: LocationId, limit: number)
300
+ => Task<Either<string, readonly Movie[]>>
301
+ }
302
+ ```
303
+
304
+ </details>
305
+
306
+ <details>
307
+ <summary>F# — abstract type</summary>
308
+
309
+ ```fsharp
310
+ type IDataAccess =
311
+ abstract member FindAttractions:
312
+ name: string * ordering: AttractionOrdering * limit: int -> Async<Attraction list>
313
+ abstract member FindArtistsFromLocation:
314
+ locationId: LocationId * limit: int -> Async<MusicArtist list>
315
+ ```
316
+
317
+ </details>
318
+
319
+ <details>
320
+ <summary>C# — interface</summary>
321
+
322
+ ```csharp
323
+ public interface IDataAccess
324
+ {
325
+ Task<Seq<Attraction>> FindAttractions(string name, AttractionOrdering ordering, int limit);
326
+ Task<Seq<MusicArtist>> FindArtistsFromLocation(LocationId locationId, int limit);
327
+ Task<Seq<Movie>> FindMoviesAboutLocation(LocationId locationId, int limit);
328
+ }
329
+ ```
330
+
331
+ </details>
332
+
333
+ <details>
334
+ <summary>Clojure — defprotocol</summary>
335
+
336
+ ```clojure
337
+ (defprotocol DataAccess
338
+ (find-attractions [this name ordering limit])
339
+ (find-artists-from-location [this location-id limit])
340
+ (find-movies-about-location [this location-id limit]))
341
+ ```
342
+
343
+ </details>
344
+
345
+ <details>
346
+ <summary>Elixir — @behaviour</summary>
347
+
348
+ ```elixir
349
+ defmodule DataAccess do
350
+ @callback find_attractions(String.t(), ordering(), pos_integer()) ::
351
+ {:ok, [Attraction.t()]} | {:error, String.t()}
352
+ @callback find_artists_from_location(LocationId.t(), pos_integer()) ::
353
+ {:ok, [MusicArtist.t()]} | {:error, String.t()}
354
+ end
355
+ ```
356
+
357
+ </details>
358
+
359
+ <details>
360
+ <summary>Ruby — module(duck typing)</summary>
361
+
362
+ ```ruby
363
+ module DataAccess
364
+ def find_attractions(name, ordering, limit)
365
+ raise NotImplementedError
366
+ end
367
+
368
+ def find_artists_from_location(location_id, limit)
369
+ raise NotImplementedError
370
+ end
371
+ end
372
+ ```
373
+
374
+ </details>
375
+
376
+ ### DI 抽象化パターンの比較
377
+
378
+ | 言語 | 抽象化手段 | エフェクト型 | エラー表現 |
379
+ |------|-----------|-------------|-----------|
380
+ | Scala | `trait` | `IO[A]` | `IO[List[A]]`(IO 内で例外) |
381
+ | Haskell | レコード型 | `IO a` | `IO (Either String [a])` |
382
+ | Rust | `#[async_trait] trait` | `async fn -> T` | `Result<Vec<A>, String>` |
383
+ | Java | `interface` | `IO<A>` | `IO<List<A>>` |
384
+ | Python | `ABC` | 直接値 / `Result` | `Result[list[A], str]` |
385
+ | TypeScript | `interface`(関数フィールド) | `Task<A>` | `Task<Either<string, A[]>>` |
386
+ | F# | `abstract type` | `Async<'a>` | `Async<'a list>` |
387
+ | C# | `interface` | `Task<T>` | `Task<Seq<T>>` |
388
+ | Clojure | `defprotocol` | 直接値 / `{:ok v}` | `{:ok v}` / `{:error msg}` |
389
+ | Elixir | `@behaviour` | 直接値 | `{:ok, v}` / `{:error, msg}` |
390
+ | Ruby | `module`(duck typing) | `IO[A]` | `{success: bool, value: A}` |
391
+
392
+ **注目すべき相違点**: Haskell はレコード型の関数フィールドとしてインターフェースを表現し、TypeScript も同様のアプローチを取ります。一方、Scala、Java、Rust、F#、C# は伝統的なインターフェース/トレイト、Clojure はプロトコル、Elixir はビヘイビアを使用します。表現形式は異なりますが、**「実装の詳細を隠蔽し、テスト時に差し替え可能にする」** という本質は全言語で共通です。
393
+
394
+ ---
395
+
396
+ ## 11.5 Resource — 安全なリソース管理の全 11 言語比較
397
+
398
+ ### 代表 3 言語の詳細比較
399
+
400
+ **Scala — cats-effect Resource:**
401
+
402
+ ```scala
403
+ def execQuery(query: String): Resource[IO, List[QuerySolution]] = {
404
+ val connection: Resource[IO, RDFConnection] =
405
+ Resource.make(
406
+ IO.blocking(RDFConnectionRemote.create().destination(endpoint).build())
407
+ )(conn => IO.blocking(conn.close()))
408
+
409
+ for {
410
+ conn <- connection
411
+ result <- Resource.eval(IO.blocking { conn.query(query).execSelect().asScala.toList })
412
+ } yield result
413
+ }
414
+ ```
415
+
416
+ **Java — Resource(自作):**
417
+
418
+ ```java
419
+ public final class Resource<A> {
420
+ private final IO<A> acquire;
421
+ private final Consumer<A> release;
422
+
423
+ public static <A extends AutoCloseable> Resource<A> fromAutoCloseable(IO<A> acquire) {
424
+ return new Resource<>(acquire, a -> { try { a.close(); } catch (Exception e) { throw new RuntimeException(e); } });
425
+ }
426
+
427
+ public <B> IO<B> use(Function<A, IO<B>> f) {
428
+ return acquire.flatMap(resource ->
429
+ f.apply(resource).guarantee(IO.effect(() -> release.accept(resource))));
430
+ }
431
+ }
432
+ ```
433
+
434
+ **Python — contextlib ベース:**
435
+
436
+ ```python
437
+ class Resource(Generic[T]):
438
+ def __init__(self, acquire: Callable[[], T], release: Callable[[T], None]) -> None:
439
+ self._acquire = acquire
440
+ self._release = release
441
+
442
+ @contextmanager
443
+ def use(self) -> Generator[T, None, None]:
444
+ resource = self._acquire()
445
+ try:
446
+ yield resource
447
+ finally:
448
+ self._release(resource)
449
+ ```
450
+
451
+ ### Resource パターンの言語別対応
452
+
453
+ | 言語 | Resource 型 | ネイティブサポート | 特徴 |
454
+ |------|------------|-------------------|------|
455
+ | Scala | `Resource[IO, A]` | cats-effect ライブラリ | for 内包表記で合成可能 |
456
+ | Haskell | `bracket` / `ResourceT` | 標準ライブラリ | `bracket acquire release use` |
457
+ | Rust | `Drop` trait(RAII) | 言語レベル | 所有権でスコープ管理 |
458
+ | Java | `Resource<A>` (自作) | `try-with-resources` | `AutoCloseable` との統合 |
459
+ | Python | `Resource[T]` (自作) | `with` 文 / `contextlib` | コンテキストマネージャ |
460
+ | TypeScript | `Resource<A>` (自作) | なし | Promise ベースの保証 |
461
+ | F# | `Resource<'a>` (自作) | `use` キーワード | `IDisposable` との統合 |
462
+ | C# | `Resource<T>` (自作) | `using` 文 | `IDisposable` / `IAsyncDisposable` |
463
+ | Clojure | `with-open` | 標準マクロ | Java の AutoCloseable を活用 |
464
+ | Elixir | `Agent` + `try/after` | OTP プロセス | プロセス終了時に自動解放 |
465
+ | Ruby | `Resource` (自作) | `ensure` | ブロック + ensure パターン |
466
+
467
+ **注目点**: Rust は RAII(Resource Acquisition Is Initialization)により、所有権がスコープを外れた時点で自動的にリソースが解放されるため、明示的な Resource 型が不要です。これは Rust の所有権システムが並行処理だけでなくリソース管理でも強力に機能することを示しています。
468
+
469
+ ---
470
+
471
+ ## 11.6 テスト用スタブ — DI の恩恵
472
+
473
+ ### 代表 3 言語のスタブ実装
474
+
475
+ DataAccess を抽象化した最大の恩恵は、テスト時にスタブを差し替えられることです。
476
+
477
+ **Scala — 匿名 trait 実装:**
478
+
479
+ ```scala
480
+ def cachedAttractions(dataAccess: DataAccess): IO[DataAccess] = for {
481
+ cache <- Ref.of[IO, Map[String, List[Attraction]]](Map.empty)
482
+ } yield new DataAccess {
483
+ def findAttractions(name: String, ordering: AttractionOrdering, limit: Int): IO[List[Attraction]] = {
484
+ val key = s"$name-$ordering-$limit"
485
+ for {
486
+ cached <- cache.get.map(_.get(key))
487
+ result <- cached match {
488
+ case Some(v) => IO.pure(v)
489
+ case None => dataAccess.findAttractions(name, ordering, limit)
490
+ .flatTap(r => cache.update(_ + (key -> r)))
491
+ }
492
+ } yield result
493
+ }
494
+ // ...
495
+ }
496
+ ```
497
+
498
+ **Haskell — レコード値の差し替え:**
499
+
500
+ ```haskell
501
+ mkTestDataAccess :: IO DataAccess
502
+ mkTestDataAccess = return DataAccess
503
+ { findAttractions = \name _ limit ->
504
+ return $ take limit [Attraction "Test" (Just "desc") testLocation | name /= ""]
505
+ , findArtistsFromLocation = \_ limit -> return $ Right $ take limit [MusicArtist "Artist"]
506
+ , findMoviesAboutLocation = \_ limit -> return $ Right $ take limit [Movie "Movie"]
507
+ }
508
+
509
+ mkFailingDataAccess :: IO DataAccess
510
+ mkFailingDataAccess = return DataAccess
511
+ { findAttractions = \_ _ limit -> return $ take limit [testAttraction]
512
+ , findArtistsFromLocation = \_ _ -> return $ Left "Network error"
513
+ , findMoviesAboutLocation = \_ _ -> return $ Left "Timeout"
514
+ }
515
+ ```
516
+
517
+ **Rust — Builder パターンのスタブ:**
518
+
519
+ ```rust
520
+ pub struct StubDataAccess {
521
+ attractions: Vec<Attraction>,
522
+ artists_error: Option<String>,
523
+ movies_error: Option<String>,
524
+ }
525
+
526
+ impl StubDataAccess {
527
+ pub fn new() -> Self { /* ... */ }
528
+ pub fn with_attractions(mut self, v: Vec<Attraction>) -> Self { self.attractions = v; self }
529
+ pub fn with_artists_error(mut self, e: &str) -> Self { self.artists_error = Some(e.into()); self }
530
+ }
531
+
532
+ #[async_trait]
533
+ impl DataAccess for StubDataAccess { /* ... */ }
534
+ ```
535
+
536
+ ### スタブパターンの比較
537
+
538
+ | 言語 | スタブ実装方式 | 特徴 |
539
+ |------|-------------|------|
540
+ | Scala | 匿名 trait / テスト用 object | `new DataAccess { ... }` |
541
+ | Haskell | レコード値の直接構築 | 関数フィールドを差し替え |
542
+ | Rust | Builder パターン + impl | `.with_attractions(...)` チェーン |
543
+ | Java | 匿名クラス / ラムダ | `new DataAccess() { ... }` |
544
+ | Python | テスト用クラス | `TestDataAccess(DataAccess)` |
545
+ | TypeScript | オブジェクトリテラル | `{ findAttractions: ... }` |
546
+ | F# | オブジェクト式 | `{ new IDataAccess with ... }` |
547
+ | C# | クラス実装 | `class TestDataAccess : IDataAccess` |
548
+ | Clojure | defrecord + protocol 実装 | `(->TestDataAccess ...)` |
549
+ | Elixir | @behaviour 実装モジュール | `defmodule TestDataAccess do` |
550
+ | Ruby | include DataAccess | `class InMemoryDataAccess` |
551
+
552
+ **注目点**: Haskell と TypeScript は DataAccess がレコード/オブジェクトリテラルのため、フィールドを直接差し替えるだけでスタブが作れます。これは「DI コンテナ」のような重い仕組みが不要であることを示しています。
553
+
554
+ ---
555
+
556
+ ## 11.7 SearchReport — 部分的失敗への対処
557
+
558
+ ### 共通パターン
559
+
560
+ TravelGuide アプリケーションでは、アーティスト検索や映画検索が失敗しても、アトラクション情報だけで旅行ガイドを生成します。失敗情報は `SearchReport` に記録されます。
561
+
562
+ **Scala のアプリケーションロジック(概要):**
563
+
564
+ ```scala
565
+ def travelGuide(da: DataAccess, name: String): IO[Option[TravelGuide]] = for {
566
+ attractions <- da.findAttractions(name, ByLocationPopulation, 3)
567
+ guide <- attractions.headOption.traverse { attraction =>
568
+ val locId = attraction.location.id
569
+ for {
570
+ artists <- da.findArtistsFromLocation(locId, 2).attempt
571
+ movies <- da.findMoviesAboutLocation(locId, 2).attempt
572
+ } yield TravelGuide(attraction, collectSubjects(artists, movies))
573
+ }
574
+ } yield guide
575
+ ```
576
+
577
+ **Haskell のアプリケーションロジック(概要):**
578
+
579
+ ```haskell
580
+ travelGuideWithReport :: DataAccess -> String -> IO (Maybe TravelGuide)
581
+ travelGuideWithReport da name = do
582
+ attractions <- findAttractions da name AttrByLocationPopulation 3
583
+ case attractions of
584
+ [] -> return Nothing
585
+ (a:_) -> do
586
+ artists <- findArtistsFromLocation da (locId $ attrLocation a) 2
587
+ movies <- findMoviesAboutLocation da (locId $ attrLocation a) 2
588
+ let errors = collectErrors [artists, movies]
589
+ return $ Just TravelGuide { tgAttraction = a, tgSubjects = ..., tgSearchReport = ... }
590
+ ```
591
+
592
+ このパターンの共通構造は以下です。
593
+
594
+ 1. アトラクションを検索(失敗 → `None` / `Nothing` を返す)
595
+ 2. アーティストと映画を検索(**失敗しても続行**)
596
+ 3. エラーを `SearchReport` に収集
597
+ 4. 成功した結果だけで `TravelGuide` を組み立て
598
+
599
+ ---
600
+
601
+ ## 11.8 比較分析
602
+
603
+ ### DI パターンの 3 つのアプローチ
604
+
605
+ 11 言語の DataAccess 抽象化は、以下の 3 つのアプローチに分類できます。
606
+
607
+ #### アプローチ 1: 型システム統合型
608
+
609
+ 言語の型システムと深く統合され、コンパイル時に実装の完全性が検証されます。
610
+
611
+ **採用言語**: Scala(trait)、Rust(trait + async_trait)、Java(interface)、F#(abstract type)、C#(interface)
612
+
613
+ **利点**: 実装漏れをコンパイル時に検出、IDE サポートが充実
614
+
615
+ #### アプローチ 2: レコード/オブジェクト型
616
+
617
+ インターフェースを関数のレコード(またはオブジェクトリテラル)として表現します。
618
+
619
+ **採用言語**: Haskell(data record)、TypeScript(interface with function fields)
620
+
621
+ **利点**: 軽量でスタブ作成が容易、部分的な差し替えが可能
622
+
623
+ #### アプローチ 3: プロトコル/ビヘイビア型
624
+
625
+ 言語固有のポリモーフィズム機構を使用します。
626
+
627
+ **採用言語**: Clojure(defprotocol)、Elixir(@behaviour)、Ruby(module/duck typing)
628
+
629
+ **利点**: 動的言語の柔軟性を活かしつつ構造を提供
630
+
631
+ ### リソース管理の安全性スペクトラム
632
+
633
+ | 安全性レベル | 言語 | 仕組み |
634
+ |-------------|------|-------|
635
+ | **所有権ベース(最高)** | Rust | RAII + Drop trait。コンパイル時にリソースリークを防止 |
636
+ | **型レベル(高)** | Scala | `Resource[IO, A]` が確実な解放を型で保証 |
637
+ | **言語構文(中〜高)** | Java, Python, F#, C# | try-with-resources / with / use / using |
638
+ | **プロセスベース(中〜高)** | Elixir | プロセス終了時に自動クリーンアップ |
639
+ | **規約ベース(低〜中)** | TypeScript, Ruby, Clojure, Haskell | 開発者が明示的に bracket / ensure を使用 |
640
+
641
+ ### テスト容易性の比較
642
+
643
+ すべての言語で DataAccess の抽象化によりテスト容易性が確保されていますが、スタブ作成の容易さには差があります。
644
+
645
+ | 容易さ | 言語 | 理由 |
646
+ |--------|------|------|
647
+ | **最も容易** | Haskell, TypeScript | レコード/オブジェクトのフィールドを直接構築 |
648
+ | **容易** | Scala, Clojure | 匿名実装 / defrecord |
649
+ | **標準** | Java, Rust, F#, C#, Elixir | クラス/モジュール実装が必要 |
650
+ | **やや煩雑** | Python, Ruby | ABC / module の実装 + ボイラープレート |
651
+
652
+ ---
653
+
654
+ ## 11.9 言語固有の特徴
655
+
656
+ ### Scala — for 内包表記による Resource の合成
657
+
658
+ Scala の `Resource` は for 内包表記で合成可能であり、複数のリソースを宣言的に扱えます。データベース接続とクエリ実行を一つの式で安全に結合できます。
659
+
660
+ ### Haskell — レコード型 DI の優雅さ
661
+
662
+ Haskell の DataAccess はレコード型の関数フィールドであり、テスト時にはレコード値を直接構築するだけです。DI フレームワークが一切不要で、純粋な関数合成だけで依存性の注入が完結します。
663
+
664
+ ### Rust — 所有権による二重の安全性
665
+
666
+ Rust では `trait DataAccess: Send + Sync` により、DataAccess 実装がスレッド安全であることをコンパイル時に保証します。さらに RAII によるリソース管理は、Resource 型を明示的に使わなくても安全です。
667
+
668
+ ### Clojure — プロトコルと REPL 駆動開発
669
+
670
+ Clojure のプロトコルは実行時にも差し替え可能であり、REPL でインタラクティブにスタブを切り替えながら開発できます。
671
+
672
+ ### Elixir — @behaviour + OTP の統合
673
+
674
+ Elixir の @behaviour はコンパイル時の警告を提供しつつ、Agent によるキャッシュ実装で OTP のプロセスモデルを自然に活用できます。
675
+
676
+ ### Java — record + sealed interface による FP 表現
677
+
678
+ Java 17 の record と sealed interface により、Scala の case class と ADT に近い表現力が得られます。`AutoCloseable` との統合で、既存の Java エコシステムとの互換性も維持しています。
679
+
680
+ ---
681
+
682
+ ## 11.10 選択指針
683
+
684
+ ### プロジェクト特性別の推奨
685
+
686
+ | プロジェクト特性 | 推奨アプローチ | 適切な言語 |
687
+ |----------------|-------------|-----------|
688
+ | 大規模エンタープライズ | 型システム統合型 DI | Scala, Java, C#, F# |
689
+ | マイクロサービス | プロセスベース DI | Elixir, Clojure |
690
+ | 高性能システム | 所有権ベースリソース管理 | Rust |
691
+ | 学術/研究 | レコード型 DI | Haskell |
692
+ | Web フロントエンド | オブジェクト型 DI | TypeScript |
693
+ | プロトタイプ/スクリプト | duck typing DI | Python, Ruby |
694
+
695
+ ### FP アプリケーション設計のベストプラクティス
696
+
697
+ 1. **ドメインモデルはイミュータブルに** — すべての言語で `case class` / `record` / `defstruct` 等を使用
698
+ 2. **外部依存はインターフェースで抽象化** — テスト容易性と差し替え可能性を確保
699
+ 3. **リソースは型で管理** — `Resource` / `bracket` / RAII で確実な解放を保証
700
+ 4. **部分的失敗に対処** — `SearchReport` パターンで失敗情報を収集しつつ処理を続行
701
+ 5. **純粋関数とエフェクトを分離** — ビジネスロジックは純粋に、IO は境界に
702
+
703
+ ---
704
+
705
+ ## 11.11 まとめ
706
+
707
+ 本章では、TravelGuide アプリケーションを題材に、11 言語の FP アプリケーション設計パターンを比較しました。
708
+
709
+ **共通する本質**: すべての言語が「イミュータブルなドメインモデル」「DataAccess の抽象化による DI」「安全なリソース管理」「部分的失敗への対処」という 4 つの柱を共有しています。FP の原則に従えば、どの言語でも同じ設計思想でアプリケーションを構築できます。
710
+
711
+ **根本的な違い**: DI の表現形式に最大の差があります。Haskell のレコード型 DI は DI フレームワーク不要の最軽量アプローチ、Scala/Java/C# の trait/interface は IDE サポートと型安全性が充実、Clojure/Elixir のプロトコル/ビヘイビアは動的言語の柔軟性を活かします。
712
+
713
+ **実践的な教訓**: FP アプリケーション設計では、DI コンテナや複雑なフレームワークは不要です。「インターフェースを引数として受け取る」というシンプルなパターンだけで、テスト容易性、保守性、拡張性のすべてが得られます。これが 11 言語を通じて見えてくる FP の最も重要な実践的価値です。
714
+
715
+ 本シリーズの全 12 章を通じて、関数型プログラミングの基礎から実践まで、11 言語の視点から包括的に比較してきました。各言語には固有の強みがありますが、**純粋関数、イミュータブルデータ、型による安全性、宣言的な合成**という FP の核心は普遍です。