@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,974 @@
1
+ # Part IV: IO とストリーム処理
2
+
3
+ 本章では、関数型プログラミングにおける副作用の扱い方を学びます。Elixir の無名関数を使って副作用を遅延実行し、Stream モジュールで無限のデータを効率的に扱う方法を習得します。
4
+
5
+ ---
6
+
7
+ ## 第8章: IO パターンとプロセス
8
+
9
+ ### 8.1 副作用の問題
10
+
11
+ 純粋関数は副作用を持ちません。しかし、実際のプログラムには副作用が必要です:
12
+
13
+ - ファイルの読み書き
14
+ - ネットワーク通信
15
+ - データベースアクセス
16
+ - 乱数生成
17
+ - 現在時刻の取得
18
+
19
+ ```plantuml
20
+ @startuml
21
+ !theme plain
22
+
23
+ rectangle "副作用の問題" {
24
+ rectangle "副作用を持つ関数" as impure {
25
+ card "テストが困難"
26
+ card "予測不可能"
27
+ card "参照透過性がない"
28
+ }
29
+
30
+ rectangle "純粋関数" as pure {
31
+ card "テストが容易"
32
+ card "予測可能"
33
+ card "参照透過性がある"
34
+ }
35
+ }
36
+
37
+ note bottom
38
+ どうすれば副作用を
39
+ 純粋関数のように扱えるか?
40
+ end note
41
+
42
+ @enduml
43
+ ```
44
+
45
+ ### 8.2 Elixir における IO パターン
46
+
47
+ Scala の `IO` モナドに相当する概念を、Elixir では**無名関数**を使って表現します。
48
+
49
+ - `fn -> expr end` は「実行すると結果を返す副作用のある計算」
50
+ - 関数を作成しただけでは副作用は発生しない
51
+ - 関数を呼び出すことで実際に実行される
52
+
53
+ ```plantuml
54
+ @startuml
55
+ !theme plain
56
+
57
+ rectangle "遅延実行パターンの仕組み" {
58
+ card "delay(fn -> 副作用 end)" as create
59
+ card "関数値" as io
60
+ card "run(io)" as run
61
+ card "副作用が実行される" as effect
62
+
63
+ create --> io : 記述を作成
64
+ io --> run : 実行を指示
65
+ run --> effect : 実際に実行
66
+ }
67
+
68
+ note bottom of io
69
+ この時点では
70
+ 副作用は発生しない
71
+ end note
72
+
73
+ @enduml
74
+ ```
75
+
76
+ ### 8.3 サイコロを振る例
77
+
78
+ **ソースファイル**: `app/elixir/lib/ch08/io_and_processes.ex`
79
+
80
+ #### 不純な関数(副作用あり)
81
+
82
+ ```elixir
83
+ def cast_the_die_impure do
84
+ :rand.uniform(6)
85
+ end
86
+
87
+ # 呼び出すたびに異なる値が返る
88
+ IO.inspect(cast_the_die_impure()) # 3
89
+ IO.inspect(cast_the_die_impure()) # 5
90
+ IO.inspect(cast_the_die_impure()) # 1
91
+ ```
92
+
93
+ #### 遅延実行パターン
94
+
95
+ ```elixir
96
+ @spec delay((-> any())) :: (-> any())
97
+ def delay(f) when is_function(f, 0), do: f
98
+
99
+ @spec run((-> any())) :: any()
100
+ def run(f) when is_function(f, 0), do: f.()
101
+
102
+ def cast_the_die do
103
+ delay(fn -> cast_the_die_impure() end)
104
+ end
105
+
106
+ # 関数を作成(この時点では実行されない)
107
+ die_cast = cast_the_die()
108
+ IO.inspect(die_cast) # #Function<...> - まだ実行されていない
109
+
110
+ # 実際に実行
111
+ IO.inspect(run(die_cast)) # 4
112
+ ```
113
+
114
+ ### 8.4 IO の作成方法
115
+
116
+ | 関数 | 用途 | 例 |
117
+ |------|------|-----|
118
+ | `delay(f)` | 副作用のある式をラップ | `delay(fn -> IO.puts("hello") end)` |
119
+ | `pure(value)` | 既存の値をラップ(副作用なし) | `pure(42)` |
120
+
121
+ ```elixir
122
+ # delay: 副作用を遅延実行
123
+ delayed_print = delay(fn -> IO.puts("Hello") end)
124
+ # IO.puts はまだ実行されていない
125
+
126
+ # pure: 既存の値をラップ
127
+ pure_value = pure(42)
128
+ # 副作用なし、即座に 42 が確定
129
+ ```
130
+
131
+ ### 8.5 IO の合成
132
+
133
+ IO 値は `flat_map` でチェーンできます。Scala の `for` 内包表記に相当します。
134
+
135
+ ```elixir
136
+ @spec flat_map((-> any()), (any() -> (-> any()))) :: (-> any())
137
+ def flat_map(io, f) do
138
+ fn ->
139
+ result = run(io)
140
+ run(f.(result))
141
+ end
142
+ end
143
+
144
+ def cast_the_die_twice do
145
+ flat_map(cast_the_die(), fn first ->
146
+ map(cast_the_die(), fn second ->
147
+ first + second
148
+ end)
149
+ end)
150
+ end
151
+
152
+ # まだ実行されていない
153
+ program = cast_the_die_twice()
154
+
155
+ # 実行
156
+ result = run(program) # 2〜12 の値
157
+ ```
158
+
159
+ ```plantuml
160
+ @startuml
161
+ !theme plain
162
+
163
+ rectangle "IO の合成" {
164
+ card "cast_the_die()" as cast1
165
+ card "cast_the_die()" as cast2
166
+ card "first + second" as combine
167
+
168
+ cast1 --> combine : flat_map
169
+ cast2 --> combine : map
170
+ }
171
+
172
+ note bottom
173
+ flat_map でチェーンして
174
+ 順序付けて合成
175
+ end note
176
+
177
+ @enduml
178
+ ```
179
+
180
+ ### 8.6 ミーティングスケジューリングの例
181
+
182
+ より実践的な例として、ミーティングのスケジューリングを見てみましょう。
183
+
184
+ ```elixir
185
+ defmodule MeetingTime do
186
+ defstruct [:start_hour, :end_hour]
187
+ end
188
+
189
+ # 副作用のある API 呼び出しを遅延実行
190
+ def calendar_entries(name) do
191
+ delay(fn ->
192
+ # モックデータ(実際には API 呼び出し)
193
+ case name do
194
+ "Alice" -> [%MeetingTime{start_hour: 9, end_hour: 10}, ...]
195
+ "Bob" -> [%MeetingTime{start_hour: 10, end_hour: 11}, ...]
196
+ _ -> []
197
+ end
198
+ end)
199
+ end
200
+ ```
201
+
202
+ #### 予定の取得
203
+
204
+ ```elixir
205
+ def scheduled_meetings(person1, person2) do
206
+ flat_map(calendar_entries(person1), fn entries1 ->
207
+ map(calendar_entries(person2), fn entries2 ->
208
+ entries1 ++ entries2
209
+ end)
210
+ end)
211
+ end
212
+
213
+ # 使用例
214
+ io = scheduled_meetings("Alice", "Bob")
215
+ meetings = run(io) # 4件の予定
216
+ ```
217
+
218
+ #### 空き時間の計算(純粋関数)
219
+
220
+ ```elixir
221
+ def possible_meetings(existing_meetings, start_hour, end_hour, length_hours) do
222
+ slots =
223
+ start_hour..(end_hour - length_hours)
224
+ |> Enum.map(fn start ->
225
+ %MeetingTime{start_hour: start, end_hour: start + length_hours}
226
+ end)
227
+
228
+ Enum.filter(slots, fn slot ->
229
+ Enum.all?(existing_meetings, fn meeting ->
230
+ not meetings_overlap?(meeting, slot)
231
+ end)
232
+ end)
233
+ end
234
+
235
+ # ミーティングが重なるかチェック
236
+ def meetings_overlap?(m1, m2) do
237
+ m1.start_hour < m2.end_hour && m2.start_hour < m1.end_hour
238
+ end
239
+ ```
240
+
241
+ ### 8.7 or_else によるエラーハンドリング
242
+
243
+ `or_else` 関数で、失敗時のフォールバックを指定できます。
244
+
245
+ ```elixir
246
+ def or_else(io, fallback) do
247
+ fn ->
248
+ try do
249
+ run(io)
250
+ rescue
251
+ _ -> run(fallback)
252
+ end
253
+ end
254
+ end
255
+
256
+ success = pure(42)
257
+ fallback = pure(0)
258
+ run(or_else(success, fallback)) # 42
259
+
260
+ failing = fn -> raise "error" end
261
+ run(or_else(failing, fallback)) # 0
262
+ ```
263
+
264
+ ```plantuml
265
+ @startuml
266
+ !theme plain
267
+
268
+ rectangle "or_else の動作" {
269
+ card "メインの IO" as main
270
+ card "フォールバック IO" as fallback
271
+ card "結果" as result
272
+
273
+ main --> result : 成功時
274
+ main --> fallback : 失敗時
275
+ fallback --> result
276
+ }
277
+
278
+ @enduml
279
+ ```
280
+
281
+ #### リトライ戦略
282
+
283
+ ```elixir
284
+ def retry(io, max_retries) do
285
+ fn ->
286
+ do_retry(io, max_retries, nil)
287
+ end
288
+ end
289
+
290
+ defp do_retry(_io, 0, last_error), do: raise(last_error || "Max retries exceeded")
291
+
292
+ defp do_retry(io, retries_left, _last_error) do
293
+ try do
294
+ run(io)
295
+ rescue
296
+ e -> do_retry(io, retries_left - 1, e)
297
+ end
298
+ end
299
+
300
+ # 使用例:API 呼び出しを5回までリトライ
301
+ result = retry(calendar_entries("Alice"), 5)
302
+ ```
303
+
304
+ ### 8.8 sequence による IO のリスト処理
305
+
306
+ `[IO]` を `IO[List]` に変換するには `sequence` を使います。
307
+
308
+ ```elixir
309
+ def sequence(ios) do
310
+ fn -> Enum.map(ios, &run/1) end
311
+ end
312
+
313
+ ios = [pure(1), pure(2), pure(3)]
314
+ combined = sequence(ios)
315
+ run(combined) # [1, 2, 3]
316
+ ```
317
+
318
+ #### 複数人の予定を取得
319
+
320
+ ```elixir
321
+ def scheduled_meetings_for(attendees) do
322
+ attendees
323
+ |> Enum.map(&calendar_entries/1)
324
+ |> sequence()
325
+ |> map(&List.flatten/1)
326
+ end
327
+
328
+ # 使用例
329
+ io = scheduled_meetings_for(["Alice", "Bob"])
330
+ meetings = run(io) # 4件の予定
331
+ ```
332
+
333
+ ### 8.9 Agent による状態管理
334
+
335
+ Elixir では `Agent` を使って状態を安全に管理できます。
336
+
337
+ ```elixir
338
+ def create_counter(initial_value) do
339
+ Agent.start_link(fn -> initial_value end)
340
+ end
341
+
342
+ def get_count(counter) do
343
+ Agent.get(counter, & &1)
344
+ end
345
+
346
+ def increment(counter) do
347
+ Agent.update(counter, &(&1 + 1))
348
+ end
349
+
350
+ # 使用例
351
+ {:ok, counter} = create_counter(0)
352
+ get_count(counter) # 0
353
+ increment(counter)
354
+ get_count(counter) # 1
355
+ ```
356
+
357
+ ### 8.10 Task による非同期処理
358
+
359
+ `Task` を使って非同期処理を実行できます。
360
+
361
+ ```elixir
362
+ def async_calendar_entries(name) do
363
+ Task.async(fn -> run(calendar_entries(name)) end)
364
+ end
365
+
366
+ def parallel_calendar_entries(names) do
367
+ names
368
+ |> Enum.map(&async_calendar_entries/1)
369
+ |> Task.await_many()
370
+ |> List.flatten()
371
+ end
372
+
373
+ # 使用例:並行でカレンダーを取得
374
+ entries = parallel_calendar_entries(["Alice", "Bob"]) # 4件
375
+ ```
376
+
377
+ ---
378
+
379
+ ## 第9章: ストリーム処理
380
+
381
+ ### 9.1 ストリームとは
382
+
383
+ **ストリーム**は、要素の(潜在的に無限の)シーケンスを遅延評価で表します。Elixir の `Stream` モジュールを使用します。
384
+
385
+ ```plantuml
386
+ @startuml
387
+ !theme plain
388
+
389
+ rectangle "List vs Stream" {
390
+ rectangle "List" as list {
391
+ card "全要素がメモリに存在"
392
+ card "有限"
393
+ card "即座に評価"
394
+ }
395
+
396
+ rectangle "Stream" as stream {
397
+ card "必要な時に要素を生成"
398
+ card "無限も可能"
399
+ card "遅延評価"
400
+ }
401
+ }
402
+
403
+ @enduml
404
+ ```
405
+
406
+ ### 9.2 基本的なストリーム操作
407
+
408
+ **ソースファイル**: `app/elixir/lib/ch09/streams.ex`
409
+
410
+ ```elixir
411
+ # 有限ストリーム
412
+ def finite_stream(list), do: Stream.map(list, & &1)
413
+
414
+ stream = finite_stream([1, 2, 3])
415
+ Enum.to_list(stream) # [1, 2, 3]
416
+
417
+ # フィルタリング
418
+ def filter_stream(stream, predicate), do: Stream.filter(stream, predicate)
419
+
420
+ odd_numbers = filter_stream([1, 2, 3, 4, 5], &(rem(&1, 2) != 0))
421
+ Enum.to_list(odd_numbers) # [1, 3, 5]
422
+ ```
423
+
424
+ ### 9.3 無限ストリーム
425
+
426
+ ```elixir
427
+ # 無限に繰り返し
428
+ def repeat(list), do: Stream.cycle(list)
429
+
430
+ infinite = repeat([1, 2, 3])
431
+ Enum.take(infinite, 8) # [1, 2, 3, 1, 2, 3, 1, 2]
432
+
433
+ # 生成関数を使用
434
+ def generate(f), do: Stream.repeatedly(f)
435
+
436
+ stream = generate(fn -> :rand.uniform(6) end)
437
+ Enum.take(stream, 5) # [4, 2, 6, 1, 3]
438
+
439
+ # カウントアップ
440
+ def count_from(start), do: Stream.iterate(start, &(&1 + 1))
441
+
442
+ stream = count_from(1)
443
+ Enum.take(stream, 5) # [1, 2, 3, 4, 5]
444
+ ```
445
+
446
+ ```plantuml
447
+ @startuml
448
+ !theme plain
449
+
450
+ rectangle "無限ストリーム" {
451
+ card "Stream.cycle([1, 2, 3])" as stream
452
+ card "1, 2, 3, 1, 2, 3, 1, 2, 3, ..." as infinite
453
+ card "Enum.take(8)" as take
454
+ card "[1, 2, 3, 1, 2, 3, 1, 2]" as result
455
+
456
+ stream --> infinite
457
+ infinite --> take
458
+ take --> result
459
+ }
460
+
461
+ note bottom
462
+ 無限のストリームから
463
+ 必要な分だけ取得
464
+ end note
465
+
466
+ @enduml
467
+ ```
468
+
469
+ ### 9.4 サイコロストリーム
470
+
471
+ ```elixir
472
+ def die_casts do
473
+ Stream.repeatedly(fn -> :rand.uniform(6) end)
474
+ end
475
+
476
+ # 最初の3回を取得
477
+ die_casts() |> Enum.take(3) # [4, 2, 6]
478
+
479
+ # 6 が出るまで振り続ける
480
+ def roll_until(target) do
481
+ die_casts()
482
+ |> Stream.take_while(fn x -> x != target end)
483
+ |> Enum.to_list()
484
+ |> Kernel.++([target])
485
+ end
486
+
487
+ roll_until(6) # [2, 4, 1, 3, 6]
488
+
489
+ # 指定回数振って合計
490
+ def roll_and_sum(n) do
491
+ die_casts()
492
+ |> Enum.take(n)
493
+ |> Enum.sum()
494
+ end
495
+
496
+ roll_and_sum(3) # 3〜18 の値
497
+ ```
498
+
499
+ ### 9.5 ストリームの主要操作
500
+
501
+ | 操作 | 説明 | 例 |
502
+ |------|------|-----|
503
+ | `Stream.take(n)` | 最初の n 要素を取得 | `stream \|> Stream.take(3)` |
504
+ | `Stream.filter(p)` | 条件を満たす要素のみ | `stream \|> Stream.filter(&(&1 > 0))` |
505
+ | `Stream.map(f)` | 各要素を変換 | `stream \|> Stream.map(&(&1 * 2))` |
506
+ | `Stream.concat(s)` | 別のストリームを結合 | `Stream.concat(s1, s2)` |
507
+ | `Stream.cycle(list)` | 無限に繰り返し | `Stream.cycle([1, 2, 3])` |
508
+ | `Stream.chunk_every(n, step)` | スライディングウィンドウ | `stream \|> Stream.chunk_every(3, 1)` |
509
+
510
+ ### 9.6 通貨交換レートの例
511
+
512
+ 為替レートを監視して、上昇トレンドを検出する例です。
513
+
514
+ #### トレンド判定(純粋関数)
515
+
516
+ ```elixir
517
+ def trending?(rates) when length(rates) <= 1, do: false
518
+
519
+ def trending?(rates) do
520
+ rates
521
+ |> Enum.zip(Enum.drop(rates, 1))
522
+ |> Enum.all?(fn {prev, curr} -> curr > prev end)
523
+ end
524
+
525
+ trending?([0.81, 0.82, 0.83]) # true (上昇トレンド)
526
+ trending?([0.81, 0.84, 0.83]) # false
527
+
528
+ def declining?(rates) when length(rates) <= 1, do: false
529
+
530
+ def declining?(rates) do
531
+ rates
532
+ |> Enum.zip(Enum.drop(rates, 1))
533
+ |> Enum.all?(fn {prev, curr} -> curr < prev end)
534
+ end
535
+
536
+ declining?([0.83, 0.82, 0.81]) # true (下降トレンド)
537
+
538
+ def stable?(values) when length(values) < 3, do: false
539
+
540
+ def stable?(values) do
541
+ values |> Enum.uniq() |> length() == 1
542
+ end
543
+
544
+ stable?([5, 5, 5]) # true (安定)
545
+ ```
546
+
547
+ #### レートのストリーム
548
+
549
+ ```elixir
550
+ def exchange_rate(from, to) do
551
+ # モックの為替レート(実際には API を呼び出す)
552
+ base_rates = %{
553
+ {:usd, :eur} => 0.85,
554
+ {:usd, :gbp} => 0.73,
555
+ # ...
556
+ }
557
+
558
+ base_rate = Map.get(base_rates, {from, to}, 1.0)
559
+ variation = (:rand.uniform() - 0.5) * 0.02
560
+ base_rate * (1 + variation)
561
+ end
562
+
563
+ def rate_stream(from, to) do
564
+ Stream.repeatedly(fn -> exchange_rate(from, to) end)
565
+ end
566
+
567
+ # 使用例
568
+ rate_stream(:usd, :eur) |> Enum.take(3)
569
+ # [0.8493, 0.8521, 0.8478]
570
+ ```
571
+
572
+ ### 9.7 スライディングウィンドウ
573
+
574
+ `Stream.chunk_every/4` で連続する要素をグループ化できます。
575
+
576
+ ```elixir
577
+ def sliding(stream, size) do
578
+ Stream.chunk_every(stream, size, 1, :discard)
579
+ end
580
+
581
+ sliding([1, 2, 3, 4, 5], 3) |> Enum.to_list()
582
+ # [[1, 2, 3], [2, 3, 4], [3, 4, 5]]
583
+ ```
584
+
585
+ ```plantuml
586
+ @startuml
587
+ !theme plain
588
+
589
+ rectangle "sliding(3) の動作" {
590
+ card "[1, 2, 3, 4, 5]" as input
591
+ card "[1, 2, 3]" as w1
592
+ card "[2, 3, 4]" as w2
593
+ card "[3, 4, 5]" as w3
594
+
595
+ input --> w1
596
+ input --> w2
597
+ input --> w3
598
+ }
599
+
600
+ @enduml
601
+ ```
602
+
603
+ #### トレンドを検出して交換
604
+
605
+ ```elixir
606
+ def exchange_if_trending(amount, from, to, window_size, max_attempts) do
607
+ result =
608
+ rate_stream(from, to)
609
+ |> sliding(window_size)
610
+ |> Stream.filter(&trending?/1) # 上昇トレンドを検出
611
+ |> Stream.map(&List.last/1) # 最新のレートを取得
612
+ |> Stream.take(1)
613
+ |> Enum.take(max_attempts)
614
+ |> List.first()
615
+
616
+ case result do
617
+ nil -> {:error, "No trending pattern found"}
618
+ rate -> {:ok, rate * amount}
619
+ end
620
+ end
621
+
622
+ # 使用例
623
+ exchange_if_trending(100.0, :usd, :eur, 3, 1000)
624
+ # {:ok, 85.23}
625
+ ```
626
+
627
+ ### 9.8 ストリームの結合
628
+
629
+ 2つのストリームを `Stream.zip` で結合できます。
630
+
631
+ ```elixir
632
+ def zip_streams(stream1, stream2) do
633
+ Stream.zip(stream1, stream2)
634
+ end
635
+
636
+ zip_streams([1, 2, 3], ["a", "b", "c"]) |> Enum.to_list()
637
+ # [{1, "a"}, {2, "b"}, {3, "c"}]
638
+
639
+ # 左側の値だけ返す
640
+ def zip_left(stream1, stream2) do
641
+ Stream.zip(stream1, stream2)
642
+ |> Stream.map(fn {left, _right} -> left end)
643
+ end
644
+
645
+ # インターバル付きで処理
646
+ def with_interval(stream, interval_ms) do
647
+ Stream.map(stream, fn item ->
648
+ Process.sleep(interval_ms)
649
+ item
650
+ end)
651
+ end
652
+
653
+ # レートを1秒間隔で取得
654
+ rate_stream(:usd, :eur)
655
+ |> with_interval(1000)
656
+ |> Enum.take(5)
657
+ ```
658
+
659
+ ```plantuml
660
+ @startuml
661
+ !theme plain
662
+
663
+ rectangle "zip_left の動作" {
664
+ card "rates: [0.81, 0.82, 0.83, ...]" as rates
665
+ card "ticks: [:tick, :tick, :tick, ...]" as ticks
666
+ card "zip_left" as zip
667
+ card "[0.81, 0.82, 0.83, ...] (1秒間隔)" as result
668
+
669
+ rates --> zip
670
+ ticks --> zip
671
+ zip --> result
672
+ }
673
+
674
+ note bottom
675
+ 左側の値を返しつつ
676
+ 右側のペースで進む
677
+ end note
678
+
679
+ @enduml
680
+ ```
681
+
682
+ ### 9.9 集計操作
683
+
684
+ ```elixir
685
+ # 合計
686
+ def sum_stream(stream), do: Enum.sum(stream)
687
+
688
+ sum_stream([1, 2, 3, 4, 5]) # 15
689
+
690
+ # 平均
691
+ def average_stream(stream) do
692
+ {sum, count} =
693
+ Enum.reduce(stream, {0, 0}, fn x, {sum, count} ->
694
+ {sum + x, count + 1}
695
+ end)
696
+
697
+ if count == 0, do: 0.0, else: sum / count
698
+ end
699
+
700
+ average_stream([1, 2, 3, 4, 5]) # 3.0
701
+
702
+ # 移動平均
703
+ def moving_average(stream, window_size) do
704
+ stream
705
+ |> sliding(window_size)
706
+ |> Stream.map(&average_stream/1)
707
+ end
708
+
709
+ moving_average([1, 2, 3, 4, 5], 3) |> Enum.to_list()
710
+ # [2.0, 3.0, 4.0]
711
+ ```
712
+
713
+ ### 9.10 unfold によるストリーム生成
714
+
715
+ `Stream.unfold/2` で状態を持つストリームを生成できます。
716
+
717
+ ```elixir
718
+ # フィボナッチ数列
719
+ def fibonacci do
720
+ Stream.unfold({0, 1}, fn {a, b} ->
721
+ {a, {b, a + b}}
722
+ end)
723
+ end
724
+
725
+ fibonacci() |> Enum.take(10)
726
+ # [0, 1, 1, 2, 3, 5, 8, 13, 21, 34]
727
+
728
+ # 等比数列
729
+ def geometric_series(start, ratio) do
730
+ Stream.unfold(start, fn current ->
731
+ {current, current * ratio}
732
+ end)
733
+ end
734
+
735
+ geometric_series(1, 2) |> Enum.take(5)
736
+ # [1, 2, 4, 8, 16]
737
+
738
+ # 等差数列
739
+ def arithmetic_series(start, diff) do
740
+ Stream.unfold(start, fn current ->
741
+ {current, current + diff}
742
+ end)
743
+ end
744
+
745
+ arithmetic_series(1, 3) |> Enum.take(5)
746
+ # [1, 4, 7, 10, 13]
747
+ ```
748
+
749
+ ---
750
+
751
+ ## まとめ
752
+
753
+ ### Part IV で学んだこと
754
+
755
+ ```plantuml
756
+ @startuml
757
+ !theme plain
758
+
759
+ rectangle "Part IV: IO とストリーム処理" {
760
+ rectangle "第8章" as ch8 {
761
+ card "遅延実行パターン"
762
+ card "delay / run"
763
+ card "flat_map / map"
764
+ card "or_else / retry"
765
+ card "sequence"
766
+ card "Agent / Task"
767
+ }
768
+
769
+ rectangle "第9章" as ch9 {
770
+ card "Elixir Stream"
771
+ card "無限ストリーム"
772
+ card "cycle / take"
773
+ card "sliding"
774
+ card "zip / zip_left"
775
+ card "unfold"
776
+ }
777
+ }
778
+
779
+ ch8 --> ch9
780
+
781
+ @enduml
782
+ ```
783
+
784
+ ### Scala との対応
785
+
786
+ | Scala | Elixir |
787
+ |-------|--------|
788
+ | `IO[A]` | `(-> any())` (無名関数) |
789
+ | `IO.delay(expr)` | `delay(fn -> expr end)` |
790
+ | `IO.pure(value)` | `pure(value)` |
791
+ | `unsafeRunSync()` | `run(io)` |
792
+ | `flatMap` | `flat_map/2` |
793
+ | `orElse` | `or_else/2` |
794
+ | `sequence` | `sequence/1` |
795
+ | `fs2.Stream` | `Stream` モジュール |
796
+ | `Stream.eval` | `Stream.repeatedly` |
797
+ | `sliding(n)` | `Stream.chunk_every(n, 1, :discard)` |
798
+ | `zipLeft` | `zip_left/2` |
799
+
800
+ ### キーポイント
801
+
802
+ 1. **遅延実行パターン**: 副作用を「記述」として扱い、実行を遅延させる
803
+ 2. **delay/run**: 副作用のある式を遅延実行
804
+ 3. **or_else**: 失敗時のフォールバックを指定
805
+ 4. **sequence**: `[IO]` → `IO[List]`
806
+ 5. **Stream**: 潜在的に無限のシーケンスを遅延評価で処理
807
+ 6. **sliding**: 連続する要素をグループ化してパターンを検出
808
+ 7. **unfold**: 状態を持つストリームを生成
809
+
810
+ ### Elixir の特徴
811
+
812
+ - **Agent**: プロセスベースの状態管理
813
+ - **Task**: 簡単な非同期処理
814
+ - **Stream**: 組み込みの遅延評価ストリーム
815
+
816
+ ### 次のステップ
817
+
818
+ Part V では、以下のトピックを学びます:
819
+
820
+ - 並行・並列処理
821
+ - GenServer による状態管理
822
+ - Supervisor による耐障害性
823
+
824
+ ---
825
+
826
+ ## 演習問題
827
+
828
+ ### 問題 1: IO の基本
829
+
830
+ 以下の関数を実装してください。
831
+
832
+ ```elixir
833
+ def print_and_return(message) do
834
+ # メッセージを出力して、そのメッセージを返す IO を作成
835
+ ???
836
+ end
837
+
838
+ # 期待される動作
839
+ # io = print_and_return("Hello")
840
+ # run(io) は "Hello" をコンソールに出力し、"Hello" を返す
841
+ ```
842
+
843
+ <details>
844
+ <summary>解答</summary>
845
+
846
+ ```elixir
847
+ def print_and_return(message) do
848
+ delay(fn ->
849
+ IO.puts(message)
850
+ message
851
+ end)
852
+ end
853
+ ```
854
+
855
+ </details>
856
+
857
+ ### 問題 2: IO の合成
858
+
859
+ 以下の関数を実装してください。2つの IO を順番に実行し、結果を結合します。
860
+
861
+ ```elixir
862
+ def combine_io(io1, io2, f) do
863
+ # io1 と io2 を実行し、結果を f で結合
864
+ ???
865
+ end
866
+
867
+ # 期待される動作
868
+ # io = combine_io(pure(1), pure(2), &(&1 + &2))
869
+ # run(io) # 3
870
+ ```
871
+
872
+ <details>
873
+ <summary>解答</summary>
874
+
875
+ ```elixir
876
+ def combine_io(io1, io2, f) do
877
+ flat_map(io1, fn a ->
878
+ map(io2, fn b ->
879
+ f.(a, b)
880
+ end)
881
+ end)
882
+ end
883
+ ```
884
+
885
+ </details>
886
+
887
+ ### 問題 3: リトライ
888
+
889
+ 以下の関数を実装してください。指定回数リトライし、全部失敗したらデフォルト値を返します。
890
+
891
+ ```elixir
892
+ def retry_with_default(io, max_retries, default) do
893
+ ???
894
+ end
895
+ ```
896
+
897
+ <details>
898
+ <summary>解答</summary>
899
+
900
+ ```elixir
901
+ def retry_with_default(io, max_retries, default) do
902
+ fn ->
903
+ try do
904
+ run(retry(io, max_retries))
905
+ rescue
906
+ _ -> default
907
+ end
908
+ end
909
+ end
910
+ ```
911
+
912
+ </details>
913
+
914
+ ### 問題 4: ストリーム操作
915
+
916
+ 以下のストリームを作成してください。
917
+
918
+ ```elixir
919
+ # 1. 1から10までの偶数のストリーム
920
+ evens = ???
921
+
922
+ # 2. 無限に交互に true/false を返すストリーム
923
+ alternating = ???
924
+
925
+ # 3. 最初の5つの要素の合計を計算
926
+ sum = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10] |> Stream.take(5) |> ???
927
+ ```
928
+
929
+ <details>
930
+ <summary>解答</summary>
931
+
932
+ ```elixir
933
+ # 1. 1から10までの偶数
934
+ evens = 1..10 |> Stream.filter(&(rem(&1, 2) == 0))
935
+ # または
936
+ evens = Stream.iterate(2, &(&1 + 2)) |> Stream.take_while(&(&1 <= 10))
937
+
938
+ # 2. 無限に交互に true/false
939
+ alternating = Stream.cycle([true, false])
940
+
941
+ # 3. 最初の5つの要素の合計
942
+ sum = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10] |> Stream.take(5) |> Enum.sum() # 15
943
+ ```
944
+
945
+ </details>
946
+
947
+ ### 問題 5: トレンド検出
948
+
949
+ 以下の関数を実装してください。直近3つの値が全て同じかどうかを判定します。
950
+
951
+ ```elixir
952
+ def stable?(values) do
953
+ ???
954
+ end
955
+
956
+ # 期待される動作
957
+ assert stable?([5, 5, 5]) == true
958
+ assert stable?([5, 5, 6]) == false
959
+ assert stable?([5, 6, 5]) == false
960
+ assert stable?([5]) == false # 3つ未満は false
961
+ ```
962
+
963
+ <details>
964
+ <summary>解答</summary>
965
+
966
+ ```elixir
967
+ def stable?(values) when length(values) < 3, do: false
968
+
969
+ def stable?(values) do
970
+ values |> Enum.uniq() |> length() == 1
971
+ end
972
+ ```
973
+
974
+ </details>