@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,729 @@
1
+ # 第14章: Abstract Server パターン
2
+
3
+ ## はじめに
4
+
5
+ Abstract Server パターンは、依存関係逆転の原則(DIP)を実現するパターンです。高レベルモジュールが低レベルモジュールの詳細に依存するのではなく、両者が抽象(プロトコル)に依存することで疎結合を実現します。
6
+
7
+ Elixir では、プロトコルを使用して Abstract Server を定義し、異なる具体的な実装を提供します。
8
+
9
+ ## 1. Switchable パターン
10
+
11
+ ### Switchable プロトコル
12
+
13
+ ```elixir
14
+ defprotocol Switchable do
15
+ @moduledoc "スイッチ可能なデバイスのプロトコル"
16
+
17
+ @doc "デバイスをオンにする"
18
+ @spec turn_on(t()) :: t()
19
+ def turn_on(device)
20
+
21
+ @doc "デバイスをオフにする"
22
+ @spec turn_off(t()) :: t()
23
+ def turn_off(device)
24
+
25
+ @doc "デバイスがオンかどうか"
26
+ @spec on?(t()) :: boolean()
27
+ def on?(device)
28
+ end
29
+ ```
30
+
31
+ ### Concrete Server: Light
32
+
33
+ ```elixir
34
+ defmodule Light do
35
+ defstruct state: :off, brightness: 100
36
+
37
+ def new, do: %__MODULE__{}
38
+ def new(brightness), do: %__MODULE__{brightness: brightness}
39
+
40
+ def set_brightness(%__MODULE__{state: :on} = light, brightness)
41
+ when brightness >= 0 and brightness <= 100 do
42
+ %{light | brightness: brightness}
43
+ end
44
+ def set_brightness(light, _), do: light
45
+ end
46
+
47
+ defimpl Switchable, for: Light do
48
+ def turn_on(%Light{} = light), do: %{light | state: :on}
49
+ def turn_off(%Light{} = light), do: %{light | state: :off}
50
+ def on?(%Light{state: state}), do: state == :on
51
+ end
52
+ ```
53
+
54
+ ### Concrete Server: Fan
55
+
56
+ ```elixir
57
+ defmodule Fan do
58
+ defstruct state: :off, speed: nil
59
+
60
+ def new, do: %__MODULE__{}
61
+
62
+ def set_speed(%__MODULE__{state: :on} = fan, speed)
63
+ when speed in [:low, :medium, :high] do
64
+ %{fan | speed: speed}
65
+ end
66
+ def set_speed(fan, _), do: fan
67
+ end
68
+
69
+ defimpl Switchable, for: Fan do
70
+ def turn_on(%Fan{} = fan), do: %{fan | state: :on, speed: :low}
71
+ def turn_off(%Fan{} = fan), do: %{fan | state: :off, speed: nil}
72
+ def on?(%Fan{state: state}), do: state == :on
73
+ end
74
+ ```
75
+
76
+ ### Concrete Server: Motor
77
+
78
+ ```elixir
79
+ defmodule Motor do
80
+ defstruct state: :off, direction: nil, rpm: 0
81
+
82
+ def new, do: %__MODULE__{}
83
+
84
+ def reverse(%__MODULE__{state: :on, direction: dir} = motor) do
85
+ new_dir = if dir == :forward, do: :reverse, else: :forward
86
+ %{motor | direction: new_dir}
87
+ end
88
+ def reverse(motor), do: motor
89
+
90
+ def set_rpm(%__MODULE__{state: :on} = motor, rpm) when rpm >= 0 do
91
+ %{motor | rpm: rpm}
92
+ end
93
+ def set_rpm(motor, _), do: motor
94
+ end
95
+
96
+ defimpl Switchable, for: Motor do
97
+ def turn_on(%Motor{} = motor), do: %{motor | state: :on, direction: :forward, rpm: 1000}
98
+ def turn_off(%Motor{} = motor), do: %{motor | state: :off, direction: nil, rpm: 0}
99
+ def on?(%Motor{state: state}), do: state == :on
100
+ end
101
+ ```
102
+
103
+ ### Client: Switch
104
+
105
+ ```elixir
106
+ defmodule Switch do
107
+ @moduledoc """
108
+ スイッチクライアント。
109
+ Switchable プロトコルを通じてデバイスを操作する。
110
+ """
111
+
112
+ @doc "スイッチを入れる"
113
+ def engage(device), do: Switchable.turn_on(device)
114
+
115
+ @doc "スイッチを切る"
116
+ def disengage(device), do: Switchable.turn_off(device)
117
+
118
+ @doc "スイッチを切り替える"
119
+ def toggle(device) do
120
+ if Switchable.on?(device) do
121
+ Switchable.turn_off(device)
122
+ else
123
+ Switchable.turn_on(device)
124
+ end
125
+ end
126
+
127
+ @doc "デバイスの状態を取得"
128
+ def status(device), do: if(Switchable.on?(device), do: :on, else: :off)
129
+ end
130
+ ```
131
+
132
+ ### 使用例
133
+
134
+ ```elixir
135
+ # Light を操作
136
+ light = Light.new()
137
+ light = Switch.engage(light) # オン
138
+ light = Switch.toggle(light) # オフ
139
+
140
+ # Fan を操作(同じ Switch コード)
141
+ fan = Fan.new()
142
+ fan = Switch.engage(fan) # オン(speed: :low)
143
+ fan = Fan.set_speed(fan, :high) # 速度を変更
144
+ fan = Switch.disengage(fan) # オフ
145
+
146
+ # Motor を操作(同じ Switch コード)
147
+ motor = Motor.new()
148
+ motor = Switch.engage(motor) # オン(direction: :forward, rpm: 1000)
149
+ motor = Motor.reverse(motor) # 方向を反転
150
+ ```
151
+
152
+ ## 2. Repository パターン
153
+
154
+ ### Repository プロトコル
155
+
156
+ ```elixir
157
+ defprotocol Repository do
158
+ @moduledoc "データリポジトリのプロトコル"
159
+
160
+ @doc "IDでエンティティを取得"
161
+ @spec find_by_id(t(), String.t()) :: {:ok, map()} | {:error, :not_found}
162
+ def find_by_id(repo, id)
163
+
164
+ @doc "全てのエンティティを取得"
165
+ @spec find_all(t()) :: [map()]
166
+ def find_all(repo)
167
+
168
+ @doc "エンティティを保存"
169
+ @spec save(t(), map()) :: {:ok, map()}
170
+ def save(repo, entity)
171
+
172
+ @doc "エンティティを削除"
173
+ @spec delete(t(), String.t()) :: {:ok, map()} | {:error, :not_found}
174
+ def delete(repo, id)
175
+ end
176
+ ```
177
+
178
+ ### Concrete Server: MemoryRepository
179
+
180
+ ```elixir
181
+ defmodule MemoryRepository do
182
+ @moduledoc "インメモリリポジトリ"
183
+
184
+ defstruct [:agent]
185
+
186
+ def new do
187
+ {:ok, agent} = Agent.start_link(fn -> %{} end)
188
+ %__MODULE__{agent: agent}
189
+ end
190
+
191
+ def stop(%__MODULE__{agent: agent}) do
192
+ if Process.alive?(agent) do
193
+ Agent.stop(agent)
194
+ else
195
+ :ok
196
+ end
197
+ end
198
+ end
199
+
200
+ defimpl Repository, for: MemoryRepository do
201
+ def find_by_id(%MemoryRepository{agent: agent}, id) do
202
+ case Agent.get(agent, &Map.get(&1, id)) do
203
+ nil -> {:error, :not_found}
204
+ entity -> {:ok, entity}
205
+ end
206
+ end
207
+
208
+ def find_all(%MemoryRepository{agent: agent}) do
209
+ Agent.get(agent, &Map.values(&1))
210
+ end
211
+
212
+ def save(%MemoryRepository{agent: agent}, entity) do
213
+ id = Map.get(entity, :id) || generate_id()
214
+ entity_with_id = Map.put(entity, :id, id)
215
+ Agent.update(agent, &Map.put(&1, id, entity_with_id))
216
+ {:ok, entity_with_id}
217
+ end
218
+
219
+ def delete(%MemoryRepository{agent: agent}, id) do
220
+ case Agent.get(agent, &Map.get(&1, id)) do
221
+ nil ->
222
+ {:error, :not_found}
223
+ entity ->
224
+ Agent.update(agent, &Map.delete(&1, id))
225
+ {:ok, entity}
226
+ end
227
+ end
228
+
229
+ defp generate_id, do: :crypto.strong_rand_bytes(8) |> Base.encode16(case: :lower)
230
+ end
231
+ ```
232
+
233
+ ### Concrete Server: MockRepository(テスト用)
234
+
235
+ ```elixir
236
+ defmodule MockRepository do
237
+ @moduledoc "モックリポジトリ(テスト用)"
238
+
239
+ defstruct data: %{}
240
+
241
+ def new(data \\ %{}), do: %__MODULE__{data: data}
242
+ end
243
+
244
+ defimpl Repository, for: MockRepository do
245
+ def find_by_id(%MockRepository{data: data}, id) do
246
+ case Map.get(data, id) do
247
+ nil -> {:error, :not_found}
248
+ entity -> {:ok, entity}
249
+ end
250
+ end
251
+
252
+ def find_all(%MockRepository{data: data}), do: Map.values(data)
253
+
254
+ def save(%MockRepository{}, entity) do
255
+ id = Map.get(entity, :id) || "mock-id"
256
+ entity_with_id = Map.put(entity, :id, id)
257
+ {:ok, entity_with_id}
258
+ end
259
+
260
+ def delete(%MockRepository{data: data}, id) do
261
+ case Map.get(data, id) do
262
+ nil -> {:error, :not_found}
263
+ entity -> {:ok, entity}
264
+ end
265
+ end
266
+ end
267
+ ```
268
+
269
+ ## 3. Logger パターン
270
+
271
+ ### Logger プロトコル
272
+
273
+ ```elixir
274
+ defprotocol Logger do
275
+ @moduledoc "ロガーのプロトコル"
276
+
277
+ @doc "デバッグログ"
278
+ @spec debug(t(), String.t()) :: t()
279
+ def debug(logger, message)
280
+
281
+ @doc "情報ログ"
282
+ @spec info(t(), String.t()) :: t()
283
+ def info(logger, message)
284
+
285
+ @doc "警告ログ"
286
+ @spec warn(t(), String.t()) :: t()
287
+ def warn(logger, message)
288
+
289
+ @doc "エラーログ"
290
+ @spec error(t(), String.t()) :: t()
291
+ def error(logger, message)
292
+ end
293
+ ```
294
+
295
+ ### Concrete Server: ConsoleLogger
296
+
297
+ ```elixir
298
+ defmodule ConsoleLogger do
299
+ @moduledoc "コンソールロガー"
300
+ defstruct level: :debug
301
+
302
+ @levels [:debug, :info, :warn, :error]
303
+
304
+ def new(level \\ :debug), do: %__MODULE__{level: level}
305
+
306
+ def should_log?(%__MODULE__{level: min_level}, level) do
307
+ min_idx = Enum.find_index(@levels, &(&1 == min_level))
308
+ level_idx = Enum.find_index(@levels, &(&1 == level))
309
+ level_idx >= min_idx
310
+ end
311
+ end
312
+
313
+ defimpl Logger, for: ConsoleLogger do
314
+ def debug(logger, message) do
315
+ if ConsoleLogger.should_log?(logger, :debug), do: IO.puts("[DEBUG] #{message}")
316
+ logger
317
+ end
318
+
319
+ def info(logger, message) do
320
+ if ConsoleLogger.should_log?(logger, :info), do: IO.puts("[INFO] #{message}")
321
+ logger
322
+ end
323
+
324
+ def warn(logger, message) do
325
+ if ConsoleLogger.should_log?(logger, :warn), do: IO.puts("[WARN] #{message}")
326
+ logger
327
+ end
328
+
329
+ def error(logger, message) do
330
+ if ConsoleLogger.should_log?(logger, :error), do: IO.puts("[ERROR] #{message}")
331
+ logger
332
+ end
333
+ end
334
+ ```
335
+
336
+ ### Concrete Server: MemoryLogger(テスト用)
337
+
338
+ ```elixir
339
+ defmodule MemoryLogger do
340
+ @moduledoc "メモリロガー(テスト用)"
341
+ defstruct entries: []
342
+
343
+ def new, do: %__MODULE__{}
344
+
345
+ def get_entries(%__MODULE__{entries: entries}), do: Enum.reverse(entries)
346
+
347
+ def clear(%__MODULE__{}), do: %__MODULE__{}
348
+ end
349
+
350
+ defimpl Logger, for: MemoryLogger do
351
+ def debug(%MemoryLogger{entries: entries} = logger, message) do
352
+ %{logger | entries: [{:debug, message} | entries]}
353
+ end
354
+
355
+ def info(%MemoryLogger{entries: entries} = logger, message) do
356
+ %{logger | entries: [{:info, message} | entries]}
357
+ end
358
+
359
+ def warn(%MemoryLogger{entries: entries} = logger, message) do
360
+ %{logger | entries: [{:warn, message} | entries]}
361
+ end
362
+
363
+ def error(%MemoryLogger{entries: entries} = logger, message) do
364
+ %{logger | entries: [{:error, message} | entries]}
365
+ end
366
+ end
367
+ ```
368
+
369
+ ## 4. Notifier パターン
370
+
371
+ ### Notifier プロトコル
372
+
373
+ ```elixir
374
+ defprotocol Notifier do
375
+ @moduledoc "通知のプロトコル"
376
+
377
+ @doc "通知を送信"
378
+ @spec notify(t(), String.t(), String.t()) :: {:ok, t()} | {:error, term()}
379
+ def notify(notifier, recipient, message)
380
+ end
381
+ ```
382
+
383
+ ### 複数の通知実装
384
+
385
+ ```elixir
386
+ defmodule EmailNotifier do
387
+ defstruct from: "noreply@example.com", sent: []
388
+
389
+ def new(from \\ "noreply@example.com"), do: %__MODULE__{from: from}
390
+ def get_sent(%__MODULE__{sent: sent}), do: Enum.reverse(sent)
391
+ end
392
+
393
+ defimpl Notifier, for: EmailNotifier do
394
+ def notify(%EmailNotifier{from: from, sent: sent} = notifier, recipient, message) do
395
+ email = %{from: from, to: recipient, subject: "Notification", body: message}
396
+ {:ok, %{notifier | sent: [email | sent]}}
397
+ end
398
+ end
399
+
400
+ defmodule SMSNotifier do
401
+ defstruct sender: "+1234567890", sent: []
402
+
403
+ def new(sender \\ "+1234567890"), do: %__MODULE__{sender: sender}
404
+ def get_sent(%__MODULE__{sent: sent}), do: Enum.reverse(sent)
405
+ end
406
+
407
+ defimpl Notifier, for: SMSNotifier do
408
+ def notify(%SMSNotifier{sender: sender, sent: sent} = notifier, recipient, message) do
409
+ truncated = String.slice(message, 0, 160) # SMS character limit
410
+ sms = %{from: sender, to: recipient, body: truncated}
411
+ {:ok, %{notifier | sent: [sms | sent]}}
412
+ end
413
+ end
414
+
415
+ defmodule PushNotifier do
416
+ defstruct app_id: "default", sent: []
417
+
418
+ def new(app_id \\ "default"), do: %__MODULE__{app_id: app_id}
419
+ def get_sent(%__MODULE__{sent: sent}), do: Enum.reverse(sent)
420
+ end
421
+
422
+ defimpl Notifier, for: PushNotifier do
423
+ def notify(%PushNotifier{app_id: app_id, sent: sent} = notifier, recipient, message) do
424
+ push = %{app_id: app_id, device_token: recipient, body: message}
425
+ {:ok, %{notifier | sent: [push | sent]}}
426
+ end
427
+ end
428
+ ```
429
+
430
+ ### CompositeNotifier(複数チャネルに送信)
431
+
432
+ ```elixir
433
+ defmodule CompositeNotifier do
434
+ @moduledoc "複合通知(複数チャネルに同時送信)"
435
+ defstruct notifiers: []
436
+
437
+ def new(notifiers), do: %__MODULE__{notifiers: notifiers}
438
+
439
+ def add(%__MODULE__{notifiers: ns} = cn, notifier) do
440
+ %{cn | notifiers: ns ++ [notifier]}
441
+ end
442
+ end
443
+
444
+ defimpl Notifier, for: CompositeNotifier do
445
+ def notify(%CompositeNotifier{notifiers: notifiers} = composite, recipient, message) do
446
+ updated_notifiers = Enum.map(notifiers, fn n ->
447
+ case Notifier.notify(n, recipient, message) do
448
+ {:ok, updated} -> updated
449
+ {:error, _} -> n
450
+ end
451
+ end)
452
+ {:ok, %{composite | notifiers: updated_notifiers}}
453
+ end
454
+ end
455
+ ```
456
+
457
+ ## 5. Service Layer
458
+
459
+ ### UserService
460
+
461
+ ```elixir
462
+ defmodule UserService do
463
+ @moduledoc """
464
+ ユーザーサービス。
465
+ Repository と Logger に依存するが、具体的な実装は知らない。
466
+ """
467
+
468
+ defstruct [:repo, :logger]
469
+
470
+ def new(repo, logger), do: %__MODULE__{repo: repo, logger: logger}
471
+
472
+ @doc "ユーザーを作成"
473
+ def create_user(%__MODULE__{repo: repo, logger: logger} = service, name, email) do
474
+ logger = Logger.info(logger, "Creating user: #{name}")
475
+ user = %{name: name, email: email, created_at: DateTime.utc_now()}
476
+
477
+ case Repository.save(repo, user) do
478
+ {:ok, saved} ->
479
+ Logger.info(logger, "User created: #{saved.id}")
480
+ {:ok, saved, %{service | logger: logger}}
481
+ error ->
482
+ Logger.error(logger, "Failed to create user: #{inspect(error)}")
483
+ error
484
+ end
485
+ end
486
+
487
+ @doc "ユーザーを取得"
488
+ def get_user(%__MODULE__{repo: repo, logger: logger} = service, id) do
489
+ logger = Logger.debug(logger, "Getting user: #{id}")
490
+
491
+ case Repository.find_by_id(repo, id) do
492
+ {:ok, user} ->
493
+ {:ok, user, %{service | logger: logger}}
494
+ {:error, :not_found} = error ->
495
+ Logger.warn(logger, "User not found: #{id}")
496
+ error
497
+ end
498
+ end
499
+
500
+ @doc "全ユーザーを取得"
501
+ def get_all_users(%__MODULE__{repo: repo, logger: logger} = service) do
502
+ logger = Logger.debug(logger, "Getting all users")
503
+ users = Repository.find_all(repo)
504
+ {:ok, users, %{service | logger: logger}}
505
+ end
506
+ end
507
+ ```
508
+
509
+ ### NotificationService
510
+
511
+ ```elixir
512
+ defmodule NotificationService do
513
+ @moduledoc """
514
+ 通知サービス。
515
+ Notifier と Logger に依存する。
516
+ """
517
+
518
+ defstruct [:notifier, :logger]
519
+
520
+ def new(notifier, logger), do: %__MODULE__{notifier: notifier, logger: logger}
521
+
522
+ @doc "ユーザーに通知を送信"
523
+ def notify_user(%__MODULE__{notifier: notifier, logger: logger} = service, user, message) do
524
+ recipient = Map.get(user, :email) || Map.get(user, :phone) || Map.get(user, :device_token)
525
+ logger = Logger.info(logger, "Sending notification to: #{recipient}")
526
+
527
+ case Notifier.notify(notifier, recipient, message) do
528
+ {:ok, updated_notifier} ->
529
+ logger = Logger.info(logger, "Notification sent successfully")
530
+ {:ok, %{service | notifier: updated_notifier, logger: logger}}
531
+ {:error, reason} = error ->
532
+ Logger.error(logger, "Failed to send notification: #{inspect(reason)}")
533
+ error
534
+ end
535
+ end
536
+
537
+ @doc "複数ユーザーに通知を送信"
538
+ def broadcast(%__MODULE__{} = service, users, message) do
539
+ Enum.reduce(users, {:ok, service}, fn user, acc ->
540
+ case acc do
541
+ {:ok, srv} -> notify_user(srv, user, message)
542
+ error -> error
543
+ end
544
+ end)
545
+ end
546
+ end
547
+ ```
548
+
549
+ ## 6. Dependency Injection Container
550
+
551
+ ```elixir
552
+ defmodule Container do
553
+ @moduledoc """
554
+ 依存性注入コンテナ。
555
+ Abstract Server の具体的な実装を管理する。
556
+ """
557
+
558
+ defstruct services: %{}
559
+
560
+ def new, do: %__MODULE__{}
561
+
562
+ @doc "サービスを登録"
563
+ def register(%__MODULE__{services: services} = container, key, service) do
564
+ %{container | services: Map.put(services, key, service)}
565
+ end
566
+
567
+ @doc "サービスを取得"
568
+ def resolve(%__MODULE__{services: services}, key) do
569
+ case Map.get(services, key) do
570
+ nil -> {:error, {:not_found, key}}
571
+ service -> {:ok, service}
572
+ end
573
+ end
574
+
575
+ @doc "サービスを取得(存在しない場合はエラー)"
576
+ def resolve!(%__MODULE__{} = container, key) do
577
+ case resolve(container, key) do
578
+ {:ok, service} -> service
579
+ {:error, _} -> raise "Service not found: #{inspect(key)}"
580
+ end
581
+ end
582
+
583
+ @doc "ファクトリ関数でサービスを登録"
584
+ def register_factory(%__MODULE__{services: services} = container, key, factory)
585
+ when is_function(factory, 1) do
586
+ %{container | services: Map.put(services, key, {:factory, factory})}
587
+ end
588
+
589
+ @doc "ファクトリを考慮してサービスを解決"
590
+ def resolve_with_factory(%__MODULE__{services: services} = container, key) do
591
+ case Map.get(services, key) do
592
+ nil -> {:error, {:not_found, key}}
593
+ {:factory, factory} -> {:ok, factory.(container)}
594
+ service -> {:ok, service}
595
+ end
596
+ end
597
+ end
598
+ ```
599
+
600
+ ## 7. Application セットアップ
601
+
602
+ ```elixir
603
+ defmodule Application do
604
+ @moduledoc "アプリケーションのセットアップ例"
605
+
606
+ @doc "開発環境用のコンテナを作成"
607
+ def development_container do
608
+ Container.new()
609
+ |> Container.register(:repository, MemoryRepository.new())
610
+ |> Container.register(:logger, ConsoleLogger.new(:debug))
611
+ |> Container.register(:notifier, EmailNotifier.new("dev@example.com"))
612
+ |> Container.register_factory(:user_service, fn container ->
613
+ repo = Container.resolve!(container, :repository)
614
+ logger = Container.resolve!(container, :logger)
615
+ UserService.new(repo, logger)
616
+ end)
617
+ |> Container.register_factory(:notification_service, fn container ->
618
+ notifier = Container.resolve!(container, :notifier)
619
+ logger = Container.resolve!(container, :logger)
620
+ NotificationService.new(notifier, logger)
621
+ end)
622
+ end
623
+
624
+ @doc "テスト環境用のコンテナを作成"
625
+ def test_container do
626
+ Container.new()
627
+ |> Container.register(:repository, MockRepository.new())
628
+ |> Container.register(:logger, MemoryLogger.new())
629
+ |> Container.register(:notifier, EmailNotifier.new("test@example.com"))
630
+ |> Container.register_factory(:user_service, fn container ->
631
+ repo = Container.resolve!(container, :repository)
632
+ logger = Container.resolve!(container, :logger)
633
+ UserService.new(repo, logger)
634
+ end)
635
+ end
636
+ end
637
+ ```
638
+
639
+ ### 使用例
640
+
641
+ ```elixir
642
+ # 開発環境でアプリケーションをセットアップ
643
+ container = Application.development_container()
644
+
645
+ # サービスを取得
646
+ {:ok, user_service} = Container.resolve_with_factory(container, :user_service)
647
+ {:ok, notification_service} = Container.resolve_with_factory(container, :notification_service)
648
+
649
+ # サービスを使用
650
+ {:ok, user, _} = UserService.create_user(user_service, "Alice", "alice@example.com")
651
+ {:ok, _} = NotificationService.notify_user(notification_service, user, "Welcome!")
652
+
653
+ # テスト環境に切り替え
654
+ test_container = Application.test_container()
655
+ {:ok, test_user_service} = Container.resolve_with_factory(test_container, :user_service)
656
+ # 同じサービスインターフェースで、異なる実装(モック)を使用
657
+ ```
658
+
659
+ ## 設計のポイント
660
+
661
+ ### 依存関係逆転の原則(DIP)
662
+
663
+ ```
664
+ 従来の依存関係:
665
+ ┌─────────────┐ ┌─────────────┐
666
+ │ UserService │ ──► │MemoryRepo │
667
+ └─────────────┘ └─────────────┘
668
+
669
+ Abstract Server による依存関係:
670
+ ┌─────────────┐ ┌─────────────┐ ┌─────────────┐
671
+ │ UserService │ ──► │ Repository │ ◄── │MemoryRepo │
672
+ └─────────────┘ │ (Protocol) │ │ MockRepo │
673
+ └─────────────┘ └─────────────┘
674
+ ```
675
+
676
+ ### テスト容易性
677
+
678
+ ```elixir
679
+ # テスト時はモック実装を注入
680
+ defmodule UserServiceTest do
681
+ test "creates user" do
682
+ mock_repo = MockRepository.new()
683
+ memory_logger = MemoryLogger.new()
684
+ service = UserService.new(mock_repo, memory_logger)
685
+
686
+ {:ok, user, updated_service} = UserService.create_user(service, "Test", "test@example.com")
687
+
688
+ assert user.name == "Test"
689
+
690
+ # ログを検証
691
+ entries = MemoryLogger.get_entries(updated_service.logger)
692
+ assert Enum.any?(entries, fn {level, _} -> level == :info end)
693
+ end
694
+ end
695
+ ```
696
+
697
+ ### 拡張性
698
+
699
+ 新しい実装を追加するには、プロトコルを実装するだけ:
700
+
701
+ ```elixir
702
+ # 新しいリポジトリ実装
703
+ defmodule PostgresRepository do
704
+ defstruct [:connection]
705
+
706
+ def new(connection), do: %__MODULE__{connection: connection}
707
+ end
708
+
709
+ defimpl Repository, for: PostgresRepository do
710
+ def find_by_id(repo, id), do: # PostgreSQL クエリ
711
+ def find_all(repo), do: # PostgreSQL クエリ
712
+ def save(repo, entity), do: # PostgreSQL INSERT/UPDATE
713
+ def delete(repo, id), do: # PostgreSQL DELETE
714
+ end
715
+
716
+ # 既存のサービスコードは変更不要
717
+ service = UserService.new(PostgresRepository.new(conn), ConsoleLogger.new())
718
+ ```
719
+
720
+ ## まとめ
721
+
722
+ Abstract Server パターンの利点:
723
+
724
+ 1. **疎結合**: 高レベルモジュールが低レベルモジュールの詳細に依存しない
725
+ 2. **テスト容易性**: モック実装を容易に注入可能
726
+ 3. **拡張性**: 新しい実装を追加しても既存コードに影響なし
727
+ 4. **設定可能性**: 環境に応じて異なる実装を選択可能
728
+
729
+ Elixir のプロトコルは、Abstract Server パターンを自然に実装するための強力な仕組みを提供します。構造体と組み合わせることで、型安全で拡張可能なシステムを構築できます。