@k2works/claude-code-booster 2.7.1 → 3.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (527) hide show
  1. package/lib/assets/docs/article/functional-desgin-ppp/all/01-immutability-and-data-transformation.md +475 -0
  2. package/lib/assets/docs/article/functional-desgin-ppp/all/02-function-composition.md +519 -0
  3. package/lib/assets/docs/article/functional-desgin-ppp/all/03-polymorphism.md +537 -0
  4. package/lib/assets/docs/article/functional-desgin-ppp/all/04-data-validation.md +300 -0
  5. package/lib/assets/docs/article/functional-desgin-ppp/all/05-property-based-testing.md +320 -0
  6. package/lib/assets/docs/article/functional-desgin-ppp/all/06-tdd-and-functional.md +498 -0
  7. package/lib/assets/docs/article/functional-desgin-ppp/all/07-composite-pattern.md +298 -0
  8. package/lib/assets/docs/article/functional-desgin-ppp/all/08-decorator-pattern.md +291 -0
  9. package/lib/assets/docs/article/functional-desgin-ppp/all/09-adapter-pattern.md +336 -0
  10. package/lib/assets/docs/article/functional-desgin-ppp/all/10-strategy-pattern.md +303 -0
  11. package/lib/assets/docs/article/functional-desgin-ppp/all/11-command-pattern.md +286 -0
  12. package/lib/assets/docs/article/functional-desgin-ppp/all/12-visitor-pattern.md +322 -0
  13. package/lib/assets/docs/article/functional-desgin-ppp/all/13-abstract-factory-pattern.md +319 -0
  14. package/lib/assets/docs/article/functional-desgin-ppp/all/14-abstract-server-pattern.md +365 -0
  15. package/lib/assets/docs/article/functional-desgin-ppp/all/15-gossiping-bus-drivers.md +156 -0
  16. package/lib/assets/docs/article/functional-desgin-ppp/all/16-payroll-system.md +178 -0
  17. package/lib/assets/docs/article/functional-desgin-ppp/all/17-video-rental-system.md +312 -0
  18. package/lib/assets/docs/article/functional-desgin-ppp/all/18-concurrency-system.md +287 -0
  19. package/lib/assets/docs/article/functional-desgin-ppp/all/19-wa-tor-simulation.md +286 -0
  20. package/lib/assets/docs/article/functional-desgin-ppp/all/20-pattern-interactions.md +274 -0
  21. package/lib/assets/docs/article/functional-desgin-ppp/all/21-best-practices.md +294 -0
  22. package/lib/assets/docs/article/functional-desgin-ppp/all/22-oo-to-fp-migration.md +337 -0
  23. package/lib/assets/docs/article/functional-desgin-ppp/all/index.md +388 -0
  24. package/lib/assets/docs/article/functional-desgin-ppp/clojure/01-immutability-and-data-transformation.md +271 -0
  25. package/lib/assets/docs/article/functional-desgin-ppp/clojure/02-function-composition.md +380 -0
  26. package/lib/assets/docs/article/functional-desgin-ppp/clojure/03-polymorphism.md +384 -0
  27. package/lib/assets/docs/article/functional-desgin-ppp/clojure/04-clojure-spec.md +350 -0
  28. package/lib/assets/docs/article/functional-desgin-ppp/clojure/05-property-based-testing.md +352 -0
  29. package/lib/assets/docs/article/functional-desgin-ppp/clojure/06-tdd-in-functional.md +383 -0
  30. package/lib/assets/docs/article/functional-desgin-ppp/clojure/07-composite-pattern.md +529 -0
  31. package/lib/assets/docs/article/functional-desgin-ppp/clojure/08-decorator-pattern.md +395 -0
  32. package/lib/assets/docs/article/functional-desgin-ppp/clojure/09-adapter-pattern.md +399 -0
  33. package/lib/assets/docs/article/functional-desgin-ppp/clojure/10-strategy-pattern.md +485 -0
  34. package/lib/assets/docs/article/functional-desgin-ppp/clojure/11-command-pattern.md +566 -0
  35. package/lib/assets/docs/article/functional-desgin-ppp/clojure/12-visitor-pattern.md +567 -0
  36. package/lib/assets/docs/article/functional-desgin-ppp/clojure/13-abstract-factory-pattern.md +475 -0
  37. package/lib/assets/docs/article/functional-desgin-ppp/clojure/14-abstract-server-pattern.md +462 -0
  38. package/lib/assets/docs/article/functional-desgin-ppp/clojure/15-gossiping-bus-drivers.md +323 -0
  39. package/lib/assets/docs/article/functional-desgin-ppp/clojure/16-payroll-system.md +401 -0
  40. package/lib/assets/docs/article/functional-desgin-ppp/clojure/17-video-rental-system.md +450 -0
  41. package/lib/assets/docs/article/functional-desgin-ppp/clojure/18-concurrency-system.md +475 -0
  42. package/lib/assets/docs/article/functional-desgin-ppp/clojure/19-wator-simulation.md +739 -0
  43. package/lib/assets/docs/article/functional-desgin-ppp/clojure/20-pattern-interactions.md +562 -0
  44. package/lib/assets/docs/article/functional-desgin-ppp/clojure/21-best-practices.md +506 -0
  45. package/lib/assets/docs/article/functional-desgin-ppp/clojure/22-oo-to-fp-migration.md +526 -0
  46. package/lib/assets/docs/article/functional-desgin-ppp/clojure/index.md +197 -0
  47. package/lib/assets/docs/article/functional-desgin-ppp/elixir/01-immutability-and-data-transformation.md +381 -0
  48. package/lib/assets/docs/article/functional-desgin-ppp/elixir/02-function-composition.md +374 -0
  49. package/lib/assets/docs/article/functional-desgin-ppp/elixir/03-polymorphism.md +375 -0
  50. package/lib/assets/docs/article/functional-desgin-ppp/elixir/04-data-validation.md +195 -0
  51. package/lib/assets/docs/article/functional-desgin-ppp/elixir/05-property-based-testing.md +268 -0
  52. package/lib/assets/docs/article/functional-desgin-ppp/elixir/06-tdd-and-fp.md +294 -0
  53. package/lib/assets/docs/article/functional-desgin-ppp/elixir/07-effects-and-pure-functions.md +164 -0
  54. package/lib/assets/docs/article/functional-desgin-ppp/elixir/08-error-handling-strategies.md +168 -0
  55. package/lib/assets/docs/article/functional-desgin-ppp/elixir/09-io-and-external-systems.md +254 -0
  56. package/lib/assets/docs/article/functional-desgin-ppp/elixir/10-concurrency-patterns.md +269 -0
  57. package/lib/assets/docs/article/functional-desgin-ppp/elixir/11-command-pattern.md +148 -0
  58. package/lib/assets/docs/article/functional-desgin-ppp/elixir/12-visitor-pattern.md +176 -0
  59. package/lib/assets/docs/article/functional-desgin-ppp/elixir/13-abstract-factory-pattern.md +604 -0
  60. package/lib/assets/docs/article/functional-desgin-ppp/elixir/14-abstract-server-pattern.md +729 -0
  61. package/lib/assets/docs/article/functional-desgin-ppp/elixir/15-gossiping-bus-drivers.md +291 -0
  62. package/lib/assets/docs/article/functional-desgin-ppp/elixir/16-payroll-system.md +420 -0
  63. package/lib/assets/docs/article/functional-desgin-ppp/elixir/17-video-rental-system.md +319 -0
  64. package/lib/assets/docs/article/functional-desgin-ppp/elixir/18-concurrency-system.md +466 -0
  65. package/lib/assets/docs/article/functional-desgin-ppp/elixir/19-wator-simulation.md +523 -0
  66. package/lib/assets/docs/article/functional-desgin-ppp/elixir/20-pattern-interactions.md +287 -0
  67. package/lib/assets/docs/article/functional-desgin-ppp/elixir/21-best-practices.md +340 -0
  68. package/lib/assets/docs/article/functional-desgin-ppp/elixir/22-oo-to-fp-migration.md +395 -0
  69. package/lib/assets/docs/article/functional-desgin-ppp/elixir/index.md +204 -0
  70. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/01-immutability-and-data-transformation.md +382 -0
  71. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/02-function-composition.md +452 -0
  72. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/03-polymorphism.md +495 -0
  73. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/04-data-validation.md +416 -0
  74. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/05-property-based-testing.md +382 -0
  75. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/06-tdd-functional.md +687 -0
  76. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/07-composite-pattern.md +442 -0
  77. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/08-decorator-pattern.md +479 -0
  78. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/09-adapter-pattern.md +479 -0
  79. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/10-strategy-pattern.md +427 -0
  80. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/11-command-pattern.md +428 -0
  81. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/12-visitor-pattern.md +339 -0
  82. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/13-abstract-factory-pattern.md +309 -0
  83. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/14-abstract-server-pattern.md +596 -0
  84. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/15-gossiping-bus-drivers.md +353 -0
  85. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/16-payroll-system.md +350 -0
  86. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/17-video-rental-system.md +412 -0
  87. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/18-concurrency-system.md +367 -0
  88. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/19-wator-simulation.md +401 -0
  89. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/20-pattern-interactions.md +291 -0
  90. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/21-best-practices.md +320 -0
  91. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/22-oo-to-fp-migration.md +322 -0
  92. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/index.md +230 -0
  93. package/lib/assets/docs/article/functional-desgin-ppp/haskell/01-immutability-and-data-transformation.md +298 -0
  94. package/lib/assets/docs/article/functional-desgin-ppp/haskell/02-function-composition.md +304 -0
  95. package/lib/assets/docs/article/functional-desgin-ppp/haskell/03-polymorphism.md +362 -0
  96. package/lib/assets/docs/article/functional-desgin-ppp/haskell/04-data-validation.md +257 -0
  97. package/lib/assets/docs/article/functional-desgin-ppp/haskell/05-property-based-testing.md +254 -0
  98. package/lib/assets/docs/article/functional-desgin-ppp/haskell/06-tdd-functional.md +283 -0
  99. package/lib/assets/docs/article/functional-desgin-ppp/haskell/07-composite-pattern.md +395 -0
  100. package/lib/assets/docs/article/functional-desgin-ppp/haskell/08-decorator-pattern.md +319 -0
  101. package/lib/assets/docs/article/functional-desgin-ppp/haskell/09-adapter-pattern.md +382 -0
  102. package/lib/assets/docs/article/functional-desgin-ppp/haskell/10-strategy-pattern.md +287 -0
  103. package/lib/assets/docs/article/functional-desgin-ppp/haskell/11-command-pattern.md +303 -0
  104. package/lib/assets/docs/article/functional-desgin-ppp/haskell/12-visitor-pattern.md +326 -0
  105. package/lib/assets/docs/article/functional-desgin-ppp/haskell/13-abstract-factory-pattern.md +332 -0
  106. package/lib/assets/docs/article/functional-desgin-ppp/haskell/14-abstract-server-pattern.md +379 -0
  107. package/lib/assets/docs/article/functional-desgin-ppp/haskell/15-gossiping-bus-drivers.md +175 -0
  108. package/lib/assets/docs/article/functional-desgin-ppp/haskell/16-payroll-system.md +219 -0
  109. package/lib/assets/docs/article/functional-desgin-ppp/haskell/17-video-rental-system.md +244 -0
  110. package/lib/assets/docs/article/functional-desgin-ppp/haskell/18-concurrency-system.md +363 -0
  111. package/lib/assets/docs/article/functional-desgin-ppp/haskell/19-wator-simulation.md +438 -0
  112. package/lib/assets/docs/article/functional-desgin-ppp/haskell/20-pattern-interactions.md +323 -0
  113. package/lib/assets/docs/article/functional-desgin-ppp/haskell/21-best-practices.md +403 -0
  114. package/lib/assets/docs/article/functional-desgin-ppp/haskell/22-oo-to-fp-migration.md +469 -0
  115. package/lib/assets/docs/article/functional-desgin-ppp/haskell/index.md +174 -0
  116. package/lib/assets/docs/article/functional-desgin-ppp/index.md +90 -0
  117. package/lib/assets/docs/article/functional-desgin-ppp/rust/01-immutability-and-data-transformation.md +448 -0
  118. package/lib/assets/docs/article/functional-desgin-ppp/rust/02-function-composition.md +463 -0
  119. package/lib/assets/docs/article/functional-desgin-ppp/rust/03-polymorphism.md +425 -0
  120. package/lib/assets/docs/article/functional-desgin-ppp/rust/04-data-validation.md +273 -0
  121. package/lib/assets/docs/article/functional-desgin-ppp/rust/05-property-based-testing.md +247 -0
  122. package/lib/assets/docs/article/functional-desgin-ppp/rust/06-tdd-and-functional.md +841 -0
  123. package/lib/assets/docs/article/functional-desgin-ppp/rust/07-composite-pattern.md +384 -0
  124. package/lib/assets/docs/article/functional-desgin-ppp/rust/08-decorator-pattern.md +383 -0
  125. package/lib/assets/docs/article/functional-desgin-ppp/rust/09-adapter-pattern.md +339 -0
  126. package/lib/assets/docs/article/functional-desgin-ppp/rust/10-strategy-pattern.md +331 -0
  127. package/lib/assets/docs/article/functional-desgin-ppp/rust/11-command-pattern.md +356 -0
  128. package/lib/assets/docs/article/functional-desgin-ppp/rust/12-visitor-pattern.md +379 -0
  129. package/lib/assets/docs/article/functional-desgin-ppp/rust/13-abstract-factory-pattern.md +361 -0
  130. package/lib/assets/docs/article/functional-desgin-ppp/rust/14-abstract-server-pattern.md +392 -0
  131. package/lib/assets/docs/article/functional-desgin-ppp/rust/15-gossiping-bus-drivers.md +300 -0
  132. package/lib/assets/docs/article/functional-desgin-ppp/rust/16-payroll-system.md +297 -0
  133. package/lib/assets/docs/article/functional-desgin-ppp/rust/17-video-rental-system.md +304 -0
  134. package/lib/assets/docs/article/functional-desgin-ppp/rust/18-concurrency-system.md +315 -0
  135. package/lib/assets/docs/article/functional-desgin-ppp/rust/19-wator-simulation.md +311 -0
  136. package/lib/assets/docs/article/functional-desgin-ppp/rust/20-pattern-interactions.md +304 -0
  137. package/lib/assets/docs/article/functional-desgin-ppp/rust/21-best-practices.md +336 -0
  138. package/lib/assets/docs/article/functional-desgin-ppp/rust/22-oo-to-fp-migration.md +349 -0
  139. package/lib/assets/docs/article/functional-desgin-ppp/rust/index.md +199 -0
  140. package/lib/assets/docs/article/functional-desgin-ppp/scala/01-immutability-and-data-transformation.md +326 -0
  141. package/lib/assets/docs/article/functional-desgin-ppp/scala/02-function-composition.md +348 -0
  142. package/lib/assets/docs/article/functional-desgin-ppp/scala/03-polymorphism.md +357 -0
  143. package/lib/assets/docs/article/functional-desgin-ppp/scala/04-data-validation.md +364 -0
  144. package/lib/assets/docs/article/functional-desgin-ppp/scala/05-property-based-testing.md +515 -0
  145. package/lib/assets/docs/article/functional-desgin-ppp/scala/06-tdd-functional.md +557 -0
  146. package/lib/assets/docs/article/functional-desgin-ppp/scala/07-composite-pattern.md +363 -0
  147. package/lib/assets/docs/article/functional-desgin-ppp/scala/08-decorator-pattern.md +327 -0
  148. package/lib/assets/docs/article/functional-desgin-ppp/scala/09-adapter-pattern.md +517 -0
  149. package/lib/assets/docs/article/functional-desgin-ppp/scala/10-strategy-pattern.md +441 -0
  150. package/lib/assets/docs/article/functional-desgin-ppp/scala/11-command-pattern.md +407 -0
  151. package/lib/assets/docs/article/functional-desgin-ppp/scala/12-visitor-pattern.md +379 -0
  152. package/lib/assets/docs/article/functional-desgin-ppp/scala/13-abstract-factory-pattern.md +398 -0
  153. package/lib/assets/docs/article/functional-desgin-ppp/scala/14-abstract-server-pattern.md +476 -0
  154. package/lib/assets/docs/article/functional-desgin-ppp/scala/15-gossiping-bus-drivers.md +389 -0
  155. package/lib/assets/docs/article/functional-desgin-ppp/scala/16-payroll-system.md +342 -0
  156. package/lib/assets/docs/article/functional-desgin-ppp/scala/17-video-rental-system.md +324 -0
  157. package/lib/assets/docs/article/functional-desgin-ppp/scala/18-concurrency-system.md +730 -0
  158. package/lib/assets/docs/article/functional-desgin-ppp/scala/19-wator-simulation.md +624 -0
  159. package/lib/assets/docs/article/functional-desgin-ppp/scala/20-pattern-interactions.md +512 -0
  160. package/lib/assets/docs/article/functional-desgin-ppp/scala/21-best-practices.md +427 -0
  161. package/lib/assets/docs/article/functional-desgin-ppp/scala/22-oo-to-fp-migration.md +682 -0
  162. package/lib/assets/docs/article/functional-desgin-ppp/scala/index.md +199 -0
  163. package/lib/assets/docs/article/getting-start-tdd/clojure/01-todo-list-and-first-test.md +166 -0
  164. package/lib/assets/docs/article/getting-start-tdd/clojure/02-fake-it-and-triangulation.md +162 -0
  165. package/lib/assets/docs/article/getting-start-tdd/clojure/03-obvious-implementation-and-refactoring.md +135 -0
  166. package/lib/assets/docs/article/getting-start-tdd/clojure/04-version-control-and-conventional-commits.md +88 -0
  167. package/lib/assets/docs/article/getting-start-tdd/clojure/05-package-management-and-static-analysis.md +299 -0
  168. package/lib/assets/docs/article/getting-start-tdd/clojure/06-task-runner-and-ci-cd.md +241 -0
  169. package/lib/assets/docs/article/getting-start-tdd/clojure/07-protocols-and-records.md +131 -0
  170. package/lib/assets/docs/article/getting-start-tdd/clojure/08-multimethods-and-design-patterns.md +130 -0
  171. package/lib/assets/docs/article/getting-start-tdd/clojure/09-namespaces-and-module-design.md +127 -0
  172. package/lib/assets/docs/article/getting-start-tdd/clojure/10-higher-order-functions-and-composition.md +114 -0
  173. package/lib/assets/docs/article/getting-start-tdd/clojure/11-persistent-data-and-pipeline.md +138 -0
  174. package/lib/assets/docs/article/getting-start-tdd/clojure/12-error-handling-and-spec.md +161 -0
  175. package/lib/assets/docs/article/getting-start-tdd/clojure/index.md +65 -0
  176. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter01.md +232 -0
  177. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter02.md +244 -0
  178. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter03.md +202 -0
  179. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter04.md +92 -0
  180. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter05.md +256 -0
  181. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter06.md +195 -0
  182. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter07.md +214 -0
  183. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter08.md +249 -0
  184. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter09.md +174 -0
  185. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter10.md +166 -0
  186. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter11.md +192 -0
  187. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter12.md +211 -0
  188. package/lib/assets/docs/article/getting-start-tdd/csharp/index.md +83 -0
  189. package/lib/assets/docs/article/getting-start-tdd/elixir/01-todo-list-and-first-test.md +87 -0
  190. package/lib/assets/docs/article/getting-start-tdd/elixir/02-fake-it-and-triangulation.md +95 -0
  191. package/lib/assets/docs/article/getting-start-tdd/elixir/03-obvious-implementation-and-refactoring.md +109 -0
  192. package/lib/assets/docs/article/getting-start-tdd/elixir/04-version-control-and-conventional-commits.md +96 -0
  193. package/lib/assets/docs/article/getting-start-tdd/elixir/05-package-management-and-static-analysis.md +88 -0
  194. package/lib/assets/docs/article/getting-start-tdd/elixir/06-task-runner-and-ci-cd.md +71 -0
  195. package/lib/assets/docs/article/getting-start-tdd/elixir/07-structs-and-protocols.md +110 -0
  196. package/lib/assets/docs/article/getting-start-tdd/elixir/08-pattern-matching-and-guards.md +108 -0
  197. package/lib/assets/docs/article/getting-start-tdd/elixir/09-module-design-and-behaviours.md +104 -0
  198. package/lib/assets/docs/article/getting-start-tdd/elixir/10-higher-order-functions-and-pipeline.md +178 -0
  199. package/lib/assets/docs/article/getting-start-tdd/elixir/11-stream-and-lazy-evaluation.md +142 -0
  200. package/lib/assets/docs/article/getting-start-tdd/elixir/12-error-handling-and-with.md +145 -0
  201. package/lib/assets/docs/article/getting-start-tdd/elixir/index.md +35 -0
  202. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter01.md +202 -0
  203. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter02.md +246 -0
  204. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter03.md +218 -0
  205. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter04.md +179 -0
  206. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter05.md +267 -0
  207. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter06.md +190 -0
  208. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter07.md +161 -0
  209. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter08.md +175 -0
  210. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter09.md +222 -0
  211. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter10.md +189 -0
  212. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter11.md +212 -0
  213. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter12.md +215 -0
  214. package/lib/assets/docs/article/getting-start-tdd/fsharp/index.md +71 -0
  215. package/lib/assets/docs/article/getting-start-tdd/go/01-todo-list-and-first-test.md +213 -0
  216. package/lib/assets/docs/article/getting-start-tdd/go/02-fake-it-and-triangulation.md +302 -0
  217. package/lib/assets/docs/article/getting-start-tdd/go/03-obvious-implementation-and-refactoring.md +339 -0
  218. package/lib/assets/docs/article/getting-start-tdd/go/04-version-control-and-conventional-commits.md +112 -0
  219. package/lib/assets/docs/article/getting-start-tdd/go/05-package-management-and-static-analysis.md +272 -0
  220. package/lib/assets/docs/article/getting-start-tdd/go/06-task-runner-and-ci-cd.md +233 -0
  221. package/lib/assets/docs/article/getting-start-tdd/go/07-encapsulation-and-polymorphism.md +394 -0
  222. package/lib/assets/docs/article/getting-start-tdd/go/08-design-patterns.md +422 -0
  223. package/lib/assets/docs/article/getting-start-tdd/go/09-solid-principles-and-module-design.md +400 -0
  224. package/lib/assets/docs/article/getting-start-tdd/go/10-higher-order-functions-and-composition.md +226 -0
  225. package/lib/assets/docs/article/getting-start-tdd/go/11-immutable-data-and-pipeline.md +296 -0
  226. package/lib/assets/docs/article/getting-start-tdd/go/12-error-handling-and-type-safety.md +411 -0
  227. package/lib/assets/docs/article/getting-start-tdd/go/index.md +83 -0
  228. package/lib/assets/docs/article/getting-start-tdd/haskell/01-todo-list-and-first-test.md +279 -0
  229. package/lib/assets/docs/article/getting-start-tdd/haskell/02-fake-it-and-triangulation.md +337 -0
  230. package/lib/assets/docs/article/getting-start-tdd/haskell/03-obvious-implementation-and-refactoring.md +257 -0
  231. package/lib/assets/docs/article/getting-start-tdd/haskell/04-version-control-and-conventional-commits.md +182 -0
  232. package/lib/assets/docs/article/getting-start-tdd/haskell/05-package-management-and-static-analysis.md +313 -0
  233. package/lib/assets/docs/article/getting-start-tdd/haskell/06-task-runner-and-ci-cd.md +309 -0
  234. package/lib/assets/docs/article/getting-start-tdd/haskell/07-algebraic-data-types-and-type-classes.md +412 -0
  235. package/lib/assets/docs/article/getting-start-tdd/haskell/08-pattern-matching-and-guards.md +390 -0
  236. package/lib/assets/docs/article/getting-start-tdd/haskell/09-module-design-and-smart-constructors.md +461 -0
  237. package/lib/assets/docs/article/getting-start-tdd/haskell/10-higher-order-functions-and-currying.md +434 -0
  238. package/lib/assets/docs/article/getting-start-tdd/haskell/11-function-composition-and-point-free.md +392 -0
  239. package/lib/assets/docs/article/getting-start-tdd/haskell/12-monad-and-error-handling.md +631 -0
  240. package/lib/assets/docs/article/getting-start-tdd/haskell/index.md +49 -0
  241. package/lib/assets/docs/article/getting-start-tdd/index.md +93 -0
  242. package/lib/assets/docs/article/getting-start-tdd/integration/01-language-overview.md +375 -0
  243. package/lib/assets/docs/article/getting-start-tdd/integration/02-test-framework-comparison.md +349 -0
  244. package/lib/assets/docs/article/getting-start-tdd/integration/03-tdd-pattern-comparison.md +445 -0
  245. package/lib/assets/docs/article/getting-start-tdd/integration/04-type-system-comparison.md +405 -0
  246. package/lib/assets/docs/article/getting-start-tdd/integration/05-dev-environment-comparison.md +330 -0
  247. package/lib/assets/docs/article/getting-start-tdd/integration/06-learning-roadmap.md +274 -0
  248. package/lib/assets/docs/article/getting-start-tdd/integration/index.md +69 -0
  249. package/lib/assets/docs/article/getting-start-tdd/java/01-todo-list-and-first-test.md +234 -0
  250. package/lib/assets/docs/article/getting-start-tdd/java/02-fake-it-and-triangulation.md +261 -0
  251. package/lib/assets/docs/article/getting-start-tdd/java/03-obvious-implementation-and-refactoring.md +185 -0
  252. package/lib/assets/docs/article/getting-start-tdd/java/04-version-control-and-conventional-commits.md +115 -0
  253. package/lib/assets/docs/article/getting-start-tdd/java/05-package-management-and-static-analysis.md +382 -0
  254. package/lib/assets/docs/article/getting-start-tdd/java/06-task-runner-and-ci-cd.md +272 -0
  255. package/lib/assets/docs/article/getting-start-tdd/java/07-encapsulation-and-polymorphism.md +626 -0
  256. package/lib/assets/docs/article/getting-start-tdd/java/08-design-patterns.md +393 -0
  257. package/lib/assets/docs/article/getting-start-tdd/java/09-solid-principles-and-module-design.md +310 -0
  258. package/lib/assets/docs/article/getting-start-tdd/java/10-higher-order-functions-and-composition.md +188 -0
  259. package/lib/assets/docs/article/getting-start-tdd/java/11-immutable-data-and-pipeline.md +167 -0
  260. package/lib/assets/docs/article/getting-start-tdd/java/12-error-handling-and-type-safety.md +205 -0
  261. package/lib/assets/docs/article/getting-start-tdd/java/index.md +61 -0
  262. package/lib/assets/docs/article/getting-start-tdd/node/01-todo-list-and-first-test.md +244 -0
  263. package/lib/assets/docs/article/getting-start-tdd/node/02-fake-it-and-triangulation.md +262 -0
  264. package/lib/assets/docs/article/getting-start-tdd/node/03-obvious-implementation-and-refactoring.md +169 -0
  265. package/lib/assets/docs/article/getting-start-tdd/node/04-version-control-and-conventional-commits.md +112 -0
  266. package/lib/assets/docs/article/getting-start-tdd/node/05-package-management-and-static-analysis.md +314 -0
  267. package/lib/assets/docs/article/getting-start-tdd/node/06-task-runner-and-ci-cd.md +235 -0
  268. package/lib/assets/docs/article/getting-start-tdd/node/07-encapsulation-and-polymorphism.md +327 -0
  269. package/lib/assets/docs/article/getting-start-tdd/node/08-design-patterns.md +322 -0
  270. package/lib/assets/docs/article/getting-start-tdd/node/09-solid-principles-and-module-design.md +285 -0
  271. package/lib/assets/docs/article/getting-start-tdd/node/10-higher-order-functions-and-composition.md +199 -0
  272. package/lib/assets/docs/article/getting-start-tdd/node/11-immutable-data-and-pipeline.md +207 -0
  273. package/lib/assets/docs/article/getting-start-tdd/node/12-error-handling-and-type-safety.md +295 -0
  274. package/lib/assets/docs/article/getting-start-tdd/node/index.md +56 -0
  275. package/lib/assets/docs/article/getting-start-tdd/php/01-todo-list-and-first-test.md +259 -0
  276. package/lib/assets/docs/article/getting-start-tdd/php/02-fake-it-and-triangulation.md +200 -0
  277. package/lib/assets/docs/article/getting-start-tdd/php/03-obvious-implementation-and-refactoring.md +248 -0
  278. package/lib/assets/docs/article/getting-start-tdd/php/04-version-control-and-conventional-commits.md +141 -0
  279. package/lib/assets/docs/article/getting-start-tdd/php/05-package-management-and-static-analysis.md +410 -0
  280. package/lib/assets/docs/article/getting-start-tdd/php/06-task-runner-and-ci-cd.md +321 -0
  281. package/lib/assets/docs/article/getting-start-tdd/php/07-encapsulation-and-polymorphism.md +372 -0
  282. package/lib/assets/docs/article/getting-start-tdd/php/08-design-patterns.md +453 -0
  283. package/lib/assets/docs/article/getting-start-tdd/php/09-solid-principles-and-module-design.md +460 -0
  284. package/lib/assets/docs/article/getting-start-tdd/php/10-higher-order-functions-and-composition.md +182 -0
  285. package/lib/assets/docs/article/getting-start-tdd/php/11-immutable-data-and-pipeline.md +266 -0
  286. package/lib/assets/docs/article/getting-start-tdd/php/12-error-handling-and-type-safety.md +308 -0
  287. package/lib/assets/docs/article/getting-start-tdd/php/index.md +84 -0
  288. package/lib/assets/docs/article/getting-start-tdd/python/01-todo-list-and-first-test.md +201 -0
  289. package/lib/assets/docs/article/getting-start-tdd/python/02-fake-it-and-triangulation.md +247 -0
  290. package/lib/assets/docs/article/getting-start-tdd/python/03-obvious-implementation-and-refactoring.md +199 -0
  291. package/lib/assets/docs/article/getting-start-tdd/python/04-version-control-and-conventional-commits.md +87 -0
  292. package/lib/assets/docs/article/getting-start-tdd/python/05-package-management-and-static-analysis.md +274 -0
  293. package/lib/assets/docs/article/getting-start-tdd/python/06-task-runner-and-ci-cd.md +190 -0
  294. package/lib/assets/docs/article/getting-start-tdd/python/07-encapsulation-and-polymorphism.md +208 -0
  295. package/lib/assets/docs/article/getting-start-tdd/python/08-design-patterns.md +172 -0
  296. package/lib/assets/docs/article/getting-start-tdd/python/09-solid-principles-and-module-design.md +130 -0
  297. package/lib/assets/docs/article/getting-start-tdd/python/10-higher-order-functions-and-composition.md +122 -0
  298. package/lib/assets/docs/article/getting-start-tdd/python/11-immutable-data-and-pipeline.md +116 -0
  299. package/lib/assets/docs/article/getting-start-tdd/python/12-error-handling-and-type-safety.md +126 -0
  300. package/lib/assets/docs/article/getting-start-tdd/python/index.md +55 -0
  301. package/lib/assets/docs/article/getting-start-tdd/ruby/01-todo-list-and-first-test.md +231 -0
  302. package/lib/assets/docs/article/getting-start-tdd/ruby/02-fake-it-and-triangulation.md +238 -0
  303. package/lib/assets/docs/article/getting-start-tdd/ruby/03-obvious-implementation-and-refactoring.md +228 -0
  304. package/lib/assets/docs/article/getting-start-tdd/ruby/04-version-control-and-conventional-commits.md +112 -0
  305. package/lib/assets/docs/article/getting-start-tdd/ruby/05-package-management-and-static-analysis.md +287 -0
  306. package/lib/assets/docs/article/getting-start-tdd/ruby/06-task-runner-and-ci-cd.md +248 -0
  307. package/lib/assets/docs/article/getting-start-tdd/ruby/07-encapsulation-and-polymorphism.md +279 -0
  308. package/lib/assets/docs/article/getting-start-tdd/ruby/08-design-patterns.md +329 -0
  309. package/lib/assets/docs/article/getting-start-tdd/ruby/09-solid-principles-and-module-design.md +196 -0
  310. package/lib/assets/docs/article/getting-start-tdd/ruby/10-higher-order-functions-and-composition.md +175 -0
  311. package/lib/assets/docs/article/getting-start-tdd/ruby/11-immutable-data-and-pipeline.md +233 -0
  312. package/lib/assets/docs/article/getting-start-tdd/ruby/12-error-handling-and-type-safety.md +398 -0
  313. package/lib/assets/docs/article/getting-start-tdd/ruby/index.md +83 -0
  314. package/lib/assets/docs/article/getting-start-tdd/rust/01-todo-list-and-first-test.md +211 -0
  315. package/lib/assets/docs/article/getting-start-tdd/rust/02-fake-it-and-triangulation.md +264 -0
  316. package/lib/assets/docs/article/getting-start-tdd/rust/03-obvious-implementation-and-refactoring.md +233 -0
  317. package/lib/assets/docs/article/getting-start-tdd/rust/04-version-control-and-conventional-commits.md +92 -0
  318. package/lib/assets/docs/article/getting-start-tdd/rust/05-package-management-and-static-analysis.md +212 -0
  319. package/lib/assets/docs/article/getting-start-tdd/rust/06-task-runner-and-ci-cd.md +164 -0
  320. package/lib/assets/docs/article/getting-start-tdd/rust/07-encapsulation-and-polymorphism.md +142 -0
  321. package/lib/assets/docs/article/getting-start-tdd/rust/08-design-patterns.md +145 -0
  322. package/lib/assets/docs/article/getting-start-tdd/rust/09-solid-principles-and-module-design.md +110 -0
  323. package/lib/assets/docs/article/getting-start-tdd/rust/10-higher-order-functions-and-composition.md +94 -0
  324. package/lib/assets/docs/article/getting-start-tdd/rust/11-immutable-data-and-pipeline.md +105 -0
  325. package/lib/assets/docs/article/getting-start-tdd/rust/12-error-handling-and-type-safety.md +112 -0
  326. package/lib/assets/docs/article/getting-start-tdd/rust/index.md +83 -0
  327. package/lib/assets/docs/article/getting-start-tdd/scala/01-todo-list-and-first-test.md +111 -0
  328. package/lib/assets/docs/article/getting-start-tdd/scala/02-fake-it-and-triangulation.md +107 -0
  329. package/lib/assets/docs/article/getting-start-tdd/scala/03-obvious-implementation-and-refactoring.md +99 -0
  330. package/lib/assets/docs/article/getting-start-tdd/scala/04-version-control-and-conventional-commits.md +123 -0
  331. package/lib/assets/docs/article/getting-start-tdd/scala/05-package-management-and-static-analysis.md +196 -0
  332. package/lib/assets/docs/article/getting-start-tdd/scala/06-task-runner-and-ci-cd.md +186 -0
  333. package/lib/assets/docs/article/getting-start-tdd/scala/07-case-classes-and-traits.md +139 -0
  334. package/lib/assets/docs/article/getting-start-tdd/scala/08-pattern-matching-and-sealed-traits.md +106 -0
  335. package/lib/assets/docs/article/getting-start-tdd/scala/09-packages-and-module-design.md +75 -0
  336. package/lib/assets/docs/article/getting-start-tdd/scala/10-higher-order-functions-and-composition.md +104 -0
  337. package/lib/assets/docs/article/getting-start-tdd/scala/11-collections-and-lazy-evaluation.md +94 -0
  338. package/lib/assets/docs/article/getting-start-tdd/scala/12-error-handling-and-type-safety.md +92 -0
  339. package/lib/assets/docs/article/getting-start-tdd/scala/index.md +65 -0
  340. package/lib/assets/docs/article/grokking-concurrency/all/index.md +404 -0
  341. package/lib/assets/docs/article/grokking-concurrency/all/part-1-ch02-sequential.md +554 -0
  342. package/lib/assets/docs/article/grokking-concurrency/all/part-2-ch04-05-threads.md +469 -0
  343. package/lib/assets/docs/article/grokking-concurrency/all/part-3-ch06-multitasking.md +520 -0
  344. package/lib/assets/docs/article/grokking-concurrency/all/part-4-ch07-parallel-patterns.md +420 -0
  345. package/lib/assets/docs/article/grokking-concurrency/all/part-5-ch08-09-synchronization.md +510 -0
  346. package/lib/assets/docs/article/grokking-concurrency/all/part-6-ch10-11-nonblocking-io.md +435 -0
  347. package/lib/assets/docs/article/grokking-concurrency/all/part-7-ch12-async.md +465 -0
  348. package/lib/assets/docs/article/grokking-concurrency/all/part-8-ch13-mapreduce.md +377 -0
  349. package/lib/assets/docs/article/grokking-concurrency/clojure/index.md +116 -0
  350. package/lib/assets/docs/article/grokking-concurrency/clojure/part-1.md +108 -0
  351. package/lib/assets/docs/article/grokking-concurrency/clojure/part-2.md +101 -0
  352. package/lib/assets/docs/article/grokking-concurrency/clojure/part-3.md +122 -0
  353. package/lib/assets/docs/article/grokking-concurrency/clojure/part-4.md +123 -0
  354. package/lib/assets/docs/article/grokking-concurrency/clojure/part-5.md +118 -0
  355. package/lib/assets/docs/article/grokking-concurrency/clojure/part-6.md +89 -0
  356. package/lib/assets/docs/article/grokking-concurrency/clojure/part-7.md +100 -0
  357. package/lib/assets/docs/article/grokking-concurrency/clojure/part-8.md +120 -0
  358. package/lib/assets/docs/article/grokking-concurrency/csharp/index.md +101 -0
  359. package/lib/assets/docs/article/grokking-concurrency/csharp/part-1.md +97 -0
  360. package/lib/assets/docs/article/grokking-concurrency/csharp/part-2.md +123 -0
  361. package/lib/assets/docs/article/grokking-concurrency/csharp/part-3.md +101 -0
  362. package/lib/assets/docs/article/grokking-concurrency/csharp/part-4.md +112 -0
  363. package/lib/assets/docs/article/grokking-concurrency/csharp/part-5.md +99 -0
  364. package/lib/assets/docs/article/grokking-concurrency/csharp/part-6.md +61 -0
  365. package/lib/assets/docs/article/grokking-concurrency/csharp/part-7.md +84 -0
  366. package/lib/assets/docs/article/grokking-concurrency/csharp/part-8.md +92 -0
  367. package/lib/assets/docs/article/grokking-concurrency/fsharp/index.md +65 -0
  368. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-1.md +80 -0
  369. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-2.md +103 -0
  370. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-3.md +94 -0
  371. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-4.md +110 -0
  372. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-5.md +104 -0
  373. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-6.md +93 -0
  374. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-7.md +121 -0
  375. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-8.md +107 -0
  376. package/lib/assets/docs/article/grokking-concurrency/haskell/index.md +248 -0
  377. package/lib/assets/docs/article/grokking-concurrency/haskell/part-1.md +96 -0
  378. package/lib/assets/docs/article/grokking-concurrency/haskell/part-2.md +96 -0
  379. package/lib/assets/docs/article/grokking-concurrency/haskell/part-3.md +91 -0
  380. package/lib/assets/docs/article/grokking-concurrency/haskell/part-4.md +106 -0
  381. package/lib/assets/docs/article/grokking-concurrency/haskell/part-5.md +99 -0
  382. package/lib/assets/docs/article/grokking-concurrency/haskell/part-6.md +95 -0
  383. package/lib/assets/docs/article/grokking-concurrency/haskell/part-7.md +111 -0
  384. package/lib/assets/docs/article/grokking-concurrency/haskell/part-8.md +118 -0
  385. package/lib/assets/docs/article/grokking-concurrency/index.md +66 -0
  386. package/lib/assets/docs/article/grokking-concurrency/java/index.md +102 -0
  387. package/lib/assets/docs/article/grokking-concurrency/java/part-1.md +308 -0
  388. package/lib/assets/docs/article/grokking-concurrency/java/part-2.md +334 -0
  389. package/lib/assets/docs/article/grokking-concurrency/java/part-3.md +221 -0
  390. package/lib/assets/docs/article/grokking-concurrency/java/part-4.md +213 -0
  391. package/lib/assets/docs/article/grokking-concurrency/java/part-5.md +112 -0
  392. package/lib/assets/docs/article/grokking-concurrency/java/part-6.md +69 -0
  393. package/lib/assets/docs/article/grokking-concurrency/java/part-7.md +101 -0
  394. package/lib/assets/docs/article/grokking-concurrency/java/part-8.md +101 -0
  395. package/lib/assets/docs/article/grokking-concurrency/python/index.md +313 -0
  396. package/lib/assets/docs/article/grokking-concurrency/python/part-1.md +239 -0
  397. package/lib/assets/docs/article/grokking-concurrency/python/part-2.md +418 -0
  398. package/lib/assets/docs/article/grokking-concurrency/python/part-3.md +227 -0
  399. package/lib/assets/docs/article/grokking-concurrency/python/part-4.md +299 -0
  400. package/lib/assets/docs/article/grokking-concurrency/python/part-5.md +315 -0
  401. package/lib/assets/docs/article/grokking-concurrency/python/part-6.md +297 -0
  402. package/lib/assets/docs/article/grokking-concurrency/python/part-7.md +314 -0
  403. package/lib/assets/docs/article/grokking-concurrency/python/part-8.md +360 -0
  404. package/lib/assets/docs/article/grokking-concurrency/rust/index.md +270 -0
  405. package/lib/assets/docs/article/grokking-concurrency/rust/part-1.md +108 -0
  406. package/lib/assets/docs/article/grokking-concurrency/rust/part-2.md +120 -0
  407. package/lib/assets/docs/article/grokking-concurrency/rust/part-3.md +126 -0
  408. package/lib/assets/docs/article/grokking-concurrency/rust/part-4.md +175 -0
  409. package/lib/assets/docs/article/grokking-concurrency/rust/part-5.md +158 -0
  410. package/lib/assets/docs/article/grokking-concurrency/rust/part-6.md +94 -0
  411. package/lib/assets/docs/article/grokking-concurrency/rust/part-7.md +133 -0
  412. package/lib/assets/docs/article/grokking-concurrency/rust/part-8.md +155 -0
  413. package/lib/assets/docs/article/grokking-concurrency/scala/index.md +69 -0
  414. package/lib/assets/docs/article/grokking-concurrency/scala/part-1.md +78 -0
  415. package/lib/assets/docs/article/grokking-concurrency/scala/part-2.md +112 -0
  416. package/lib/assets/docs/article/grokking-concurrency/scala/part-3.md +93 -0
  417. package/lib/assets/docs/article/grokking-concurrency/scala/part-4.md +110 -0
  418. package/lib/assets/docs/article/grokking-concurrency/scala/part-5.md +119 -0
  419. package/lib/assets/docs/article/grokking-concurrency/scala/part-6.md +83 -0
  420. package/lib/assets/docs/article/grokking-concurrency/scala/part-7.md +131 -0
  421. package/lib/assets/docs/article/grokking-concurrency/scala/part-8.md +129 -0
  422. package/lib/assets/docs/article/grokkingfp/all/index.md +368 -0
  423. package/lib/assets/docs/article/grokkingfp/all/part-1-ch01-fp-introduction.md +530 -0
  424. package/lib/assets/docs/article/grokkingfp/all/part-1-ch02-pure-functions.md +923 -0
  425. package/lib/assets/docs/article/grokkingfp/all/part-2-ch03-immutable-data.md +1122 -0
  426. package/lib/assets/docs/article/grokkingfp/all/part-2-ch04-higher-order-functions.md +1104 -0
  427. package/lib/assets/docs/article/grokkingfp/all/part-2-ch05-flatmap.md +1026 -0
  428. package/lib/assets/docs/article/grokkingfp/all/part-3-ch06-option.md +777 -0
  429. package/lib/assets/docs/article/grokkingfp/all/part-3-ch07-either-adt.md +871 -0
  430. package/lib/assets/docs/article/grokkingfp/all/part-4-ch08-io-monad.md +972 -0
  431. package/lib/assets/docs/article/grokkingfp/all/part-4-ch09-streams.md +926 -0
  432. package/lib/assets/docs/article/grokkingfp/all/part-5-ch10-concurrency.md +870 -0
  433. package/lib/assets/docs/article/grokkingfp/all/part-6-ch11-application.md +715 -0
  434. package/lib/assets/docs/article/grokkingfp/all/part-6-ch12-testing.md +626 -0
  435. package/lib/assets/docs/article/grokkingfp/all/writing-plan.md +696 -0
  436. package/lib/assets/docs/article/grokkingfp/clojure/index.md +276 -0
  437. package/lib/assets/docs/article/grokkingfp/clojure/part-1.md +667 -0
  438. package/lib/assets/docs/article/grokkingfp/clojure/part-2.md +643 -0
  439. package/lib/assets/docs/article/grokkingfp/clojure/part-3.md +620 -0
  440. package/lib/assets/docs/article/grokkingfp/clojure/part-4.md +697 -0
  441. package/lib/assets/docs/article/grokkingfp/clojure/part-5.md +751 -0
  442. package/lib/assets/docs/article/grokkingfp/clojure/part-6.md +721 -0
  443. package/lib/assets/docs/article/grokkingfp/csharp/index.md +246 -0
  444. package/lib/assets/docs/article/grokkingfp/csharp/part-1.md +811 -0
  445. package/lib/assets/docs/article/grokkingfp/csharp/part-2.md +971 -0
  446. package/lib/assets/docs/article/grokkingfp/csharp/part-3.md +981 -0
  447. package/lib/assets/docs/article/grokkingfp/csharp/part-4.md +949 -0
  448. package/lib/assets/docs/article/grokkingfp/csharp/part-5.md +947 -0
  449. package/lib/assets/docs/article/grokkingfp/csharp/part-6.md +739 -0
  450. package/lib/assets/docs/article/grokkingfp/elixir/index.md +203 -0
  451. package/lib/assets/docs/article/grokkingfp/elixir/part-1.md +710 -0
  452. package/lib/assets/docs/article/grokkingfp/elixir/part-2.md +838 -0
  453. package/lib/assets/docs/article/grokkingfp/elixir/part-3.md +985 -0
  454. package/lib/assets/docs/article/grokkingfp/elixir/part-4.md +974 -0
  455. package/lib/assets/docs/article/grokkingfp/elixir/part-5.md +1284 -0
  456. package/lib/assets/docs/article/grokkingfp/elixir/part-6.md +1047 -0
  457. package/lib/assets/docs/article/grokkingfp/fsharp/index.md +210 -0
  458. package/lib/assets/docs/article/grokkingfp/fsharp/part-1.md +714 -0
  459. package/lib/assets/docs/article/grokkingfp/fsharp/part-2.md +961 -0
  460. package/lib/assets/docs/article/grokkingfp/fsharp/part-3.md +972 -0
  461. package/lib/assets/docs/article/grokkingfp/fsharp/part-4.md +832 -0
  462. package/lib/assets/docs/article/grokkingfp/fsharp/part-5.md +911 -0
  463. package/lib/assets/docs/article/grokkingfp/fsharp/part-6.md +920 -0
  464. package/lib/assets/docs/article/grokkingfp/haskell/index.md +234 -0
  465. package/lib/assets/docs/article/grokkingfp/haskell/part-1.md +591 -0
  466. package/lib/assets/docs/article/grokkingfp/haskell/part-2.md +866 -0
  467. package/lib/assets/docs/article/grokkingfp/haskell/part-3.md +915 -0
  468. package/lib/assets/docs/article/grokkingfp/haskell/part-4.md +876 -0
  469. package/lib/assets/docs/article/grokkingfp/haskell/part-5.md +845 -0
  470. package/lib/assets/docs/article/grokkingfp/haskell/part-6.md +842 -0
  471. package/lib/assets/docs/article/grokkingfp/index.md +143 -0
  472. package/lib/assets/docs/article/grokkingfp/java/index.md +211 -0
  473. package/lib/assets/docs/article/grokkingfp/java/part-1.md +646 -0
  474. package/lib/assets/docs/article/grokkingfp/java/part-2.md +667 -0
  475. package/lib/assets/docs/article/grokkingfp/java/part-3.md +672 -0
  476. package/lib/assets/docs/article/grokkingfp/java/part-4.md +771 -0
  477. package/lib/assets/docs/article/grokkingfp/java/part-5.md +959 -0
  478. package/lib/assets/docs/article/grokkingfp/java/part-6.md +1324 -0
  479. package/lib/assets/docs/article/grokkingfp/python/index.md +258 -0
  480. package/lib/assets/docs/article/grokkingfp/python/part-1.md +437 -0
  481. package/lib/assets/docs/article/grokkingfp/python/part-2.md +958 -0
  482. package/lib/assets/docs/article/grokkingfp/python/part-3.md +1004 -0
  483. package/lib/assets/docs/article/grokkingfp/python/part-4.md +765 -0
  484. package/lib/assets/docs/article/grokkingfp/python/part-5.md +747 -0
  485. package/lib/assets/docs/article/grokkingfp/python/part-6.md +861 -0
  486. package/lib/assets/docs/article/grokkingfp/ruby/index.md +330 -0
  487. package/lib/assets/docs/article/grokkingfp/ruby/part-1.md +753 -0
  488. package/lib/assets/docs/article/grokkingfp/ruby/part-2.md +938 -0
  489. package/lib/assets/docs/article/grokkingfp/ruby/part-3.md +946 -0
  490. package/lib/assets/docs/article/grokkingfp/ruby/part-4.md +921 -0
  491. package/lib/assets/docs/article/grokkingfp/ruby/part-5.md +908 -0
  492. package/lib/assets/docs/article/grokkingfp/ruby/part-6.md +1410 -0
  493. package/lib/assets/docs/article/grokkingfp/rust/index.md +242 -0
  494. package/lib/assets/docs/article/grokkingfp/rust/part-1.md +634 -0
  495. package/lib/assets/docs/article/grokkingfp/rust/part-2.md +1060 -0
  496. package/lib/assets/docs/article/grokkingfp/rust/part-3.md +994 -0
  497. package/lib/assets/docs/article/grokkingfp/rust/part-4.md +571 -0
  498. package/lib/assets/docs/article/grokkingfp/rust/part-5.md +705 -0
  499. package/lib/assets/docs/article/grokkingfp/rust/part-6.md +508 -0
  500. package/lib/assets/docs/article/grokkingfp/scala/index.md +171 -0
  501. package/lib/assets/docs/article/grokkingfp/scala/part-1.md +541 -0
  502. package/lib/assets/docs/article/grokkingfp/scala/part-2.md +946 -0
  503. package/lib/assets/docs/article/grokkingfp/scala/part-3.md +917 -0
  504. package/lib/assets/docs/article/grokkingfp/scala/part-4.md +742 -0
  505. package/lib/assets/docs/article/grokkingfp/scala/part-5.md +722 -0
  506. package/lib/assets/docs/article/grokkingfp/scala/part-6.md +865 -0
  507. package/lib/assets/docs/article/grokkingfp/typescript/index.md +273 -0
  508. package/lib/assets/docs/article/grokkingfp/typescript/part-1.md +559 -0
  509. package/lib/assets/docs/article/grokkingfp/typescript/part-2.md +1129 -0
  510. package/lib/assets/docs/article/grokkingfp/typescript/part-3.md +842 -0
  511. package/lib/assets/docs/article/grokkingfp/typescript/part-4.md +1085 -0
  512. package/lib/assets/docs/article/grokkingfp/typescript/part-5.md +717 -0
  513. package/lib/assets/docs/article/grokkingfp/typescript/part-6.md +980 -0
  514. package/lib/assets/docs/article/index.md +36 -0
  515. package/lib/assets/docs/design/index.md +39 -27
  516. package/lib/assets/docs/development/index.md +11 -1
  517. package/lib/assets/docs/index.md +33 -103
  518. package/lib/assets/docs/operation/index.md +16 -6
  519. package/lib/assets/docs/reference/index.md +5 -4
  520. package/lib/assets/docs/requirements/index.md +13 -6
  521. package/lib/assets/docs/review/index.md +5 -0
  522. package/lib/assets/docs/strategy/index.md +15 -0
  523. package/lib/assets/docs/template/index.md +9 -5
  524. package/lib/assets/mkdocs.yml +33 -19
  525. package/package.json +1 -1
  526. package/lib/assets/docs/analysis/index.md +0 -8
  527. /package/lib/assets/docs/{analysis → strategy}/slide/.gitkeep +0 -0
