@k2works/claude-code-booster 3.4.1 → 3.6.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.
- package/LICENSE +21 -21
- package/README.md +42 -42
- package/bin/claude-code-booster +90 -90
- package/lib/assets/.claude/README.md +239 -239
- package/lib/assets/.claude/scripts/generate-inception-deck.mjs +911 -911
- package/lib/assets/.claude/settings.json +11 -11
- package/lib/assets/.claude/skills/ai-agent-guidelines/SKILL.md +111 -111
- package/lib/assets/.claude/skills/analyzing-architecture/SKILL.md +83 -83
- package/lib/assets/.claude/skills/analyzing-business/SKILL.md +95 -95
- package/lib/assets/.claude/skills/analyzing-data-model/SKILL.md +77 -77
- package/lib/assets/.claude/skills/analyzing-domain-model/SKILL.md +117 -88
- package/lib/assets/.claude/skills/analyzing-inception-deck/SKILL.md +84 -84
- package/lib/assets/.claude/skills/analyzing-non-functional/SKILL.md +95 -95
- package/lib/assets/.claude/skills/analyzing-operation/SKILL.md +95 -95
- package/lib/assets/.claude/skills/analyzing-requirements/SKILL.md +91 -91
- package/lib/assets/.claude/skills/analyzing-tech-stack/SKILL.md +101 -101
- package/lib/assets/.claude/skills/analyzing-test-strategy/SKILL.md +89 -89
- package/lib/assets/.claude/skills/analyzing-ui-design/SKILL.md +80 -80
- package/lib/assets/.claude/skills/analyzing-usecases/SKILL.md +72 -72
- package/lib/assets/.claude/skills/creating-adr/SKILL.md +113 -113
- package/lib/assets/.claude/skills/developing-backend/SKILL.md +100 -100
- package/lib/assets/.claude/skills/developing-frontend/SKILL.md +93 -93
- package/lib/assets/.claude/skills/developing-release/SKILL.md +120 -120
- package/lib/assets/.claude/skills/generating-slides/SKILL.md +94 -94
- package/lib/assets/.claude/skills/git-commit/SKILL.md +81 -81
- package/lib/assets/.claude/skills/killing-processes/SKILL.md +44 -44
- package/lib/assets/.claude/skills/operating-backup/SKILL.md +59 -59
- package/lib/assets/.claude/skills/operating-cicd/SKILL.md +54 -54
- package/lib/assets/.claude/skills/operating-deploy/SKILL.md +67 -67
- package/lib/assets/.claude/skills/operating-docs/SKILL.md +219 -219
- package/lib/assets/.claude/skills/operating-provision/SKILL.md +77 -77
- package/lib/assets/.claude/skills/operating-setup/SKILL.md +63 -63
- package/lib/assets/.claude/skills/orchestrating-analysis/SKILL.md +104 -104
- package/lib/assets/.claude/skills/orchestrating-development/SKILL.md +162 -161
- package/lib/assets/.claude/skills/orchestrating-operation/SKILL.md +158 -158
- package/lib/assets/.claude/skills/orchestrating-project/SKILL.md +144 -144
- package/lib/assets/.claude/skills/planning-releases/SKILL.md +119 -119
- package/lib/assets/.claude/skills/syncing-github-project/SKILL.md +151 -151
- package/lib/assets/.claude/skills/tracking-progress/SKILL.md +91 -91
- package/lib/assets/.claude/skills/validating-iteration-plan/SKILL.md +29 -1
- package/lib/assets/.devcontainer/devcontainer.json +34 -34
- package/lib/assets/.env.example +17 -17
- package/lib/assets/.gitattributes +4 -4
- package/lib/assets/.github/workflows/docker-publish.yml +77 -77
- package/lib/assets/.github/workflows/mkdocs.yml +39 -39
- package/lib/assets/AGENTS.md +94 -94
- package/lib/assets/CLAUDE.md +183 -183
- package/lib/assets/README.md +254 -254
- package/lib/assets/docker-compose.yml +33 -33
- package/lib/assets/docs/adr/index.md +10 -10
- package/lib/assets/docs/article/functional-desgin-ppp/all/01-immutability-and-data-transformation.md +475 -475
- package/lib/assets/docs/article/functional-desgin-ppp/all/02-function-composition.md +519 -519
- package/lib/assets/docs/article/functional-desgin-ppp/all/03-polymorphism.md +537 -537
- package/lib/assets/docs/article/functional-desgin-ppp/all/04-data-validation.md +300 -300
- package/lib/assets/docs/article/functional-desgin-ppp/all/05-property-based-testing.md +320 -320
- package/lib/assets/docs/article/functional-desgin-ppp/all/06-tdd-and-functional.md +498 -498
- package/lib/assets/docs/article/functional-desgin-ppp/all/07-composite-pattern.md +298 -298
- package/lib/assets/docs/article/functional-desgin-ppp/all/08-decorator-pattern.md +291 -291
- package/lib/assets/docs/article/functional-desgin-ppp/all/09-adapter-pattern.md +336 -336
- package/lib/assets/docs/article/functional-desgin-ppp/all/10-strategy-pattern.md +303 -303
- package/lib/assets/docs/article/functional-desgin-ppp/all/11-command-pattern.md +286 -286
- package/lib/assets/docs/article/functional-desgin-ppp/all/12-visitor-pattern.md +322 -322
- package/lib/assets/docs/article/functional-desgin-ppp/all/13-abstract-factory-pattern.md +319 -319
- package/lib/assets/docs/article/functional-desgin-ppp/all/14-abstract-server-pattern.md +365 -365
- package/lib/assets/docs/article/functional-desgin-ppp/all/15-gossiping-bus-drivers.md +156 -156
- package/lib/assets/docs/article/functional-desgin-ppp/all/16-payroll-system.md +178 -178
- package/lib/assets/docs/article/functional-desgin-ppp/all/17-video-rental-system.md +312 -312
- package/lib/assets/docs/article/functional-desgin-ppp/all/18-concurrency-system.md +287 -287
- package/lib/assets/docs/article/functional-desgin-ppp/all/19-wa-tor-simulation.md +286 -286
- package/lib/assets/docs/article/functional-desgin-ppp/all/20-pattern-interactions.md +274 -274
- package/lib/assets/docs/article/functional-desgin-ppp/all/21-best-practices.md +294 -294
- package/lib/assets/docs/article/functional-desgin-ppp/all/22-oo-to-fp-migration.md +337 -337
- package/lib/assets/docs/article/functional-desgin-ppp/all/index.md +388 -388
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/01-immutability-and-data-transformation.md +273 -273
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/02-function-composition.md +380 -380
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/03-polymorphism.md +384 -384
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/04-clojure-spec.md +350 -350
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/05-property-based-testing.md +352 -352
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/06-tdd-in-functional.md +383 -383
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/07-composite-pattern.md +529 -529
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/08-decorator-pattern.md +395 -395
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/09-adapter-pattern.md +399 -399
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/10-strategy-pattern.md +485 -485
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/11-command-pattern.md +566 -566
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/12-visitor-pattern.md +567 -567
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/13-abstract-factory-pattern.md +475 -475
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/14-abstract-server-pattern.md +462 -462
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/15-gossiping-bus-drivers.md +325 -325
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/16-payroll-system.md +401 -401
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/17-video-rental-system.md +450 -450
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/18-concurrency-system.md +475 -475
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/19-wator-simulation.md +739 -739
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/20-pattern-interactions.md +567 -567
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/21-best-practices.md +518 -518
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/22-oo-to-fp-migration.md +532 -532
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/index.md +241 -241
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/01-immutability-and-data-transformation.md +383 -383
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/02-function-composition.md +374 -374
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/03-polymorphism.md +375 -375
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/04-data-validation.md +195 -195
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/05-property-based-testing.md +268 -268
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/06-tdd-and-fp.md +294 -294
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/07-effects-and-pure-functions.md +164 -164
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/08-error-handling-strategies.md +168 -168
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/09-io-and-external-systems.md +254 -254
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/10-concurrency-patterns.md +269 -269
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/11-command-pattern.md +148 -148
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/12-visitor-pattern.md +176 -176
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/13-abstract-factory-pattern.md +604 -604
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/14-abstract-server-pattern.md +729 -729
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/15-gossiping-bus-drivers.md +291 -291
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/16-payroll-system.md +420 -420
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/17-video-rental-system.md +319 -319
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/18-concurrency-system.md +466 -466
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/19-wator-simulation.md +523 -523
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/20-pattern-interactions.md +287 -287
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/21-best-practices.md +340 -340
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/22-oo-to-fp-migration.md +395 -395
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/index.md +248 -248
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/01-immutability-and-data-transformation.md +384 -384
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/02-function-composition.md +452 -452
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/03-polymorphism.md +495 -495
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/04-data-validation.md +416 -416
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/05-property-based-testing.md +382 -382
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/06-tdd-functional.md +687 -687
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/07-composite-pattern.md +442 -442
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/08-decorator-pattern.md +479 -479
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/09-adapter-pattern.md +479 -479
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/10-strategy-pattern.md +427 -427
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/11-command-pattern.md +428 -428
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/12-visitor-pattern.md +339 -339
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/13-abstract-factory-pattern.md +309 -309
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/14-abstract-server-pattern.md +596 -596
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/15-gossiping-bus-drivers.md +355 -355
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/16-payroll-system.md +350 -350
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/17-video-rental-system.md +414 -414
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/18-concurrency-system.md +367 -367
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/19-wator-simulation.md +403 -403
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/20-pattern-interactions.md +291 -291
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/21-best-practices.md +324 -324
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/22-oo-to-fp-migration.md +332 -332
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/index.md +274 -274
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/01-immutability-and-data-transformation.md +298 -298
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/02-function-composition.md +304 -304
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/03-polymorphism.md +362 -362
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/04-data-validation.md +257 -257
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/05-property-based-testing.md +254 -254
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/06-tdd-functional.md +283 -283
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/07-composite-pattern.md +395 -395
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/08-decorator-pattern.md +319 -319
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/09-adapter-pattern.md +382 -382
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/10-strategy-pattern.md +287 -287
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/11-command-pattern.md +303 -303
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/12-visitor-pattern.md +326 -326
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/13-abstract-factory-pattern.md +332 -332
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/14-abstract-server-pattern.md +379 -379
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/15-gossiping-bus-drivers.md +177 -177
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/16-payroll-system.md +219 -219
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/17-video-rental-system.md +244 -244
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/18-concurrency-system.md +363 -363
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/19-wator-simulation.md +438 -438
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/20-pattern-interactions.md +325 -325
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/21-best-practices.md +403 -403
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/22-oo-to-fp-migration.md +469 -469
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/index.md +174 -174
- package/lib/assets/docs/article/functional-desgin-ppp/index.md +90 -90
- package/lib/assets/docs/article/functional-desgin-ppp/rust/01-immutability-and-data-transformation.md +450 -450
- package/lib/assets/docs/article/functional-desgin-ppp/rust/02-function-composition.md +463 -463
- package/lib/assets/docs/article/functional-desgin-ppp/rust/03-polymorphism.md +425 -425
- package/lib/assets/docs/article/functional-desgin-ppp/rust/04-data-validation.md +273 -273
- package/lib/assets/docs/article/functional-desgin-ppp/rust/05-property-based-testing.md +247 -247
- package/lib/assets/docs/article/functional-desgin-ppp/rust/06-tdd-and-functional.md +841 -841
- package/lib/assets/docs/article/functional-desgin-ppp/rust/07-composite-pattern.md +384 -384
- package/lib/assets/docs/article/functional-desgin-ppp/rust/08-decorator-pattern.md +383 -383
- package/lib/assets/docs/article/functional-desgin-ppp/rust/09-adapter-pattern.md +339 -339
- package/lib/assets/docs/article/functional-desgin-ppp/rust/10-strategy-pattern.md +331 -331
- package/lib/assets/docs/article/functional-desgin-ppp/rust/11-command-pattern.md +356 -356
- package/lib/assets/docs/article/functional-desgin-ppp/rust/12-visitor-pattern.md +379 -379
- package/lib/assets/docs/article/functional-desgin-ppp/rust/13-abstract-factory-pattern.md +361 -361
- package/lib/assets/docs/article/functional-desgin-ppp/rust/14-abstract-server-pattern.md +392 -392
- package/lib/assets/docs/article/functional-desgin-ppp/rust/15-gossiping-bus-drivers.md +300 -300
- package/lib/assets/docs/article/functional-desgin-ppp/rust/16-payroll-system.md +297 -297
- package/lib/assets/docs/article/functional-desgin-ppp/rust/17-video-rental-system.md +304 -304
- package/lib/assets/docs/article/functional-desgin-ppp/rust/18-concurrency-system.md +315 -315
- package/lib/assets/docs/article/functional-desgin-ppp/rust/19-wator-simulation.md +311 -311
- package/lib/assets/docs/article/functional-desgin-ppp/rust/20-pattern-interactions.md +304 -304
- package/lib/assets/docs/article/functional-desgin-ppp/rust/21-best-practices.md +336 -336
- package/lib/assets/docs/article/functional-desgin-ppp/rust/22-oo-to-fp-migration.md +349 -349
- package/lib/assets/docs/article/functional-desgin-ppp/rust/index.md +243 -243
- package/lib/assets/docs/article/functional-desgin-ppp/scala/01-immutability-and-data-transformation.md +328 -328
- package/lib/assets/docs/article/functional-desgin-ppp/scala/02-function-composition.md +348 -348
- package/lib/assets/docs/article/functional-desgin-ppp/scala/03-polymorphism.md +357 -357
- package/lib/assets/docs/article/functional-desgin-ppp/scala/04-data-validation.md +364 -364
- package/lib/assets/docs/article/functional-desgin-ppp/scala/05-property-based-testing.md +515 -515
- package/lib/assets/docs/article/functional-desgin-ppp/scala/06-tdd-functional.md +557 -557
- package/lib/assets/docs/article/functional-desgin-ppp/scala/07-composite-pattern.md +363 -363
- package/lib/assets/docs/article/functional-desgin-ppp/scala/08-decorator-pattern.md +327 -327
- package/lib/assets/docs/article/functional-desgin-ppp/scala/09-adapter-pattern.md +517 -517
- package/lib/assets/docs/article/functional-desgin-ppp/scala/10-strategy-pattern.md +441 -441
- package/lib/assets/docs/article/functional-desgin-ppp/scala/11-command-pattern.md +407 -407
- package/lib/assets/docs/article/functional-desgin-ppp/scala/12-visitor-pattern.md +379 -379
- package/lib/assets/docs/article/functional-desgin-ppp/scala/13-abstract-factory-pattern.md +398 -398
- package/lib/assets/docs/article/functional-desgin-ppp/scala/14-abstract-server-pattern.md +476 -476
- package/lib/assets/docs/article/functional-desgin-ppp/scala/15-gossiping-bus-drivers.md +391 -391
- package/lib/assets/docs/article/functional-desgin-ppp/scala/16-payroll-system.md +342 -342
- package/lib/assets/docs/article/functional-desgin-ppp/scala/17-video-rental-system.md +324 -324
- package/lib/assets/docs/article/functional-desgin-ppp/scala/18-concurrency-system.md +730 -730
- package/lib/assets/docs/article/functional-desgin-ppp/scala/19-wator-simulation.md +624 -624
- package/lib/assets/docs/article/functional-desgin-ppp/scala/20-pattern-interactions.md +512 -512
- package/lib/assets/docs/article/functional-desgin-ppp/scala/21-best-practices.md +433 -433
- package/lib/assets/docs/article/functional-desgin-ppp/scala/22-oo-to-fp-migration.md +688 -688
- package/lib/assets/docs/article/functional-desgin-ppp/scala/index.md +243 -243
- package/lib/assets/docs/article/getting-start-tdd/clojure/01-todo-list-and-first-test.md +166 -166
- package/lib/assets/docs/article/getting-start-tdd/clojure/02-fake-it-and-triangulation.md +162 -162
- package/lib/assets/docs/article/getting-start-tdd/clojure/03-obvious-implementation-and-refactoring.md +135 -135
- package/lib/assets/docs/article/getting-start-tdd/clojure/04-version-control-and-conventional-commits.md +88 -88
- package/lib/assets/docs/article/getting-start-tdd/clojure/05-package-management-and-static-analysis.md +299 -299
- package/lib/assets/docs/article/getting-start-tdd/clojure/06-task-runner-and-ci-cd.md +241 -241
- package/lib/assets/docs/article/getting-start-tdd/clojure/07-protocols-and-records.md +131 -131
- package/lib/assets/docs/article/getting-start-tdd/clojure/08-multimethods-and-design-patterns.md +130 -130
- package/lib/assets/docs/article/getting-start-tdd/clojure/09-namespaces-and-module-design.md +127 -127
- package/lib/assets/docs/article/getting-start-tdd/clojure/10-higher-order-functions-and-composition.md +114 -114
- package/lib/assets/docs/article/getting-start-tdd/clojure/11-persistent-data-and-pipeline.md +138 -138
- package/lib/assets/docs/article/getting-start-tdd/clojure/12-error-handling-and-spec.md +161 -161
- package/lib/assets/docs/article/getting-start-tdd/clojure/index.md +65 -65
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter01.md +232 -232
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter02.md +244 -244
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter03.md +202 -202
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter04.md +92 -92
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter05.md +256 -256
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter06.md +195 -195
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter07.md +214 -214
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter08.md +249 -249
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter09.md +174 -174
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter10.md +166 -166
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter11.md +192 -192
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter12.md +211 -211
- package/lib/assets/docs/article/getting-start-tdd/csharp/index.md +83 -83
- package/lib/assets/docs/article/getting-start-tdd/elixir/01-todo-list-and-first-test.md +87 -87
- package/lib/assets/docs/article/getting-start-tdd/elixir/02-fake-it-and-triangulation.md +95 -95
- package/lib/assets/docs/article/getting-start-tdd/elixir/03-obvious-implementation-and-refactoring.md +109 -109
- package/lib/assets/docs/article/getting-start-tdd/elixir/04-version-control-and-conventional-commits.md +96 -96
- package/lib/assets/docs/article/getting-start-tdd/elixir/05-package-management-and-static-analysis.md +88 -88
- package/lib/assets/docs/article/getting-start-tdd/elixir/06-task-runner-and-ci-cd.md +71 -71
- package/lib/assets/docs/article/getting-start-tdd/elixir/07-structs-and-protocols.md +110 -110
- package/lib/assets/docs/article/getting-start-tdd/elixir/08-pattern-matching-and-guards.md +108 -108
- package/lib/assets/docs/article/getting-start-tdd/elixir/09-module-design-and-behaviours.md +104 -104
- package/lib/assets/docs/article/getting-start-tdd/elixir/10-higher-order-functions-and-pipeline.md +178 -178
- package/lib/assets/docs/article/getting-start-tdd/elixir/11-stream-and-lazy-evaluation.md +142 -142
- package/lib/assets/docs/article/getting-start-tdd/elixir/12-error-handling-and-with.md +145 -145
- package/lib/assets/docs/article/getting-start-tdd/elixir/index.md +35 -35
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter01.md +202 -202
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter02.md +246 -246
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter03.md +218 -218
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter04.md +179 -179
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter05.md +267 -267
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter06.md +190 -190
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter07.md +161 -161
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter08.md +175 -175
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter09.md +222 -222
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter10.md +189 -189
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter11.md +212 -212
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter12.md +215 -215
- package/lib/assets/docs/article/getting-start-tdd/fsharp/index.md +71 -71
- package/lib/assets/docs/article/getting-start-tdd/go/01-todo-list-and-first-test.md +213 -213
- package/lib/assets/docs/article/getting-start-tdd/go/02-fake-it-and-triangulation.md +302 -302
- package/lib/assets/docs/article/getting-start-tdd/go/03-obvious-implementation-and-refactoring.md +339 -339
- package/lib/assets/docs/article/getting-start-tdd/go/04-version-control-and-conventional-commits.md +112 -112
- package/lib/assets/docs/article/getting-start-tdd/go/05-package-management-and-static-analysis.md +272 -272
- package/lib/assets/docs/article/getting-start-tdd/go/06-task-runner-and-ci-cd.md +233 -233
- package/lib/assets/docs/article/getting-start-tdd/go/07-encapsulation-and-polymorphism.md +394 -394
- package/lib/assets/docs/article/getting-start-tdd/go/08-design-patterns.md +422 -422
- package/lib/assets/docs/article/getting-start-tdd/go/09-solid-principles-and-module-design.md +400 -400
- package/lib/assets/docs/article/getting-start-tdd/go/10-higher-order-functions-and-composition.md +226 -226
- package/lib/assets/docs/article/getting-start-tdd/go/11-immutable-data-and-pipeline.md +296 -296
- package/lib/assets/docs/article/getting-start-tdd/go/12-error-handling-and-type-safety.md +411 -411
- package/lib/assets/docs/article/getting-start-tdd/go/index.md +83 -83
- package/lib/assets/docs/article/getting-start-tdd/haskell/01-todo-list-and-first-test.md +279 -279
- package/lib/assets/docs/article/getting-start-tdd/haskell/02-fake-it-and-triangulation.md +337 -337
- package/lib/assets/docs/article/getting-start-tdd/haskell/03-obvious-implementation-and-refactoring.md +257 -257
- package/lib/assets/docs/article/getting-start-tdd/haskell/04-version-control-and-conventional-commits.md +182 -182
- package/lib/assets/docs/article/getting-start-tdd/haskell/05-package-management-and-static-analysis.md +313 -313
- package/lib/assets/docs/article/getting-start-tdd/haskell/06-task-runner-and-ci-cd.md +309 -309
- package/lib/assets/docs/article/getting-start-tdd/haskell/07-algebraic-data-types-and-type-classes.md +412 -412
- package/lib/assets/docs/article/getting-start-tdd/haskell/08-pattern-matching-and-guards.md +390 -390
- package/lib/assets/docs/article/getting-start-tdd/haskell/09-module-design-and-smart-constructors.md +461 -461
- package/lib/assets/docs/article/getting-start-tdd/haskell/10-higher-order-functions-and-currying.md +434 -434
- package/lib/assets/docs/article/getting-start-tdd/haskell/11-function-composition-and-point-free.md +392 -392
- package/lib/assets/docs/article/getting-start-tdd/haskell/12-monad-and-error-handling.md +631 -631
- package/lib/assets/docs/article/getting-start-tdd/haskell/index.md +49 -49
- package/lib/assets/docs/article/getting-start-tdd/index.md +93 -93
- package/lib/assets/docs/article/getting-start-tdd/integration/01-language-overview.md +375 -375
- package/lib/assets/docs/article/getting-start-tdd/integration/02-test-framework-comparison.md +349 -349
- package/lib/assets/docs/article/getting-start-tdd/integration/03-tdd-pattern-comparison.md +445 -445
- package/lib/assets/docs/article/getting-start-tdd/integration/04-type-system-comparison.md +409 -409
- package/lib/assets/docs/article/getting-start-tdd/integration/05-dev-environment-comparison.md +330 -330
- package/lib/assets/docs/article/getting-start-tdd/integration/06-learning-roadmap.md +290 -290
- package/lib/assets/docs/article/getting-start-tdd/integration/index.md +69 -69
- package/lib/assets/docs/article/getting-start-tdd/java/01-todo-list-and-first-test.md +234 -234
- package/lib/assets/docs/article/getting-start-tdd/java/02-fake-it-and-triangulation.md +261 -261
- package/lib/assets/docs/article/getting-start-tdd/java/03-obvious-implementation-and-refactoring.md +185 -185
- package/lib/assets/docs/article/getting-start-tdd/java/04-version-control-and-conventional-commits.md +115 -115
- package/lib/assets/docs/article/getting-start-tdd/java/05-package-management-and-static-analysis.md +382 -382
- package/lib/assets/docs/article/getting-start-tdd/java/06-task-runner-and-ci-cd.md +272 -272
- package/lib/assets/docs/article/getting-start-tdd/java/07-encapsulation-and-polymorphism.md +626 -626
- package/lib/assets/docs/article/getting-start-tdd/java/08-design-patterns.md +393 -393
- package/lib/assets/docs/article/getting-start-tdd/java/09-solid-principles-and-module-design.md +310 -310
- package/lib/assets/docs/article/getting-start-tdd/java/10-higher-order-functions-and-composition.md +188 -188
- package/lib/assets/docs/article/getting-start-tdd/java/11-immutable-data-and-pipeline.md +167 -167
- package/lib/assets/docs/article/getting-start-tdd/java/12-error-handling-and-type-safety.md +205 -205
- package/lib/assets/docs/article/getting-start-tdd/java/index.md +61 -61
- package/lib/assets/docs/article/getting-start-tdd/node/01-todo-list-and-first-test.md +244 -244
- package/lib/assets/docs/article/getting-start-tdd/node/02-fake-it-and-triangulation.md +262 -262
- package/lib/assets/docs/article/getting-start-tdd/node/03-obvious-implementation-and-refactoring.md +169 -169
- package/lib/assets/docs/article/getting-start-tdd/node/04-version-control-and-conventional-commits.md +112 -112
- package/lib/assets/docs/article/getting-start-tdd/node/05-package-management-and-static-analysis.md +314 -314
- package/lib/assets/docs/article/getting-start-tdd/node/06-task-runner-and-ci-cd.md +235 -235
- package/lib/assets/docs/article/getting-start-tdd/node/07-encapsulation-and-polymorphism.md +327 -327
- package/lib/assets/docs/article/getting-start-tdd/node/08-design-patterns.md +322 -322
- package/lib/assets/docs/article/getting-start-tdd/node/09-solid-principles-and-module-design.md +285 -285
- package/lib/assets/docs/article/getting-start-tdd/node/10-higher-order-functions-and-composition.md +199 -199
- package/lib/assets/docs/article/getting-start-tdd/node/11-immutable-data-and-pipeline.md +207 -207
- package/lib/assets/docs/article/getting-start-tdd/node/12-error-handling-and-type-safety.md +295 -295
- package/lib/assets/docs/article/getting-start-tdd/node/index.md +56 -56
- package/lib/assets/docs/article/getting-start-tdd/php/01-todo-list-and-first-test.md +259 -259
- package/lib/assets/docs/article/getting-start-tdd/php/02-fake-it-and-triangulation.md +200 -200
- package/lib/assets/docs/article/getting-start-tdd/php/03-obvious-implementation-and-refactoring.md +248 -248
- package/lib/assets/docs/article/getting-start-tdd/php/04-version-control-and-conventional-commits.md +141 -141
- package/lib/assets/docs/article/getting-start-tdd/php/05-package-management-and-static-analysis.md +410 -410
- package/lib/assets/docs/article/getting-start-tdd/php/06-task-runner-and-ci-cd.md +321 -321
- package/lib/assets/docs/article/getting-start-tdd/php/07-encapsulation-and-polymorphism.md +372 -372
- package/lib/assets/docs/article/getting-start-tdd/php/08-design-patterns.md +453 -453
- package/lib/assets/docs/article/getting-start-tdd/php/09-solid-principles-and-module-design.md +460 -460
- package/lib/assets/docs/article/getting-start-tdd/php/10-higher-order-functions-and-composition.md +182 -182
- package/lib/assets/docs/article/getting-start-tdd/php/11-immutable-data-and-pipeline.md +266 -266
- package/lib/assets/docs/article/getting-start-tdd/php/12-error-handling-and-type-safety.md +308 -308
- package/lib/assets/docs/article/getting-start-tdd/php/index.md +84 -84
- package/lib/assets/docs/article/getting-start-tdd/python/01-todo-list-and-first-test.md +201 -201
- package/lib/assets/docs/article/getting-start-tdd/python/02-fake-it-and-triangulation.md +247 -247
- package/lib/assets/docs/article/getting-start-tdd/python/03-obvious-implementation-and-refactoring.md +199 -199
- package/lib/assets/docs/article/getting-start-tdd/python/04-version-control-and-conventional-commits.md +87 -87
- package/lib/assets/docs/article/getting-start-tdd/python/05-package-management-and-static-analysis.md +274 -274
- package/lib/assets/docs/article/getting-start-tdd/python/06-task-runner-and-ci-cd.md +190 -190
- package/lib/assets/docs/article/getting-start-tdd/python/07-encapsulation-and-polymorphism.md +208 -208
- package/lib/assets/docs/article/getting-start-tdd/python/08-design-patterns.md +172 -172
- package/lib/assets/docs/article/getting-start-tdd/python/09-solid-principles-and-module-design.md +130 -130
- package/lib/assets/docs/article/getting-start-tdd/python/10-higher-order-functions-and-composition.md +122 -122
- package/lib/assets/docs/article/getting-start-tdd/python/11-immutable-data-and-pipeline.md +116 -116
- package/lib/assets/docs/article/getting-start-tdd/python/12-error-handling-and-type-safety.md +126 -126
- package/lib/assets/docs/article/getting-start-tdd/python/index.md +55 -55
- package/lib/assets/docs/article/getting-start-tdd/ruby/01-todo-list-and-first-test.md +231 -231
- package/lib/assets/docs/article/getting-start-tdd/ruby/02-fake-it-and-triangulation.md +238 -238
- package/lib/assets/docs/article/getting-start-tdd/ruby/03-obvious-implementation-and-refactoring.md +228 -228
- package/lib/assets/docs/article/getting-start-tdd/ruby/04-version-control-and-conventional-commits.md +112 -112
- package/lib/assets/docs/article/getting-start-tdd/ruby/05-package-management-and-static-analysis.md +287 -287
- package/lib/assets/docs/article/getting-start-tdd/ruby/06-task-runner-and-ci-cd.md +248 -248
- package/lib/assets/docs/article/getting-start-tdd/ruby/07-encapsulation-and-polymorphism.md +279 -279
- package/lib/assets/docs/article/getting-start-tdd/ruby/08-design-patterns.md +329 -329
- package/lib/assets/docs/article/getting-start-tdd/ruby/09-solid-principles-and-module-design.md +196 -196
- package/lib/assets/docs/article/getting-start-tdd/ruby/10-higher-order-functions-and-composition.md +175 -175
- package/lib/assets/docs/article/getting-start-tdd/ruby/11-immutable-data-and-pipeline.md +237 -237
- package/lib/assets/docs/article/getting-start-tdd/ruby/12-error-handling-and-type-safety.md +398 -398
- package/lib/assets/docs/article/getting-start-tdd/ruby/index.md +83 -83
- package/lib/assets/docs/article/getting-start-tdd/rust/01-todo-list-and-first-test.md +211 -211
- package/lib/assets/docs/article/getting-start-tdd/rust/02-fake-it-and-triangulation.md +264 -264
- package/lib/assets/docs/article/getting-start-tdd/rust/03-obvious-implementation-and-refactoring.md +233 -233
- package/lib/assets/docs/article/getting-start-tdd/rust/04-version-control-and-conventional-commits.md +92 -92
- package/lib/assets/docs/article/getting-start-tdd/rust/05-package-management-and-static-analysis.md +212 -212
- package/lib/assets/docs/article/getting-start-tdd/rust/06-task-runner-and-ci-cd.md +164 -164
- package/lib/assets/docs/article/getting-start-tdd/rust/07-encapsulation-and-polymorphism.md +142 -142
- package/lib/assets/docs/article/getting-start-tdd/rust/08-design-patterns.md +145 -145
- package/lib/assets/docs/article/getting-start-tdd/rust/09-solid-principles-and-module-design.md +110 -110
- package/lib/assets/docs/article/getting-start-tdd/rust/10-higher-order-functions-and-composition.md +94 -94
- package/lib/assets/docs/article/getting-start-tdd/rust/11-immutable-data-and-pipeline.md +105 -105
- package/lib/assets/docs/article/getting-start-tdd/rust/12-error-handling-and-type-safety.md +112 -112
- package/lib/assets/docs/article/getting-start-tdd/rust/index.md +83 -83
- package/lib/assets/docs/article/getting-start-tdd/scala/01-todo-list-and-first-test.md +111 -111
- package/lib/assets/docs/article/getting-start-tdd/scala/02-fake-it-and-triangulation.md +107 -107
- package/lib/assets/docs/article/getting-start-tdd/scala/03-obvious-implementation-and-refactoring.md +99 -99
- package/lib/assets/docs/article/getting-start-tdd/scala/04-version-control-and-conventional-commits.md +123 -123
- package/lib/assets/docs/article/getting-start-tdd/scala/05-package-management-and-static-analysis.md +196 -196
- package/lib/assets/docs/article/getting-start-tdd/scala/06-task-runner-and-ci-cd.md +186 -186
- package/lib/assets/docs/article/getting-start-tdd/scala/07-case-classes-and-traits.md +139 -139
- package/lib/assets/docs/article/getting-start-tdd/scala/08-pattern-matching-and-sealed-traits.md +106 -106
- package/lib/assets/docs/article/getting-start-tdd/scala/09-packages-and-module-design.md +75 -75
- package/lib/assets/docs/article/getting-start-tdd/scala/10-higher-order-functions-and-composition.md +104 -104
- package/lib/assets/docs/article/getting-start-tdd/scala/11-collections-and-lazy-evaluation.md +94 -94
- package/lib/assets/docs/article/getting-start-tdd/scala/12-error-handling-and-type-safety.md +92 -92
- package/lib/assets/docs/article/getting-start-tdd/scala/index.md +65 -65
- package/lib/assets/docs/article/grokking-concurrency/all/index.md +404 -404
- package/lib/assets/docs/article/grokking-concurrency/all/part-1-ch02-sequential.md +554 -554
- package/lib/assets/docs/article/grokking-concurrency/all/part-2-ch04-05-threads.md +469 -469
- package/lib/assets/docs/article/grokking-concurrency/all/part-3-ch06-multitasking.md +520 -520
- package/lib/assets/docs/article/grokking-concurrency/all/part-4-ch07-parallel-patterns.md +420 -420
- package/lib/assets/docs/article/grokking-concurrency/all/part-5-ch08-09-synchronization.md +510 -510
- package/lib/assets/docs/article/grokking-concurrency/all/part-6-ch10-11-nonblocking-io.md +435 -435
- package/lib/assets/docs/article/grokking-concurrency/all/part-7-ch12-async.md +465 -465
- package/lib/assets/docs/article/grokking-concurrency/all/part-8-ch13-mapreduce.md +377 -377
- package/lib/assets/docs/article/grokking-concurrency/clojure/index.md +116 -116
- package/lib/assets/docs/article/grokking-concurrency/clojure/part-1.md +108 -108
- package/lib/assets/docs/article/grokking-concurrency/clojure/part-2.md +101 -101
- package/lib/assets/docs/article/grokking-concurrency/clojure/part-3.md +122 -122
- package/lib/assets/docs/article/grokking-concurrency/clojure/part-4.md +123 -123
- package/lib/assets/docs/article/grokking-concurrency/clojure/part-5.md +118 -118
- package/lib/assets/docs/article/grokking-concurrency/clojure/part-6.md +89 -89
- package/lib/assets/docs/article/grokking-concurrency/clojure/part-7.md +100 -100
- package/lib/assets/docs/article/grokking-concurrency/clojure/part-8.md +120 -120
- package/lib/assets/docs/article/grokking-concurrency/csharp/index.md +101 -101
- package/lib/assets/docs/article/grokking-concurrency/csharp/part-1.md +97 -97
- package/lib/assets/docs/article/grokking-concurrency/csharp/part-2.md +123 -123
- package/lib/assets/docs/article/grokking-concurrency/csharp/part-3.md +101 -101
- package/lib/assets/docs/article/grokking-concurrency/csharp/part-4.md +112 -112
- package/lib/assets/docs/article/grokking-concurrency/csharp/part-5.md +99 -99
- package/lib/assets/docs/article/grokking-concurrency/csharp/part-6.md +61 -61
- package/lib/assets/docs/article/grokking-concurrency/csharp/part-7.md +84 -84
- package/lib/assets/docs/article/grokking-concurrency/csharp/part-8.md +92 -92
- package/lib/assets/docs/article/grokking-concurrency/fsharp/index.md +65 -65
- package/lib/assets/docs/article/grokking-concurrency/fsharp/part-1.md +80 -80
- package/lib/assets/docs/article/grokking-concurrency/fsharp/part-2.md +103 -103
- package/lib/assets/docs/article/grokking-concurrency/fsharp/part-3.md +94 -94
- package/lib/assets/docs/article/grokking-concurrency/fsharp/part-4.md +110 -110
- package/lib/assets/docs/article/grokking-concurrency/fsharp/part-5.md +104 -104
- package/lib/assets/docs/article/grokking-concurrency/fsharp/part-6.md +93 -93
- package/lib/assets/docs/article/grokking-concurrency/fsharp/part-7.md +121 -121
- package/lib/assets/docs/article/grokking-concurrency/fsharp/part-8.md +107 -107
- package/lib/assets/docs/article/grokking-concurrency/haskell/index.md +248 -248
- package/lib/assets/docs/article/grokking-concurrency/haskell/part-1.md +96 -96
- package/lib/assets/docs/article/grokking-concurrency/haskell/part-2.md +96 -96
- package/lib/assets/docs/article/grokking-concurrency/haskell/part-3.md +91 -91
- package/lib/assets/docs/article/grokking-concurrency/haskell/part-4.md +106 -106
- package/lib/assets/docs/article/grokking-concurrency/haskell/part-5.md +99 -99
- package/lib/assets/docs/article/grokking-concurrency/haskell/part-6.md +95 -95
- package/lib/assets/docs/article/grokking-concurrency/haskell/part-7.md +111 -111
- package/lib/assets/docs/article/grokking-concurrency/haskell/part-8.md +118 -118
- package/lib/assets/docs/article/grokking-concurrency/index.md +66 -66
- package/lib/assets/docs/article/grokking-concurrency/java/index.md +102 -102
- package/lib/assets/docs/article/grokking-concurrency/java/part-1.md +308 -308
- package/lib/assets/docs/article/grokking-concurrency/java/part-2.md +334 -334
- package/lib/assets/docs/article/grokking-concurrency/java/part-3.md +221 -221
- package/lib/assets/docs/article/grokking-concurrency/java/part-4.md +213 -213
- package/lib/assets/docs/article/grokking-concurrency/java/part-5.md +112 -112
- package/lib/assets/docs/article/grokking-concurrency/java/part-6.md +69 -69
- package/lib/assets/docs/article/grokking-concurrency/java/part-7.md +101 -101
- package/lib/assets/docs/article/grokking-concurrency/java/part-8.md +101 -101
- package/lib/assets/docs/article/grokking-concurrency/python/index.md +313 -313
- package/lib/assets/docs/article/grokking-concurrency/python/part-1.md +239 -239
- package/lib/assets/docs/article/grokking-concurrency/python/part-2.md +418 -418
- package/lib/assets/docs/article/grokking-concurrency/python/part-3.md +227 -227
- package/lib/assets/docs/article/grokking-concurrency/python/part-4.md +299 -299
- package/lib/assets/docs/article/grokking-concurrency/python/part-5.md +315 -315
- package/lib/assets/docs/article/grokking-concurrency/python/part-6.md +297 -297
- package/lib/assets/docs/article/grokking-concurrency/python/part-7.md +314 -314
- package/lib/assets/docs/article/grokking-concurrency/python/part-8.md +360 -360
- package/lib/assets/docs/article/grokking-concurrency/rust/index.md +270 -270
- package/lib/assets/docs/article/grokking-concurrency/rust/part-1.md +108 -108
- package/lib/assets/docs/article/grokking-concurrency/rust/part-2.md +120 -120
- package/lib/assets/docs/article/grokking-concurrency/rust/part-3.md +126 -126
- package/lib/assets/docs/article/grokking-concurrency/rust/part-4.md +175 -175
- package/lib/assets/docs/article/grokking-concurrency/rust/part-5.md +158 -158
- package/lib/assets/docs/article/grokking-concurrency/rust/part-6.md +94 -94
- package/lib/assets/docs/article/grokking-concurrency/rust/part-7.md +133 -133
- package/lib/assets/docs/article/grokking-concurrency/rust/part-8.md +155 -155
- package/lib/assets/docs/article/grokking-concurrency/scala/index.md +69 -69
- package/lib/assets/docs/article/grokking-concurrency/scala/part-1.md +78 -78
- package/lib/assets/docs/article/grokking-concurrency/scala/part-2.md +112 -112
- package/lib/assets/docs/article/grokking-concurrency/scala/part-3.md +93 -93
- package/lib/assets/docs/article/grokking-concurrency/scala/part-4.md +110 -110
- package/lib/assets/docs/article/grokking-concurrency/scala/part-5.md +119 -119
- package/lib/assets/docs/article/grokking-concurrency/scala/part-6.md +83 -83
- package/lib/assets/docs/article/grokking-concurrency/scala/part-7.md +131 -131
- package/lib/assets/docs/article/grokking-concurrency/scala/part-8.md +129 -129
- package/lib/assets/docs/article/grokkingfp/all/index.md +368 -368
- package/lib/assets/docs/article/grokkingfp/all/part-1-ch01-fp-introduction.md +530 -530
- package/lib/assets/docs/article/grokkingfp/all/part-1-ch02-pure-functions.md +923 -923
- package/lib/assets/docs/article/grokkingfp/all/part-2-ch03-immutable-data.md +1128 -1128
- package/lib/assets/docs/article/grokkingfp/all/part-2-ch04-higher-order-functions.md +1104 -1104
- package/lib/assets/docs/article/grokkingfp/all/part-2-ch05-flatmap.md +1026 -1026
- package/lib/assets/docs/article/grokkingfp/all/part-3-ch06-option.md +785 -785
- package/lib/assets/docs/article/grokkingfp/all/part-3-ch07-either-adt.md +871 -871
- package/lib/assets/docs/article/grokkingfp/all/part-4-ch08-io-monad.md +972 -972
- package/lib/assets/docs/article/grokkingfp/all/part-4-ch09-streams.md +926 -926
- package/lib/assets/docs/article/grokkingfp/all/part-5-ch10-concurrency.md +870 -870
- package/lib/assets/docs/article/grokkingfp/all/part-6-ch11-application.md +715 -715
- package/lib/assets/docs/article/grokkingfp/all/part-6-ch12-testing.md +626 -626
- package/lib/assets/docs/article/grokkingfp/all/writing-plan.md +712 -712
- package/lib/assets/docs/article/grokkingfp/clojure/index.md +276 -276
- package/lib/assets/docs/article/grokkingfp/clojure/part-1.md +667 -667
- package/lib/assets/docs/article/grokkingfp/clojure/part-2.md +643 -643
- package/lib/assets/docs/article/grokkingfp/clojure/part-3.md +620 -620
- package/lib/assets/docs/article/grokkingfp/clojure/part-4.md +697 -697
- package/lib/assets/docs/article/grokkingfp/clojure/part-5.md +751 -751
- package/lib/assets/docs/article/grokkingfp/clojure/part-6.md +721 -721
- package/lib/assets/docs/article/grokkingfp/csharp/index.md +246 -246
- package/lib/assets/docs/article/grokkingfp/csharp/part-1.md +811 -811
- package/lib/assets/docs/article/grokkingfp/csharp/part-2.md +971 -971
- package/lib/assets/docs/article/grokkingfp/csharp/part-3.md +981 -981
- package/lib/assets/docs/article/grokkingfp/csharp/part-4.md +949 -949
- package/lib/assets/docs/article/grokkingfp/csharp/part-5.md +947 -947
- package/lib/assets/docs/article/grokkingfp/csharp/part-6.md +739 -739
- package/lib/assets/docs/article/grokkingfp/elixir/index.md +203 -203
- package/lib/assets/docs/article/grokkingfp/elixir/part-1.md +712 -712
- package/lib/assets/docs/article/grokkingfp/elixir/part-2.md +838 -838
- package/lib/assets/docs/article/grokkingfp/elixir/part-3.md +985 -985
- package/lib/assets/docs/article/grokkingfp/elixir/part-4.md +974 -974
- package/lib/assets/docs/article/grokkingfp/elixir/part-5.md +1286 -1286
- package/lib/assets/docs/article/grokkingfp/elixir/part-6.md +1049 -1049
- package/lib/assets/docs/article/grokkingfp/fsharp/index.md +210 -210
- package/lib/assets/docs/article/grokkingfp/fsharp/part-1.md +714 -714
- package/lib/assets/docs/article/grokkingfp/fsharp/part-2.md +961 -961
- package/lib/assets/docs/article/grokkingfp/fsharp/part-3.md +972 -972
- package/lib/assets/docs/article/grokkingfp/fsharp/part-4.md +832 -832
- package/lib/assets/docs/article/grokkingfp/fsharp/part-5.md +911 -911
- package/lib/assets/docs/article/grokkingfp/fsharp/part-6.md +922 -922
- package/lib/assets/docs/article/grokkingfp/haskell/index.md +234 -234
- package/lib/assets/docs/article/grokkingfp/haskell/part-1.md +591 -591
- package/lib/assets/docs/article/grokkingfp/haskell/part-2.md +866 -866
- package/lib/assets/docs/article/grokkingfp/haskell/part-3.md +915 -915
- package/lib/assets/docs/article/grokkingfp/haskell/part-4.md +878 -878
- package/lib/assets/docs/article/grokkingfp/haskell/part-5.md +845 -845
- package/lib/assets/docs/article/grokkingfp/haskell/part-6.md +844 -844
- package/lib/assets/docs/article/grokkingfp/index.md +143 -143
- package/lib/assets/docs/article/grokkingfp/java/index.md +211 -211
- package/lib/assets/docs/article/grokkingfp/java/part-1.md +648 -648
- package/lib/assets/docs/article/grokkingfp/java/part-2.md +675 -675
- package/lib/assets/docs/article/grokkingfp/java/part-3.md +672 -672
- package/lib/assets/docs/article/grokkingfp/java/part-4.md +771 -771
- package/lib/assets/docs/article/grokkingfp/java/part-5.md +959 -959
- package/lib/assets/docs/article/grokkingfp/java/part-6.md +1328 -1328
- package/lib/assets/docs/article/grokkingfp/python/index.md +258 -258
- package/lib/assets/docs/article/grokkingfp/python/part-1.md +443 -443
- package/lib/assets/docs/article/grokkingfp/python/part-2.md +958 -958
- package/lib/assets/docs/article/grokkingfp/python/part-3.md +1004 -1004
- package/lib/assets/docs/article/grokkingfp/python/part-4.md +765 -765
- package/lib/assets/docs/article/grokkingfp/python/part-5.md +747 -747
- package/lib/assets/docs/article/grokkingfp/python/part-6.md +861 -861
- package/lib/assets/docs/article/grokkingfp/ruby/index.md +330 -330
- package/lib/assets/docs/article/grokkingfp/ruby/part-1.md +755 -755
- package/lib/assets/docs/article/grokkingfp/ruby/part-2.md +938 -938
- package/lib/assets/docs/article/grokkingfp/ruby/part-3.md +946 -946
- package/lib/assets/docs/article/grokkingfp/ruby/part-4.md +921 -921
- package/lib/assets/docs/article/grokkingfp/ruby/part-5.md +908 -908
- package/lib/assets/docs/article/grokkingfp/ruby/part-6.md +1412 -1412
- package/lib/assets/docs/article/grokkingfp/rust/index.md +242 -242
- package/lib/assets/docs/article/grokkingfp/rust/part-1.md +634 -634
- package/lib/assets/docs/article/grokkingfp/rust/part-2.md +1060 -1060
- package/lib/assets/docs/article/grokkingfp/rust/part-3.md +994 -994
- package/lib/assets/docs/article/grokkingfp/rust/part-4.md +573 -573
- package/lib/assets/docs/article/grokkingfp/rust/part-5.md +705 -705
- package/lib/assets/docs/article/grokkingfp/rust/part-6.md +508 -508
- package/lib/assets/docs/article/grokkingfp/scala/index.md +171 -171
- package/lib/assets/docs/article/grokkingfp/scala/part-1.md +543 -543
- package/lib/assets/docs/article/grokkingfp/scala/part-2.md +946 -946
- package/lib/assets/docs/article/grokkingfp/scala/part-3.md +919 -919
- package/lib/assets/docs/article/grokkingfp/scala/part-4.md +742 -742
- package/lib/assets/docs/article/grokkingfp/scala/part-5.md +722 -722
- package/lib/assets/docs/article/grokkingfp/scala/part-6.md +867 -867
- package/lib/assets/docs/article/grokkingfp/typescript/index.md +273 -273
- package/lib/assets/docs/article/grokkingfp/typescript/part-1.md +561 -561
- package/lib/assets/docs/article/grokkingfp/typescript/part-2.md +1129 -1129
- package/lib/assets/docs/article/grokkingfp/typescript/part-3.md +842 -842
- package/lib/assets/docs/article/grokkingfp/typescript/part-4.md +1087 -1087
- package/lib/assets/docs/article/grokkingfp/typescript/part-5.md +717 -717
- package/lib/assets/docs/article/grokkingfp/typescript/part-6.md +982 -982
- package/lib/assets/docs/article/practical-database-design/index.md +121 -121
- package/lib/assets/docs/article/practical-database-design/part1/chapter01.md +288 -288
- package/lib/assets/docs/article/practical-database-design/part1/chapter02.md +518 -518
- package/lib/assets/docs/article/practical-database-design/part1/chapter03.md +557 -557
- package/lib/assets/docs/article/practical-database-design/part2/chapter04.md +924 -924
- package/lib/assets/docs/article/practical-database-design/part2/chapter05.md +1627 -1627
- package/lib/assets/docs/article/practical-database-design/part2/chapter06.md +2716 -2716
- package/lib/assets/docs/article/practical-database-design/part2/chapter07.md +2082 -2082
- package/lib/assets/docs/article/practical-database-design/part2/chapter08.md +2105 -2105
- package/lib/assets/docs/article/practical-database-design/part2/chapter09.md +2031 -2031
- package/lib/assets/docs/article/practical-database-design/part2/chapter10.md +1387 -1387
- package/lib/assets/docs/article/practical-database-design/part2/chapter11.md +1677 -1677
- package/lib/assets/docs/article/practical-database-design/part2/chapter12.md +1417 -1417
- package/lib/assets/docs/article/practical-database-design/part2/chapter13.md +1434 -1434
- package/lib/assets/docs/article/practical-database-design/part3/chapter14.md +667 -667
- package/lib/assets/docs/article/practical-database-design/part3/chapter15.md +1625 -1625
- package/lib/assets/docs/article/practical-database-design/part3/chapter16.md +1915 -1915
- package/lib/assets/docs/article/practical-database-design/part3/chapter17.md +1708 -1708
- package/lib/assets/docs/article/practical-database-design/part3/chapter18.md +2095 -2095
- package/lib/assets/docs/article/practical-database-design/part3/chapter19.md +1123 -1123
- package/lib/assets/docs/article/practical-database-design/part3/chapter20.md +1031 -1031
- package/lib/assets/docs/article/practical-database-design/part3/chapter21.md +1382 -1382
- package/lib/assets/docs/article/practical-database-design/part3-orm/chapter14-orm.md +991 -991
- package/lib/assets/docs/article/practical-database-design/part3-orm/chapter15-orm.md +1300 -1300
- package/lib/assets/docs/article/practical-database-design/part3-orm/chapter16-orm.md +1166 -1166
- package/lib/assets/docs/article/practical-database-design/part3-orm/chapter17-orm.md +1584 -1584
- package/lib/assets/docs/article/practical-database-design/part3-orm/chapter18-orm.md +1183 -1183
- package/lib/assets/docs/article/practical-database-design/part3-orm/chapter19-orm.md +1016 -1016
- package/lib/assets/docs/article/practical-database-design/part3-orm/chapter20-orm.md +1753 -1753
- package/lib/assets/docs/article/practical-database-design/part3-orm/chapter21-orm.md +1447 -1447
- package/lib/assets/docs/article/practical-database-design/part3-orm/chapter22-orm.md +1878 -1878
- package/lib/assets/docs/article/practical-database-design/part4/chapter22.md +965 -965
- package/lib/assets/docs/article/practical-database-design/part4/chapter23.md +2069 -2069
- package/lib/assets/docs/article/practical-database-design/part4/chapter24.md +2439 -2439
- package/lib/assets/docs/article/practical-database-design/part4/chapter25.md +3661 -3661
- package/lib/assets/docs/article/practical-database-design/part4/chapter26.md +2916 -2916
- package/lib/assets/docs/article/practical-database-design/part4/chapter27.md +3105 -3105
- package/lib/assets/docs/article/practical-database-design/part4/chapter28.md +2697 -2697
- package/lib/assets/docs/article/practical-database-design/part4/chapter29.md +2544 -2544
- package/lib/assets/docs/article/practical-database-design/part4/chapter30.md +2180 -2180
- package/lib/assets/docs/article/practical-database-design/part4/chapter31.md +1192 -1192
- package/lib/assets/docs/article/practical-database-design/part4/chapter32.md +2101 -2101
- package/lib/assets/docs/article/practical-database-design/part5/chapter33.md +1032 -1032
- package/lib/assets/docs/article/practical-database-design/part5/chapter34.md +1609 -1609
- package/lib/assets/docs/article/practical-database-design/part5/chapter35.md +1453 -1453
- package/lib/assets/docs/article/practical-database-design/part5/chapter36.md +1292 -1292
- package/lib/assets/docs/article/practical-database-design/part5/chapter37.md +1470 -1470
- package/lib/assets/docs/article/practical-database-design/part5/chapter38.md +1698 -1698
- package/lib/assets/docs/article/practical-database-design/part5/chapter39.md +2334 -2334
- package/lib/assets/docs/article/practical-database-design/study/study2-1.md +1693 -1693
- package/lib/assets/docs/article/practical-database-design/study/study2-2.md +1347 -1347
- package/lib/assets/docs/article/practical-database-design/study/study2-3.md +2044 -2044
- package/lib/assets/docs/article/practical-database-design/study/study2-4.md +2229 -2229
- package/lib/assets/docs/article/practical-database-design/study/study2-5.md +2418 -2418
- package/lib/assets/docs/article/practical-database-design/study/study3-1.md +2205 -2205
- package/lib/assets/docs/article/practical-database-design/study/study3-2.md +2221 -2221
- package/lib/assets/docs/article/practical-database-design/study/study3-3.md +2253 -2253
- package/lib/assets/docs/article/practical-database-design/study/study3-4.md +2106 -2106
- package/lib/assets/docs/article/practical-database-design/study/study3-5.md +2507 -2507
- package/lib/assets/docs/article/practical-database-design/study/study4-1.md +2587 -2587
- package/lib/assets/docs/article/practical-database-design/study/study4-2.md +2075 -2075
- package/lib/assets/docs/article/practical-database-design/study/study4-3.md +1805 -1805
- package/lib/assets/docs/article/practical-database-design/study/study4-4.md +1895 -1895
- package/lib/assets/docs/article/practical-database-design/study/study4-5.md +2878 -2878
- package/lib/assets/docs/assets/css/extra.css +29 -29
- package/lib/assets/docs/assets/js/extra.js +44 -44
- package/lib/assets/docs/development/index.md +39 -39
- package/lib/assets/docs/operation/index.md +11 -11
- package/lib/assets/docs/reference/CodexCLIMCP/343/202/242/343/203/227/343/203/252/343/202/261/343/203/274/343/202/267/343/203/247/343/203/263/351/226/213/347/231/272/343/203/225/343/203/255/343/203/274.md +532 -532
- package/lib/assets/docs/reference/CodexCLIMCP/343/202/265/343/203/274/343/203/220/343/203/274/350/250/255/345/256/232/346/211/213/351/240/206.md +341 -341
- package/lib/assets/docs/reference/Java/343/202/242/343/203/227/343/203/252/343/202/261/343/203/274/343/202/267/343/203/247/343/203/263/347/222/260/345/242/203/346/247/213/347/257/211/343/202/254/343/202/244/343/203/211.md +581 -580
- package/lib/assets/docs/reference/SonarQube/343/203/255/343/203/274/343/202/253/343/203/253/347/222/260/345/242/203/343/202/273/343/203/203/343/203/210/343/202/242/343/203/203/343/203/227/346/211/213/351/240/206/346/233/270.md +642 -642
- package/lib/assets/docs/reference/TypeScript/343/202/242/343/203/227/343/203/252/343/202/261/343/203/274/343/202/267/343/203/247/343/203/263/347/222/260/345/242/203/346/247/213/347/257/211/343/202/254/343/202/244/343/203/211.md +465 -465
- package/lib/assets/docs/reference/UI/350/250/255/350/250/210/343/202/254/343/202/244/343/203/211.md +450 -450
- package/lib/assets/docs/reference/images/Ansoff.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/BrandBasicStrategy.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/BrandCategorization.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/BrandRecurutementStrategy.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/BrandValue.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/BusinessActivitiy.svg +3 -3
- package/lib/assets/docs/reference/images/HRM.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/MarketingStructure.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/OrganizationElemnts.svg +3 -3
- package/lib/assets/docs/reference/images/PPM.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/PositioningMap.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/ProductLayer.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/ProductMix.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/SWOT.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/TargetMarket.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/ThreeGenericStrategies.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/VRIO.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/ValueChain.drawio.svg +3 -3
- package/lib/assets/docs/reference/index.md +52 -52
- package/lib/assets/docs/reference//343/202/210/343/201/204/343/202/275/343/203/225/343/203/210/343/202/246/343/202/247/343/202/242/343/201/250/343/201/257.md +250 -242
- package/lib/assets/docs/reference//343/202/242/343/203/274/343/202/255/343/203/206/343/202/257/343/203/201/343/203/243/350/250/255/350/250/210/343/202/254/343/202/244/343/203/211.md +2216 -2216
- package/lib/assets/docs/reference//343/202/244/343/203/263/343/203/225/343/203/251/350/250/255/350/250/210/343/202/254/343/202/244/343/203/211.md +1878 -1878
- package/lib/assets/docs/reference//343/202/250/343/202/257/343/202/271/343/203/210/343/203/252/343/203/274/343/203/240/343/203/227/343/203/255/343/202/260/343/203/251/343/203/237/343/203/263/343/202/260.md +550 -544
- package/lib/assets/docs/reference//343/202/263/343/203/274/343/203/207/343/202/243/343/203/263/343/202/260/343/201/250/343/203/206/343/202/271/343/203/210/343/202/254/343/202/244/343/203/211.md +705 -705
- package/lib/assets/docs/reference//343/203/206/343/202/271/343/203/210/346/210/246/347/225/245/343/202/254/343/202/244/343/203/211.md +1313 -1313
- package/lib/assets/docs/reference//343/203/207/343/203/274/343/202/277/343/203/242/343/203/207/343/203/253/350/250/255/350/250/210/343/202/254/343/202/244/343/203/211.md +311 -311
- package/lib/assets/docs/reference//343/203/211/343/203/241/343/202/244/343/203/263/343/203/242/343/203/207/343/203/253/350/250/255/350/250/210/343/202/254/343/202/244/343/203/211.md +599 -599
- package/lib/assets/docs/reference//343/203/223/343/202/270/343/203/215/343/202/271/343/202/242/343/203/274/343/202/255/343/203/206/343/202/257/343/203/201/343/203/243/345/210/206/346/236/220/343/202/254/343/202/244/343/203/211.md +528 -528
- package/lib/assets/docs/reference//343/203/246/343/203/274/343/202/271/343/202/261/343/203/274/343/202/271/344/275/234/346/210/220/343/202/254/343/202/244/343/203/211.md +689 -682
- package/lib/assets/docs/reference//343/203/252/343/203/252/343/203/274/343/202/271/343/202/254/343/202/244/343/203/211.md +461 -461
- package/lib/assets/docs/reference//343/203/252/343/203/252/343/203/274/343/202/271/343/203/273/343/202/244/343/203/206/343/203/254/343/203/274/343/202/267/343/203/247/343/203/263/350/250/210/347/224/273/343/202/254/343/202/244/343/203/211.md +580 -560
- package/lib/assets/docs/reference//343/203/255/343/202/270/343/202/253/343/203/253/343/202/267/343/203/263/343/202/255/343/203/263/343/202/260.md +1367 -1367
- package/lib/assets/docs/reference//344/274/201/346/245/255/347/265/214/345/226/266/350/253/226.md +2637 -2636
- package/lib/assets/docs/reference//347/222/260/345/242/203/345/244/211/346/225/260/347/256/241/347/220/206/343/202/254/343/202/244/343/203/211.md +665 -663
- package/lib/assets/docs/reference//350/246/201/344/273/266/345/256/232/347/276/251/343/202/254/343/202/244/343/203/211.md +1248 -1248
- package/lib/assets/docs/reference//350/250/200/350/252/236/345/210/245/351/226/213/347/231/272/343/202/254/343/202/244/343/203/211.md +28 -0
- package/lib/assets/docs/reference//351/201/213/345/226/266/347/256/241/347/220/206.md +1482 -1482
- package/lib/assets/docs/reference//351/201/213/347/224/250/343/202/271/343/202/257/343/203/252/343/203/227/343/203/210/344/275/234/346/210/220/343/202/254/343/202/244/343/203/211.md +421 -421
- package/lib/assets/docs/reference//351/201/213/347/224/250/350/246/201/344/273/266/345/256/232/347/276/251/343/202/254/343/202/244/343/203/211.md +392 -392
- package/lib/assets/docs/reference//351/226/213/347/231/272/343/202/254/343/202/244/343/203/211.md +299 -299
- package/lib/assets/docs/reference//351/235/236/346/251/237/350/203/275/350/246/201/344/273/266/345/256/232/347/276/251/343/202/254/343/202/244/343/203/211.md +1236 -1236
- package/lib/assets/docs/review/index.md +5 -5
- package/lib/assets/docs/strategy/index.md +1 -1
- package/lib/assets/docs/template/ADR.md +30 -30
- package/lib/assets/docs/template/AWS/343/202/271/343/203/206/343/203/274/343/202/270/343/203/263/343/202/260/347/222/260/345/242/203/343/202/273/343/203/203/343/203/210/343/202/242/343/203/203/343/203/227/346/211/213/351/240/206/346/233/270.md +1366 -1366
- package/lib/assets/docs/template/AWS/343/203/227/343/203/255/343/203/200/343/202/257/343/202/267/343/203/247/343/203/263/347/222/260/345/242/203/343/202/273/343/203/203/343/203/210/343/202/242/343/203/203/343/203/227/346/211/213/351/240/206/346/233/270.md +634 -634
- package/lib/assets/docs/template/README.md +50 -50
- package/lib/assets/docs/template/index.md +23 -23
- package/lib/assets/docs/template//343/201/276/343/201/232/343/201/223/343/202/214/343/202/222/350/252/255/343/202/202/343/201/206/343/203/252/343/202/271/343/203/210.md +12 -12
- package/lib/assets/docs/template//343/202/242/343/203/227/343/203/252/343/202/261/343/203/274/343/202/267/343/203/247/343/203/263/351/226/213/347/231/272/347/222/260/345/242/203/343/202/273/343/203/203/343/203/210/343/202/242/343/203/203/343/203/227/346/211/213/351/240/206/346/233/270.md +547 -547
- package/lib/assets/docs/template//343/202/244/343/203/206/343/203/254/343/203/274/343/202/267/343/203/247/343/203/263/345/256/214/344/272/206/345/240/261/345/221/212/346/233/270.md +58 -58
- package/lib/assets/docs/template//343/202/244/343/203/263/343/202/273/343/203/227/343/202/267/343/203/247/343/203/263/343/203/207/343/203/203/343/202/255.md +13 -13
- package/lib/assets/docs/template//343/203/223/343/202/270/343/203/215/343/202/271/343/202/242/343/203/274/343/202/255/343/203/206/343/202/257/343/203/201/343/203/243.md +379 -379
- package/lib/assets/docs/template//344/274/201/346/245/255/345/210/206/346/236/220.md +573 -573
- package/lib/assets/docs/template//345/256/214/345/205/250/345/275/242/345/274/217/343/201/256/343/203/246/343/203/274/343/202/271/343/202/261/343/203/274/343/202/271.md +69 -68
- package/lib/assets/docs/template//350/246/201/344/273/266/345/256/232/347/276/251.md +669 -669
- package/lib/assets/docs/template//350/250/255/350/250/210.md +173 -173
- package/lib/assets/docs/template//351/226/213/347/231/272/347/222/260/345/242/203/343/202/273/343/203/203/343/203/210/343/202/242/343/203/203/343/203/227/346/211/213/351/240/206/346/233/270.md +688 -688
- package/lib/assets/gulpfile.js +25 -25
- package/lib/assets/mkdocs.yml +136 -135
- package/lib/assets/ops/docker/mkdoc/Dockerfile +19 -19
- package/lib/assets/ops/scripts/journal.js +180 -180
- package/lib/assets/ops/scripts/mkdocs.js +82 -82
- package/lib/assets/ops/scripts/release.js +431 -431
- package/lib/assets/ops/scripts/sonar_local.js +726 -726
- package/lib/assets/ops/scripts/ssh.js +190 -190
- package/lib/assets/ops/scripts/vault.js +299 -299
- package/lib/assets/package-lock.json +1653 -1653
- package/lib/assets/package.json +40 -40
- package/lib/gulpfile.js +37 -37
- package/package.json +41 -41
- package/lib/assets/.claude/agent-memory/xp-programmer/MEMORY.md +0 -6
- package/lib/assets/.claude/agent-memory/xp-programmer/project_cargo_tracker.md +0 -11
- package/lib/assets/.claude/agent-memory/xp-programmer/project_ddd_patterns.md +0 -27
- package/lib/assets/.claude/agent-memory/xp-programmer/project_us07_route_assignment.md +0 -19
|
@@ -1,390 +1,390 @@
|
|
|
1
|
-
# 第 8 章: パターンマッチとガード
|
|
2
|
-
|
|
3
|
-
## 8.1 はじめに
|
|
4
|
-
|
|
5
|
-
前章では代数的データ型と型クラスを使って FizzBuzz のタイプ別ロジックを構造化しました。この章では、Haskell のパターンマッチとガード式の仕組みを深掘りし、`Either` 型によるエラーハンドリングを導入します。
|
|
6
|
-
|
|
7
|
-
前章の TODO リストから続けます。
|
|
8
|
-
|
|
9
|
-
**TODO リスト**:
|
|
10
|
-
|
|
11
|
-
- [x] タイプ 1 の場合(通常の FizzBuzz)
|
|
12
|
-
- [x] タイプ 2 の場合(数字のみ)
|
|
13
|
-
- [x] タイプ 3 の場合(FizzBuzz の場合のみ)
|
|
14
|
-
- [ ] それ以外のタイプの場合(エラーを返す)
|
|
15
|
-
- [x] 値オブジェクト FizzBuzzValue の導入
|
|
16
|
-
- [x] コレクション FizzBuzzList の導入
|
|
17
|
-
|
|
18
|
-
## 8.2 パターンマッチの基本
|
|
19
|
-
|
|
20
|
-
### 関数引数のパターンマッチ
|
|
21
|
-
|
|
22
|
-
Haskell では関数定義の引数にパターンを直接記述できます。前章の `generate` 関数を見てみましょう。
|
|
23
|
-
|
|
24
|
-
```haskell
|
|
25
|
-
instance Generatable FizzBuzzType where
|
|
26
|
-
generate Type01 n = ...
|
|
27
|
-
generate Type02 n = ...
|
|
28
|
-
generate Type03 n = ...
|
|
29
|
-
```
|
|
30
|
-
|
|
31
|
-
この定義では、第 1 引数に対して 3 つのパターン(`Type01`、`Type02`、`Type03`)が列挙されています。Haskell のランタイムは引数の値を上から順にパターンと照合し、最初にマッチしたパターンの右辺を実行します。
|
|
32
|
-
|
|
33
|
-
これは Rust の `match` 式に似ていますが、Haskell では関数定義そのものにパターンマッチが組み込まれている点が特徴です。
|
|
34
|
-
|
|
35
|
-
| 言語 | パターンマッチ | 記法 |
|
|
36
|
-
|------|--------------|------|
|
|
37
|
-
| Haskell | 関数定義に直接記述 | `f Pattern1 = ...; f Pattern2 = ...` |
|
|
38
|
-
| Rust | `match` 式 | `match x { Pattern1 => ..., Pattern2 => ... }` |
|
|
39
|
-
| Scala | `match` 式 | `x match { case Pattern1 => ... }` |
|
|
40
|
-
| Clojure | `condp` / マルチメソッド | `(condp = x pattern1 ...)` |
|
|
41
|
-
|
|
42
|
-
### パターンの種類
|
|
43
|
-
|
|
44
|
-
Haskell のパターンマッチで使える主なパターンを整理します。
|
|
45
|
-
|
|
46
|
-
```haskell
|
|
47
|
-
-- (1) リテラルパターン: 具体的な値にマッチ
|
|
48
|
-
createType 1 = Right Type01
|
|
49
|
-
createType 2 = Right Type02
|
|
50
|
-
|
|
51
|
-
-- (2) コンストラクタパターン: データコンストラクタにマッチ
|
|
52
|
-
generate Type01 n = ...
|
|
53
|
-
generate Type02 n = ...
|
|
54
|
-
|
|
55
|
-
-- (3) ワイルドカードパターン: 任意の値にマッチ(値を使わない)
|
|
56
|
-
createType _ = Left "未定義のタイプです"
|
|
57
|
-
|
|
58
|
-
-- (4) 変数パターン: 任意の値にマッチし、変数に束縛
|
|
59
|
-
generate Type02 n = show n
|
|
60
|
-
|
|
61
|
-
-- (5) レコードパターン: レコードのフィールドにマッチ
|
|
62
|
-
executeValue cmd = ... vcNumber cmd ... vcType cmd ...
|
|
63
|
-
```
|
|
64
|
-
|
|
65
|
-
### 評価順序
|
|
66
|
-
|
|
67
|
-
パターンは **上から順に** 評価されます。最初にマッチしたパターンが採用され、残りは評価されません。
|
|
68
|
-
|
|
69
|
-
```haskell
|
|
70
|
-
createType 1 = Right Type01 -- (1) まず 1 と比較
|
|
71
|
-
createType 2 = Right Type02 -- (2) 次に 2 と比較
|
|
72
|
-
createType 3 = Right Type03 -- (3) 次に 3 と比較
|
|
73
|
-
createType _ = Left "未定義のタイプです" -- (4) どれにもマッチしなければここ
|
|
74
|
-
```
|
|
75
|
-
|
|
76
|
-
`createType 2` を呼び出すと、(1) で `2 == 1` が `False` となり、(2) で `2 == 2` が `True` となるため、`Right Type02` が返されます。
|
|
77
|
-
|
|
78
|
-
ワイルドカード `_` は必ず最後に置きます。`_` より下のパターンには到達できないため、コンパイラが警告を出します。
|
|
79
|
-
|
|
80
|
-
## 8.3 ガード式
|
|
81
|
-
|
|
82
|
-
### ガードの構文
|
|
83
|
-
|
|
84
|
-
ガード式は、パターンマッチに条件分岐を追加する構文です。`|` で始まり、条件式と結果を `=` で結びます。
|
|
85
|
-
|
|
86
|
-
```haskell
|
|
87
|
-
generate Type01 n
|
|
88
|
-
| n `mod` 15 == 0 = "FizzBuzz"
|
|
89
|
-
| n `mod` 3 == 0 = "Fizz"
|
|
90
|
-
| n `mod` 5 == 0 = "Buzz"
|
|
91
|
-
| otherwise = show n
|
|
92
|
-
```
|
|
93
|
-
|
|
94
|
-
この定義は次のように読めます。
|
|
95
|
-
|
|
96
|
-
1. 第 1 引数が `Type01` にマッチした場合、さらにガード条件を評価する
|
|
97
|
-
2. `n `mod` 15 == 0` が `True` なら `"FizzBuzz"` を返す
|
|
98
|
-
3. `n `mod` 3 == 0` が `True` なら `"Fizz"` を返す
|
|
99
|
-
4. `n `mod` 5 == 0` が `True` なら `"Buzz"` を返す
|
|
100
|
-
5. `otherwise` は常に `True`(`otherwise = True` と定義されている)なのでデフォルト値を返す
|
|
101
|
-
|
|
102
|
-
### ガードの評価順序
|
|
103
|
-
|
|
104
|
-
ガードもパターンと同様に **上から順に** 評価されます。これが FizzBuzz で重要な意味を持ちます。
|
|
105
|
-
|
|
106
|
-
```haskell
|
|
107
|
-
generate Type01 n
|
|
108
|
-
| n `mod` 15 == 0 = "FizzBuzz" -- 15 の倍数チェックが最初
|
|
109
|
-
| n `mod` 3 == 0 = "Fizz" -- 3 の倍数チェックが次
|
|
110
|
-
| n `mod` 5 == 0 = "Buzz" -- 5 の倍数チェックが最後
|
|
111
|
-
| otherwise = show n
|
|
112
|
-
```
|
|
113
|
-
|
|
114
|
-
15 は 3 と 5 の公倍数なので、`n `mod` 15 == 0` のガードは `n `mod` 3 == 0` と `n `mod` 5 == 0` より **前に** 配置する必要があります。順序を入れ替えると、15 の倍数が `"Fizz"` や `"Buzz"` として処理されてしまいます。
|
|
115
|
-
|
|
116
|
-
```haskell
|
|
117
|
-
-- 誤った順序(15 の倍数が "Fizz" になる)
|
|
118
|
-
generate Type01 n
|
|
119
|
-
| n `mod` 3 == 0 = "Fizz" -- 15 も 3 の倍数なのでここでマッチ
|
|
120
|
-
| n `mod` 5 == 0 = "Buzz"
|
|
121
|
-
| n `mod` 15 == 0 = "FizzBuzz" -- ここに到達しない
|
|
122
|
-
| otherwise = show n
|
|
123
|
-
```
|
|
124
|
-
|
|
125
|
-
### otherwise の正体
|
|
126
|
-
|
|
127
|
-
`otherwise` は特別なキーワードではなく、`Prelude` モジュールで定義された単なる定数です。
|
|
128
|
-
|
|
129
|
-
```haskell
|
|
130
|
-
-- GHC.Base での定義
|
|
131
|
-
otherwise :: Bool
|
|
132
|
-
otherwise = True
|
|
133
|
-
```
|
|
134
|
-
|
|
135
|
-
`otherwise` は常に `True` なので、すべてのガード条件が `False` だった場合の **フォールバック** として機能します。名前が意図を明確にするため慣例的に使われますが、技術的には `True` と書いても同じです。
|
|
136
|
-
|
|
137
|
-
### パターンマッチとガードの組み合わせ
|
|
138
|
-
|
|
139
|
-
タイプ 3 の定義を見ると、パターンマッチとガードの組み合わせの良い例です。
|
|
140
|
-
|
|
141
|
-
```haskell
|
|
142
|
-
generate Type03 n
|
|
143
|
-
| n `mod` 15 == 0 = "FizzBuzz"
|
|
144
|
-
| n `mod` 3 == 0 = "Fizz"
|
|
145
|
-
| otherwise = show n
|
|
146
|
-
```
|
|
147
|
-
|
|
148
|
-
タイプ 3 は `Buzz` を返しません。タイプ 1 との違いは `n `mod` 5 == 0` のガードがない点だけです。パターンマッチ(`Type03`)で分岐先を選び、ガード式で細かい条件分岐を行うという 2 段階の分岐が自然に表現されています。
|
|
149
|
-
|
|
150
|
-
## 8.4 case 式との比較
|
|
151
|
-
|
|
152
|
-
### case 式の構文
|
|
153
|
-
|
|
154
|
-
Haskell にはパターンマッチのもう 1 つの形式として `case` 式があります。
|
|
155
|
-
|
|
156
|
-
```haskell
|
|
157
|
-
-- case 式を使った場合
|
|
158
|
-
createType' :: Int -> Either String FizzBuzzType
|
|
159
|
-
createType' n = case n of
|
|
160
|
-
1 -> Right Type01
|
|
161
|
-
2 -> Right Type02
|
|
162
|
-
3 -> Right Type03
|
|
163
|
-
_ -> Left "未定義のタイプです"
|
|
164
|
-
```
|
|
165
|
-
|
|
166
|
-
これは関数引数でのパターンマッチと等価です。
|
|
167
|
-
|
|
168
|
-
```haskell
|
|
169
|
-
-- 関数引数のパターンマッチ(プロジェクトで採用している形式)
|
|
170
|
-
createType :: Int -> Either String FizzBuzzType
|
|
171
|
-
createType 1 = Right Type01
|
|
172
|
-
createType 2 = Right Type02
|
|
173
|
-
createType 3 = Right Type03
|
|
174
|
-
createType _ = Left "未定義のタイプです"
|
|
175
|
-
```
|
|
176
|
-
|
|
177
|
-
### 使い分けの指針
|
|
178
|
-
|
|
179
|
-
| 形式 | 適する場面 | 例 |
|
|
180
|
-
|------|-----------|-----|
|
|
181
|
-
| 関数引数パターンマッチ | トップレベル関数の定義 | `createType 1 = Right Type01` |
|
|
182
|
-
| case 式 | 関数の途中で分岐したい場合 | `case result of Right v -> ...; Left e -> ...` |
|
|
183
|
-
|
|
184
|
-
本プロジェクトでは、関数引数のパターンマッチを優先的に使用しています。関数の定義が複数の等式として並ぶため、各ケースの対応が視覚的に分かりやすくなります。
|
|
185
|
-
|
|
186
|
-
### case 式でのガード
|
|
187
|
-
|
|
188
|
-
`case` 式でもガードを使えます。
|
|
189
|
-
|
|
190
|
-
```haskell
|
|
191
|
-
describe :: FizzBuzzType -> Int -> String
|
|
192
|
-
describe fbType n = case fbType of
|
|
193
|
-
Type01
|
|
194
|
-
| n `mod` 15 == 0 -> "Type01: FizzBuzz"
|
|
195
|
-
| n `mod` 3 == 0 -> "Type01: Fizz"
|
|
196
|
-
| n `mod` 5 == 0 -> "Type01: Buzz"
|
|
197
|
-
| otherwise -> "Type01: " ++ show n
|
|
198
|
-
Type02 -> "Type02: " ++ show n
|
|
199
|
-
Type03
|
|
200
|
-
| n `mod` 15 == 0 -> "Type03: FizzBuzz"
|
|
201
|
-
| n `mod` 3 == 0 -> "Type03: Fizz"
|
|
202
|
-
| otherwise -> "Type03: " ++ show n
|
|
203
|
-
```
|
|
204
|
-
|
|
205
|
-
ただし、この形式は関数引数パターンマッチ + ガードに比べてネストが深くなるため、通常はトップレベル関数で使うことを推奨します。
|
|
206
|
-
|
|
207
|
-
## 8.5 Either 型によるエラーハンドリング
|
|
208
|
-
|
|
209
|
-
### Either 型とは
|
|
210
|
-
|
|
211
|
-
`Either` は Haskell の標準ライブラリで定義された代数的データ型です。
|
|
212
|
-
|
|
213
|
-
```haskell
|
|
214
|
-
data Either a b = Left a | Right b
|
|
215
|
-
```
|
|
216
|
-
|
|
217
|
-
慣例として、`Left` はエラー(失敗)、`Right` は正常値(成功)を表します。`Right` が「正しい」(right)という英語の語呂合わせにもなっています。
|
|
218
|
-
|
|
219
|
-
Java の例外や Go の `error` 戻り値と異なり、`Either` は **型安全** なエラーハンドリングを提供します。コンパイラがエラーケースの処理を強制するため、エラーを無視するコードを書くことが難しくなります。
|
|
220
|
-
|
|
221
|
-
| 言語 | エラーハンドリング | 型安全性 |
|
|
222
|
-
|------|------------------|---------|
|
|
223
|
-
| Haskell | `Either String a` | コンパイル時に検査 |
|
|
224
|
-
| Rust | `Result<T, E>` | コンパイル時に検査 |
|
|
225
|
-
| Java | `throws Exception` | チェック例外のみ |
|
|
226
|
-
| Go | `(value, error)` | 型安全だが無視可能 |
|
|
227
|
-
| Python | `raise Exception` | ランタイムのみ |
|
|
228
|
-
|
|
229
|
-
### Red: createType のテスト
|
|
230
|
-
|
|
231
|
-
`createType` 関数のテストを追加します。
|
|
232
|
-
|
|
233
|
-
```haskell
|
|
234
|
-
describe "createType" $ do
|
|
235
|
-
it "タイプ 1 を生成できる" $
|
|
236
|
-
createType 1 `shouldBe` Right Type01
|
|
237
|
-
|
|
238
|
-
it "タイプ 2 を生成できる" $
|
|
239
|
-
createType 2 `shouldBe` Right Type02
|
|
240
|
-
|
|
241
|
-
it "タイプ 3 を生成できる" $
|
|
242
|
-
createType 3 `shouldBe` Right Type03
|
|
243
|
-
|
|
244
|
-
it "未定義のタイプでエラーを返す" $
|
|
245
|
-
createType 4 `shouldBe` Left "未定義のタイプです"
|
|
246
|
-
```
|
|
247
|
-
|
|
248
|
-
### Green: createType の実装
|
|
249
|
-
|
|
250
|
-
```haskell
|
|
251
|
-
createType :: Int -> Either String FizzBuzzType
|
|
252
|
-
createType 1 = Right Type01
|
|
253
|
-
createType 2 = Right Type02
|
|
254
|
-
createType 3 = Right Type03
|
|
255
|
-
createType _ = Left "未定義のタイプです"
|
|
256
|
-
```
|
|
257
|
-
|
|
258
|
-
この関数は 4 つの点で注目に値します。
|
|
259
|
-
|
|
260
|
-
1. **リテラルパターンマッチ**: 引数が `1`、`2`、`3` のいずれかにマッチする
|
|
261
|
-
2. **ワイルドカード**: `_` でそれ以外のすべての値をキャッチする
|
|
262
|
-
3. **Right で成功を表現**: 正常な値は `Right` で包む
|
|
263
|
-
4. **Left でエラーを表現**: エラーメッセージは `Left` で包む
|
|
264
|
-
|
|
265
|
-
モジュールのエクスポートリストに `createType` を追加します。
|
|
266
|
-
|
|
267
|
-
```haskell
|
|
268
|
-
module FizzBuzz.Type
|
|
269
|
-
( FizzBuzzType(..)
|
|
270
|
-
, Generatable(..)
|
|
271
|
-
, createType
|
|
272
|
-
) where
|
|
273
|
-
```
|
|
274
|
-
|
|
275
|
-
テストを実行します。
|
|
276
|
-
|
|
277
|
-
```bash
|
|
278
|
-
$ stack test
|
|
279
|
-
FizzBuzz.TypeSpec
|
|
280
|
-
Type01
|
|
281
|
-
1 を渡すと '1' を返す
|
|
282
|
-
3 の倍数を渡すと 'Fizz' を返す
|
|
283
|
-
5 の倍数を渡すと 'Buzz' を返す
|
|
284
|
-
15 の倍数を渡すと 'FizzBuzz' を返す
|
|
285
|
-
Type02
|
|
286
|
-
数値を文字列に変換する
|
|
287
|
-
Type03
|
|
288
|
-
1 を渡すと '1' を返す
|
|
289
|
-
3 を渡すと 'Fizz' を返す
|
|
290
|
-
15 の倍数で 'FizzBuzz' を返す
|
|
291
|
-
createType
|
|
292
|
-
タイプ 1 を生成できる
|
|
293
|
-
タイプ 2 を生成できる
|
|
294
|
-
タイプ 3 を生成できる
|
|
295
|
-
未定義のタイプでエラーを返す
|
|
296
|
-
|
|
297
|
-
Finished in 0.0001 seconds
|
|
298
|
-
12 examples, 0 failures
|
|
299
|
-
```
|
|
300
|
-
|
|
301
|
-
**TODO リスト**:
|
|
302
|
-
|
|
303
|
-
- [x] タイプ 1 の場合(通常の FizzBuzz)
|
|
304
|
-
- [x] タイプ 2 の場合(数字のみ)
|
|
305
|
-
- [x] タイプ 3 の場合(FizzBuzz の場合のみ)
|
|
306
|
-
- [x] それ以外のタイプの場合(エラーを返す)
|
|
307
|
-
- [x] 値オブジェクト FizzBuzzValue の導入
|
|
308
|
-
- [x] コレクション FizzBuzzList の導入
|
|
309
|
-
|
|
310
|
-
## 8.6 パターンマッチの網羅性チェック
|
|
311
|
-
|
|
312
|
-
### GHC の網羅性警告
|
|
313
|
-
|
|
314
|
-
GHC はパターンマッチが網羅的でない場合に警告を出します。例えば、`createType` からワイルドカードパターンを削除すると以下のようになります。
|
|
315
|
-
|
|
316
|
-
```haskell
|
|
317
|
-
-- 網羅的でない定義(警告が出る)
|
|
318
|
-
createType :: Int -> Either String FizzBuzzType
|
|
319
|
-
createType 1 = Right Type01
|
|
320
|
-
createType 2 = Right Type02
|
|
321
|
-
createType 3 = Right Type03
|
|
322
|
-
-- createType _ = Left "..." を削除
|
|
323
|
-
```
|
|
324
|
-
|
|
325
|
-
```bash
|
|
326
|
-
$ stack build
|
|
327
|
-
warning: [-Wincomplete-patterns]
|
|
328
|
-
Pattern match(es) are non-exhaustive
|
|
329
|
-
In an equation for 'createType':
|
|
330
|
-
Patterns of type 'Int' not matched: _
|
|
331
|
-
```
|
|
332
|
-
|
|
333
|
-
`Int` 型は無限の値を持つため、`1`、`2`、`3` だけではすべてのケースを網羅できません。ワイルドカード `_` を最後に追加することで、すべての値をカバーできます。
|
|
334
|
-
|
|
335
|
-
### 代数的データ型の網羅性
|
|
336
|
-
|
|
337
|
-
代数的データ型の場合、パターンの網羅性チェックがさらに有効です。
|
|
338
|
-
|
|
339
|
-
```haskell
|
|
340
|
-
-- 網羅的でない定義(Type03 が欠けている)
|
|
341
|
-
generate' :: FizzBuzzType -> Int -> String
|
|
342
|
-
generate' Type01 n = show n
|
|
343
|
-
generate' Type02 n = show n
|
|
344
|
-
-- Type03 のケースがない
|
|
345
|
-
```
|
|
346
|
-
|
|
347
|
-
```bash
|
|
348
|
-
warning: [-Wincomplete-patterns]
|
|
349
|
-
Pattern match(es) are non-exhaustive
|
|
350
|
-
In an equation for 'generate'':
|
|
351
|
-
Patterns of type 'FizzBuzzType' not matched: Type03
|
|
352
|
-
```
|
|
353
|
-
|
|
354
|
-
GHC は `FizzBuzzType` が `Type01`、`Type02`、`Type03` の 3 つの値コンストラクタを持つことを知っているため、`Type03` が欠けていることを指摘できます。これは、後で新しいタイプ(`Type04` など)を追加した場合にも、すべてのパターンマッチ箇所で警告が出るため、実装漏れを防げます。
|
|
355
|
-
|
|
356
|
-
### -Wall オプション
|
|
357
|
-
|
|
358
|
-
`package.yaml` で `-Wall` を有効にすると、網羅性警告を含むすべての警告が有効になります。
|
|
359
|
-
|
|
360
|
-
```yaml
|
|
361
|
-
ghc-options:
|
|
362
|
-
- -Wall
|
|
363
|
-
```
|
|
364
|
-
|
|
365
|
-
本プロジェクトでは `-Wall` を有効にして、パターンマッチの網羅性を含む潜在的な問題を早期に検出しています。
|
|
366
|
-
|
|
367
|
-
## 8.7 パターンマッチのベストプラクティス
|
|
368
|
-
|
|
369
|
-
この章で扱ったパターンマッチの知見をベストプラクティスとしてまとめます。
|
|
370
|
-
|
|
371
|
-
1. **網羅性を確保する**: すべてのコンストラクタをカバーするか、ワイルドカード `_` でフォールバックを用意する
|
|
372
|
-
2. **ガードの順序に注意**: より具体的な条件を先に書く(15 の倍数 → 3 の倍数 → 5 の倍数)
|
|
373
|
-
3. **ワイルドカードは最後に**: `_` より下のパターンには到達できない
|
|
374
|
-
4. **`-Wall` を有効にする**: コンパイラの網羅性チェックを活用する
|
|
375
|
-
5. **`Either` でエラーを型安全に扱う**: 例外ではなく戻り値でエラーを表現する
|
|
376
|
-
6. **関数引数パターンマッチを優先**: トップレベル関数では `case` 式より関数定義のパターンマッチを使う
|
|
377
|
-
|
|
378
|
-
## 8.8 まとめ
|
|
379
|
-
|
|
380
|
-
この章では以下のことを学びました。
|
|
381
|
-
|
|
382
|
-
- **関数引数のパターンマッチ** で値コンストラクタごとに処理を分岐する
|
|
383
|
-
- **ガード式** で条件分岐を追加し、FizzBuzz ロジックを自然に記述する
|
|
384
|
-
- **ガードの評価順序** と **otherwise** の仕組みを理解する
|
|
385
|
-
- **case 式** と関数引数パターンマッチの使い分け
|
|
386
|
-
- **Either 型** による型安全なエラーハンドリング(`createType` のスマートコンストラクタ)
|
|
387
|
-
- GHC の **パターンマッチ網羅性チェック** による安全性の確保
|
|
388
|
-
- **`-Wall`** オプションによる警告の有効化
|
|
389
|
-
|
|
390
|
-
次章では、モジュール設計とスマートコンストラクタのパターンを詳しく見ていきます。
|
|
1
|
+
# 第 8 章: パターンマッチとガード
|
|
2
|
+
|
|
3
|
+
## 8.1 はじめに
|
|
4
|
+
|
|
5
|
+
前章では代数的データ型と型クラスを使って FizzBuzz のタイプ別ロジックを構造化しました。この章では、Haskell のパターンマッチとガード式の仕組みを深掘りし、`Either` 型によるエラーハンドリングを導入します。
|
|
6
|
+
|
|
7
|
+
前章の TODO リストから続けます。
|
|
8
|
+
|
|
9
|
+
**TODO リスト**:
|
|
10
|
+
|
|
11
|
+
- [x] タイプ 1 の場合(通常の FizzBuzz)
|
|
12
|
+
- [x] タイプ 2 の場合(数字のみ)
|
|
13
|
+
- [x] タイプ 3 の場合(FizzBuzz の場合のみ)
|
|
14
|
+
- [ ] それ以外のタイプの場合(エラーを返す)
|
|
15
|
+
- [x] 値オブジェクト FizzBuzzValue の導入
|
|
16
|
+
- [x] コレクション FizzBuzzList の導入
|
|
17
|
+
|
|
18
|
+
## 8.2 パターンマッチの基本
|
|
19
|
+
|
|
20
|
+
### 関数引数のパターンマッチ
|
|
21
|
+
|
|
22
|
+
Haskell では関数定義の引数にパターンを直接記述できます。前章の `generate` 関数を見てみましょう。
|
|
23
|
+
|
|
24
|
+
```haskell
|
|
25
|
+
instance Generatable FizzBuzzType where
|
|
26
|
+
generate Type01 n = ...
|
|
27
|
+
generate Type02 n = ...
|
|
28
|
+
generate Type03 n = ...
|
|
29
|
+
```
|
|
30
|
+
|
|
31
|
+
この定義では、第 1 引数に対して 3 つのパターン(`Type01`、`Type02`、`Type03`)が列挙されています。Haskell のランタイムは引数の値を上から順にパターンと照合し、最初にマッチしたパターンの右辺を実行します。
|
|
32
|
+
|
|
33
|
+
これは Rust の `match` 式に似ていますが、Haskell では関数定義そのものにパターンマッチが組み込まれている点が特徴です。
|
|
34
|
+
|
|
35
|
+
| 言語 | パターンマッチ | 記法 |
|
|
36
|
+
|------|--------------|------|
|
|
37
|
+
| Haskell | 関数定義に直接記述 | `f Pattern1 = ...; f Pattern2 = ...` |
|
|
38
|
+
| Rust | `match` 式 | `match x { Pattern1 => ..., Pattern2 => ... }` |
|
|
39
|
+
| Scala | `match` 式 | `x match { case Pattern1 => ... }` |
|
|
40
|
+
| Clojure | `condp` / マルチメソッド | `(condp = x pattern1 ...)` |
|
|
41
|
+
|
|
42
|
+
### パターンの種類
|
|
43
|
+
|
|
44
|
+
Haskell のパターンマッチで使える主なパターンを整理します。
|
|
45
|
+
|
|
46
|
+
```haskell
|
|
47
|
+
-- (1) リテラルパターン: 具体的な値にマッチ
|
|
48
|
+
createType 1 = Right Type01
|
|
49
|
+
createType 2 = Right Type02
|
|
50
|
+
|
|
51
|
+
-- (2) コンストラクタパターン: データコンストラクタにマッチ
|
|
52
|
+
generate Type01 n = ...
|
|
53
|
+
generate Type02 n = ...
|
|
54
|
+
|
|
55
|
+
-- (3) ワイルドカードパターン: 任意の値にマッチ(値を使わない)
|
|
56
|
+
createType _ = Left "未定義のタイプです"
|
|
57
|
+
|
|
58
|
+
-- (4) 変数パターン: 任意の値にマッチし、変数に束縛
|
|
59
|
+
generate Type02 n = show n
|
|
60
|
+
|
|
61
|
+
-- (5) レコードパターン: レコードのフィールドにマッチ
|
|
62
|
+
executeValue cmd = ... vcNumber cmd ... vcType cmd ...
|
|
63
|
+
```
|
|
64
|
+
|
|
65
|
+
### 評価順序
|
|
66
|
+
|
|
67
|
+
パターンは **上から順に** 評価されます。最初にマッチしたパターンが採用され、残りは評価されません。
|
|
68
|
+
|
|
69
|
+
```haskell
|
|
70
|
+
createType 1 = Right Type01 -- (1) まず 1 と比較
|
|
71
|
+
createType 2 = Right Type02 -- (2) 次に 2 と比較
|
|
72
|
+
createType 3 = Right Type03 -- (3) 次に 3 と比較
|
|
73
|
+
createType _ = Left "未定義のタイプです" -- (4) どれにもマッチしなければここ
|
|
74
|
+
```
|
|
75
|
+
|
|
76
|
+
`createType 2` を呼び出すと、(1) で `2 == 1` が `False` となり、(2) で `2 == 2` が `True` となるため、`Right Type02` が返されます。
|
|
77
|
+
|
|
78
|
+
ワイルドカード `_` は必ず最後に置きます。`_` より下のパターンには到達できないため、コンパイラが警告を出します。
|
|
79
|
+
|
|
80
|
+
## 8.3 ガード式
|
|
81
|
+
|
|
82
|
+
### ガードの構文
|
|
83
|
+
|
|
84
|
+
ガード式は、パターンマッチに条件分岐を追加する構文です。`|` で始まり、条件式と結果を `=` で結びます。
|
|
85
|
+
|
|
86
|
+
```haskell
|
|
87
|
+
generate Type01 n
|
|
88
|
+
| n `mod` 15 == 0 = "FizzBuzz"
|
|
89
|
+
| n `mod` 3 == 0 = "Fizz"
|
|
90
|
+
| n `mod` 5 == 0 = "Buzz"
|
|
91
|
+
| otherwise = show n
|
|
92
|
+
```
|
|
93
|
+
|
|
94
|
+
この定義は次のように読めます。
|
|
95
|
+
|
|
96
|
+
1. 第 1 引数が `Type01` にマッチした場合、さらにガード条件を評価する
|
|
97
|
+
2. `n `mod` 15 == 0` が `True` なら `"FizzBuzz"` を返す
|
|
98
|
+
3. `n `mod` 3 == 0` が `True` なら `"Fizz"` を返す
|
|
99
|
+
4. `n `mod` 5 == 0` が `True` なら `"Buzz"` を返す
|
|
100
|
+
5. `otherwise` は常に `True`(`otherwise = True` と定義されている)なのでデフォルト値を返す
|
|
101
|
+
|
|
102
|
+
### ガードの評価順序
|
|
103
|
+
|
|
104
|
+
ガードもパターンと同様に **上から順に** 評価されます。これが FizzBuzz で重要な意味を持ちます。
|
|
105
|
+
|
|
106
|
+
```haskell
|
|
107
|
+
generate Type01 n
|
|
108
|
+
| n `mod` 15 == 0 = "FizzBuzz" -- 15 の倍数チェックが最初
|
|
109
|
+
| n `mod` 3 == 0 = "Fizz" -- 3 の倍数チェックが次
|
|
110
|
+
| n `mod` 5 == 0 = "Buzz" -- 5 の倍数チェックが最後
|
|
111
|
+
| otherwise = show n
|
|
112
|
+
```
|
|
113
|
+
|
|
114
|
+
15 は 3 と 5 の公倍数なので、`n `mod` 15 == 0` のガードは `n `mod` 3 == 0` と `n `mod` 5 == 0` より **前に** 配置する必要があります。順序を入れ替えると、15 の倍数が `"Fizz"` や `"Buzz"` として処理されてしまいます。
|
|
115
|
+
|
|
116
|
+
```haskell
|
|
117
|
+
-- 誤った順序(15 の倍数が "Fizz" になる)
|
|
118
|
+
generate Type01 n
|
|
119
|
+
| n `mod` 3 == 0 = "Fizz" -- 15 も 3 の倍数なのでここでマッチ
|
|
120
|
+
| n `mod` 5 == 0 = "Buzz"
|
|
121
|
+
| n `mod` 15 == 0 = "FizzBuzz" -- ここに到達しない
|
|
122
|
+
| otherwise = show n
|
|
123
|
+
```
|
|
124
|
+
|
|
125
|
+
### otherwise の正体
|
|
126
|
+
|
|
127
|
+
`otherwise` は特別なキーワードではなく、`Prelude` モジュールで定義された単なる定数です。
|
|
128
|
+
|
|
129
|
+
```haskell
|
|
130
|
+
-- GHC.Base での定義
|
|
131
|
+
otherwise :: Bool
|
|
132
|
+
otherwise = True
|
|
133
|
+
```
|
|
134
|
+
|
|
135
|
+
`otherwise` は常に `True` なので、すべてのガード条件が `False` だった場合の **フォールバック** として機能します。名前が意図を明確にするため慣例的に使われますが、技術的には `True` と書いても同じです。
|
|
136
|
+
|
|
137
|
+
### パターンマッチとガードの組み合わせ
|
|
138
|
+
|
|
139
|
+
タイプ 3 の定義を見ると、パターンマッチとガードの組み合わせの良い例です。
|
|
140
|
+
|
|
141
|
+
```haskell
|
|
142
|
+
generate Type03 n
|
|
143
|
+
| n `mod` 15 == 0 = "FizzBuzz"
|
|
144
|
+
| n `mod` 3 == 0 = "Fizz"
|
|
145
|
+
| otherwise = show n
|
|
146
|
+
```
|
|
147
|
+
|
|
148
|
+
タイプ 3 は `Buzz` を返しません。タイプ 1 との違いは `n `mod` 5 == 0` のガードがない点だけです。パターンマッチ(`Type03`)で分岐先を選び、ガード式で細かい条件分岐を行うという 2 段階の分岐が自然に表現されています。
|
|
149
|
+
|
|
150
|
+
## 8.4 case 式との比較
|
|
151
|
+
|
|
152
|
+
### case 式の構文
|
|
153
|
+
|
|
154
|
+
Haskell にはパターンマッチのもう 1 つの形式として `case` 式があります。
|
|
155
|
+
|
|
156
|
+
```haskell
|
|
157
|
+
-- case 式を使った場合
|
|
158
|
+
createType' :: Int -> Either String FizzBuzzType
|
|
159
|
+
createType' n = case n of
|
|
160
|
+
1 -> Right Type01
|
|
161
|
+
2 -> Right Type02
|
|
162
|
+
3 -> Right Type03
|
|
163
|
+
_ -> Left "未定義のタイプです"
|
|
164
|
+
```
|
|
165
|
+
|
|
166
|
+
これは関数引数でのパターンマッチと等価です。
|
|
167
|
+
|
|
168
|
+
```haskell
|
|
169
|
+
-- 関数引数のパターンマッチ(プロジェクトで採用している形式)
|
|
170
|
+
createType :: Int -> Either String FizzBuzzType
|
|
171
|
+
createType 1 = Right Type01
|
|
172
|
+
createType 2 = Right Type02
|
|
173
|
+
createType 3 = Right Type03
|
|
174
|
+
createType _ = Left "未定義のタイプです"
|
|
175
|
+
```
|
|
176
|
+
|
|
177
|
+
### 使い分けの指針
|
|
178
|
+
|
|
179
|
+
| 形式 | 適する場面 | 例 |
|
|
180
|
+
|------|-----------|-----|
|
|
181
|
+
| 関数引数パターンマッチ | トップレベル関数の定義 | `createType 1 = Right Type01` |
|
|
182
|
+
| case 式 | 関数の途中で分岐したい場合 | `case result of Right v -> ...; Left e -> ...` |
|
|
183
|
+
|
|
184
|
+
本プロジェクトでは、関数引数のパターンマッチを優先的に使用しています。関数の定義が複数の等式として並ぶため、各ケースの対応が視覚的に分かりやすくなります。
|
|
185
|
+
|
|
186
|
+
### case 式でのガード
|
|
187
|
+
|
|
188
|
+
`case` 式でもガードを使えます。
|
|
189
|
+
|
|
190
|
+
```haskell
|
|
191
|
+
describe :: FizzBuzzType -> Int -> String
|
|
192
|
+
describe fbType n = case fbType of
|
|
193
|
+
Type01
|
|
194
|
+
| n `mod` 15 == 0 -> "Type01: FizzBuzz"
|
|
195
|
+
| n `mod` 3 == 0 -> "Type01: Fizz"
|
|
196
|
+
| n `mod` 5 == 0 -> "Type01: Buzz"
|
|
197
|
+
| otherwise -> "Type01: " ++ show n
|
|
198
|
+
Type02 -> "Type02: " ++ show n
|
|
199
|
+
Type03
|
|
200
|
+
| n `mod` 15 == 0 -> "Type03: FizzBuzz"
|
|
201
|
+
| n `mod` 3 == 0 -> "Type03: Fizz"
|
|
202
|
+
| otherwise -> "Type03: " ++ show n
|
|
203
|
+
```
|
|
204
|
+
|
|
205
|
+
ただし、この形式は関数引数パターンマッチ + ガードに比べてネストが深くなるため、通常はトップレベル関数で使うことを推奨します。
|
|
206
|
+
|
|
207
|
+
## 8.5 Either 型によるエラーハンドリング
|
|
208
|
+
|
|
209
|
+
### Either 型とは
|
|
210
|
+
|
|
211
|
+
`Either` は Haskell の標準ライブラリで定義された代数的データ型です。
|
|
212
|
+
|
|
213
|
+
```haskell
|
|
214
|
+
data Either a b = Left a | Right b
|
|
215
|
+
```
|
|
216
|
+
|
|
217
|
+
慣例として、`Left` はエラー(失敗)、`Right` は正常値(成功)を表します。`Right` が「正しい」(right)という英語の語呂合わせにもなっています。
|
|
218
|
+
|
|
219
|
+
Java の例外や Go の `error` 戻り値と異なり、`Either` は **型安全** なエラーハンドリングを提供します。コンパイラがエラーケースの処理を強制するため、エラーを無視するコードを書くことが難しくなります。
|
|
220
|
+
|
|
221
|
+
| 言語 | エラーハンドリング | 型安全性 |
|
|
222
|
+
|------|------------------|---------|
|
|
223
|
+
| Haskell | `Either String a` | コンパイル時に検査 |
|
|
224
|
+
| Rust | `Result<T, E>` | コンパイル時に検査 |
|
|
225
|
+
| Java | `throws Exception` | チェック例外のみ |
|
|
226
|
+
| Go | `(value, error)` | 型安全だが無視可能 |
|
|
227
|
+
| Python | `raise Exception` | ランタイムのみ |
|
|
228
|
+
|
|
229
|
+
### Red: createType のテスト
|
|
230
|
+
|
|
231
|
+
`createType` 関数のテストを追加します。
|
|
232
|
+
|
|
233
|
+
```haskell
|
|
234
|
+
describe "createType" $ do
|
|
235
|
+
it "タイプ 1 を生成できる" $
|
|
236
|
+
createType 1 `shouldBe` Right Type01
|
|
237
|
+
|
|
238
|
+
it "タイプ 2 を生成できる" $
|
|
239
|
+
createType 2 `shouldBe` Right Type02
|
|
240
|
+
|
|
241
|
+
it "タイプ 3 を生成できる" $
|
|
242
|
+
createType 3 `shouldBe` Right Type03
|
|
243
|
+
|
|
244
|
+
it "未定義のタイプでエラーを返す" $
|
|
245
|
+
createType 4 `shouldBe` Left "未定義のタイプです"
|
|
246
|
+
```
|
|
247
|
+
|
|
248
|
+
### Green: createType の実装
|
|
249
|
+
|
|
250
|
+
```haskell
|
|
251
|
+
createType :: Int -> Either String FizzBuzzType
|
|
252
|
+
createType 1 = Right Type01
|
|
253
|
+
createType 2 = Right Type02
|
|
254
|
+
createType 3 = Right Type03
|
|
255
|
+
createType _ = Left "未定義のタイプです"
|
|
256
|
+
```
|
|
257
|
+
|
|
258
|
+
この関数は 4 つの点で注目に値します。
|
|
259
|
+
|
|
260
|
+
1. **リテラルパターンマッチ**: 引数が `1`、`2`、`3` のいずれかにマッチする
|
|
261
|
+
2. **ワイルドカード**: `_` でそれ以外のすべての値をキャッチする
|
|
262
|
+
3. **Right で成功を表現**: 正常な値は `Right` で包む
|
|
263
|
+
4. **Left でエラーを表現**: エラーメッセージは `Left` で包む
|
|
264
|
+
|
|
265
|
+
モジュールのエクスポートリストに `createType` を追加します。
|
|
266
|
+
|
|
267
|
+
```haskell
|
|
268
|
+
module FizzBuzz.Type
|
|
269
|
+
( FizzBuzzType(..)
|
|
270
|
+
, Generatable(..)
|
|
271
|
+
, createType
|
|
272
|
+
) where
|
|
273
|
+
```
|
|
274
|
+
|
|
275
|
+
テストを実行します。
|
|
276
|
+
|
|
277
|
+
```bash
|
|
278
|
+
$ stack test
|
|
279
|
+
FizzBuzz.TypeSpec
|
|
280
|
+
Type01
|
|
281
|
+
1 を渡すと '1' を返す
|
|
282
|
+
3 の倍数を渡すと 'Fizz' を返す
|
|
283
|
+
5 の倍数を渡すと 'Buzz' を返す
|
|
284
|
+
15 の倍数を渡すと 'FizzBuzz' を返す
|
|
285
|
+
Type02
|
|
286
|
+
数値を文字列に変換する
|
|
287
|
+
Type03
|
|
288
|
+
1 を渡すと '1' を返す
|
|
289
|
+
3 を渡すと 'Fizz' を返す
|
|
290
|
+
15 の倍数で 'FizzBuzz' を返す
|
|
291
|
+
createType
|
|
292
|
+
タイプ 1 を生成できる
|
|
293
|
+
タイプ 2 を生成できる
|
|
294
|
+
タイプ 3 を生成できる
|
|
295
|
+
未定義のタイプでエラーを返す
|
|
296
|
+
|
|
297
|
+
Finished in 0.0001 seconds
|
|
298
|
+
12 examples, 0 failures
|
|
299
|
+
```
|
|
300
|
+
|
|
301
|
+
**TODO リスト**:
|
|
302
|
+
|
|
303
|
+
- [x] タイプ 1 の場合(通常の FizzBuzz)
|
|
304
|
+
- [x] タイプ 2 の場合(数字のみ)
|
|
305
|
+
- [x] タイプ 3 の場合(FizzBuzz の場合のみ)
|
|
306
|
+
- [x] それ以外のタイプの場合(エラーを返す)
|
|
307
|
+
- [x] 値オブジェクト FizzBuzzValue の導入
|
|
308
|
+
- [x] コレクション FizzBuzzList の導入
|
|
309
|
+
|
|
310
|
+
## 8.6 パターンマッチの網羅性チェック
|
|
311
|
+
|
|
312
|
+
### GHC の網羅性警告
|
|
313
|
+
|
|
314
|
+
GHC はパターンマッチが網羅的でない場合に警告を出します。例えば、`createType` からワイルドカードパターンを削除すると以下のようになります。
|
|
315
|
+
|
|
316
|
+
```haskell
|
|
317
|
+
-- 網羅的でない定義(警告が出る)
|
|
318
|
+
createType :: Int -> Either String FizzBuzzType
|
|
319
|
+
createType 1 = Right Type01
|
|
320
|
+
createType 2 = Right Type02
|
|
321
|
+
createType 3 = Right Type03
|
|
322
|
+
-- createType _ = Left "..." を削除
|
|
323
|
+
```
|
|
324
|
+
|
|
325
|
+
```bash
|
|
326
|
+
$ stack build
|
|
327
|
+
warning: [-Wincomplete-patterns]
|
|
328
|
+
Pattern match(es) are non-exhaustive
|
|
329
|
+
In an equation for 'createType':
|
|
330
|
+
Patterns of type 'Int' not matched: _
|
|
331
|
+
```
|
|
332
|
+
|
|
333
|
+
`Int` 型は無限の値を持つため、`1`、`2`、`3` だけではすべてのケースを網羅できません。ワイルドカード `_` を最後に追加することで、すべての値をカバーできます。
|
|
334
|
+
|
|
335
|
+
### 代数的データ型の網羅性
|
|
336
|
+
|
|
337
|
+
代数的データ型の場合、パターンの網羅性チェックがさらに有効です。
|
|
338
|
+
|
|
339
|
+
```haskell
|
|
340
|
+
-- 網羅的でない定義(Type03 が欠けている)
|
|
341
|
+
generate' :: FizzBuzzType -> Int -> String
|
|
342
|
+
generate' Type01 n = show n
|
|
343
|
+
generate' Type02 n = show n
|
|
344
|
+
-- Type03 のケースがない
|
|
345
|
+
```
|
|
346
|
+
|
|
347
|
+
```bash
|
|
348
|
+
warning: [-Wincomplete-patterns]
|
|
349
|
+
Pattern match(es) are non-exhaustive
|
|
350
|
+
In an equation for 'generate'':
|
|
351
|
+
Patterns of type 'FizzBuzzType' not matched: Type03
|
|
352
|
+
```
|
|
353
|
+
|
|
354
|
+
GHC は `FizzBuzzType` が `Type01`、`Type02`、`Type03` の 3 つの値コンストラクタを持つことを知っているため、`Type03` が欠けていることを指摘できます。これは、後で新しいタイプ(`Type04` など)を追加した場合にも、すべてのパターンマッチ箇所で警告が出るため、実装漏れを防げます。
|
|
355
|
+
|
|
356
|
+
### -Wall オプション
|
|
357
|
+
|
|
358
|
+
`package.yaml` で `-Wall` を有効にすると、網羅性警告を含むすべての警告が有効になります。
|
|
359
|
+
|
|
360
|
+
```yaml
|
|
361
|
+
ghc-options:
|
|
362
|
+
- -Wall
|
|
363
|
+
```
|
|
364
|
+
|
|
365
|
+
本プロジェクトでは `-Wall` を有効にして、パターンマッチの網羅性を含む潜在的な問題を早期に検出しています。
|
|
366
|
+
|
|
367
|
+
## 8.7 パターンマッチのベストプラクティス
|
|
368
|
+
|
|
369
|
+
この章で扱ったパターンマッチの知見をベストプラクティスとしてまとめます。
|
|
370
|
+
|
|
371
|
+
1. **網羅性を確保する**: すべてのコンストラクタをカバーするか、ワイルドカード `_` でフォールバックを用意する
|
|
372
|
+
2. **ガードの順序に注意**: より具体的な条件を先に書く(15 の倍数 → 3 の倍数 → 5 の倍数)
|
|
373
|
+
3. **ワイルドカードは最後に**: `_` より下のパターンには到達できない
|
|
374
|
+
4. **`-Wall` を有効にする**: コンパイラの網羅性チェックを活用する
|
|
375
|
+
5. **`Either` でエラーを型安全に扱う**: 例外ではなく戻り値でエラーを表現する
|
|
376
|
+
6. **関数引数パターンマッチを優先**: トップレベル関数では `case` 式より関数定義のパターンマッチを使う
|
|
377
|
+
|
|
378
|
+
## 8.8 まとめ
|
|
379
|
+
|
|
380
|
+
この章では以下のことを学びました。
|
|
381
|
+
|
|
382
|
+
- **関数引数のパターンマッチ** で値コンストラクタごとに処理を分岐する
|
|
383
|
+
- **ガード式** で条件分岐を追加し、FizzBuzz ロジックを自然に記述する
|
|
384
|
+
- **ガードの評価順序** と **otherwise** の仕組みを理解する
|
|
385
|
+
- **case 式** と関数引数パターンマッチの使い分け
|
|
386
|
+
- **Either 型** による型安全なエラーハンドリング(`createType` のスマートコンストラクタ)
|
|
387
|
+
- GHC の **パターンマッチ網羅性チェック** による安全性の確保
|
|
388
|
+
- **`-Wall`** オプションによる警告の有効化
|
|
389
|
+
|
|
390
|
+
次章では、モジュール設計とスマートコンストラクタのパターンを詳しく見ていきます。
|