@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,1085 @@
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
+ ### 8.2 IO モナドとは
46
+
47
+ **IO モナド**は「副作用を持つ計算の**記述**」を表す型です。
48
+
49
+ fp-ts では:
50
+ - `IO<A>` は「実行すると `A` 型の値を返す同期的な副作用のある計算」
51
+ - `Task<A>` は「実行すると `Promise<A>` を返す非同期的な副作用のある計算」
52
+ - IO/Task 値を作成しただけでは副作用は発生しない
53
+ - `()` で実行すると副作用が発生する
54
+
55
+ ```plantuml
56
+ @startuml
57
+ !theme plain
58
+
59
+ rectangle "IO の仕組み" {
60
+ card "IO.of(value) / () => value" as create
61
+ card "IO<A> 値" as io
62
+ card "io()" as run
63
+ card "副作用が実行される" as effect
64
+
65
+ create --> io : 記述を作成
66
+ io --> run : 実行を指示
67
+ run --> effect : 実際に実行
68
+ }
69
+
70
+ note bottom of io
71
+ この時点では
72
+ 副作用は発生しない
73
+ end note
74
+
75
+ @enduml
76
+ ```
77
+
78
+ ### 8.3 IO の基本
79
+
80
+ **ソースファイル**: `app/typescript/src/ch08_io.ts`
81
+
82
+ ```typescript
83
+ import * as IO from 'fp-ts/IO'
84
+
85
+ // 純粋な値を IO にラップする
86
+ const pureIO = <A>(a: A): IO.IO<A> => IO.of(a)
87
+
88
+ const io = pureIO(42)
89
+ io() // => 42
90
+
91
+ // 副作用のある計算を IO にラップする
92
+ const delayIO = <A>(thunk: () => A): IO.IO<A> => thunk
93
+
94
+ const log = delayIO(() => console.log("Hello"))
95
+ // この時点では何も出力されない
96
+ log() // => "Hello" が出力される
97
+ ```
98
+
99
+ #### 現在時刻と乱数
100
+
101
+ ```typescript
102
+ // 現在時刻を取得する IO
103
+ const getCurrentTime: IO.IO<number> = () => Date.now()
104
+
105
+ // 乱数を生成する IO
106
+ const getRandomNumber: IO.IO<number> = () => Math.random()
107
+ ```
108
+
109
+ ### 8.4 サイコロを振る例
110
+
111
+ ```plantuml
112
+ @startuml
113
+ !theme plain
114
+
115
+ rectangle "不純な関数 vs IO" {
116
+ rectangle "不純な関数" as impure {
117
+ card "castTheDieImpure()"
118
+ card "呼び出すたびに異なる値"
119
+ card "テストが困難"
120
+ }
121
+
122
+ rectangle "IO を使った純粋な記述" as pure_io {
123
+ card "castTheDie: IO<number>"
124
+ card "実行まで値は確定しない"
125
+ card "記述と実行が分離"
126
+ }
127
+ }
128
+
129
+ impure --> pure_io : IO でラップ
130
+
131
+ @enduml
132
+ ```
133
+
134
+ ```typescript
135
+ // 不純な関数(副作用あり)- 直接乱数を返す
136
+ const castTheDieImpure = (): number => Math.floor(Math.random() * 6) + 1
137
+
138
+ castTheDieImpure() // 3
139
+ castTheDieImpure() // 5
140
+ castTheDieImpure() // 1
141
+
142
+ // IO を使った純粋な記述
143
+ const castTheDie: IO.IO<number> = () => castTheDieImpure()
144
+
145
+ // まだ実行されていない
146
+ const dieCast = castTheDie
147
+ console.log(dieCast) // [Function]
148
+
149
+ // 実行
150
+ dieCast() // 4
151
+ ```
152
+
153
+ ### 8.5 IO の合成
154
+
155
+ IO 値は `map` と `chain`(flatMap)で合成できます。
156
+
157
+ ```typescript
158
+ import { pipe } from 'fp-ts/function'
159
+ import * as IO from 'fp-ts/IO'
160
+
161
+ // サイコロを2回振って合計を返す
162
+ const castTheDieTwice: IO.IO<number> = pipe(
163
+ castTheDie,
164
+ IO.chain((first) =>
165
+ pipe(
166
+ castTheDie,
167
+ IO.map((second) => first + second)
168
+ )
169
+ )
170
+ )
171
+
172
+ // まだ実行されていない
173
+ const program = castTheDieTwice
174
+
175
+ // 実行
176
+ const result = program() // 2-12 のランダムな値
177
+ ```
178
+
179
+ ```plantuml
180
+ @startuml
181
+ !theme plain
182
+
183
+ rectangle "IO の合成" {
184
+ card "castTheDie" as cast1
185
+ card "castTheDie" as cast2
186
+ card "first + second" as combine
187
+
188
+ cast1 --> combine : chain
189
+ cast2 --> combine : map
190
+ }
191
+
192
+ note bottom
193
+ pipe と chain/map で
194
+ 順序付けて合成
195
+ end note
196
+
197
+ @enduml
198
+ ```
199
+
200
+ #### サイコロを n 回振る
201
+
202
+ ```typescript
203
+ import * as RA from 'fp-ts/ReadonlyArray'
204
+
205
+ const castTheDieNTimes = (n: number): IO.IO<number> => {
206
+ if (n <= 0) return IO.of(0)
207
+ const casts = RA.replicate(n, castTheDie)
208
+ return pipe(
209
+ casts,
210
+ IO.sequenceArray,
211
+ IO.map(RA.reduce(0, (acc, val) => acc + val))
212
+ )
213
+ }
214
+
215
+ const threeRolls = castTheDieNTimes(3)
216
+ threeRolls() // 3-18 のランダムな値
217
+ ```
218
+
219
+ ### 8.6 Task(非同期 IO)
220
+
221
+ `Task<A>` は非同期版の IO です。実行すると `Promise<A>` を返します。
222
+
223
+ ```typescript
224
+ import * as T from 'fp-ts/Task'
225
+
226
+ // 純粋な値を Task にラップする
227
+ const pureTask = <A>(a: A): T.Task<A> => T.of(a)
228
+
229
+ const task = pureTask(42)
230
+ await task() // => 42
231
+
232
+ // 指定ミリ秒後に値を返す Task
233
+ const delayedTask = <A>(ms: number, value: A): T.Task<A> =>
234
+ () => new Promise((resolve) => setTimeout(() => resolve(value), ms))
235
+
236
+ const delayed = delayedTask(1000, "Hello")
237
+ await delayed() // 1秒後に "Hello"
238
+ ```
239
+
240
+ ### 8.7 TaskEither(非同期 + エラーハンドリング)
241
+
242
+ `TaskEither<E, A>` は「実行すると `Promise<Either<E, A>>` を返す非同期計算」です。
243
+
244
+ ```typescript
245
+ import * as TE from 'fp-ts/TaskEither'
246
+
247
+ // 成功値を TaskEither にラップする
248
+ const rightTask = <E, A>(a: A): TE.TaskEither<E, A> => TE.right(a)
249
+
250
+ // エラー値を TaskEither にラップする
251
+ const leftTask = <E, A>(e: E): TE.TaskEither<E, A> => TE.left(e)
252
+
253
+ // Promise を TaskEither に変換する
254
+ const fromPromise = <E, A>(
255
+ promise: () => Promise<A>,
256
+ onRejected: (reason: unknown) => E
257
+ ): TE.TaskEither<E, A> => TE.tryCatch(promise, onRejected)
258
+
259
+ const fetchData = fromPromise(
260
+ () => fetch('/api/data').then(r => r.json()),
261
+ (e) => `Fetch error: ${e}`
262
+ )
263
+ ```
264
+
265
+ ### 8.8 ミーティングスケジューリングの例
266
+
267
+ より実践的な例として、ミーティングのスケジューリングを見てみましょう。
268
+
269
+ ```typescript
270
+ interface MeetingTime {
271
+ readonly startHour: number
272
+ readonly endHour: number
273
+ }
274
+
275
+ const createMeetingTime = (
276
+ startHour: number,
277
+ endHour: number
278
+ ): MeetingTime => ({ startHour, endHour })
279
+
280
+ // 2つのミーティングが重複しているかを判定する(純粋関数)
281
+ const meetingsOverlap = (
282
+ meeting1: MeetingTime,
283
+ meeting2: MeetingTime
284
+ ): boolean =>
285
+ meeting1.startHour < meeting2.endHour &&
286
+ meeting2.startHour < meeting1.endHour
287
+ ```
288
+
289
+ #### 空き時間の計算(純粋関数)
290
+
291
+ ```typescript
292
+ const possibleMeetings = (
293
+ existingMeetings: readonly MeetingTime[],
294
+ startHour: number,
295
+ endHour: number,
296
+ lengthHours: number
297
+ ): readonly MeetingTime[] => {
298
+ const slots = pipe(
299
+ RA.makeBy(endHour - lengthHours - startHour + 1, (i) => startHour + i),
300
+ RA.map((start) => createMeetingTime(start, start + lengthHours))
301
+ )
302
+ return pipe(
303
+ slots,
304
+ RA.filter((slot) =>
305
+ existingMeetings.every((meeting) => !meetingsOverlap(meeting, slot))
306
+ )
307
+ )
308
+ }
309
+
310
+ const existing = [createMeetingTime(9, 10), createMeetingTime(14, 15)]
311
+ possibleMeetings(existing, 8, 17, 1)
312
+ // => 8-9, 10-11, 11-12, 12-13, 13-14, 15-16, 16-17
313
+ ```
314
+
315
+ #### API 呼び出しを TaskEither でラップ
316
+
317
+ ```typescript
318
+ // API からカレンダーエントリを取得(シミュレーション)
319
+ const calendarEntriesApi = (
320
+ name: string
321
+ ): TE.TaskEither<string, readonly MeetingTime[]> => {
322
+ const mockData: Record<string, readonly MeetingTime[]> = {
323
+ Alice: [createMeetingTime(9, 10), createMeetingTime(14, 15)],
324
+ Bob: [createMeetingTime(10, 11), createMeetingTime(15, 16)],
325
+ }
326
+ return name in mockData
327
+ ? TE.right(mockData[name])
328
+ : TE.left(`Calendar not found for ${name}`)
329
+ }
330
+
331
+ // 複数の参加者の予定を取得して結合
332
+ const scheduledMeetings = (
333
+ attendees: readonly string[]
334
+ ): TE.TaskEither<string, readonly MeetingTime[]> =>
335
+ pipe(
336
+ attendees,
337
+ RA.traverse(TE.ApplicativeSeq)(calendarEntriesApi),
338
+ TE.map(RA.flatten)
339
+ )
340
+ ```
341
+
342
+ ### 8.9 orElse によるエラーハンドリング
343
+
344
+ TaskEither の `orElse` メソッドで、失敗時のフォールバックを指定できます。
345
+
346
+ ```typescript
347
+ const orElseTE = <E, A>(
348
+ te: TE.TaskEither<E, A>,
349
+ alternative: (e: E) => TE.TaskEither<E, A>
350
+ ): TE.TaskEither<E, A> => pipe(te, TE.orElse(alternative))
351
+
352
+ const year = rightTask<string, number>(996)
353
+ const noYear = leftTask<string, number>("no year")
354
+
355
+ await pipe(year, TE.orElse(() => rightTask(2020)))() // right(996)
356
+ await pipe(noYear, TE.orElse(() => rightTask(2020)))() // right(2020)
357
+ ```
358
+
359
+ ```plantuml
360
+ @startuml
361
+ !theme plain
362
+
363
+ rectangle "orElse の動作" {
364
+ card "メインの TaskEither" as main
365
+ card "フォールバック TaskEither" as fallback
366
+ card "結果" as result
367
+
368
+ main --> result : 成功時
369
+ main --> fallback : 失敗時
370
+ fallback --> result
371
+ }
372
+
373
+ @enduml
374
+ ```
375
+
376
+ #### リトライ戦略
377
+
378
+ ```typescript
379
+ // リトライ関数。指定回数まで再試行する
380
+ const retry = <E, A>(
381
+ action: TE.TaskEither<E, A>,
382
+ maxRetries: number
383
+ ): TE.TaskEither<E, A> => {
384
+ if (maxRetries <= 0) return action
385
+ return pipe(
386
+ action,
387
+ TE.orElse(() => retry(action, maxRetries - 1))
388
+ )
389
+ }
390
+
391
+ // リトライしてもダメならデフォルト値を返す
392
+ const retryWithDefault = <E, A>(
393
+ action: TE.TaskEither<E, A>,
394
+ maxRetries: number,
395
+ defaultValue: A
396
+ ): T.Task<A> =>
397
+ pipe(
398
+ retry(action, maxRetries),
399
+ TE.getOrElse(() => T.of(defaultValue))
400
+ )
401
+
402
+ // 使用例
403
+ const result = await retryWithDefault(
404
+ calendarEntriesApi("Unknown"),
405
+ 3,
406
+ []
407
+ )() // => []
408
+ ```
409
+
410
+ ### 8.10 IO のユーティリティ
411
+
412
+ ```typescript
413
+ // 値をコンソールに出力する IO
414
+ const consoleLog = (message: string): IO.IO<void> => () =>
415
+ console.log(message)
416
+
417
+ // メッセージを出力して同じ値を返す IO(デバッグ用)
418
+ const printAndReturn = (message: string): IO.IO<string> => () => {
419
+ console.log(message)
420
+ return message
421
+ }
422
+
423
+ // 複数の IO を順番に実行し、最後の結果を返す
424
+ const chainAll = <A>(...ios: readonly IO.IO<A>[]): IO.IO<A> => {
425
+ if (ios.length === 0) throw new Error('At least one IO required')
426
+ return ios.reduce((acc, io) =>
427
+ pipe(acc, IO.chain(() => io))
428
+ )
429
+ }
430
+ ```
431
+
432
+ ### 8.11 IOEither
433
+
434
+ `IOEither` は同期的だがエラーを返す可能性のある計算を表します。
435
+
436
+ ```typescript
437
+ type IOEither<E, A> = IO.IO<E.Either<E, A>>
438
+
439
+ // 例外をスローする可能性のある関数を IOEither にラップする
440
+ const tryCatchIO = <E, A>(
441
+ thunk: () => A,
442
+ onThrow: (error: unknown) => E
443
+ ): IOEither<E, A> => () => {
444
+ try {
445
+ return E.right(thunk())
446
+ } catch (e) {
447
+ return E.left(onThrow(e))
448
+ }
449
+ }
450
+
451
+ const parseJson = tryCatchIO(
452
+ () => JSON.parse('{"a": 1}'),
453
+ (e) => `Parse error: ${e}`
454
+ )
455
+ parseJson() // right({ a: 1 })
456
+
457
+ const parseInvalid = tryCatchIO(
458
+ () => JSON.parse('invalid'),
459
+ (e) => `Parse error: ${e}`
460
+ )
461
+ parseInvalid() // left("Parse error: ...")
462
+ ```
463
+
464
+ ### 8.12 メモ化
465
+
466
+ IO の結果をキャッシュして、2回目以降は同じ値を返すようにできます。
467
+
468
+ ```typescript
469
+ import * as O from 'fp-ts/Option'
470
+
471
+ const memoizeIO = <A>(io: IO.IO<A>): IO.IO<A> => {
472
+ let cached: O.Option<A> = O.none
473
+ return () =>
474
+ pipe(
475
+ cached,
476
+ O.getOrElse(() => {
477
+ const result = io()
478
+ cached = O.some(result)
479
+ return result
480
+ })
481
+ )
482
+ }
483
+
484
+ let count = 0
485
+ const io = memoizeIO(() => { count++; return count })
486
+ io() // => 1
487
+ io() // => 1 (キャッシュされた値)
488
+ io() // => 1
489
+ ```
490
+
491
+ ---
492
+
493
+ ## 第9章: ストリーム処理
494
+
495
+ ### 9.1 ストリームとは
496
+
497
+ **ストリーム**は、要素の(潜在的に無限の)シーケンスを表します。TypeScript では Generator 関数を使って実装します。
498
+
499
+ ```plantuml
500
+ @startuml
501
+ !theme plain
502
+
503
+ rectangle "Array vs LazyList" {
504
+ rectangle "Array" as array {
505
+ card "全要素がメモリに存在"
506
+ card "有限"
507
+ card "即座に評価"
508
+ }
509
+
510
+ rectangle "LazyList (Generator)" as lazylist {
511
+ card "必要な時に要素を生成"
512
+ card "無限も可能"
513
+ card "遅延評価"
514
+ }
515
+ }
516
+
517
+ @enduml
518
+ ```
519
+
520
+ ### 9.2 LazyList の基本
521
+
522
+ **ソースファイル**: `app/typescript/src/ch09_streams.ts`
523
+
524
+ ```typescript
525
+ // LazyList は Generator を使って表現
526
+ type LazyList<A> = Generator<A, void, undefined>
527
+
528
+ // 配列から LazyList を作成
529
+ function* fromArray<A>(arr: readonly A[]): LazyList<A> {
530
+ for (const item of arr) {
531
+ yield item
532
+ }
533
+ }
534
+
535
+ const list = fromArray([1, 2, 3])
536
+ [...list] // => [1, 2, 3]
537
+
538
+ // 範囲から LazyList を作成
539
+ function* range(start: number, end: number): LazyList<number> {
540
+ for (let i = start; i < end; i++) {
541
+ yield i
542
+ }
543
+ }
544
+
545
+ [...range(1, 5)] // => [1, 2, 3, 4]
546
+ ```
547
+
548
+ ### 9.3 無限ストリーム
549
+
550
+ ```typescript
551
+ // 値を無限に繰り返すストリーム
552
+ function* repeatValue<A>(value: A): LazyList<A> {
553
+ while (true) {
554
+ yield value
555
+ }
556
+ }
557
+
558
+ // 配列を無限に繰り返すストリーム
559
+ function* cycleArray<A>(arr: readonly A[]): LazyList<A> {
560
+ if (arr.length === 0) return
561
+ while (true) {
562
+ for (const item of arr) {
563
+ yield item
564
+ }
565
+ }
566
+ }
567
+
568
+ // 初期値と生成関数から無限ストリームを作成(unfold)
569
+ function* iterate<A>(initial: A, fn: (a: A) => A): LazyList<A> {
570
+ let current = initial
571
+ while (true) {
572
+ yield current
573
+ current = fn(current)
574
+ }
575
+ }
576
+
577
+ const powers = iterate(1, n => n * 2)
578
+ take(powers, 5) // => [1, 2, 4, 8, 16]
579
+ ```
580
+
581
+ ```plantuml
582
+ @startuml
583
+ !theme plain
584
+
585
+ rectangle "無限ストリーム" {
586
+ card "cycleArray([1, 2, 3])" as stream
587
+ card "1, 2, 3, 1, 2, 3, 1, 2, 3, ..." as infinite
588
+ card "take(8)" as take
589
+ card "[1, 2, 3, 1, 2, 3, 1, 2]" as result
590
+
591
+ stream --> infinite
592
+ infinite --> take
593
+ take --> result
594
+ }
595
+
596
+ note bottom
597
+ 無限のストリームから
598
+ 必要な分だけ取得
599
+ end note
600
+
601
+ @enduml
602
+ ```
603
+
604
+ ### 9.4 ストリームの主要操作
605
+
606
+ | 操作 | 説明 | 例 |
607
+ |------|------|-----|
608
+ | `take(n)` | 最初の n 要素を取得 | `take(stream, 3)` |
609
+ | `drop(n)` | 最初の n 要素をスキップ | `drop(stream, 2)` |
610
+ | `filter(p)` | 条件を満たす要素のみ | `filterStream(stream, x => x > 0)` |
611
+ | `map(f)` | 各要素を変換 | `mapStream(stream, x => x * 2)` |
612
+ | `takeWhile(p)` | 条件を満たす間取得 | `takeWhile(stream, x => x < 10)` |
613
+ | `sliding(n)` | スライディングウィンドウ | `sliding(stream, 3)` |
614
+
615
+ ```typescript
616
+ // 最初の n 要素を取得
617
+ const take = <A>(gen: LazyList<A>, n: number): readonly A[] => {
618
+ const result: A[] = []
619
+ let count = 0
620
+ for (const item of gen) {
621
+ if (count >= n) break
622
+ result.push(item)
623
+ count++
624
+ }
625
+ return result
626
+ }
627
+
628
+ // 各要素を変換
629
+ function* mapStream<A, B>(
630
+ gen: LazyList<A>,
631
+ fn: (a: A) => B
632
+ ): LazyList<B> {
633
+ for (const item of gen) {
634
+ yield fn(item)
635
+ }
636
+ }
637
+
638
+ // 条件を満たす要素のみ
639
+ function* filterStream<A>(
640
+ gen: LazyList<A>,
641
+ predicate: (a: A) => boolean
642
+ ): LazyList<A> {
643
+ for (const item of gen) {
644
+ if (predicate(item)) {
645
+ yield item
646
+ }
647
+ }
648
+ }
649
+ ```
650
+
651
+ ### 9.5 ストリームの結合
652
+
653
+ ```typescript
654
+ // 2つのストリームを結合
655
+ function* append<A>(gen1: LazyList<A>, gen2: LazyList<A>): LazyList<A> {
656
+ yield* gen1
657
+ yield* gen2
658
+ }
659
+
660
+ // 2つのストリームをペアにして結合
661
+ function* zipStream<A, B>(
662
+ gen1: LazyList<A>,
663
+ gen2: LazyList<B>
664
+ ): LazyList<readonly [A, B]> {
665
+ const iter1 = gen1[Symbol.iterator]()
666
+ const iter2 = gen2[Symbol.iterator]()
667
+ while (true) {
668
+ const next1 = iter1.next()
669
+ const next2 = iter2.next()
670
+ if (next1.done || next2.done) break
671
+ yield [next1.value, next2.value] as const
672
+ }
673
+ }
674
+
675
+ const list1 = fromArray([1, 2, 3])
676
+ const list2 = fromArray(['a', 'b', 'c'])
677
+ [...zipStream(list1, list2)] // => [[1, 'a'], [2, 'b'], [3, 'c']]
678
+ ```
679
+
680
+ ### 9.6 スライディングウィンドウ
681
+
682
+ `sliding` で連続する要素をグループ化できます。
683
+
684
+ ```typescript
685
+ function* sliding<A>(
686
+ gen: LazyList<A>,
687
+ windowSize: number
688
+ ): LazyList<readonly A[]> {
689
+ const buffer: A[] = []
690
+ for (const item of gen) {
691
+ buffer.push(item)
692
+ if (buffer.length === windowSize) {
693
+ yield [...buffer]
694
+ buffer.shift()
695
+ }
696
+ }
697
+ }
698
+
699
+ const stream = fromArray([1, 2, 3, 4, 5])
700
+ [...sliding(stream, 3)]
701
+ // => [[1, 2, 3], [2, 3, 4], [3, 4, 5]]
702
+ ```
703
+
704
+ ```plantuml
705
+ @startuml
706
+ !theme plain
707
+
708
+ rectangle "sliding(3) の動作" {
709
+ card "[1, 2, 3, 4, 5]" as input
710
+ card "[1, 2, 3]" as w1
711
+ card "[2, 3, 4]" as w2
712
+ card "[3, 4, 5]" as w3
713
+
714
+ input --> w1
715
+ input --> w2
716
+ input --> w3
717
+ }
718
+
719
+ @enduml
720
+ ```
721
+
722
+ ### 9.7 トレンド検出(為替レートの例)
723
+
724
+ 為替レートを監視して、上昇トレンドを検出する例です。
725
+
726
+ #### トレンド判定(純粋関数)
727
+
728
+ ```typescript
729
+ const trending = (rates: readonly number[]): boolean => {
730
+ if (rates.length <= 1) return false
731
+ const prev = RA.dropRight(1)(rates)
732
+ const curr = RA.dropLeft(1)(rates)
733
+ return pipe(
734
+ RA.zip(curr)(prev),
735
+ RA.every(([p, c]) => c > p)
736
+ )
737
+ }
738
+
739
+ trending([0.81, 0.82, 0.83]) // true (上昇トレンド)
740
+ trending([0.81, 0.84, 0.83]) // false
741
+
742
+ // 下降トレンドの検出
743
+ const downtrending = (rates: readonly number[]): boolean => {
744
+ if (rates.length <= 1) return false
745
+ const prev = RA.dropRight(1)(rates)
746
+ const curr = RA.dropLeft(1)(rates)
747
+ return pipe(
748
+ RA.zip(curr)(prev),
749
+ RA.every(([p, c]) => c < p)
750
+ )
751
+ }
752
+
753
+ // 安定しているかどうか
754
+ const isStable = (values: readonly number[]): boolean => {
755
+ if (values.length < 2) return false
756
+ const first = values[0]
757
+ return values.every((v) => v === first)
758
+ }
759
+ ```
760
+
761
+ #### スライディングウィンドウでトレンドを検出
762
+
763
+ ```typescript
764
+ function* detectTrends(
765
+ gen: LazyList<number>,
766
+ windowSize: number
767
+ ): LazyList<{ window: readonly number[]; trending: boolean }> {
768
+ for (const window of sliding(gen, windowSize)) {
769
+ yield { window, trending: trending(window) }
770
+ }
771
+ }
772
+
773
+ const rates = fromArray([0.80, 0.81, 0.82, 0.83, 0.82])
774
+ const trends = [...detectTrends(rates, 3)]
775
+ // [
776
+ // { window: [0.80, 0.81, 0.82], trending: true },
777
+ // { window: [0.81, 0.82, 0.83], trending: true },
778
+ // { window: [0.82, 0.83, 0.82], trending: false }
779
+ // ]
780
+ ```
781
+
782
+ ### 9.8 サイコロストリーム
783
+
784
+ ```typescript
785
+ // サイコロを振る無限ストリーム
786
+ function* dieStream(): LazyList<number> {
787
+ while (true) {
788
+ yield Math.floor(Math.random() * 6) + 1
789
+ }
790
+ }
791
+
792
+ take(dieStream(), 5) // => [3, 1, 6, 2, 4] (ランダム)
793
+
794
+ // 指定した値が出るまでサイコロを振り続ける
795
+ function* dieUntil(target: number): LazyList<number> {
796
+ while (true) {
797
+ const roll = Math.floor(Math.random() * 6) + 1
798
+ yield roll
799
+ if (roll === target) break
800
+ }
801
+ }
802
+
803
+ [...dieUntil(6)] // => [3, 1, 5, 2, 6] (6が出るまで)
804
+ ```
805
+
806
+ ### 9.9 便利なストリームユーティリティ
807
+
808
+ ```typescript
809
+ // フィボナッチ数列
810
+ function* fibonacci(): LazyList<number> {
811
+ let [a, b] = [0, 1]
812
+ while (true) {
813
+ yield a
814
+ ;[a, b] = [b, a + b]
815
+ }
816
+ }
817
+
818
+ take(fibonacci(), 10) // => [0, 1, 1, 2, 3, 5, 8, 13, 21, 34]
819
+
820
+ // 素数
821
+ function* primes(): LazyList<number> {
822
+ const sieve: Set<number> = new Set()
823
+ let n = 2
824
+ while (true) {
825
+ if (!sieve.has(n)) {
826
+ yield n
827
+ for (let i = n * n; i < n * n + n * 100; i += n) {
828
+ sieve.add(i)
829
+ }
830
+ }
831
+ n++
832
+ }
833
+ }
834
+
835
+ take(primes(), 10) // => [2, 3, 5, 7, 11, 13, 17, 19, 23, 29]
836
+
837
+ // 累積和
838
+ function* scan<A, B>(
839
+ gen: LazyList<A>,
840
+ initial: B,
841
+ fn: (acc: B, a: A) => B
842
+ ): LazyList<B> {
843
+ let acc = initial
844
+ yield acc
845
+ for (const item of gen) {
846
+ acc = fn(acc, item)
847
+ yield acc
848
+ }
849
+ }
850
+
851
+ [...scan(fromArray([1, 2, 3, 4]), 0, (a, b) => a + b)]
852
+ // => [0, 1, 3, 6, 10]
853
+ ```
854
+
855
+ ---
856
+
857
+ ## まとめ
858
+
859
+ ### Part IV で学んだこと
860
+
861
+ ```plantuml
862
+ @startuml
863
+ !theme plain
864
+
865
+ rectangle "Part IV: IO と副作用の管理" {
866
+ rectangle "第8章" as ch8 {
867
+ card "IO モナド"
868
+ card "Task (非同期)"
869
+ card "TaskEither"
870
+ card "orElse / retry"
871
+ card "メモ化"
872
+ }
873
+
874
+ rectangle "第9章" as ch9 {
875
+ card "LazyList (Generator)"
876
+ card "無限ストリーム"
877
+ card "take / filter / map"
878
+ card "sliding"
879
+ card "トレンド検出"
880
+ }
881
+ }
882
+
883
+ ch8 --> ch9
884
+
885
+ @enduml
886
+ ```
887
+
888
+ ### fp-ts の IO 関連型の比較
889
+
890
+ | 型 | 同期/非同期 | エラー | 用途 |
891
+ |------|-------------|--------|------|
892
+ | `IO<A>` | 同期 | なし | 単純な同期副作用 |
893
+ | `IOEither<E, A>` | 同期 | あり | 失敗する可能性のある同期処理 |
894
+ | `Task<A>` | 非同期 | なし | Promise ベースの非同期処理 |
895
+ | `TaskEither<E, A>` | 非同期 | あり | 失敗する可能性のある非同期処理 |
896
+
897
+ ### キーポイント
898
+
899
+ 1. **IO モナド**: 副作用を「記述」として扱い、実行を遅延させる
900
+ 2. **Task**: 非同期処理を純粋な関数として扱う
901
+ 3. **TaskEither**: 非同期処理とエラーハンドリングを組み合わせる
902
+ 4. **orElse / retry**: 失敗時のフォールバック戦略
903
+ 5. **LazyList**: Generator を使った遅延評価のシーケンス
904
+ 6. **sliding**: 連続する要素をグループ化してパターンを検出
905
+
906
+ ### 次のステップ
907
+
908
+ Part V では、以下のトピックを学びます:
909
+
910
+ - 並行・並列処理
911
+ - より高度なモナド変換子
912
+
913
+ ---
914
+
915
+ ## 演習問題
916
+
917
+ ### 問題 1: IO の基本
918
+
919
+ 以下の関数を実装してください。
920
+
921
+ ```typescript
922
+ import * as IO from 'fp-ts/IO'
923
+
924
+ const printAndReturn = (message: string): IO.IO<string> => ???
925
+
926
+ // 期待される動作
927
+ // printAndReturn("Hello")() は
928
+ // "Hello" をコンソールに出力し、"Hello" を返す
929
+ ```
930
+
931
+ <details>
932
+ <summary>解答</summary>
933
+
934
+ ```typescript
935
+ const printAndReturn = (message: string): IO.IO<string> => () => {
936
+ console.log(message)
937
+ return message
938
+ }
939
+ ```
940
+
941
+ </details>
942
+
943
+ ### 問題 2: IO の合成
944
+
945
+ 以下の関数を実装してください。2つの IO を順番に実行し、結果を結合します。
946
+
947
+ ```typescript
948
+ const combineIO = <A, B, C>(
949
+ io1: IO.IO<A>,
950
+ io2: IO.IO<B>,
951
+ f: (a: A, b: B) => C
952
+ ): IO.IO<C> => ???
953
+
954
+ // 期待される動作
955
+ const result = combineIO(IO.of(1), IO.of(2), (a, b) => a + b)
956
+ result() // 3
957
+ ```
958
+
959
+ <details>
960
+ <summary>解答</summary>
961
+
962
+ ```typescript
963
+ import { pipe } from 'fp-ts/function'
964
+
965
+ const combineIO = <A, B, C>(
966
+ io1: IO.IO<A>,
967
+ io2: IO.IO<B>,
968
+ f: (a: A, b: B) => C
969
+ ): IO.IO<C> =>
970
+ pipe(
971
+ io1,
972
+ IO.chain((a) =>
973
+ pipe(
974
+ io2,
975
+ IO.map((b) => f(a, b))
976
+ )
977
+ )
978
+ )
979
+ ```
980
+
981
+ </details>
982
+
983
+ ### 問題 3: リトライ
984
+
985
+ 以下の関数を実装してください。指定回数リトライし、全部失敗したらデフォルト値を返します。
986
+
987
+ ```typescript
988
+ const retryWithDefault = <E, A>(
989
+ action: TE.TaskEither<E, A>,
990
+ maxRetries: number,
991
+ defaultValue: A
992
+ ): T.Task<A> => ???
993
+ ```
994
+
995
+ <details>
996
+ <summary>解答</summary>
997
+
998
+ ```typescript
999
+ import { pipe } from 'fp-ts/function'
1000
+ import * as TE from 'fp-ts/TaskEither'
1001
+ import * as T from 'fp-ts/Task'
1002
+
1003
+ const retry = <E, A>(
1004
+ action: TE.TaskEither<E, A>,
1005
+ maxRetries: number
1006
+ ): TE.TaskEither<E, A> => {
1007
+ if (maxRetries <= 0) return action
1008
+ return pipe(
1009
+ action,
1010
+ TE.orElse(() => retry(action, maxRetries - 1))
1011
+ )
1012
+ }
1013
+
1014
+ const retryWithDefault = <E, A>(
1015
+ action: TE.TaskEither<E, A>,
1016
+ maxRetries: number,
1017
+ defaultValue: A
1018
+ ): T.Task<A> =>
1019
+ pipe(
1020
+ retry(action, maxRetries),
1021
+ TE.getOrElse(() => T.of(defaultValue))
1022
+ )
1023
+ ```
1024
+
1025
+ </details>
1026
+
1027
+ ### 問題 4: ストリーム操作
1028
+
1029
+ 以下のストリームを作成してください。
1030
+
1031
+ ```typescript
1032
+ // 1. 1から10までの偶数のストリーム
1033
+ const evens: LazyList<number> = ???
1034
+
1035
+ // 2. 無限に交互に true/false を返すストリーム
1036
+ const alternating: LazyList<boolean> = ???
1037
+
1038
+ // 3. 最初の5つの要素の合計を計算
1039
+ const sum: number = take(range(1, 100), 5).???
1040
+ ```
1041
+
1042
+ <details>
1043
+ <summary>解答</summary>
1044
+
1045
+ ```typescript
1046
+ // 1. 1から10までの偶数
1047
+ const evens = filterStream(rangeInclusive(1, 10), x => x % 2 === 0)
1048
+ // または
1049
+ const evens = fromArray([2, 4, 6, 8, 10])
1050
+
1051
+ // 2. 無限に交互に true/false
1052
+ const alternating = cycleArray([true, false])
1053
+
1054
+ // 3. 最初の5つの要素の合計
1055
+ const sum = take(range(1, 100), 5).reduce((a, b) => a + b, 0) // 15
1056
+ ```
1057
+
1058
+ </details>
1059
+
1060
+ ### 問題 5: トレンド検出
1061
+
1062
+ 以下の関数を実装してください。直近3つの値が全て同じかどうかを判定します。
1063
+
1064
+ ```typescript
1065
+ const isStable = (values: readonly number[]): boolean => ???
1066
+
1067
+ // 期待される動作
1068
+ assert(isStable([5, 5, 5]) === true)
1069
+ assert(isStable([5, 5, 6]) === false)
1070
+ assert(isStable([5, 6, 5]) === false)
1071
+ assert(isStable([5]) === false) // 2つ未満は false
1072
+ ```
1073
+
1074
+ <details>
1075
+ <summary>解答</summary>
1076
+
1077
+ ```typescript
1078
+ const isStable = (values: readonly number[]): boolean => {
1079
+ if (values.length < 2) return false
1080
+ const first = values[0]
1081
+ return values.every((v) => v === first)
1082
+ }
1083
+ ```
1084
+
1085
+ </details>