@@ -0,0 +1,765 @@
1
+ # Part IV: IO と副作用の管理
2
+
3
+ 本章では、関数型プログラミングにおける副作用の扱い方を学びます。IO モナドを使って副作用を純粋関数内で安全に記述し、ストリーム処理で無限のデータを扱う方法を習得します。
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
+ **ソースファイル**: `app/python/src/grokking_fp/ch08_io.py`
46
+
47
+ ```python
48
+ import random
49
+
50
+ def cast_the_die_impure() -> int:
51
+ """不純な関数: 呼び出すたびに異なる値を返す。"""
52
+ return random.randint(1, 6)
53
+
54
+ # 呼び出すたびに異なる値が返る
55
+ print(cast_the_die_impure()) # 3
56
+ print(cast_the_die_impure()) # 5
57
+ print(cast_the_die_impure()) # 1
58
+ ```
59
+
60
+ ### 8.2 LazyIO モナドとは
61
+
62
+ **LazyIO モナド**は「副作用を持つ計算の**記述**」を表す型です。
63
+
64
+ - `LazyIO[A]` は「実行すると `A` 型の値を返す副作用のある計算」
65
+ - LazyIO 値を作成しただけでは副作用は発生しない
66
+ - `.run()` で実際に実行される
67
+
68
+ ```plantuml
69
+ @startuml
70
+ !theme plain
71
+
72
+ rectangle "LazyIO の仕組み" {
73
+ card "LazyIO(lambda: 副作用)" as create
74
+ card "LazyIO[A] 値" as io
75
+ card ".run()" as run
76
+ card "副作用が実行される" as effect
77
+
78
+ create --> io : 記述を作成
79
+ io --> run : 実行を指示
80
+ run --> effect : 実際に実行
81
+ }
82
+
83
+ note bottom of io
84
+ この時点では
85
+ 副作用は発生しない
86
+ end note
87
+
88
+ @enduml
89
+ ```
90
+
91
+ **ソースファイル**: `app/python/src/grokking_fp/ch08_io.py`
92
+
93
+ #### LazyIO の実装
94
+
95
+ ```python
96
+ from collections.abc import Callable
97
+ from typing import Generic, TypeVar
98
+
99
+ T = TypeVar("T")
100
+ U = TypeVar("U")
101
+
102
+ class LazyIO(Generic[T]):
103
+ """遅延評価する IO モナド。"""
104
+
105
+ def __init__(self, effect: Callable[[], T]) -> None:
106
+ self._effect = effect
107
+
108
+ def run(self) -> T:
109
+ """副作用を実行して結果を返す。"""
110
+ return self._effect()
111
+
112
+ def map(self, f: Callable[[T], U]) -> "LazyIO[U]":
113
+ """値を変換する。"""
114
+ return LazyIO(lambda: f(self.run()))
115
+
116
+ def bind(self, f: Callable[[T], "LazyIO[U]"]) -> "LazyIO[U]":
117
+ """IO を返す関数で変換する(flatMap)。"""
118
+ return LazyIO(lambda: f(self.run()).run())
119
+ ```
120
+
121
+ ### 8.3 サイコロを振る例
122
+
123
+ #### LazyIO を使った純粋な記述
124
+
125
+ ```python
126
+ def cast_the_die() -> LazyIO[int]:
127
+ """LazyIO でラップした純粋な記述。"""
128
+ return LazyIO(lambda: random.randint(1, 6))
129
+
130
+ # LazyIO 値を作成(この時点では実行されない)
131
+ die_cast = cast_the_die()
132
+ print(die_cast) # LazyIO(...)
133
+
134
+ # 実際に実行
135
+ print(die_cast.run()) # 4
136
+ ```
137
+
138
+ ### 8.4 LazyIO の作成方法
139
+
140
+ | 関数 | 用途 | 例 |
141
+ |------|------|-----|
142
+ | `io_pure(value)` | 既存の値をラップ(副作用なし) | `io_pure(42)` |
143
+ | `io_delay(func)` | 副作用のある関数をラップ | `io_delay(lambda: print("hello"))` |
144
+
145
+ ```python
146
+ def io_pure(value: T) -> LazyIO[T]:
147
+ """既存の値を LazyIO にラップする。"""
148
+ return LazyIO(lambda: value)
149
+
150
+ def io_delay(func: Callable[[], T]) -> LazyIO[T]:
151
+ """副作用のある関数を LazyIO にラップする。"""
152
+ return LazyIO(func)
153
+
154
+ # io_pure: 既存の値をラップ
155
+ pure_value = io_pure(42)
156
+ print(pure_value.run()) # 42
157
+
158
+ # io_delay: 副作用を遅延実行
159
+ counter = [0]
160
+ delayed = io_delay(lambda: counter.__setitem__(0, counter[0] + 1) or counter[0])
161
+ print(counter[0]) # 0 - まだ実行されていない
162
+ print(delayed.run()) # 1 - 実行された
163
+ ```
164
+
165
+ ### 8.5 LazyIO の合成
166
+
167
+ LazyIO 値は `bind` や `map` で合成できます。
168
+
169
+ ```python
170
+ def cast_the_die_twice() -> LazyIO[int]:
171
+ """サイコロを2回振って合計を返す。"""
172
+ return cast_the_die().bind(
173
+ lambda first: cast_the_die().map(lambda second: first + second)
174
+ )
175
+
176
+ # まだ実行されていない
177
+ program = cast_the_die_twice()
178
+
179
+ # 実行
180
+ result = program.run()
181
+ print(result) # 2 <= result <= 12
182
+ ```
183
+
184
+ ```plantuml
185
+ @startuml
186
+ !theme plain
187
+
188
+ rectangle "LazyIO の合成" {
189
+ card "cast_the_die()" as cast1
190
+ card "cast_the_die()" as cast2
191
+ card "first + second" as combine
192
+
193
+ cast1 --> combine : bind
194
+ cast2 --> combine : map
195
+ }
196
+
197
+ note bottom
198
+ bind と map で
199
+ 順序付けて合成
200
+ end note
201
+
202
+ @enduml
203
+ ```
204
+
205
+ ### 8.6 ミーティングスケジューリングの例
206
+
207
+ **ソースファイル**: `app/python/src/grokking_fp/ch08_io.py`
208
+
209
+ より実践的な例として、ミーティングのスケジューリングを見てみましょう。
210
+
211
+ ```python
212
+ from dataclasses import dataclass
213
+
214
+ @dataclass(frozen=True)
215
+ class MeetingTime:
216
+ """ミーティングの時間枠。"""
217
+ start_hour: int
218
+ end_hour: int
219
+
220
+ def meetings_overlap(meeting1: MeetingTime, meeting2: MeetingTime) -> bool:
221
+ """2つのミーティングが重複しているかを判定する。"""
222
+ return meeting1.start_hour < meeting2.end_hour and meeting2.start_hour < meeting1.end_hour
223
+ ```
224
+
225
+ #### 空き時間の計算(純粋関数)
226
+
227
+ ```python
228
+ def possible_meetings(
229
+ existing_meetings: list[MeetingTime],
230
+ start_hour: int,
231
+ end_hour: int,
232
+ length_hours: int,
233
+ ) -> list[MeetingTime]:
234
+ """空いている時間枠を計算する(純粋関数)。"""
235
+ slots = [
236
+ MeetingTime(start, start + length_hours)
237
+ for start in range(start_hour, end_hour - length_hours + 1)
238
+ ]
239
+ return [
240
+ slot
241
+ for slot in slots
242
+ if all(not meetings_overlap(meeting, slot) for meeting in existing_meetings)
243
+ ]
244
+ ```
245
+
246
+ #### カレンダー API の呼び出しを LazyIO でラップ
247
+
248
+ ```python
249
+ def calendar_entries(name: str) -> LazyIO[list[MeetingTime]]:
250
+ """カレンダー API 呼び出しを LazyIO でラップ。"""
251
+ return LazyIO(lambda: calendar_entries_impure(name))
252
+
253
+ def scheduled_meetings(person1: str, person2: str) -> LazyIO[list[MeetingTime]]:
254
+ """2人の予定を取得して結合する。"""
255
+ return calendar_entries(person1).bind(
256
+ lambda entries1: calendar_entries(person2).map(
257
+ lambda entries2: entries1 + entries2
258
+ )
259
+ )
260
+ ```
261
+
262
+ ### 8.7 エラーハンドリングと orElse
263
+
264
+ LazyIO の `io_or_else` 関数で、失敗時のフォールバックを指定できます。
265
+
266
+ ```python
267
+ def io_or_else(io: LazyIO[T], fallback: LazyIO[T]) -> LazyIO[T]:
268
+ """IO が失敗した場合にフォールバックを使う。"""
269
+ def run() -> T:
270
+ try:
271
+ return io.run()
272
+ except Exception:
273
+ return fallback.run()
274
+ return LazyIO(run)
275
+
276
+ # 使用例
277
+ success = io_pure(42)
278
+ fallback = io_pure(0)
279
+ io_or_else(success, fallback).run() # 42
280
+
281
+ def fail() -> int:
282
+ raise ValueError("error")
283
+
284
+ failure = LazyIO(fail)
285
+ io_or_else(failure, fallback).run() # 0
286
+ ```
287
+
288
+ #### リトライ戦略
289
+
290
+ ```python
291
+ def retry(action: LazyIO[T], max_retries: int, default: T) -> LazyIO[T]:
292
+ """指定回数リトライし、全部失敗したらデフォルト値を返す。"""
293
+ def run() -> T:
294
+ for _ in range(max_retries):
295
+ try:
296
+ return action.run()
297
+ except Exception:
298
+ continue
299
+ return default
300
+ return LazyIO(run)
301
+
302
+ # 使用例
303
+ retry(calendar_entries("Alice"), 10, [])
304
+ ```
305
+
306
+ ### 8.8 sequence による LazyIO のリスト処理
307
+
308
+ `list[LazyIO[A]]` を `LazyIO[list[A]]` に変換するには `sequence_io` を使います。
309
+
310
+ ```python
311
+ def sequence_io(ios: list[LazyIO[T]]) -> LazyIO[list[T]]:
312
+ """IO のリストを、リストの IO に変換する。"""
313
+ def run() -> list[T]:
314
+ return [io.run() for io in ios]
315
+ return LazyIO(run)
316
+
317
+ def traverse_io(items: list[T], f: Callable[[T], LazyIO[U]]) -> LazyIO[list[U]]:
318
+ """リストの各要素に IO を返す関数を適用し、結果をまとめる。"""
319
+ return sequence_io([f(item) for item in items])
320
+
321
+ # 使用例
322
+ ios = [io_pure(1), io_pure(2), io_pure(3)]
323
+ sequence_io(ios).run() # [1, 2, 3]
324
+ ```
325
+
326
+ ---
327
+
328
+ ## 第9章: ストリーム処理
329
+
330
+ ### 9.1 ストリームとは
331
+
332
+ **ストリーム**は、要素の(潜在的に無限の)シーケンスを表します。Python ではジェネレータを使って実現します。
333
+
334
+ ```plantuml
335
+ @startuml
336
+ !theme plain
337
+
338
+ rectangle "List vs Stream(ジェネレータ)" {
339
+ rectangle "List" as list {
340
+ card "全要素がメモリに存在"
341
+ card "有限"
342
+ card "即座に評価"
343
+ }
344
+
345
+ rectangle "ジェネレータ" as stream {
346
+ card "必要な時に要素を生成"
347
+ card "無限も可能"
348
+ card "遅延評価"
349
+ }
350
+ }
351
+
352
+ @enduml
353
+ ```
354
+
355
+ **ソースファイル**: `app/python/src/grokking_fp/ch09_streams.py`
356
+
357
+ ### 9.2 基本的なストリーム
358
+
359
+ ```python
360
+ from collections.abc import Generator
361
+ from itertools import count, islice
362
+
363
+ def stream_of(*args: T) -> Generator[T, None, None]:
364
+ """有限ストリームを作成する。"""
365
+ yield from args
366
+
367
+ def stream_range(start: int, end: int) -> Generator[int, None, None]:
368
+ """範囲のストリームを作成する。"""
369
+ yield from range(start, end)
370
+
371
+ # 使用例
372
+ list(stream_of(1, 2, 3)) # [1, 2, 3]
373
+ list(stream_range(1, 5)) # [1, 2, 3, 4]
374
+ ```
375
+
376
+ ### 9.3 無限ストリーム
377
+
378
+ ```python
379
+ def infinite_stream(start: int = 0) -> Generator[int, None, None]:
380
+ """無限の整数ストリームを作成する。"""
381
+ yield from count(start)
382
+
383
+ def cycle_stream(items: list[T]) -> Generator[T, None, None]:
384
+ """リストの要素を無限に繰り返すストリーム。"""
385
+ while True:
386
+ yield from items
387
+
388
+ # take で有限個を取得
389
+ list(islice(infinite_stream(), 5)) # [0, 1, 2, 3, 4]
390
+ list(islice(cycle_stream([1, 2, 3]), 8)) # [1, 2, 3, 1, 2, 3, 1, 2]
391
+ ```
392
+
393
+ ```plantuml
394
+ @startuml
395
+ !theme plain
396
+
397
+ rectangle "無限ストリーム" {
398
+ card "cycle_stream([1, 2, 3])" as stream
399
+ card "1, 2, 3, 1, 2, 3, 1, 2, 3, ..." as infinite
400
+ card "islice(stream, 8)" as take
401
+ card "[1, 2, 3, 1, 2, 3, 1, 2]" as result
402
+
403
+ stream --> infinite
404
+ infinite --> take
405
+ take --> result
406
+ }
407
+
408
+ note bottom
409
+ 無限のストリームから
410
+ 必要な分だけ取得
411
+ end note
412
+
413
+ @enduml
414
+ ```
415
+
416
+ ### 9.4 ストリームの主要操作
417
+
418
+ | 操作 | 説明 | 例 |
419
+ |------|------|-----|
420
+ | `stream_take(n)` | 最初の n 要素を取得 | `stream_take(stream, 3)` |
421
+ | `stream_filter(p)` | 条件を満たす要素のみ | `stream_filter(stream, lambda x: x > 0)` |
422
+ | `stream_map(f)` | 各要素を変換 | `stream_map(stream, lambda x: x * 2)` |
423
+ | `stream_append(s)` | 別のストリームを結合 | `stream_append(s1, s2)` |
424
+ | `stream_sliding(n)` | スライディングウィンドウ | `stream_sliding(stream, 3)` |
425
+
426
+ ```python
427
+ def stream_take(stream: Iterator[T], n: int) -> Generator[T, None, None]:
428
+ """ストリームから最初の n 要素を取得する。"""
429
+ yield from islice(stream, n)
430
+
431
+ def stream_filter(stream: Iterator[T], predicate: Callable[[T], bool]) -> Generator[T, None, None]:
432
+ """条件を満たす要素だけを返すストリーム。"""
433
+ for item in stream:
434
+ if predicate(item):
435
+ yield item
436
+
437
+ def stream_map(stream: Iterator[T], f: Callable[[T], U]) -> Generator[U, None, None]:
438
+ """各要素を変換するストリーム。"""
439
+ for item in stream:
440
+ yield f(item)
441
+ ```
442
+
443
+ ### 9.5 スライディングウィンドウ
444
+
445
+ `stream_sliding` で連続する要素をグループ化できます。
446
+
447
+ ```python
448
+ from collections import deque
449
+
450
+ def stream_sliding(stream: Iterator[T], window_size: int) -> Generator[list[T], None, None]:
451
+ """スライディングウィンドウを生成する。"""
452
+ window: deque[T] = deque(maxlen=window_size)
453
+ for item in stream:
454
+ window.append(item)
455
+ if len(window) == window_size:
456
+ yield list(window)
457
+
458
+ # 使用例
459
+ list(stream_sliding(stream_of(1, 2, 3, 4, 5), 3))
460
+ # [[1, 2, 3], [2, 3, 4], [3, 4, 5]]
461
+ ```
462
+
463
+ ```plantuml
464
+ @startuml
465
+ !theme plain
466
+
467
+ rectangle "sliding(3) の動作" {
468
+ card "[1, 2, 3, 4, 5]" as input
469
+ card "[1, 2, 3]" as w1
470
+ card "[2, 3, 4]" as w2
471
+ card "[3, 4, 5]" as w3
472
+
473
+ input --> w1
474
+ input --> w2
475
+ input --> w3
476
+ }
477
+
478
+ @enduml
479
+ ```
480
+
481
+ ### 9.6 サイコロの例
482
+
483
+ **ソースファイル**: `app/python/src/grokking_fp/ch09_streams.py`
484
+
485
+ ```python
486
+ def cast_die_stream() -> Generator[int, None, None]:
487
+ """サイコロを無限に振るストリーム。"""
488
+ while True:
489
+ yield random.randint(1, 6)
490
+
491
+ def first_six() -> Generator[int, None, None]:
492
+ """6 が出るまでサイコロを振り続ける。"""
493
+ for die in cast_die_stream():
494
+ yield die
495
+ if die == 6:
496
+ break
497
+
498
+ # 使用例
499
+ list(stream_take(cast_die_stream(), 10)) # [4, 2, 6, 1, 3, 5, 2, 4, 6, 1]
500
+ list(first_six()) # [..., 6]
501
+ ```
502
+
503
+ ### 9.7 通貨交換レートの例
504
+
505
+ **ソースファイル**: `app/python/src/grokking_fp/ch09_streams.py`
506
+
507
+ 為替レートを監視して、上昇トレンドを検出する例です。
508
+
509
+ ```python
510
+ from decimal import Decimal
511
+ from enum import Enum
512
+
513
+ class Currency(Enum):
514
+ USD = "USD"
515
+ EUR = "EUR"
516
+ JPY = "JPY"
517
+ GBP = "GBP"
518
+
519
+ def exchange_rate_stream(
520
+ from_currency: Currency, to_currency: Currency
521
+ ) -> Generator[Decimal, None, None]:
522
+ """為替レートを継続的に取得するストリーム。"""
523
+ while True:
524
+ yield get_exchange_rate(from_currency, to_currency)
525
+ ```
526
+
527
+ #### トレンド判定(純粋関数)
528
+
529
+ ```python
530
+ def trending(rates: list[Decimal]) -> bool:
531
+ """レートが上昇トレンドかどうかを判定する。"""
532
+ if len(rates) < 2:
533
+ return False
534
+ return all(rates[i] < rates[i + 1] for i in range(len(rates) - 1))
535
+
536
+ trending([Decimal("0.81"), Decimal("0.82"), Decimal("0.83")]) # True
537
+ trending([Decimal("0.81"), Decimal("0.84"), Decimal("0.83")]) # False
538
+ ```
539
+
540
+ #### トレンドを検出して交換
541
+
542
+ ```python
543
+ def find_trending_window(
544
+ from_currency: Currency, to_currency: Currency, window_size: int = 3
545
+ ) -> list[Decimal]:
546
+ """上昇トレンドのウィンドウを見つける。"""
547
+ rate_stream = exchange_rate_stream(from_currency, to_currency)
548
+ windows = stream_sliding(rate_stream, window_size)
549
+
550
+ for window in stream_take(windows, 100):
551
+ if trending(window):
552
+ return window
553
+ return []
554
+ ```
555
+
556
+ ### 9.8 ストリームの結合
557
+
558
+ 2つのストリームを `stream_zip` で結合できます。
559
+
560
+ ```python
561
+ def stream_zip(stream1: Iterator[T], stream2: Iterator[U]) -> Generator[tuple[T, U], None, None]:
562
+ """2つのストリームを zip する。"""
563
+ yield from zip(stream1, stream2)
564
+
565
+ # 使用例
566
+ list(stream_zip(stream_of(1, 2, 3), stream_of("a", "b", "c")))
567
+ # [(1, 'a'), (2, 'b'), (3, 'c')]
568
+ ```
569
+
570
+ ### 9.9 ストリームの集約
571
+
572
+ ```python
573
+ def stream_sum(stream: Iterator[int | float | Decimal]) -> int | float | Decimal:
574
+ """ストリームの合計を計算する。"""
575
+ return sum(stream)
576
+
577
+ def stream_reduce(stream: Iterator[T], f: Callable[[U, T], U], initial: U) -> U:
578
+ """ストリームを畳み込む。"""
579
+ result = initial
580
+ for item in stream:
581
+ result = f(result, item)
582
+ return result
583
+
584
+ def stream_find(stream: Iterator[T], predicate: Callable[[T], bool]) -> T | None:
585
+ """条件を満たす最初の要素を見つける。"""
586
+ for item in stream:
587
+ if predicate(item):
588
+ return item
589
+ return None
590
+ ```
591
+
592
+ ---
593
+
594
+ ## まとめ
595
+
596
+ ### Part IV で学んだこと
597
+
598
+ ```plantuml
599
+ @startuml
600
+ !theme plain
601
+
602
+ rectangle "Part IV: IO と副作用の管理" {
603
+ rectangle "第8章" as ch8 {
604
+ card "LazyIO モナド"
605
+ card "io_pure / io_delay"
606
+ card "bind / map"
607
+ card "io_or_else"
608
+ card "retry"
609
+ card "sequence_io"
610
+ }
611
+
612
+ rectangle "第9章" as ch9 {
613
+ card "ジェネレータ"
614
+ card "無限ストリーム"
615
+ card "islice / take"
616
+ card "sliding"
617
+ card "zip"
618
+ }
619
+ }
620
+
621
+ ch8 --> ch9
622
+
623
+ @enduml
624
+ ```
625
+
626
+ ### LazyIO と ジェネレータの比較
627
+
628
+ | 特性 | LazyIO[A] | Generator[A] |
629
+ |------|-----------|--------------|
630
+ | 要素数 | 1つ | 0個以上(無限も可) |
631
+ | 実行 | `.run()` | `list()` または `next()` |
632
+ | 用途 | 単一の副作用 | 連続した値の生成 |
633
+
634
+ ### キーポイント
635
+
636
+ 1. **LazyIO モナド**: 副作用を「記述」として扱い、実行を遅延させる
637
+ 2. **io_delay**: 副作用のある式を LazyIO にラップ
638
+ 3. **io_or_else**: 失敗時のフォールバックを指定
639
+ 4. **sequence_io**: `list[LazyIO[A]]` → `LazyIO[list[A]]`
640
+ 5. **ジェネレータ**: 潜在的に無限のシーケンスを遅延評価で処理
641
+ 6. **sliding**: 連続する要素をグループ化してパターンを検出
642
+
643
+ ### Python 特有のポイント
644
+
645
+ - **ジェネレータ**: `yield` を使った遅延評価
646
+ - **`itertools`**: `count`, `islice`, `repeat` などの便利な関数
647
+ - **`deque`**: スライディングウィンドウの効率的な実装
648
+ - **`Decimal`**: 金融計算での精度保証
649
+
650
+ ---
651
+
652
+ ## 演習問題
653
+
654
+ ### 問題 1: LazyIO の基本
655
+
656
+ 以下の関数を実装してください。
657
+
658
+ ```python
659
+ def print_and_return(message: str) -> LazyIO[str]:
660
+ """メッセージを出力して返す。"""
661
+ ...
662
+
663
+ # 期待される動作
664
+ # print_and_return("Hello").run() は
665
+ # "Hello" をコンソールに出力し、"Hello" を返す
666
+ ```
667
+
668
+ <details>
669
+ <summary>解答</summary>
670
+
671
+ ```python
672
+ def print_and_return(message: str) -> LazyIO[str]:
673
+ return LazyIO(lambda: (print(message), message)[1])
674
+
675
+ # または
676
+ def print_and_return(message: str) -> LazyIO[str]:
677
+ def effect() -> str:
678
+ print(message)
679
+ return message
680
+ return LazyIO(effect)
681
+ ```
682
+
683
+ </details>
684
+
685
+ ### 問題 2: LazyIO の合成
686
+
687
+ 以下の関数を実装してください。2つの LazyIO を順番に実行し、結果を結合します。
688
+
689
+ ```python
690
+ def combine_io(io1: LazyIO[T], io2: LazyIO[U], f: Callable[[T, U], V]) -> LazyIO[V]:
691
+ ...
692
+
693
+ # 期待される動作
694
+ result = combine_io(io_pure(1), io_pure(2), lambda a, b: a + b)
695
+ result.run() # 3
696
+ ```
697
+
698
+ <details>
699
+ <summary>解答</summary>
700
+
701
+ ```python
702
+ def combine_io(io1: LazyIO[T], io2: LazyIO[U], f: Callable[[T, U], V]) -> LazyIO[V]:
703
+ return io1.bind(lambda a: io2.map(lambda b: f(a, b)))
704
+ ```
705
+
706
+ </details>
707
+
708
+ ### 問題 3: ストリーム操作
709
+
710
+ 以下のストリームを作成してください。
711
+
712
+ ```python
713
+ # 1. 1から10までの偶数のストリーム
714
+ evens = ???
715
+
716
+ # 2. 無限に交互に true/false を返すストリーム
717
+ alternating = ???
718
+
719
+ # 3. 最初の5つの要素の合計を計算
720
+ total = stream_sum(stream_take(stream_range(1, 100), 5))
721
+ ```
722
+
723
+ <details>
724
+ <summary>解答</summary>
725
+
726
+ ```python
727
+ # 1. 1から10までの偶数
728
+ evens = stream_filter(stream_range(1, 11), lambda x: x % 2 == 0)
729
+ # または
730
+ evens = stream_of(2, 4, 6, 8, 10)
731
+
732
+ # 2. 無限に交互に true/false
733
+ alternating = cycle_stream([True, False])
734
+
735
+ # 3. 最初の5つの要素の合計
736
+ total = stream_sum(stream_take(stream_range(1, 100), 5)) # 15
737
+ ```
738
+
739
+ </details>
740
+
741
+ ### 問題 4: トレンド検出
742
+
743
+ 直近3つの値が全て同じかどうかを判定する関数を実装してください。
744
+
745
+ ```python
746
+ def is_stable(values: list[Decimal]) -> bool:
747
+ ...
748
+
749
+ # 期待される動作
750
+ assert is_stable([Decimal("5"), Decimal("5"), Decimal("5")]) == True
751
+ assert is_stable([Decimal("5"), Decimal("5"), Decimal("6")]) == False
752
+ assert is_stable([Decimal("5")]) == False # 2つ未満は False
753
+ ```
754
+
755
+ <details>
756
+ <summary>解答</summary>
757
+
758
+ ```python
759
+ def is_stable(values: list[Decimal]) -> bool:
760
+ if len(values) < 2:
761
+ return False
762
+ return len(set(values)) == 1
763
+ ```
764
+
765
+ </details>