@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,842 @@
1
+ # Part III: エラーハンドリングと Option/Either
2
+
3
+ 本章では、関数型プログラミングにおける安全なエラーハンドリングを学びます。`null` や例外に頼らず、`Option` と `Either` を使って型安全にエラーを扱う方法を習得します。
4
+
5
+ ---
6
+
7
+ ## 第6章: Option 型による安全なエラーハンドリング
8
+
9
+ ### 6.1 なぜ Option が必要か
10
+
11
+ 従来のエラーハンドリングには問題があります。
12
+
13
+ ```plantuml
14
+ @startuml
15
+ !theme plain
16
+
17
+ rectangle "従来のエラーハンドリング" {
18
+ rectangle "null/undefined を返す" as null_ret {
19
+ card "TypeError の危険"
20
+ card "コンパイル時に検出できない"
21
+ }
22
+
23
+ rectangle "例外をスローする" as exception {
24
+ card "制御フローが複雑化"
25
+ card "純粋関数ではなくなる"
26
+ }
27
+ }
28
+
29
+ rectangle "Option による解決" as option {
30
+ card "型で「値がないかもしれない」を表現"
31
+ card "コンパイル時にチェック"
32
+ card "純粋関数のまま"
33
+ }
34
+
35
+ null_ret --> option : 置き換え
36
+ exception --> option : 置き換え
37
+
38
+ @enduml
39
+ ```
40
+
41
+ ### 6.2 Option の基本
42
+
43
+ fp-ts の `Option<A>` は「`A` 型の値があるか、ないか」を表す型です。
44
+
45
+ ```plantuml
46
+ @startuml
47
+ !theme plain
48
+
49
+ rectangle "Option<A>" as option {
50
+ rectangle "some(value)" as some {
51
+ card "値が存在する"
52
+ card "value: A を保持"
53
+ }
54
+
55
+ rectangle "none" as none {
56
+ card "値が存在しない"
57
+ card "null/undefined の代わり"
58
+ }
59
+ }
60
+
61
+ @enduml
62
+ ```
63
+
64
+ **ソースファイル**: `app/typescript/src/ch06_error_handling.ts`
65
+
66
+ ```typescript
67
+ import * as O from 'fp-ts/Option'
68
+
69
+ // 安全な除算
70
+ const safeDivide = (a: number, b: number): O.Option<number> =>
71
+ b === 0 ? O.none : O.some(Math.floor(a / b))
72
+
73
+ safeDivide(10, 2) // some(5)
74
+ safeDivide(10, 0) // none
75
+
76
+ // 文字列を数値にパース
77
+ const parseNumber = (s: string): O.Option<number> => {
78
+ const n = parseInt(s, 10)
79
+ return isNaN(n) ? O.none : O.some(n)
80
+ }
81
+
82
+ parseNumber("42") // some(42)
83
+ parseNumber("abc") // none
84
+ ```
85
+
86
+ ### 6.3 TV番組のパース例
87
+
88
+ TV番組の文字列をパースする例で Option の使い方を学びます。
89
+
90
+ ```typescript
91
+ interface TvShow {
92
+ readonly title: string
93
+ readonly start: number
94
+ readonly end: number
95
+ }
96
+
97
+ // 入力例: "Breaking Bad (2008-2013)"
98
+ // 期待する出力: { title: "Breaking Bad", start: 2008, end: 2013 }
99
+ ```
100
+
101
+ #### 例外を使う方法(問題あり)
102
+
103
+ ```typescript
104
+ // 問題のあるコード - 例外がスローされる可能性
105
+ const parseShowUnsafe = (rawShow: string): TvShow => {
106
+ const bracketOpen = rawShow.indexOf('(')
107
+ const bracketClose = rawShow.indexOf(')')
108
+ const dash = rawShow.indexOf('-')
109
+
110
+ const name = rawShow.substring(0, bracketOpen).trim()
111
+ const yearStart = parseInt(rawShow.substring(bracketOpen + 1, dash))
112
+ const yearEnd = parseInt(rawShow.substring(dash + 1, bracketClose))
113
+
114
+ return { title: name, start: yearStart, end: yearEnd }
115
+ }
116
+
117
+ // 無効な入力で NaN やおかしな結果になる
118
+ ```
119
+
120
+ #### Option を使う方法
121
+
122
+ ```typescript
123
+ import { pipe } from 'fp-ts/function'
124
+ import * as O from 'fp-ts/Option'
125
+
126
+ const parseShow = (rawShow: string): O.Option<TvShow> =>
127
+ pipe(
128
+ O.Do,
129
+ O.bind('title', () => extractName(rawShow)),
130
+ O.bind('start', () =>
131
+ pipe(
132
+ extractYearStart(rawShow),
133
+ O.orElse(() => extractSingleYear(rawShow))
134
+ )
135
+ ),
136
+ O.bind('end', () =>
137
+ pipe(
138
+ extractYearEnd(rawShow),
139
+ O.orElse(() => extractSingleYear(rawShow))
140
+ )
141
+ ),
142
+ O.map(({ title, start, end }) => ({ title, start, end }))
143
+ )
144
+
145
+ // 正常ケース
146
+ parseShow("Breaking Bad (2008-2013)")
147
+ // some({ title: "Breaking Bad", start: 2008, end: 2013 })
148
+
149
+ // 異常ケース → none が返される(例外なし)
150
+ parseShow("The Wire 2002-2008") // none
151
+ ```
152
+
153
+ ### 6.4 小さな関数から組み立てる
154
+
155
+ 複雑なパース処理を小さな関数に分解します。
156
+
157
+ ```typescript
158
+ // 名前を抽出
159
+ const extractName = (rawShow: string): O.Option<string> => {
160
+ const bracketOpen = rawShow.indexOf('(')
161
+ if (bracketOpen > 0) {
162
+ return O.some(rawShow.substring(0, bracketOpen).trim())
163
+ }
164
+ return O.none
165
+ }
166
+
167
+ // 開始年を抽出
168
+ const extractYearStart = (rawShow: string): O.Option<number> => {
169
+ const bracketOpen = rawShow.indexOf('(')
170
+ const dash = rawShow.indexOf('-')
171
+ if (bracketOpen !== -1 && dash > bracketOpen + 1) {
172
+ const yearStr = rawShow.substring(bracketOpen + 1, dash)
173
+ return parseNumber(yearStr)
174
+ }
175
+ return O.none
176
+ }
177
+
178
+ // 単年を抽出("Chernobyl (2019)" のような形式用)
179
+ const extractSingleYear = (rawShow: string): O.Option<number> => {
180
+ const dash = rawShow.indexOf('-')
181
+ const bracketOpen = rawShow.indexOf('(')
182
+ const bracketClose = rawShow.indexOf(')')
183
+ if (dash === -1 && bracketOpen !== -1 && bracketClose > bracketOpen + 1) {
184
+ const yearStr = rawShow.substring(bracketOpen + 1, bracketClose)
185
+ return parseNumber(yearStr)
186
+ }
187
+ return O.none
188
+ }
189
+ ```
190
+
191
+ ```plantuml
192
+ @startuml
193
+ !theme plain
194
+
195
+ rectangle "パース処理の分解" {
196
+ card "\"Breaking Bad (2008-2013)\"" as input
197
+
198
+ rectangle "小さな関数" as funcs {
199
+ card "extractName" as name
200
+ card "extractYearStart" as start
201
+ card "extractYearEnd" as end
202
+ }
203
+
204
+ card "TvShow" as output
205
+
206
+ input --> funcs
207
+ funcs --> output : O.Do で合成
208
+ }
209
+
210
+ @enduml
211
+ ```
212
+
213
+ ### 6.5 orElse によるフォールバック
214
+
215
+ `orElse` を使って、最初の Option が `none` の場合に代替を試すことができます。
216
+
217
+ ```typescript
218
+ const seven: O.Option<number> = O.some(7)
219
+ const eight: O.Option<number> = O.some(8)
220
+ const none: O.Option<number> = O.none
221
+
222
+ pipe(seven, O.orElse(() => eight)) // some(7) - 最初が some なのでそのまま
223
+ pipe(none, O.orElse(() => eight)) // some(8) - 最初が none なので代替を使用
224
+ ```
225
+
226
+ #### 単年の番組に対応する
227
+
228
+ ```typescript
229
+ // これで単年の番組もパースできる
230
+ parseShow("Chernobyl (2019)")
231
+ // some({ title: "Chernobyl", start: 2019, end: 2019 })
232
+ ```
233
+
234
+ ### 6.6 Option の主要メソッド
235
+
236
+ | メソッド | 説明 | 例 |
237
+ |----------|------|-----|
238
+ | `O.map` | 値があれば変換 | `pipe(O.some(5), O.map(x => x * 2))` → `some(10)` |
239
+ | `O.chain` | 値があれば Option を返す関数を適用 | `pipe(O.some(5), O.chain(x => O.some(x * 2)))` → `some(10)` |
240
+ | `O.filter` | 条件を満たさなければ none | `pipe(O.some(5), O.filter(x => x > 10))` → `none` |
241
+ | `O.orElse` | none なら代替を使用 | `pipe(O.none, O.orElse(() => O.some(5)))` → `some(5)` |
242
+ | `O.getOrElse` | none ならデフォルト値 | `pipe(O.none, O.getOrElse(() => 0))` → `0` |
243
+
244
+ ```typescript
245
+ const year: O.Option<number> = O.some(996)
246
+ const noYear: O.Option<number> = O.none
247
+
248
+ // map
249
+ pipe(year, O.map(y => y * 2)) // some(1992)
250
+ pipe(noYear, O.map(y => y * 2)) // none
251
+
252
+ // chain
253
+ pipe(year, O.chain(y => O.some(y * 2))) // some(1992)
254
+
255
+ // filter
256
+ pipe(year, O.filter(y => y < 2020)) // some(996)
257
+ pipe(year, O.filter(y => y > 2020)) // none
258
+
259
+ // getOrElse
260
+ pipe(year, O.getOrElse(() => 2020)) // 996
261
+ pipe(noYear, O.getOrElse(() => 2020)) // 2020
262
+ ```
263
+
264
+ ### 6.7 エラーハンドリング戦略
265
+
266
+ 複数の要素をパースする場合、2つの戦略があります。
267
+
268
+ ```plantuml
269
+ @startuml
270
+ !theme plain
271
+
272
+ rectangle "エラーハンドリング戦略" {
273
+ rectangle "Best-effort 戦略" as best {
274
+ card "パースできたものだけ返す"
275
+ card "エラーは無視"
276
+ card "readonly TvShow[] を返す"
277
+ }
278
+
279
+ rectangle "All-or-nothing 戦略" as all {
280
+ card "全部成功するか、全部失敗"
281
+ card "一つでも失敗したら none"
282
+ card "Option<readonly TvShow[]> を返す"
283
+ }
284
+ }
285
+
286
+ @enduml
287
+ ```
288
+
289
+ #### Best-effort 戦略
290
+
291
+ ```typescript
292
+ import * as RA from 'fp-ts/ReadonlyArray'
293
+
294
+ const parseShowsBestEffort = (
295
+ rawShows: readonly string[]
296
+ ): readonly TvShow[] =>
297
+ pipe(
298
+ rawShows,
299
+ RA.map(parseShow),
300
+ RA.compact // none を除去して some の値だけを抽出
301
+ )
302
+
303
+ const rawShows = [
304
+ "Breaking Bad (2008-2013)",
305
+ "The Wire 2002 2008", // 無効な形式
306
+ "Mad Men (2007-2015)"
307
+ ]
308
+
309
+ parseShowsBestEffort(rawShows)
310
+ // [TvShow("Breaking Bad", ...), TvShow("Mad Men", ...)]
311
+ // 無効なものは無視される
312
+ ```
313
+
314
+ #### All-or-nothing 戦略
315
+
316
+ ```typescript
317
+ const parseShowsAllOrNothing = (
318
+ rawShows: readonly string[]
319
+ ): O.Option<readonly TvShow[]> =>
320
+ pipe(rawShows, RA.traverse(O.Applicative)(parseShow))
321
+
322
+ // 全部成功 → some([...])
323
+ parseShowsAllOrNothing(["Breaking Bad (2008-2013)", "Mad Men (2007-2015)"])
324
+ // some([TvShow(...), TvShow(...)])
325
+
326
+ // 一つでも失敗 → none
327
+ parseShowsAllOrNothing(["Breaking Bad (2008-2013)", "Invalid"])
328
+ // none
329
+ ```
330
+
331
+ ---
332
+
333
+ ## 第7章: Either 型と複合的なエラー処理
334
+
335
+ ### 7.1 Option の限界
336
+
337
+ `Option` は「値があるかないか」しか表現できません。**なぜ失敗したのか**を伝えられません。
338
+
339
+ ```plantuml
340
+ @startuml
341
+ !theme plain
342
+
343
+ rectangle "Option vs Either" {
344
+ rectangle "Option<A>" as opt {
345
+ card "some(value)" as some
346
+ card "none" as none
347
+ none -[hidden]-> some
348
+ }
349
+
350
+ rectangle "Either<E, A>" as either {
351
+ card "right(value)" as right
352
+ card "left(error)" as left
353
+ left -[hidden]-> right
354
+ }
355
+
356
+ opt --> either : 進化
357
+ }
358
+
359
+ note bottom of none
360
+ 失敗理由が分からない
361
+ end note
362
+
363
+ note bottom of left
364
+ エラー情報を保持できる
365
+ end note
366
+
367
+ @enduml
368
+ ```
369
+
370
+ ### 7.2 Either の基本
371
+
372
+ **ソースファイル**: `app/typescript/src/ch07_either.ts`
373
+
374
+ `Either<E, A>` は「`E` 型のエラーか、`A` 型の成功値か」を表す型です。
375
+
376
+ - `right(value)`: 成功(慣例的に「正しい」= right)
377
+ - `left(error)`: 失敗(エラー情報を保持)
378
+
379
+ ```typescript
380
+ import * as E from 'fp-ts/Either'
381
+
382
+ const safeDivideE = (a: number, b: number): E.Either<string, number> =>
383
+ b === 0 ? E.left('Division by zero') : E.right(Math.floor(a / b))
384
+
385
+ safeDivideE(10, 2) // right(5)
386
+ safeDivideE(10, 0) // left("Division by zero")
387
+
388
+ const parseNumberE = (s: string): E.Either<string, number> => {
389
+ const n = parseInt(s, 10)
390
+ return isNaN(n) ? E.left(`Cannot parse '${s}' as number`) : E.right(n)
391
+ }
392
+
393
+ parseNumberE("42") // right(42)
394
+ parseNumberE("abc") // left("Cannot parse 'abc' as number")
395
+ ```
396
+
397
+ ### 7.3 Either を使ったパース
398
+
399
+ ```typescript
400
+ const extractNameE = (rawShow: string): E.Either<string, string> => {
401
+ const bracketOpen = rawShow.indexOf('(')
402
+ if (bracketOpen > 0) {
403
+ return E.right(rawShow.substring(0, bracketOpen).trim())
404
+ }
405
+ return E.left(`Can't extract name from ${rawShow}`)
406
+ }
407
+
408
+ extractNameE("The Wire (2002-2008)") // right("The Wire")
409
+ extractNameE("(2022)") // left("Can't extract name from (2022)")
410
+ ```
411
+
412
+ ### 7.4 Option から Either への変換
413
+
414
+ `E.fromOption` で `Option` を `Either` に変換できます。
415
+
416
+ ```typescript
417
+ const optionToEither = <E, A>(
418
+ opt: O.Option<A>,
419
+ errorValue: E
420
+ ): E.Either<E, A> =>
421
+ pipe(opt, E.fromOption(() => errorValue))
422
+
423
+ optionToEither(O.some(5), "no value") // right(5)
424
+ optionToEither(O.none, "no value") // left("no value")
425
+ ```
426
+
427
+ ### 7.5 代数的データ型(ADT)
428
+
429
+ TypeScript では **discriminated union**(タグ付きユニオン)を使って ADT を表現します。
430
+
431
+ #### 直和型(Sum Type)
432
+
433
+ 複数の選択肢を **OR** で表す型です。
434
+
435
+ ```typescript
436
+ // 活動期間を表す直和型
437
+ type YearsActive =
438
+ | { readonly _tag: 'StillActive'; readonly since: number }
439
+ | { readonly _tag: 'ActiveBetween'; readonly start: number; readonly end: number }
440
+
441
+ // コンストラクタ関数
442
+ const stillActive = (since: number): YearsActive => ({
443
+ _tag: 'StillActive',
444
+ since,
445
+ })
446
+
447
+ const activeBetween = (start: number, end: number): YearsActive => ({
448
+ _tag: 'ActiveBetween',
449
+ start,
450
+ end,
451
+ })
452
+
453
+ // 音楽ジャンルを表す直和型
454
+ type MusicGenre = 'HeavyMetal' | 'Pop' | 'HardRock' | 'Grunge'
455
+ ```
456
+
457
+ #### 直積型(Product Type)
458
+
459
+ 複数のフィールドを **AND** で組み合わせる型です。
460
+
461
+ ```typescript
462
+ interface Artist {
463
+ readonly name: string
464
+ readonly genre: MusicGenre
465
+ readonly origin: Location
466
+ readonly yearsActive: YearsActive
467
+ }
468
+
469
+ const metallica = createArtist('Metallica', 'HeavyMetal', 'US', stillActive(1981))
470
+ const ledZeppelin = createArtist('Led Zeppelin', 'HardRock', 'England', activeBetween(1968, 1980))
471
+ ```
472
+
473
+ ```plantuml
474
+ @startuml
475
+ !theme plain
476
+
477
+ rectangle "代数的データ型(ADT)" {
478
+ rectangle "直積型(Product Type)" as product {
479
+ card "interface / type"
480
+ card "A AND B AND C"
481
+ card "フィールドの組み合わせ"
482
+ }
483
+
484
+ rectangle "直和型(Sum Type)" as sum {
485
+ card "discriminated union"
486
+ card "A OR B OR C"
487
+ card "選択肢のいずれか"
488
+ }
489
+ }
490
+
491
+ note bottom of product
492
+ Artist = name AND genre AND origin AND yearsActive
493
+ end note
494
+
495
+ note bottom of sum
496
+ YearsActive = StillActive OR ActiveBetween
497
+ end note
498
+
499
+ @enduml
500
+ ```
501
+
502
+ ### 7.6 パターンマッチング
503
+
504
+ TypeScript では `switch` 文と `_tag` プロパティで安全にパターンマッチングができます。
505
+
506
+ ```typescript
507
+ const wasArtistActive = (
508
+ artist: Artist,
509
+ yearStart: number,
510
+ yearEnd: number
511
+ ): boolean => {
512
+ const years = artist.yearsActive
513
+ switch (years._tag) {
514
+ case 'StillActive':
515
+ return years.since <= yearEnd
516
+ case 'ActiveBetween':
517
+ return years.start <= yearEnd && years.end >= yearStart
518
+ }
519
+ }
520
+
521
+ const activeLength = (artist: Artist, currentYear: number): number => {
522
+ const years = artist.yearsActive
523
+ switch (years._tag) {
524
+ case 'StillActive':
525
+ return currentYear - years.since
526
+ case 'ActiveBetween':
527
+ return years.end - years.start
528
+ }
529
+ }
530
+ ```
531
+
532
+ ```plantuml
533
+ @startuml
534
+ !theme plain
535
+
536
+ rectangle "パターンマッチング" {
537
+ card "artist.yearsActive" as input
538
+
539
+ rectangle "switch (years._tag)" as match_block {
540
+ card "case 'StillActive' =>" as case1
541
+ card "case 'ActiveBetween' =>" as case2
542
+ }
543
+
544
+ card "結果" as result
545
+
546
+ input --> match_block
547
+ match_block --> result
548
+ }
549
+
550
+ note bottom
551
+ TypeScript は全てのケースを
552
+ 網羅していないと警告
553
+ end note
554
+
555
+ @enduml
556
+ ```
557
+
558
+ ### 7.7 検索条件のモデリング
559
+
560
+ 検索条件も ADT でモデリングできます。
561
+
562
+ ```typescript
563
+ type SearchCondition =
564
+ | { readonly _tag: 'SearchByGenre'; readonly genres: readonly MusicGenre[] }
565
+ | { readonly _tag: 'SearchByOrigin'; readonly locations: readonly Location[] }
566
+ | { readonly _tag: 'SearchByActiveYears'; readonly start: number; readonly end: number }
567
+
568
+ const searchArtists = (
569
+ artists: readonly Artist[],
570
+ conditions: readonly SearchCondition[]
571
+ ): readonly Artist[] =>
572
+ pipe(
573
+ artists,
574
+ RA.filter((artist) =>
575
+ conditions.every((condition) => matchesCondition(artist, condition))
576
+ )
577
+ )
578
+
579
+ // 使用例
580
+ searchArtists(artists, [
581
+ searchByGenre(['HeavyMetal']),
582
+ searchByOrigin(['US'])
583
+ ])
584
+ ```
585
+
586
+ ### 7.8 Either を使ったバリデーション
587
+
588
+ ```typescript
589
+ const validateAge = (age: number): E.Either<string, number> => {
590
+ if (age < 0) return E.left('Age cannot be negative')
591
+ if (age > 150) return E.left('Age cannot be greater than 150')
592
+ return E.right(age)
593
+ }
594
+
595
+ const validateUsername = (username: string): E.Either<string, string> => {
596
+ if (username.length === 0) return E.left('Username cannot be empty')
597
+ if (username.length < 3) return E.left('Username must be at least 3 characters')
598
+ return E.right(username)
599
+ }
600
+
601
+ // 複数のバリデーションを組み合わせる
602
+ const validateUser = (
603
+ username: string,
604
+ email: string,
605
+ age: number
606
+ ): E.Either<string, User> =>
607
+ pipe(
608
+ E.Do,
609
+ E.bind('username', () => validateUsername(username)),
610
+ E.bind('email', () => validateEmail(email)),
611
+ E.bind('age', () => validateAge(age))
612
+ )
613
+
614
+ validateUser('alice', 'alice@example.com', 25)
615
+ // right({ username: 'alice', email: 'alice@example.com', age: 25 })
616
+
617
+ validateUser('', 'alice@example.com', 25)
618
+ // left('Username cannot be empty')
619
+ ```
620
+
621
+ ### 7.9 支払い方法の例(ADT)
622
+
623
+ ```typescript
624
+ type PaymentMethod =
625
+ | { readonly _tag: 'CreditCard'; readonly number: string; readonly expiry: string }
626
+ | { readonly _tag: 'BankTransfer'; readonly accountNumber: string }
627
+ | { readonly _tag: 'Cash' }
628
+
629
+ const describePayment = (method: PaymentMethod): string => {
630
+ switch (method._tag) {
631
+ case 'CreditCard':
632
+ return `Credit card ending in ${method.number}`
633
+ case 'BankTransfer':
634
+ return `Bank transfer to account ${method.accountNumber}`
635
+ case 'Cash':
636
+ return 'Cash payment'
637
+ }
638
+ }
639
+
640
+ describePayment(creditCard('1234', '12/25'))
641
+ // "Credit card ending in 1234"
642
+ ```
643
+
644
+ ---
645
+
646
+ ## まとめ
647
+
648
+ ### Part III で学んだこと
649
+
650
+ ```plantuml
651
+ @startuml
652
+ !theme plain
653
+
654
+ rectangle "Part III: エラーハンドリング" {
655
+ rectangle "第6章" as ch6 {
656
+ card "Option<A>"
657
+ card "some / none"
658
+ card "orElse"
659
+ card "O.Do による合成"
660
+ card "エラーハンドリング戦略"
661
+ }
662
+
663
+ rectangle "第7章" as ch7 {
664
+ card "Either<E, A>"
665
+ card "right / left"
666
+ card "エラーメッセージの保持"
667
+ card "ADT(代数的データ型)"
668
+ card "パターンマッチング"
669
+ }
670
+ }
671
+
672
+ ch6 --> ch7
673
+
674
+ @enduml
675
+ ```
676
+
677
+ ### Option vs Either の使い分け
678
+
679
+ | 状況 | 使用する型 |
680
+ |------|------------|
681
+ | 値があるかないかだけが重要 | `Option<A>` |
682
+ | 失敗理由を伝える必要がある | `Either<E, A>` |
683
+ | 検索結果が見つからない | `Option<A>` |
684
+ | バリデーションエラーを伝える | `Either<string, A>` |
685
+ | 複数のエラー種別がある | `Either<ErrorType, A>` |
686
+
687
+ ### Scala vs TypeScript fp-ts 比較
688
+
689
+ | Scala | TypeScript (fp-ts) | 説明 |
690
+ |-------|-------------------|------|
691
+ | `Some(value)` | `O.some(value)` | Option の some |
692
+ | `None` | `O.none` | Option の none |
693
+ | `Right(value)` | `E.right(value)` | Either の right |
694
+ | `Left(error)` | `E.left(error)` | Either の left |
695
+ | `option.map(f)` | `pipe(option, O.map(f))` | Option の map |
696
+ | `either.flatMap(f)` | `pipe(either, E.chain(f))` | Either の flatMap |
697
+ | `option.toRight(error)` | `E.fromOption(() => error)(option)` | Option → Either |
698
+ | `enum` | discriminated union | 直和型 |
699
+ | `case class` | `interface` | 直積型 |
700
+ | `match { case ... }` | `switch (x._tag)` | パターンマッチング |
701
+
702
+ ### キーポイント
703
+
704
+ 1. **Option**: 値の有無を型で表現する
705
+ 2. **Either**: 成功/失敗とエラー情報を型で表現する
706
+ 3. **O.Do / E.Do**: Option/Either を組み合わせて使う
707
+ 4. **orElse**: フォールバックを提供する
708
+ 5. **ADT**: 直積型と直和型でドメインを正確にモデリング
709
+ 6. **discriminated union**: TypeScript で直和型を表現する方法
710
+ 7. **パターンマッチング**: switch と _tag で安全に処理する
711
+
712
+ ### 次のステップ
713
+
714
+ Part IV では、以下のトピックを学びます:
715
+
716
+ - IO モナドの導入
717
+ - 副作用の管理
718
+ - Task による非同期処理
719
+
720
+ ---
721
+
722
+ ## 演習問題
723
+
724
+ ### 問題 1: Option の基本
725
+
726
+ 以下の関数を実装してください。
727
+
728
+ ```typescript
729
+ const safeDivide = (a: number, b: number): O.Option<number> => ???
730
+
731
+ // 期待される動作
732
+ // safeDivide(10, 2) === some(5)
733
+ // safeDivide(10, 0) === none
734
+ // safeDivide(7, 2) === some(3)
735
+ ```
736
+
737
+ <details>
738
+ <summary>解答</summary>
739
+
740
+ ```typescript
741
+ import * as O from 'fp-ts/Option'
742
+
743
+ const safeDivide = (a: number, b: number): O.Option<number> =>
744
+ b === 0 ? O.none : O.some(Math.floor(a / b))
745
+ ```
746
+
747
+ </details>
748
+
749
+ ### 問題 2: Option の合成
750
+
751
+ 以下の関数を実装してください。2つの数値文字列を受け取り、その合計を返します。
752
+
753
+ ```typescript
754
+ const addStrings = (a: string, b: string): O.Option<number> => ???
755
+
756
+ // 期待される動作
757
+ // addStrings("10", "20") === some(30)
758
+ // addStrings("10", "abc") === none
759
+ // addStrings("xyz", "20") === none
760
+ ```
761
+
762
+ <details>
763
+ <summary>解答</summary>
764
+
765
+ ```typescript
766
+ import { pipe } from 'fp-ts/function'
767
+ import * as O from 'fp-ts/Option'
768
+
769
+ const addStrings = (a: string, b: string): O.Option<number> =>
770
+ pipe(
771
+ O.Do,
772
+ O.bind('x', () => parseNumber(a)),
773
+ O.bind('y', () => parseNumber(b)),
774
+ O.map(({ x, y }) => x + y)
775
+ )
776
+ ```
777
+
778
+ </details>
779
+
780
+ ### 問題 3: Either によるバリデーション
781
+
782
+ 以下の関数を実装してください。年齢を検証し、エラーメッセージを返します。
783
+
784
+ ```typescript
785
+ const validateAge = (age: number): E.Either<string, number> => ???
786
+
787
+ // 期待される動作
788
+ // validateAge(25) === right(25)
789
+ // validateAge(-5) === left("Age cannot be negative")
790
+ // validateAge(200) === left("Age cannot be greater than 150")
791
+ ```
792
+
793
+ <details>
794
+ <summary>解答</summary>
795
+
796
+ ```typescript
797
+ import * as E from 'fp-ts/Either'
798
+
799
+ const validateAge = (age: number): E.Either<string, number> => {
800
+ if (age < 0) return E.left('Age cannot be negative')
801
+ if (age > 150) return E.left('Age cannot be greater than 150')
802
+ return E.right(age)
803
+ }
804
+ ```
805
+
806
+ </details>
807
+
808
+ ### 問題 4: パターンマッチング
809
+
810
+ 以下の discriminated union とパターンマッチングを使った関数を実装してください。
811
+
812
+ ```typescript
813
+ type PaymentMethod =
814
+ | { readonly _tag: 'CreditCard'; readonly number: string; readonly expiry: string }
815
+ | { readonly _tag: 'BankTransfer'; readonly accountNumber: string }
816
+ | { readonly _tag: 'Cash' }
817
+
818
+ const describePayment = (method: PaymentMethod): string => ???
819
+
820
+ // 期待される動作
821
+ // describePayment(creditCard("1234", "12/25")) === "Credit card ending in 1234"
822
+ // describePayment(bankTransfer("9876")) === "Bank transfer to account 9876"
823
+ // describePayment(cash()) === "Cash payment"
824
+ ```
825
+
826
+ <details>
827
+ <summary>解答</summary>
828
+
829
+ ```typescript
830
+ const describePayment = (method: PaymentMethod): string => {
831
+ switch (method._tag) {
832
+ case 'CreditCard':
833
+ return `Credit card ending in ${method.number}`
834
+ case 'BankTransfer':
835
+ return `Bank transfer to account ${method.accountNumber}`
836
+ case 'Cash':
837
+ return 'Cash payment'
838
+ }
839
+ }
840
+ ```
841
+
842
+ </details>