@k2works/claude-code-booster 3.6.1 → 3.8.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (713) hide show
  1. package/LICENSE +21 -21
  2. package/README.md +42 -42
  3. package/bin/claude-code-booster +90 -90
  4. package/lib/assets/.claude/README.md +258 -239
  5. package/lib/assets/.claude/agent-memory/xp-programmer/MEMORY.md +6 -0
  6. package/lib/assets/.claude/agent-memory/xp-programmer/project_cargo_tracker.md +11 -0
  7. package/lib/assets/.claude/agent-memory/xp-programmer/project_ddd_patterns.md +27 -0
  8. package/lib/assets/.claude/agent-memory/xp-programmer/project_us07_route_assignment.md +19 -0
  9. package/lib/assets/.claude/scripts/generate-inception-deck.mjs +911 -911
  10. package/lib/assets/.claude/settings.json +11 -11
  11. package/lib/assets/.claude/skills/ai-agent-guidelines/SKILL.md +111 -111
  12. package/lib/assets/.claude/skills/analyzing-architecture/SKILL.md +83 -83
  13. package/lib/assets/.claude/skills/analyzing-business/SKILL.md +95 -95
  14. package/lib/assets/.claude/skills/analyzing-data-model/SKILL.md +77 -77
  15. package/lib/assets/.claude/skills/analyzing-domain-model/SKILL.md +117 -117
  16. package/lib/assets/.claude/skills/analyzing-inception-deck/SKILL.md +84 -84
  17. package/lib/assets/.claude/skills/analyzing-non-functional/SKILL.md +95 -95
  18. package/lib/assets/.claude/skills/analyzing-operation/SKILL.md +95 -95
  19. package/lib/assets/.claude/skills/analyzing-requirements/SKILL.md +91 -91
  20. package/lib/assets/.claude/skills/analyzing-tech-stack/SKILL.md +101 -101
  21. package/lib/assets/.claude/skills/analyzing-test-strategy/SKILL.md +89 -89
  22. package/lib/assets/.claude/skills/analyzing-ui-design/SKILL.md +80 -80
  23. package/lib/assets/.claude/skills/analyzing-usecases/SKILL.md +72 -72
  24. package/lib/assets/.claude/skills/creating-adr/SKILL.md +113 -113
  25. package/lib/assets/.claude/skills/developing-backend/SKILL.md +100 -100
  26. package/lib/assets/.claude/skills/developing-frontend/SKILL.md +93 -93
  27. package/lib/assets/.claude/skills/developing-release/SKILL.md +120 -120
  28. package/lib/assets/.claude/skills/generating-bmc/SKILL.md +97 -0
  29. package/lib/assets/.claude/skills/generating-slides/SKILL.md +94 -94
  30. package/lib/assets/.claude/skills/git-commit/SKILL.md +81 -81
  31. package/lib/assets/.claude/skills/killing-processes/SKILL.md +44 -44
  32. package/lib/assets/.claude/skills/operating-backup/SKILL.md +59 -59
  33. package/lib/assets/.claude/skills/operating-cicd/SKILL.md +54 -54
  34. package/lib/assets/.claude/skills/operating-deploy/SKILL.md +67 -67
  35. package/lib/assets/.claude/skills/operating-docs/SKILL.md +219 -219
  36. package/lib/assets/.claude/skills/operating-provision/SKILL.md +77 -77
  37. package/lib/assets/.claude/skills/operating-setup/SKILL.md +63 -63
  38. package/lib/assets/.claude/skills/orchestrating-analysis/SKILL.md +104 -104
  39. package/lib/assets/.claude/skills/orchestrating-development/SKILL.md +27 -21
  40. package/lib/assets/.claude/skills/orchestrating-operation/SKILL.md +158 -158
  41. package/lib/assets/.claude/skills/orchestrating-project/SKILL.md +144 -144
  42. package/lib/assets/.claude/skills/planning-releases/SKILL.md +119 -119
  43. package/lib/assets/.claude/skills/syncing-github-project/SKILL.md +151 -151
  44. package/lib/assets/.claude/skills/tracking-progress/SKILL.md +91 -91
  45. package/lib/assets/.claude/skills/validating-iteration-plan/SKILL.md +215 -215
  46. package/lib/assets/.devcontainer/devcontainer.json +34 -34
  47. package/lib/assets/.env.example +17 -17
  48. package/lib/assets/.gitattributes +4 -4
  49. package/lib/assets/.github/workflows/docker-publish.yml +77 -77
  50. package/lib/assets/.github/workflows/mkdocs.yml +39 -39
  51. package/lib/assets/AGENTS.md +94 -94
  52. package/lib/assets/CLAUDE.md +1 -0
  53. package/lib/assets/README.md +254 -254
  54. package/lib/assets/docker-compose.yml +33 -33
  55. package/lib/assets/docs/adr/index.md +10 -10
  56. package/lib/assets/docs/article/functional-desgin-ppp/all/01-immutability-and-data-transformation.md +475 -475
  57. package/lib/assets/docs/article/functional-desgin-ppp/all/02-function-composition.md +519 -519
  58. package/lib/assets/docs/article/functional-desgin-ppp/all/03-polymorphism.md +537 -537
  59. package/lib/assets/docs/article/functional-desgin-ppp/all/04-data-validation.md +300 -300
  60. package/lib/assets/docs/article/functional-desgin-ppp/all/05-property-based-testing.md +320 -320
  61. package/lib/assets/docs/article/functional-desgin-ppp/all/06-tdd-and-functional.md +498 -498
  62. package/lib/assets/docs/article/functional-desgin-ppp/all/07-composite-pattern.md +298 -298
  63. package/lib/assets/docs/article/functional-desgin-ppp/all/08-decorator-pattern.md +291 -291
  64. package/lib/assets/docs/article/functional-desgin-ppp/all/09-adapter-pattern.md +336 -336
  65. package/lib/assets/docs/article/functional-desgin-ppp/all/10-strategy-pattern.md +303 -303
  66. package/lib/assets/docs/article/functional-desgin-ppp/all/11-command-pattern.md +286 -286
  67. package/lib/assets/docs/article/functional-desgin-ppp/all/12-visitor-pattern.md +322 -322
  68. package/lib/assets/docs/article/functional-desgin-ppp/all/13-abstract-factory-pattern.md +319 -319
  69. package/lib/assets/docs/article/functional-desgin-ppp/all/14-abstract-server-pattern.md +365 -365
  70. package/lib/assets/docs/article/functional-desgin-ppp/all/15-gossiping-bus-drivers.md +156 -156
  71. package/lib/assets/docs/article/functional-desgin-ppp/all/16-payroll-system.md +178 -178
  72. package/lib/assets/docs/article/functional-desgin-ppp/all/17-video-rental-system.md +312 -312
  73. package/lib/assets/docs/article/functional-desgin-ppp/all/18-concurrency-system.md +287 -287
  74. package/lib/assets/docs/article/functional-desgin-ppp/all/19-wa-tor-simulation.md +286 -286
  75. package/lib/assets/docs/article/functional-desgin-ppp/all/20-pattern-interactions.md +274 -274
  76. package/lib/assets/docs/article/functional-desgin-ppp/all/21-best-practices.md +294 -294
  77. package/lib/assets/docs/article/functional-desgin-ppp/all/22-oo-to-fp-migration.md +337 -337
  78. package/lib/assets/docs/article/functional-desgin-ppp/all/index.md +388 -388
  79. package/lib/assets/docs/article/functional-desgin-ppp/clojure/01-immutability-and-data-transformation.md +273 -273
  80. package/lib/assets/docs/article/functional-desgin-ppp/clojure/02-function-composition.md +380 -380
  81. package/lib/assets/docs/article/functional-desgin-ppp/clojure/03-polymorphism.md +384 -384
  82. package/lib/assets/docs/article/functional-desgin-ppp/clojure/04-clojure-spec.md +350 -350
  83. package/lib/assets/docs/article/functional-desgin-ppp/clojure/05-property-based-testing.md +352 -352
  84. package/lib/assets/docs/article/functional-desgin-ppp/clojure/06-tdd-in-functional.md +383 -383
  85. package/lib/assets/docs/article/functional-desgin-ppp/clojure/07-composite-pattern.md +529 -529
  86. package/lib/assets/docs/article/functional-desgin-ppp/clojure/08-decorator-pattern.md +395 -395
  87. package/lib/assets/docs/article/functional-desgin-ppp/clojure/09-adapter-pattern.md +399 -399
  88. package/lib/assets/docs/article/functional-desgin-ppp/clojure/10-strategy-pattern.md +485 -485
  89. package/lib/assets/docs/article/functional-desgin-ppp/clojure/11-command-pattern.md +566 -566
  90. package/lib/assets/docs/article/functional-desgin-ppp/clojure/12-visitor-pattern.md +567 -567
  91. package/lib/assets/docs/article/functional-desgin-ppp/clojure/13-abstract-factory-pattern.md +475 -475
  92. package/lib/assets/docs/article/functional-desgin-ppp/clojure/14-abstract-server-pattern.md +462 -462
  93. package/lib/assets/docs/article/functional-desgin-ppp/clojure/15-gossiping-bus-drivers.md +325 -325
  94. package/lib/assets/docs/article/functional-desgin-ppp/clojure/16-payroll-system.md +401 -401
  95. package/lib/assets/docs/article/functional-desgin-ppp/clojure/17-video-rental-system.md +450 -450
  96. package/lib/assets/docs/article/functional-desgin-ppp/clojure/18-concurrency-system.md +475 -475
  97. package/lib/assets/docs/article/functional-desgin-ppp/clojure/19-wator-simulation.md +739 -739
  98. package/lib/assets/docs/article/functional-desgin-ppp/clojure/20-pattern-interactions.md +567 -567
  99. package/lib/assets/docs/article/functional-desgin-ppp/clojure/21-best-practices.md +518 -518
  100. package/lib/assets/docs/article/functional-desgin-ppp/clojure/22-oo-to-fp-migration.md +532 -532
  101. package/lib/assets/docs/article/functional-desgin-ppp/clojure/index.md +241 -241
  102. package/lib/assets/docs/article/functional-desgin-ppp/elixir/01-immutability-and-data-transformation.md +383 -383
  103. package/lib/assets/docs/article/functional-desgin-ppp/elixir/02-function-composition.md +374 -374
  104. package/lib/assets/docs/article/functional-desgin-ppp/elixir/03-polymorphism.md +375 -375
  105. package/lib/assets/docs/article/functional-desgin-ppp/elixir/04-data-validation.md +195 -195
  106. package/lib/assets/docs/article/functional-desgin-ppp/elixir/05-property-based-testing.md +268 -268
  107. package/lib/assets/docs/article/functional-desgin-ppp/elixir/06-tdd-and-fp.md +294 -294
  108. package/lib/assets/docs/article/functional-desgin-ppp/elixir/07-effects-and-pure-functions.md +164 -164
  109. package/lib/assets/docs/article/functional-desgin-ppp/elixir/08-error-handling-strategies.md +168 -168
  110. package/lib/assets/docs/article/functional-desgin-ppp/elixir/09-io-and-external-systems.md +254 -254
  111. package/lib/assets/docs/article/functional-desgin-ppp/elixir/10-concurrency-patterns.md +269 -269
  112. package/lib/assets/docs/article/functional-desgin-ppp/elixir/11-command-pattern.md +148 -148
  113. package/lib/assets/docs/article/functional-desgin-ppp/elixir/12-visitor-pattern.md +176 -176
  114. package/lib/assets/docs/article/functional-desgin-ppp/elixir/13-abstract-factory-pattern.md +604 -604
  115. package/lib/assets/docs/article/functional-desgin-ppp/elixir/14-abstract-server-pattern.md +729 -729
  116. package/lib/assets/docs/article/functional-desgin-ppp/elixir/15-gossiping-bus-drivers.md +291 -291
  117. package/lib/assets/docs/article/functional-desgin-ppp/elixir/16-payroll-system.md +420 -420
  118. package/lib/assets/docs/article/functional-desgin-ppp/elixir/17-video-rental-system.md +319 -319
  119. package/lib/assets/docs/article/functional-desgin-ppp/elixir/18-concurrency-system.md +466 -466
  120. package/lib/assets/docs/article/functional-desgin-ppp/elixir/19-wator-simulation.md +523 -523
  121. package/lib/assets/docs/article/functional-desgin-ppp/elixir/20-pattern-interactions.md +287 -287
  122. package/lib/assets/docs/article/functional-desgin-ppp/elixir/21-best-practices.md +340 -340
  123. package/lib/assets/docs/article/functional-desgin-ppp/elixir/22-oo-to-fp-migration.md +395 -395
  124. package/lib/assets/docs/article/functional-desgin-ppp/elixir/index.md +248 -248
  125. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/01-immutability-and-data-transformation.md +384 -384
  126. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/02-function-composition.md +452 -452
  127. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/03-polymorphism.md +495 -495
  128. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/04-data-validation.md +416 -416
  129. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/05-property-based-testing.md +382 -382
  130. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/06-tdd-functional.md +687 -687
  131. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/07-composite-pattern.md +442 -442
  132. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/08-decorator-pattern.md +479 -479
  133. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/09-adapter-pattern.md +479 -479
  134. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/10-strategy-pattern.md +427 -427
  135. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/11-command-pattern.md +428 -428
  136. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/12-visitor-pattern.md +339 -339
  137. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/13-abstract-factory-pattern.md +309 -309
  138. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/14-abstract-server-pattern.md +596 -596
  139. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/15-gossiping-bus-drivers.md +355 -355
  140. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/16-payroll-system.md +350 -350
  141. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/17-video-rental-system.md +414 -414
  142. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/18-concurrency-system.md +367 -367
  143. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/19-wator-simulation.md +403 -403
  144. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/20-pattern-interactions.md +291 -291
  145. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/21-best-practices.md +324 -324
  146. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/22-oo-to-fp-migration.md +332 -332
  147. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/index.md +274 -274
  148. package/lib/assets/docs/article/functional-desgin-ppp/haskell/01-immutability-and-data-transformation.md +298 -298
  149. package/lib/assets/docs/article/functional-desgin-ppp/haskell/02-function-composition.md +304 -304
  150. package/lib/assets/docs/article/functional-desgin-ppp/haskell/03-polymorphism.md +362 -362
  151. package/lib/assets/docs/article/functional-desgin-ppp/haskell/04-data-validation.md +257 -257
  152. package/lib/assets/docs/article/functional-desgin-ppp/haskell/05-property-based-testing.md +254 -254
  153. package/lib/assets/docs/article/functional-desgin-ppp/haskell/06-tdd-functional.md +283 -283
  154. package/lib/assets/docs/article/functional-desgin-ppp/haskell/07-composite-pattern.md +395 -395
  155. package/lib/assets/docs/article/functional-desgin-ppp/haskell/08-decorator-pattern.md +319 -319
  156. package/lib/assets/docs/article/functional-desgin-ppp/haskell/09-adapter-pattern.md +382 -382
  157. package/lib/assets/docs/article/functional-desgin-ppp/haskell/10-strategy-pattern.md +287 -287
  158. package/lib/assets/docs/article/functional-desgin-ppp/haskell/11-command-pattern.md +303 -303
  159. package/lib/assets/docs/article/functional-desgin-ppp/haskell/12-visitor-pattern.md +326 -326
  160. package/lib/assets/docs/article/functional-desgin-ppp/haskell/13-abstract-factory-pattern.md +332 -332
  161. package/lib/assets/docs/article/functional-desgin-ppp/haskell/14-abstract-server-pattern.md +379 -379
  162. package/lib/assets/docs/article/functional-desgin-ppp/haskell/15-gossiping-bus-drivers.md +177 -177
  163. package/lib/assets/docs/article/functional-desgin-ppp/haskell/16-payroll-system.md +219 -219
  164. package/lib/assets/docs/article/functional-desgin-ppp/haskell/17-video-rental-system.md +244 -244
  165. package/lib/assets/docs/article/functional-desgin-ppp/haskell/18-concurrency-system.md +363 -363
  166. package/lib/assets/docs/article/functional-desgin-ppp/haskell/19-wator-simulation.md +438 -438
  167. package/lib/assets/docs/article/functional-desgin-ppp/haskell/20-pattern-interactions.md +325 -325
  168. package/lib/assets/docs/article/functional-desgin-ppp/haskell/21-best-practices.md +403 -403
  169. package/lib/assets/docs/article/functional-desgin-ppp/haskell/22-oo-to-fp-migration.md +469 -469
  170. package/lib/assets/docs/article/functional-desgin-ppp/haskell/index.md +174 -174
  171. package/lib/assets/docs/article/functional-desgin-ppp/index.md +90 -90
  172. package/lib/assets/docs/article/functional-desgin-ppp/rust/01-immutability-and-data-transformation.md +450 -450
  173. package/lib/assets/docs/article/functional-desgin-ppp/rust/02-function-composition.md +463 -463
  174. package/lib/assets/docs/article/functional-desgin-ppp/rust/03-polymorphism.md +425 -425
  175. package/lib/assets/docs/article/functional-desgin-ppp/rust/04-data-validation.md +273 -273
  176. package/lib/assets/docs/article/functional-desgin-ppp/rust/05-property-based-testing.md +247 -247
  177. package/lib/assets/docs/article/functional-desgin-ppp/rust/06-tdd-and-functional.md +841 -841
  178. package/lib/assets/docs/article/functional-desgin-ppp/rust/07-composite-pattern.md +384 -384
  179. package/lib/assets/docs/article/functional-desgin-ppp/rust/08-decorator-pattern.md +383 -383
  180. package/lib/assets/docs/article/functional-desgin-ppp/rust/09-adapter-pattern.md +339 -339
  181. package/lib/assets/docs/article/functional-desgin-ppp/rust/10-strategy-pattern.md +331 -331
  182. package/lib/assets/docs/article/functional-desgin-ppp/rust/11-command-pattern.md +356 -356
  183. package/lib/assets/docs/article/functional-desgin-ppp/rust/12-visitor-pattern.md +379 -379
  184. package/lib/assets/docs/article/functional-desgin-ppp/rust/13-abstract-factory-pattern.md +361 -361
  185. package/lib/assets/docs/article/functional-desgin-ppp/rust/14-abstract-server-pattern.md +392 -392
  186. package/lib/assets/docs/article/functional-desgin-ppp/rust/15-gossiping-bus-drivers.md +300 -300
  187. package/lib/assets/docs/article/functional-desgin-ppp/rust/16-payroll-system.md +297 -297
  188. package/lib/assets/docs/article/functional-desgin-ppp/rust/17-video-rental-system.md +304 -304
  189. package/lib/assets/docs/article/functional-desgin-ppp/rust/18-concurrency-system.md +315 -315
  190. package/lib/assets/docs/article/functional-desgin-ppp/rust/19-wator-simulation.md +311 -311
  191. package/lib/assets/docs/article/functional-desgin-ppp/rust/20-pattern-interactions.md +304 -304
  192. package/lib/assets/docs/article/functional-desgin-ppp/rust/21-best-practices.md +336 -336
  193. package/lib/assets/docs/article/functional-desgin-ppp/rust/22-oo-to-fp-migration.md +349 -349
  194. package/lib/assets/docs/article/functional-desgin-ppp/rust/index.md +243 -243
  195. package/lib/assets/docs/article/functional-desgin-ppp/scala/01-immutability-and-data-transformation.md +328 -328
  196. package/lib/assets/docs/article/functional-desgin-ppp/scala/02-function-composition.md +348 -348
  197. package/lib/assets/docs/article/functional-desgin-ppp/scala/03-polymorphism.md +357 -357
  198. package/lib/assets/docs/article/functional-desgin-ppp/scala/04-data-validation.md +364 -364
  199. package/lib/assets/docs/article/functional-desgin-ppp/scala/05-property-based-testing.md +515 -515
  200. package/lib/assets/docs/article/functional-desgin-ppp/scala/06-tdd-functional.md +557 -557
  201. package/lib/assets/docs/article/functional-desgin-ppp/scala/07-composite-pattern.md +363 -363
  202. package/lib/assets/docs/article/functional-desgin-ppp/scala/08-decorator-pattern.md +327 -327
  203. package/lib/assets/docs/article/functional-desgin-ppp/scala/09-adapter-pattern.md +517 -517
  204. package/lib/assets/docs/article/functional-desgin-ppp/scala/10-strategy-pattern.md +441 -441
  205. package/lib/assets/docs/article/functional-desgin-ppp/scala/11-command-pattern.md +407 -407
  206. package/lib/assets/docs/article/functional-desgin-ppp/scala/12-visitor-pattern.md +379 -379
  207. package/lib/assets/docs/article/functional-desgin-ppp/scala/13-abstract-factory-pattern.md +398 -398
  208. package/lib/assets/docs/article/functional-desgin-ppp/scala/14-abstract-server-pattern.md +476 -476
  209. package/lib/assets/docs/article/functional-desgin-ppp/scala/15-gossiping-bus-drivers.md +391 -391
  210. package/lib/assets/docs/article/functional-desgin-ppp/scala/16-payroll-system.md +342 -342
  211. package/lib/assets/docs/article/functional-desgin-ppp/scala/17-video-rental-system.md +324 -324
  212. package/lib/assets/docs/article/functional-desgin-ppp/scala/18-concurrency-system.md +730 -730
  213. package/lib/assets/docs/article/functional-desgin-ppp/scala/19-wator-simulation.md +624 -624
  214. package/lib/assets/docs/article/functional-desgin-ppp/scala/20-pattern-interactions.md +512 -512
  215. package/lib/assets/docs/article/functional-desgin-ppp/scala/21-best-practices.md +433 -433
  216. package/lib/assets/docs/article/functional-desgin-ppp/scala/22-oo-to-fp-migration.md +688 -688
  217. package/lib/assets/docs/article/functional-desgin-ppp/scala/index.md +243 -243
  218. package/lib/assets/docs/article/getting-start-tdd/clojure/01-todo-list-and-first-test.md +166 -166
  219. package/lib/assets/docs/article/getting-start-tdd/clojure/02-fake-it-and-triangulation.md +162 -162
  220. package/lib/assets/docs/article/getting-start-tdd/clojure/03-obvious-implementation-and-refactoring.md +135 -135
  221. package/lib/assets/docs/article/getting-start-tdd/clojure/04-version-control-and-conventional-commits.md +88 -88
  222. package/lib/assets/docs/article/getting-start-tdd/clojure/05-package-management-and-static-analysis.md +299 -299
  223. package/lib/assets/docs/article/getting-start-tdd/clojure/06-task-runner-and-ci-cd.md +241 -241
  224. package/lib/assets/docs/article/getting-start-tdd/clojure/07-protocols-and-records.md +131 -131
  225. package/lib/assets/docs/article/getting-start-tdd/clojure/08-multimethods-and-design-patterns.md +130 -130
  226. package/lib/assets/docs/article/getting-start-tdd/clojure/09-namespaces-and-module-design.md +127 -127
  227. package/lib/assets/docs/article/getting-start-tdd/clojure/10-higher-order-functions-and-composition.md +114 -114
  228. package/lib/assets/docs/article/getting-start-tdd/clojure/11-persistent-data-and-pipeline.md +138 -138
  229. package/lib/assets/docs/article/getting-start-tdd/clojure/12-error-handling-and-spec.md +161 -161
  230. package/lib/assets/docs/article/getting-start-tdd/clojure/index.md +65 -65
  231. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter01.md +232 -232
  232. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter02.md +244 -244
  233. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter03.md +202 -202
  234. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter04.md +92 -92
  235. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter05.md +256 -256
  236. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter06.md +195 -195
  237. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter07.md +214 -214
  238. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter08.md +249 -249
  239. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter09.md +174 -174
  240. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter10.md +166 -166
  241. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter11.md +192 -192
  242. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter12.md +211 -211
  243. package/lib/assets/docs/article/getting-start-tdd/csharp/index.md +83 -83
  244. package/lib/assets/docs/article/getting-start-tdd/elixir/01-todo-list-and-first-test.md +87 -87
  245. package/lib/assets/docs/article/getting-start-tdd/elixir/02-fake-it-and-triangulation.md +95 -95
  246. package/lib/assets/docs/article/getting-start-tdd/elixir/03-obvious-implementation-and-refactoring.md +109 -109
  247. package/lib/assets/docs/article/getting-start-tdd/elixir/04-version-control-and-conventional-commits.md +96 -96
  248. package/lib/assets/docs/article/getting-start-tdd/elixir/05-package-management-and-static-analysis.md +88 -88
  249. package/lib/assets/docs/article/getting-start-tdd/elixir/06-task-runner-and-ci-cd.md +71 -71
  250. package/lib/assets/docs/article/getting-start-tdd/elixir/07-structs-and-protocols.md +110 -110
  251. package/lib/assets/docs/article/getting-start-tdd/elixir/08-pattern-matching-and-guards.md +108 -108
  252. package/lib/assets/docs/article/getting-start-tdd/elixir/09-module-design-and-behaviours.md +104 -104
  253. package/lib/assets/docs/article/getting-start-tdd/elixir/10-higher-order-functions-and-pipeline.md +178 -178
  254. package/lib/assets/docs/article/getting-start-tdd/elixir/11-stream-and-lazy-evaluation.md +142 -142
  255. package/lib/assets/docs/article/getting-start-tdd/elixir/12-error-handling-and-with.md +145 -145
  256. package/lib/assets/docs/article/getting-start-tdd/elixir/index.md +35 -35
  257. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter01.md +202 -202
  258. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter02.md +246 -246
  259. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter03.md +218 -218
  260. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter04.md +179 -179
  261. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter05.md +267 -267
  262. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter06.md +190 -190
  263. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter07.md +161 -161
  264. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter08.md +175 -175
  265. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter09.md +222 -222
  266. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter10.md +189 -189
  267. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter11.md +212 -212
  268. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter12.md +215 -215
  269. package/lib/assets/docs/article/getting-start-tdd/fsharp/index.md +71 -71
  270. package/lib/assets/docs/article/getting-start-tdd/go/01-todo-list-and-first-test.md +213 -213
  271. package/lib/assets/docs/article/getting-start-tdd/go/02-fake-it-and-triangulation.md +302 -302
  272. package/lib/assets/docs/article/getting-start-tdd/go/03-obvious-implementation-and-refactoring.md +339 -339
  273. package/lib/assets/docs/article/getting-start-tdd/go/04-version-control-and-conventional-commits.md +112 -112
  274. package/lib/assets/docs/article/getting-start-tdd/go/05-package-management-and-static-analysis.md +272 -272
  275. package/lib/assets/docs/article/getting-start-tdd/go/06-task-runner-and-ci-cd.md +233 -233
  276. package/lib/assets/docs/article/getting-start-tdd/go/07-encapsulation-and-polymorphism.md +394 -394
  277. package/lib/assets/docs/article/getting-start-tdd/go/08-design-patterns.md +422 -422
  278. package/lib/assets/docs/article/getting-start-tdd/go/09-solid-principles-and-module-design.md +400 -400
  279. package/lib/assets/docs/article/getting-start-tdd/go/10-higher-order-functions-and-composition.md +226 -226
  280. package/lib/assets/docs/article/getting-start-tdd/go/11-immutable-data-and-pipeline.md +296 -296
  281. package/lib/assets/docs/article/getting-start-tdd/go/12-error-handling-and-type-safety.md +411 -411
  282. package/lib/assets/docs/article/getting-start-tdd/go/index.md +83 -83
  283. package/lib/assets/docs/article/getting-start-tdd/haskell/01-todo-list-and-first-test.md +279 -279
  284. package/lib/assets/docs/article/getting-start-tdd/haskell/02-fake-it-and-triangulation.md +337 -337
  285. package/lib/assets/docs/article/getting-start-tdd/haskell/03-obvious-implementation-and-refactoring.md +257 -257
  286. package/lib/assets/docs/article/getting-start-tdd/haskell/04-version-control-and-conventional-commits.md +182 -182
  287. package/lib/assets/docs/article/getting-start-tdd/haskell/05-package-management-and-static-analysis.md +313 -313
  288. package/lib/assets/docs/article/getting-start-tdd/haskell/06-task-runner-and-ci-cd.md +309 -309
  289. package/lib/assets/docs/article/getting-start-tdd/haskell/07-algebraic-data-types-and-type-classes.md +412 -412
  290. package/lib/assets/docs/article/getting-start-tdd/haskell/08-pattern-matching-and-guards.md +390 -390
  291. package/lib/assets/docs/article/getting-start-tdd/haskell/09-module-design-and-smart-constructors.md +461 -461
  292. package/lib/assets/docs/article/getting-start-tdd/haskell/10-higher-order-functions-and-currying.md +434 -434
  293. package/lib/assets/docs/article/getting-start-tdd/haskell/11-function-composition-and-point-free.md +392 -392
  294. package/lib/assets/docs/article/getting-start-tdd/haskell/12-monad-and-error-handling.md +631 -631
  295. package/lib/assets/docs/article/getting-start-tdd/haskell/index.md +49 -49
  296. package/lib/assets/docs/article/getting-start-tdd/index.md +93 -93
  297. package/lib/assets/docs/article/getting-start-tdd/integration/01-language-overview.md +375 -375
  298. package/lib/assets/docs/article/getting-start-tdd/integration/02-test-framework-comparison.md +349 -349
  299. package/lib/assets/docs/article/getting-start-tdd/integration/03-tdd-pattern-comparison.md +445 -445
  300. package/lib/assets/docs/article/getting-start-tdd/integration/04-type-system-comparison.md +409 -409
  301. package/lib/assets/docs/article/getting-start-tdd/integration/05-dev-environment-comparison.md +330 -330
  302. package/lib/assets/docs/article/getting-start-tdd/integration/06-learning-roadmap.md +290 -290
  303. package/lib/assets/docs/article/getting-start-tdd/integration/index.md +69 -69
  304. package/lib/assets/docs/article/getting-start-tdd/java/01-todo-list-and-first-test.md +234 -234
  305. package/lib/assets/docs/article/getting-start-tdd/java/02-fake-it-and-triangulation.md +261 -261
  306. package/lib/assets/docs/article/getting-start-tdd/java/03-obvious-implementation-and-refactoring.md +185 -185
  307. package/lib/assets/docs/article/getting-start-tdd/java/04-version-control-and-conventional-commits.md +115 -115
  308. package/lib/assets/docs/article/getting-start-tdd/java/05-package-management-and-static-analysis.md +382 -382
  309. package/lib/assets/docs/article/getting-start-tdd/java/06-task-runner-and-ci-cd.md +272 -272
  310. package/lib/assets/docs/article/getting-start-tdd/java/07-encapsulation-and-polymorphism.md +626 -626
  311. package/lib/assets/docs/article/getting-start-tdd/java/08-design-patterns.md +393 -393
  312. package/lib/assets/docs/article/getting-start-tdd/java/09-solid-principles-and-module-design.md +310 -310
  313. package/lib/assets/docs/article/getting-start-tdd/java/10-higher-order-functions-and-composition.md +188 -188
  314. package/lib/assets/docs/article/getting-start-tdd/java/11-immutable-data-and-pipeline.md +167 -167
  315. package/lib/assets/docs/article/getting-start-tdd/java/12-error-handling-and-type-safety.md +205 -205
  316. package/lib/assets/docs/article/getting-start-tdd/java/index.md +61 -61
  317. package/lib/assets/docs/article/getting-start-tdd/node/01-todo-list-and-first-test.md +244 -244
  318. package/lib/assets/docs/article/getting-start-tdd/node/02-fake-it-and-triangulation.md +262 -262
  319. package/lib/assets/docs/article/getting-start-tdd/node/03-obvious-implementation-and-refactoring.md +169 -169
  320. package/lib/assets/docs/article/getting-start-tdd/node/04-version-control-and-conventional-commits.md +112 -112
  321. package/lib/assets/docs/article/getting-start-tdd/node/05-package-management-and-static-analysis.md +314 -314
  322. package/lib/assets/docs/article/getting-start-tdd/node/06-task-runner-and-ci-cd.md +235 -235
  323. package/lib/assets/docs/article/getting-start-tdd/node/07-encapsulation-and-polymorphism.md +327 -327
  324. package/lib/assets/docs/article/getting-start-tdd/node/08-design-patterns.md +322 -322
  325. package/lib/assets/docs/article/getting-start-tdd/node/09-solid-principles-and-module-design.md +285 -285
  326. package/lib/assets/docs/article/getting-start-tdd/node/10-higher-order-functions-and-composition.md +199 -199
  327. package/lib/assets/docs/article/getting-start-tdd/node/11-immutable-data-and-pipeline.md +207 -207
  328. package/lib/assets/docs/article/getting-start-tdd/node/12-error-handling-and-type-safety.md +295 -295
  329. package/lib/assets/docs/article/getting-start-tdd/node/index.md +56 -56
  330. package/lib/assets/docs/article/getting-start-tdd/php/01-todo-list-and-first-test.md +259 -259
  331. package/lib/assets/docs/article/getting-start-tdd/php/02-fake-it-and-triangulation.md +200 -200
  332. package/lib/assets/docs/article/getting-start-tdd/php/03-obvious-implementation-and-refactoring.md +248 -248
  333. package/lib/assets/docs/article/getting-start-tdd/php/04-version-control-and-conventional-commits.md +141 -141
  334. package/lib/assets/docs/article/getting-start-tdd/php/05-package-management-and-static-analysis.md +410 -410
  335. package/lib/assets/docs/article/getting-start-tdd/php/06-task-runner-and-ci-cd.md +321 -321
  336. package/lib/assets/docs/article/getting-start-tdd/php/07-encapsulation-and-polymorphism.md +372 -372
  337. package/lib/assets/docs/article/getting-start-tdd/php/08-design-patterns.md +453 -453
  338. package/lib/assets/docs/article/getting-start-tdd/php/09-solid-principles-and-module-design.md +460 -460
  339. package/lib/assets/docs/article/getting-start-tdd/php/10-higher-order-functions-and-composition.md +182 -182
  340. package/lib/assets/docs/article/getting-start-tdd/php/11-immutable-data-and-pipeline.md +266 -266
  341. package/lib/assets/docs/article/getting-start-tdd/php/12-error-handling-and-type-safety.md +308 -308
  342. package/lib/assets/docs/article/getting-start-tdd/php/index.md +84 -84
  343. package/lib/assets/docs/article/getting-start-tdd/python/01-todo-list-and-first-test.md +201 -201
  344. package/lib/assets/docs/article/getting-start-tdd/python/02-fake-it-and-triangulation.md +247 -247
  345. package/lib/assets/docs/article/getting-start-tdd/python/03-obvious-implementation-and-refactoring.md +199 -199
  346. package/lib/assets/docs/article/getting-start-tdd/python/04-version-control-and-conventional-commits.md +87 -87
  347. package/lib/assets/docs/article/getting-start-tdd/python/05-package-management-and-static-analysis.md +274 -274
  348. package/lib/assets/docs/article/getting-start-tdd/python/06-task-runner-and-ci-cd.md +190 -190
  349. package/lib/assets/docs/article/getting-start-tdd/python/07-encapsulation-and-polymorphism.md +208 -208
  350. package/lib/assets/docs/article/getting-start-tdd/python/08-design-patterns.md +172 -172
  351. package/lib/assets/docs/article/getting-start-tdd/python/09-solid-principles-and-module-design.md +130 -130
  352. package/lib/assets/docs/article/getting-start-tdd/python/10-higher-order-functions-and-composition.md +122 -122
  353. package/lib/assets/docs/article/getting-start-tdd/python/11-immutable-data-and-pipeline.md +116 -116
  354. package/lib/assets/docs/article/getting-start-tdd/python/12-error-handling-and-type-safety.md +126 -126
  355. package/lib/assets/docs/article/getting-start-tdd/python/index.md +55 -55
  356. package/lib/assets/docs/article/getting-start-tdd/ruby/01-todo-list-and-first-test.md +231 -231
  357. package/lib/assets/docs/article/getting-start-tdd/ruby/02-fake-it-and-triangulation.md +238 -238
  358. package/lib/assets/docs/article/getting-start-tdd/ruby/03-obvious-implementation-and-refactoring.md +228 -228
  359. package/lib/assets/docs/article/getting-start-tdd/ruby/04-version-control-and-conventional-commits.md +112 -112
  360. package/lib/assets/docs/article/getting-start-tdd/ruby/05-package-management-and-static-analysis.md +287 -287
  361. package/lib/assets/docs/article/getting-start-tdd/ruby/06-task-runner-and-ci-cd.md +248 -248
  362. package/lib/assets/docs/article/getting-start-tdd/ruby/07-encapsulation-and-polymorphism.md +279 -279
  363. package/lib/assets/docs/article/getting-start-tdd/ruby/08-design-patterns.md +329 -329
  364. package/lib/assets/docs/article/getting-start-tdd/ruby/09-solid-principles-and-module-design.md +196 -196
  365. package/lib/assets/docs/article/getting-start-tdd/ruby/10-higher-order-functions-and-composition.md +175 -175
  366. package/lib/assets/docs/article/getting-start-tdd/ruby/11-immutable-data-and-pipeline.md +237 -237
  367. package/lib/assets/docs/article/getting-start-tdd/ruby/12-error-handling-and-type-safety.md +398 -398
  368. package/lib/assets/docs/article/getting-start-tdd/ruby/index.md +83 -83
  369. package/lib/assets/docs/article/getting-start-tdd/rust/01-todo-list-and-first-test.md +211 -211
  370. package/lib/assets/docs/article/getting-start-tdd/rust/02-fake-it-and-triangulation.md +264 -264
  371. package/lib/assets/docs/article/getting-start-tdd/rust/03-obvious-implementation-and-refactoring.md +233 -233
  372. package/lib/assets/docs/article/getting-start-tdd/rust/04-version-control-and-conventional-commits.md +92 -92
  373. package/lib/assets/docs/article/getting-start-tdd/rust/05-package-management-and-static-analysis.md +212 -212
  374. package/lib/assets/docs/article/getting-start-tdd/rust/06-task-runner-and-ci-cd.md +164 -164
  375. package/lib/assets/docs/article/getting-start-tdd/rust/07-encapsulation-and-polymorphism.md +142 -142
  376. package/lib/assets/docs/article/getting-start-tdd/rust/08-design-patterns.md +145 -145
  377. package/lib/assets/docs/article/getting-start-tdd/rust/09-solid-principles-and-module-design.md +110 -110
  378. package/lib/assets/docs/article/getting-start-tdd/rust/10-higher-order-functions-and-composition.md +94 -94
  379. package/lib/assets/docs/article/getting-start-tdd/rust/11-immutable-data-and-pipeline.md +105 -105
  380. package/lib/assets/docs/article/getting-start-tdd/rust/12-error-handling-and-type-safety.md +112 -112
  381. package/lib/assets/docs/article/getting-start-tdd/rust/index.md +83 -83
  382. package/lib/assets/docs/article/getting-start-tdd/scala/01-todo-list-and-first-test.md +111 -111
  383. package/lib/assets/docs/article/getting-start-tdd/scala/02-fake-it-and-triangulation.md +107 -107
  384. package/lib/assets/docs/article/getting-start-tdd/scala/03-obvious-implementation-and-refactoring.md +99 -99
  385. package/lib/assets/docs/article/getting-start-tdd/scala/04-version-control-and-conventional-commits.md +123 -123
  386. package/lib/assets/docs/article/getting-start-tdd/scala/05-package-management-and-static-analysis.md +196 -196
  387. package/lib/assets/docs/article/getting-start-tdd/scala/06-task-runner-and-ci-cd.md +186 -186
  388. package/lib/assets/docs/article/getting-start-tdd/scala/07-case-classes-and-traits.md +139 -139
  389. package/lib/assets/docs/article/getting-start-tdd/scala/08-pattern-matching-and-sealed-traits.md +106 -106
  390. package/lib/assets/docs/article/getting-start-tdd/scala/09-packages-and-module-design.md +75 -75
  391. package/lib/assets/docs/article/getting-start-tdd/scala/10-higher-order-functions-and-composition.md +104 -104
  392. package/lib/assets/docs/article/getting-start-tdd/scala/11-collections-and-lazy-evaluation.md +94 -94
  393. package/lib/assets/docs/article/getting-start-tdd/scala/12-error-handling-and-type-safety.md +92 -92
  394. package/lib/assets/docs/article/getting-start-tdd/scala/index.md +65 -65
  395. package/lib/assets/docs/article/grokking-concurrency/all/index.md +404 -404
  396. package/lib/assets/docs/article/grokking-concurrency/all/part-1-ch02-sequential.md +554 -554
  397. package/lib/assets/docs/article/grokking-concurrency/all/part-2-ch04-05-threads.md +469 -469
  398. package/lib/assets/docs/article/grokking-concurrency/all/part-3-ch06-multitasking.md +520 -520
  399. package/lib/assets/docs/article/grokking-concurrency/all/part-4-ch07-parallel-patterns.md +420 -420
  400. package/lib/assets/docs/article/grokking-concurrency/all/part-5-ch08-09-synchronization.md +510 -510
  401. package/lib/assets/docs/article/grokking-concurrency/all/part-6-ch10-11-nonblocking-io.md +435 -435
  402. package/lib/assets/docs/article/grokking-concurrency/all/part-7-ch12-async.md +465 -465
  403. package/lib/assets/docs/article/grokking-concurrency/all/part-8-ch13-mapreduce.md +377 -377
  404. package/lib/assets/docs/article/grokking-concurrency/clojure/index.md +116 -116
  405. package/lib/assets/docs/article/grokking-concurrency/clojure/part-1.md +108 -108
  406. package/lib/assets/docs/article/grokking-concurrency/clojure/part-2.md +101 -101
  407. package/lib/assets/docs/article/grokking-concurrency/clojure/part-3.md +122 -122
  408. package/lib/assets/docs/article/grokking-concurrency/clojure/part-4.md +123 -123
  409. package/lib/assets/docs/article/grokking-concurrency/clojure/part-5.md +118 -118
  410. package/lib/assets/docs/article/grokking-concurrency/clojure/part-6.md +89 -89
  411. package/lib/assets/docs/article/grokking-concurrency/clojure/part-7.md +100 -100
  412. package/lib/assets/docs/article/grokking-concurrency/clojure/part-8.md +120 -120
  413. package/lib/assets/docs/article/grokking-concurrency/csharp/index.md +101 -101
  414. package/lib/assets/docs/article/grokking-concurrency/csharp/part-1.md +97 -97
  415. package/lib/assets/docs/article/grokking-concurrency/csharp/part-2.md +123 -123
  416. package/lib/assets/docs/article/grokking-concurrency/csharp/part-3.md +101 -101
  417. package/lib/assets/docs/article/grokking-concurrency/csharp/part-4.md +112 -112
  418. package/lib/assets/docs/article/grokking-concurrency/csharp/part-5.md +99 -99
  419. package/lib/assets/docs/article/grokking-concurrency/csharp/part-6.md +61 -61
  420. package/lib/assets/docs/article/grokking-concurrency/csharp/part-7.md +84 -84
  421. package/lib/assets/docs/article/grokking-concurrency/csharp/part-8.md +92 -92
  422. package/lib/assets/docs/article/grokking-concurrency/fsharp/index.md +65 -65
  423. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-1.md +80 -80
  424. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-2.md +103 -103
  425. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-3.md +94 -94
  426. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-4.md +110 -110
  427. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-5.md +104 -104
  428. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-6.md +93 -93
  429. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-7.md +121 -121
  430. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-8.md +107 -107
  431. package/lib/assets/docs/article/grokking-concurrency/haskell/index.md +248 -248
  432. package/lib/assets/docs/article/grokking-concurrency/haskell/part-1.md +96 -96
  433. package/lib/assets/docs/article/grokking-concurrency/haskell/part-2.md +96 -96
  434. package/lib/assets/docs/article/grokking-concurrency/haskell/part-3.md +91 -91
  435. package/lib/assets/docs/article/grokking-concurrency/haskell/part-4.md +106 -106
  436. package/lib/assets/docs/article/grokking-concurrency/haskell/part-5.md +99 -99
  437. package/lib/assets/docs/article/grokking-concurrency/haskell/part-6.md +95 -95
  438. package/lib/assets/docs/article/grokking-concurrency/haskell/part-7.md +111 -111
  439. package/lib/assets/docs/article/grokking-concurrency/haskell/part-8.md +118 -118
  440. package/lib/assets/docs/article/grokking-concurrency/index.md +66 -66
  441. package/lib/assets/docs/article/grokking-concurrency/java/index.md +102 -102
  442. package/lib/assets/docs/article/grokking-concurrency/java/part-1.md +308 -308
  443. package/lib/assets/docs/article/grokking-concurrency/java/part-2.md +334 -334
  444. package/lib/assets/docs/article/grokking-concurrency/java/part-3.md +221 -221
  445. package/lib/assets/docs/article/grokking-concurrency/java/part-4.md +213 -213
  446. package/lib/assets/docs/article/grokking-concurrency/java/part-5.md +112 -112
  447. package/lib/assets/docs/article/grokking-concurrency/java/part-6.md +69 -69
  448. package/lib/assets/docs/article/grokking-concurrency/java/part-7.md +101 -101
  449. package/lib/assets/docs/article/grokking-concurrency/java/part-8.md +101 -101
  450. package/lib/assets/docs/article/grokking-concurrency/python/index.md +313 -313
  451. package/lib/assets/docs/article/grokking-concurrency/python/part-1.md +239 -239
  452. package/lib/assets/docs/article/grokking-concurrency/python/part-2.md +418 -418
  453. package/lib/assets/docs/article/grokking-concurrency/python/part-3.md +227 -227
  454. package/lib/assets/docs/article/grokking-concurrency/python/part-4.md +299 -299
  455. package/lib/assets/docs/article/grokking-concurrency/python/part-5.md +315 -315
  456. package/lib/assets/docs/article/grokking-concurrency/python/part-6.md +297 -297
  457. package/lib/assets/docs/article/grokking-concurrency/python/part-7.md +314 -314
  458. package/lib/assets/docs/article/grokking-concurrency/python/part-8.md +360 -360
  459. package/lib/assets/docs/article/grokking-concurrency/rust/index.md +270 -270
  460. package/lib/assets/docs/article/grokking-concurrency/rust/part-1.md +108 -108
  461. package/lib/assets/docs/article/grokking-concurrency/rust/part-2.md +120 -120
  462. package/lib/assets/docs/article/grokking-concurrency/rust/part-3.md +126 -126
  463. package/lib/assets/docs/article/grokking-concurrency/rust/part-4.md +175 -175
  464. package/lib/assets/docs/article/grokking-concurrency/rust/part-5.md +158 -158
  465. package/lib/assets/docs/article/grokking-concurrency/rust/part-6.md +94 -94
  466. package/lib/assets/docs/article/grokking-concurrency/rust/part-7.md +133 -133
  467. package/lib/assets/docs/article/grokking-concurrency/rust/part-8.md +155 -155
  468. package/lib/assets/docs/article/grokking-concurrency/scala/index.md +69 -69
  469. package/lib/assets/docs/article/grokking-concurrency/scala/part-1.md +78 -78
  470. package/lib/assets/docs/article/grokking-concurrency/scala/part-2.md +112 -112
  471. package/lib/assets/docs/article/grokking-concurrency/scala/part-3.md +93 -93
  472. package/lib/assets/docs/article/grokking-concurrency/scala/part-4.md +110 -110
  473. package/lib/assets/docs/article/grokking-concurrency/scala/part-5.md +119 -119
  474. package/lib/assets/docs/article/grokking-concurrency/scala/part-6.md +83 -83
  475. package/lib/assets/docs/article/grokking-concurrency/scala/part-7.md +131 -131
  476. package/lib/assets/docs/article/grokking-concurrency/scala/part-8.md +129 -129
  477. package/lib/assets/docs/article/grokkingfp/all/index.md +368 -368
  478. package/lib/assets/docs/article/grokkingfp/all/part-1-ch01-fp-introduction.md +530 -530
  479. package/lib/assets/docs/article/grokkingfp/all/part-1-ch02-pure-functions.md +923 -923
  480. package/lib/assets/docs/article/grokkingfp/all/part-2-ch03-immutable-data.md +1128 -1128
  481. package/lib/assets/docs/article/grokkingfp/all/part-2-ch04-higher-order-functions.md +1104 -1104
  482. package/lib/assets/docs/article/grokkingfp/all/part-2-ch05-flatmap.md +1026 -1026
  483. package/lib/assets/docs/article/grokkingfp/all/part-3-ch06-option.md +785 -785
  484. package/lib/assets/docs/article/grokkingfp/all/part-3-ch07-either-adt.md +871 -871
  485. package/lib/assets/docs/article/grokkingfp/all/part-4-ch08-io-monad.md +972 -972
  486. package/lib/assets/docs/article/grokkingfp/all/part-4-ch09-streams.md +926 -926
  487. package/lib/assets/docs/article/grokkingfp/all/part-5-ch10-concurrency.md +870 -870
  488. package/lib/assets/docs/article/grokkingfp/all/part-6-ch11-application.md +715 -715
  489. package/lib/assets/docs/article/grokkingfp/all/part-6-ch12-testing.md +626 -626
  490. package/lib/assets/docs/article/grokkingfp/all/writing-plan.md +712 -712
  491. package/lib/assets/docs/article/grokkingfp/clojure/index.md +276 -276
  492. package/lib/assets/docs/article/grokkingfp/clojure/part-1.md +667 -667
  493. package/lib/assets/docs/article/grokkingfp/clojure/part-2.md +643 -643
  494. package/lib/assets/docs/article/grokkingfp/clojure/part-3.md +620 -620
  495. package/lib/assets/docs/article/grokkingfp/clojure/part-4.md +697 -697
  496. package/lib/assets/docs/article/grokkingfp/clojure/part-5.md +751 -751
  497. package/lib/assets/docs/article/grokkingfp/clojure/part-6.md +721 -721
  498. package/lib/assets/docs/article/grokkingfp/csharp/index.md +246 -246
  499. package/lib/assets/docs/article/grokkingfp/csharp/part-1.md +811 -811
  500. package/lib/assets/docs/article/grokkingfp/csharp/part-2.md +971 -971
  501. package/lib/assets/docs/article/grokkingfp/csharp/part-3.md +981 -981
  502. package/lib/assets/docs/article/grokkingfp/csharp/part-4.md +949 -949
  503. package/lib/assets/docs/article/grokkingfp/csharp/part-5.md +947 -947
  504. package/lib/assets/docs/article/grokkingfp/csharp/part-6.md +739 -739
  505. package/lib/assets/docs/article/grokkingfp/elixir/index.md +203 -203
  506. package/lib/assets/docs/article/grokkingfp/elixir/part-1.md +712 -712
  507. package/lib/assets/docs/article/grokkingfp/elixir/part-2.md +838 -838
  508. package/lib/assets/docs/article/grokkingfp/elixir/part-3.md +985 -985
  509. package/lib/assets/docs/article/grokkingfp/elixir/part-4.md +974 -974
  510. package/lib/assets/docs/article/grokkingfp/elixir/part-5.md +1286 -1286
  511. package/lib/assets/docs/article/grokkingfp/elixir/part-6.md +1049 -1049
  512. package/lib/assets/docs/article/grokkingfp/fsharp/index.md +210 -210
  513. package/lib/assets/docs/article/grokkingfp/fsharp/part-1.md +714 -714
  514. package/lib/assets/docs/article/grokkingfp/fsharp/part-2.md +961 -961
  515. package/lib/assets/docs/article/grokkingfp/fsharp/part-3.md +972 -972
  516. package/lib/assets/docs/article/grokkingfp/fsharp/part-4.md +832 -832
  517. package/lib/assets/docs/article/grokkingfp/fsharp/part-5.md +911 -911
  518. package/lib/assets/docs/article/grokkingfp/fsharp/part-6.md +922 -922
  519. package/lib/assets/docs/article/grokkingfp/haskell/index.md +234 -234
  520. package/lib/assets/docs/article/grokkingfp/haskell/part-1.md +591 -591
  521. package/lib/assets/docs/article/grokkingfp/haskell/part-2.md +866 -866
  522. package/lib/assets/docs/article/grokkingfp/haskell/part-3.md +915 -915
  523. package/lib/assets/docs/article/grokkingfp/haskell/part-4.md +878 -878
  524. package/lib/assets/docs/article/grokkingfp/haskell/part-5.md +845 -845
  525. package/lib/assets/docs/article/grokkingfp/haskell/part-6.md +844 -844
  526. package/lib/assets/docs/article/grokkingfp/index.md +143 -143
  527. package/lib/assets/docs/article/grokkingfp/java/index.md +211 -211
  528. package/lib/assets/docs/article/grokkingfp/java/part-1.md +648 -648
  529. package/lib/assets/docs/article/grokkingfp/java/part-2.md +675 -675
  530. package/lib/assets/docs/article/grokkingfp/java/part-3.md +672 -672
  531. package/lib/assets/docs/article/grokkingfp/java/part-4.md +771 -771
  532. package/lib/assets/docs/article/grokkingfp/java/part-5.md +959 -959
  533. package/lib/assets/docs/article/grokkingfp/java/part-6.md +1328 -1328
  534. package/lib/assets/docs/article/grokkingfp/python/index.md +258 -258
  535. package/lib/assets/docs/article/grokkingfp/python/part-1.md +443 -443
  536. package/lib/assets/docs/article/grokkingfp/python/part-2.md +958 -958
  537. package/lib/assets/docs/article/grokkingfp/python/part-3.md +1004 -1004
  538. package/lib/assets/docs/article/grokkingfp/python/part-4.md +765 -765
  539. package/lib/assets/docs/article/grokkingfp/python/part-5.md +747 -747
  540. package/lib/assets/docs/article/grokkingfp/python/part-6.md +861 -861
  541. package/lib/assets/docs/article/grokkingfp/ruby/index.md +330 -330
  542. package/lib/assets/docs/article/grokkingfp/ruby/part-1.md +755 -755
  543. package/lib/assets/docs/article/grokkingfp/ruby/part-2.md +938 -938
  544. package/lib/assets/docs/article/grokkingfp/ruby/part-3.md +946 -946
  545. package/lib/assets/docs/article/grokkingfp/ruby/part-4.md +921 -921
  546. package/lib/assets/docs/article/grokkingfp/ruby/part-5.md +908 -908
  547. package/lib/assets/docs/article/grokkingfp/ruby/part-6.md +1412 -1412
  548. package/lib/assets/docs/article/grokkingfp/rust/index.md +242 -242
  549. package/lib/assets/docs/article/grokkingfp/rust/part-1.md +634 -634
  550. package/lib/assets/docs/article/grokkingfp/rust/part-2.md +1060 -1060
  551. package/lib/assets/docs/article/grokkingfp/rust/part-3.md +994 -994
  552. package/lib/assets/docs/article/grokkingfp/rust/part-4.md +573 -573
  553. package/lib/assets/docs/article/grokkingfp/rust/part-5.md +705 -705
  554. package/lib/assets/docs/article/grokkingfp/rust/part-6.md +508 -508
  555. package/lib/assets/docs/article/grokkingfp/scala/index.md +171 -171
  556. package/lib/assets/docs/article/grokkingfp/scala/part-1.md +543 -543
  557. package/lib/assets/docs/article/grokkingfp/scala/part-2.md +946 -946
  558. package/lib/assets/docs/article/grokkingfp/scala/part-3.md +919 -919
  559. package/lib/assets/docs/article/grokkingfp/scala/part-4.md +742 -742
  560. package/lib/assets/docs/article/grokkingfp/scala/part-5.md +722 -722
  561. package/lib/assets/docs/article/grokkingfp/scala/part-6.md +867 -867
  562. package/lib/assets/docs/article/grokkingfp/typescript/index.md +273 -273
  563. package/lib/assets/docs/article/grokkingfp/typescript/part-1.md +561 -561
  564. package/lib/assets/docs/article/grokkingfp/typescript/part-2.md +1129 -1129
  565. package/lib/assets/docs/article/grokkingfp/typescript/part-3.md +842 -842
  566. package/lib/assets/docs/article/grokkingfp/typescript/part-4.md +1087 -1087
  567. package/lib/assets/docs/article/grokkingfp/typescript/part-5.md +717 -717
  568. package/lib/assets/docs/article/grokkingfp/typescript/part-6.md +982 -982
  569. package/lib/assets/docs/article/practical-database-design/index.md +121 -121
  570. package/lib/assets/docs/article/practical-database-design/part1/chapter01.md +288 -288
  571. package/lib/assets/docs/article/practical-database-design/part1/chapter02.md +518 -518
  572. package/lib/assets/docs/article/practical-database-design/part1/chapter03.md +557 -557
  573. package/lib/assets/docs/article/practical-database-design/part2/chapter04.md +924 -924
  574. package/lib/assets/docs/article/practical-database-design/part2/chapter05.md +1627 -1627
  575. package/lib/assets/docs/article/practical-database-design/part2/chapter06.md +2716 -2716
  576. package/lib/assets/docs/article/practical-database-design/part2/chapter07.md +2082 -2082
  577. package/lib/assets/docs/article/practical-database-design/part2/chapter08.md +2105 -2105
  578. package/lib/assets/docs/article/practical-database-design/part2/chapter09.md +2031 -2031
  579. package/lib/assets/docs/article/practical-database-design/part2/chapter10.md +1387 -1387
  580. package/lib/assets/docs/article/practical-database-design/part2/chapter11.md +1677 -1677
  581. package/lib/assets/docs/article/practical-database-design/part2/chapter12.md +1417 -1417
  582. package/lib/assets/docs/article/practical-database-design/part2/chapter13.md +1434 -1434
  583. package/lib/assets/docs/article/practical-database-design/part3/chapter14.md +667 -667
  584. package/lib/assets/docs/article/practical-database-design/part3/chapter15.md +1625 -1625
  585. package/lib/assets/docs/article/practical-database-design/part3/chapter16.md +1915 -1915
  586. package/lib/assets/docs/article/practical-database-design/part3/chapter17.md +1708 -1708
  587. package/lib/assets/docs/article/practical-database-design/part3/chapter18.md +2095 -2095
  588. package/lib/assets/docs/article/practical-database-design/part3/chapter19.md +1123 -1123
  589. package/lib/assets/docs/article/practical-database-design/part3/chapter20.md +1031 -1031
  590. package/lib/assets/docs/article/practical-database-design/part3/chapter21.md +1382 -1382
  591. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter14-orm.md +991 -991
  592. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter15-orm.md +1300 -1300
  593. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter16-orm.md +1166 -1166
  594. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter17-orm.md +1584 -1584
  595. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter18-orm.md +1183 -1183
  596. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter19-orm.md +1016 -1016
  597. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter20-orm.md +1753 -1753
  598. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter21-orm.md +1447 -1447
  599. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter22-orm.md +1878 -1878
  600. package/lib/assets/docs/article/practical-database-design/part4/chapter22.md +965 -965
  601. package/lib/assets/docs/article/practical-database-design/part4/chapter23.md +2069 -2069
  602. package/lib/assets/docs/article/practical-database-design/part4/chapter24.md +2439 -2439
  603. package/lib/assets/docs/article/practical-database-design/part4/chapter25.md +3661 -3661
  604. package/lib/assets/docs/article/practical-database-design/part4/chapter26.md +2916 -2916
  605. package/lib/assets/docs/article/practical-database-design/part4/chapter27.md +3105 -3105
  606. package/lib/assets/docs/article/practical-database-design/part4/chapter28.md +2697 -2697
  607. package/lib/assets/docs/article/practical-database-design/part4/chapter29.md +2544 -2544
  608. package/lib/assets/docs/article/practical-database-design/part4/chapter30.md +2180 -2180
  609. package/lib/assets/docs/article/practical-database-design/part4/chapter31.md +1192 -1192
  610. package/lib/assets/docs/article/practical-database-design/part4/chapter32.md +2101 -2101
  611. package/lib/assets/docs/article/practical-database-design/part5/chapter33.md +1032 -1032
  612. package/lib/assets/docs/article/practical-database-design/part5/chapter34.md +1609 -1609
  613. package/lib/assets/docs/article/practical-database-design/part5/chapter35.md +1453 -1453
  614. package/lib/assets/docs/article/practical-database-design/part5/chapter36.md +1292 -1292
  615. package/lib/assets/docs/article/practical-database-design/part5/chapter37.md +1470 -1470
  616. package/lib/assets/docs/article/practical-database-design/part5/chapter38.md +1698 -1698
  617. package/lib/assets/docs/article/practical-database-design/part5/chapter39.md +2334 -2334
  618. package/lib/assets/docs/article/practical-database-design/study/study2-1.md +1693 -1693
  619. package/lib/assets/docs/article/practical-database-design/study/study2-2.md +1347 -1347
  620. package/lib/assets/docs/article/practical-database-design/study/study2-3.md +2044 -2044
  621. package/lib/assets/docs/article/practical-database-design/study/study2-4.md +2229 -2229
  622. package/lib/assets/docs/article/practical-database-design/study/study2-5.md +2418 -2418
  623. package/lib/assets/docs/article/practical-database-design/study/study3-1.md +2205 -2205
  624. package/lib/assets/docs/article/practical-database-design/study/study3-2.md +2221 -2221
  625. package/lib/assets/docs/article/practical-database-design/study/study3-3.md +2253 -2253
  626. package/lib/assets/docs/article/practical-database-design/study/study3-4.md +2106 -2106
  627. package/lib/assets/docs/article/practical-database-design/study/study3-5.md +2507 -2507
  628. package/lib/assets/docs/article/practical-database-design/study/study4-1.md +2587 -2587
  629. package/lib/assets/docs/article/practical-database-design/study/study4-2.md +2075 -2075
  630. package/lib/assets/docs/article/practical-database-design/study/study4-3.md +1805 -1805
  631. package/lib/assets/docs/article/practical-database-design/study/study4-4.md +1895 -1895
  632. package/lib/assets/docs/article/practical-database-design/study/study4-5.md +2878 -2878
  633. package/lib/assets/docs/assets/css/extra.css +29 -29
  634. package/lib/assets/docs/assets/js/extra.js +44 -44
  635. package/lib/assets/docs/development/index.md +39 -39
  636. package/lib/assets/docs/operation/index.md +11 -11
  637. 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 +19 -5
  638. 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
  639. 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 -581
  640. 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
  641. 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
  642. package/lib/assets/docs/reference/UI/350/250/255/350/250/210/343/202/254/343/202/244/343/203/211.md +450 -450
  643. package/lib/assets/docs/reference/images/Ansoff.drawio.svg +3 -3
  644. package/lib/assets/docs/reference/images/BrandBasicStrategy.drawio.svg +3 -3
  645. package/lib/assets/docs/reference/images/BrandCategorization.drawio.svg +3 -3
  646. package/lib/assets/docs/reference/images/BrandRecurutementStrategy.drawio.svg +3 -3
  647. package/lib/assets/docs/reference/images/BrandValue.drawio.svg +3 -3
  648. package/lib/assets/docs/reference/images/BusinessActivitiy.svg +3 -3
  649. package/lib/assets/docs/reference/images/HRM.drawio.svg +3 -3
  650. package/lib/assets/docs/reference/images/MarketingStructure.drawio.svg +3 -3
  651. package/lib/assets/docs/reference/images/OrganizationElemnts.svg +3 -3
  652. package/lib/assets/docs/reference/images/PPM.drawio.svg +3 -3
  653. package/lib/assets/docs/reference/images/PositioningMap.drawio.svg +3 -3
  654. package/lib/assets/docs/reference/images/ProductLayer.drawio.svg +3 -3
  655. package/lib/assets/docs/reference/images/ProductMix.drawio.svg +3 -3
  656. package/lib/assets/docs/reference/images/SWOT.drawio.svg +3 -3
  657. package/lib/assets/docs/reference/images/TargetMarket.drawio.svg +3 -3
  658. package/lib/assets/docs/reference/images/ThreeGenericStrategies.drawio.svg +3 -3
  659. package/lib/assets/docs/reference/images/VRIO.drawio.svg +3 -3
  660. package/lib/assets/docs/reference/images/ValueChain.drawio.svg +3 -3
  661. package/lib/assets/docs/reference/index.md +52 -52
  662. 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 -250
  663. 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
  664. 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
  665. 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 -550
  666. 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
  667. 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
  668. 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
  669. 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
  670. 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
  671. 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 -689
  672. 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
  673. 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 -580
  674. 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
  675. package/lib/assets/docs/reference//344/274/201/346/245/255/347/265/214/345/226/266/350/253/226.md +2637 -2637
  676. 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 -665
  677. 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
  678. 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 +518 -518
  679. package/lib/assets/docs/reference//351/201/213/345/226/266/347/256/241/347/220/206.md +1482 -1482
  680. 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
  681. 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
  682. package/lib/assets/docs/reference//351/226/213/347/231/272/343/202/254/343/202/244/343/203/211.md +299 -299
  683. 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
  684. package/lib/assets/docs/review/index.md +5 -5
  685. package/lib/assets/docs/strategy/index.md +1 -1
  686. package/lib/assets/docs/template/ADR.md +30 -30
  687. 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
  688. 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
  689. package/lib/assets/docs/template/README.md +50 -50
  690. package/lib/assets/docs/template/index.md +23 -23
  691. 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
  692. 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
  693. 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
  694. 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
  695. 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
  696. package/lib/assets/docs/template//344/274/201/346/245/255/345/210/206/346/236/220.md +573 -573
  697. 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 -69
  698. package/lib/assets/docs/template//350/246/201/344/273/266/345/256/232/347/276/251.md +669 -669
  699. package/lib/assets/docs/template//350/250/255/350/250/210.md +173 -173
  700. 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
  701. package/lib/assets/gulpfile.js +25 -25
  702. package/lib/assets/mkdocs.yml +136 -136
  703. package/lib/assets/ops/docker/mkdoc/Dockerfile +19 -19
  704. package/lib/assets/ops/scripts/journal.js +180 -180
  705. package/lib/assets/ops/scripts/mkdocs.js +82 -82
  706. package/lib/assets/ops/scripts/release.js +431 -431
  707. package/lib/assets/ops/scripts/sonar_local.js +726 -726
  708. package/lib/assets/ops/scripts/ssh.js +190 -190
  709. package/lib/assets/ops/scripts/vault.js +299 -299
  710. package/lib/assets/package-lock.json +1653 -1653
  711. package/lib/assets/package.json +40 -40
  712. package/lib/gulpfile.js +37 -37
  713. package/package.json +41 -41
@@ -1,1387 +1,1387 @@
1
- # 第10章:債務管理の設計
2
-
3
- 販売管理システムにおいて、債務管理は仕入先への支払を適切に管理するための重要な機能です。本章では、支払締処理、支払方法の管理、買掛金残高管理のデータベース設計と実装を行います。債権管理(第7章)と対をなす概念として、仕入側からの視点でデータ構造を設計します。
4
-
5
- ## 支払業務の全体像
6
-
7
- 支払業務は「仕入計上 → 支払締処理 → 支払承認 → 支払実行 → 買掛金消込」という一連のフローで構成されます。
8
-
9
- ```plantuml
10
- @startuml
11
-
12
- title 支払業務の全体フロー
13
-
14
- |調達部門|
15
- start
16
- :発注;
17
- :入荷・検収;
18
- :仕入計上;
19
- note right: 仕入データから\n買掛金が発生
20
-
21
- |経理部門|
22
- :支払締処理;
23
- note right: 締日ごとに\n仕入金額を集計
24
-
25
- :支払予定作成;
26
- note right: 支払日・支払方法\nを決定
27
-
28
- :支払承認;
29
-
30
- |財務部門|
31
- if (支払方法?) then (振込)
32
- :振込データ作成;
33
- :銀行振込実行;
34
- elseif (支払方法?) then (手形)
35
- :手形発行;
36
- :手形交付;
37
- else (現金)
38
- :現金支払;
39
- endif
40
-
41
- |経理部門|
42
- :支払消込;
43
- note right: 買掛金残高から\n支払額を減少
44
-
45
- :買掛金残高更新;
46
-
47
- stop
48
-
49
- @enduml
50
- ```
51
-
52
- ### 支払管理で扱うデータ
53
-
54
- | データ | 説明 |
55
- |-------|------|
56
- | **支払データ** | 支払締処理で作成されるヘッダ情報 |
57
- | **支払明細データ** | 仕入データとの紐付け |
58
- | **買掛金残高データ** | 仕入先別・月次の残高管理 |
59
- | **支払予定データ** | 支払スケジュールの管理 |
60
-
61
- ---
62
-
63
- ## 10.1 支払業務の DB 設計
64
-
65
- ### 支払締処理の仕組み
66
-
67
- 支払締処理は、債権管理の請求締処理と対になる仕組みです。仕入先マスタに設定された締日に基づいて、仕入データを集計し支払データを作成します。
68
-
69
- ```plantuml
70
- @startuml
71
-
72
- title 支払締処理の流れ
73
-
74
- |経理部門|
75
- start
76
- :締日到来確認;
77
- note right: 仕入先マスタの\n締日を参照
78
-
79
- :対象仕入データ抽出;
80
- note right: 前回締日翌日~今回締日\nの仕入を対象
81
-
82
- :支払データ作成;
83
- note right: 支払ヘッダ・明細を\n自動生成
84
-
85
- :支払金額計算;
86
- fork
87
- :仕入金額合計;
88
- fork again
89
- :消費税計算;
90
- fork again
91
- :源泉徴収計算;
92
- fork end
93
-
94
- :支払予定日算出;
95
- note right: 支払サイト(月数)\nから計算
96
-
97
- :支払データ確定;
98
-
99
- stop
100
-
101
- @enduml
102
- ```
103
-
104
- ### 支払ステータスの遷移
105
-
106
- 支払データは承認ワークフローに従ってステータスが変化します。
107
-
108
- ```plantuml
109
- @startuml
110
-
111
- title 支払ステータスの遷移
112
-
113
- [*] --> 作成中
114
-
115
- 作成中 --> 承認待ち : 承認申請
116
-
117
- 承認待ち --> 承認済 : 承認
118
-
119
- 承認済 --> 支払済 : 支払実行
120
-
121
- 作成中 --> 取消 : 取消
122
- 承認待ち --> 取消 : 取消
123
-
124
- 取消 --> [*]
125
- 支払済 --> [*]
126
-
127
- @enduml
128
- ```
129
-
130
- | ステータス | 説明 | 次のアクション |
131
- |-----------|------|---------------|
132
- | **作成中** | 支払データ作成直後 | 承認申請または取消 |
133
- | **承認待ち** | 承認申請済み | 承認または取消 |
134
- | **承認済** | 管理者が承認 | 支払実行 |
135
- | **支払済** | 支払処理完了 | 終了状態 |
136
- | **取消** | 支払を取り消し | 終了状態 |
137
-
138
- ### 支払方法の種類
139
-
140
- | 支払方法 | 説明 | 特徴 |
141
- |---------|------|------|
142
- | **振込** | 銀行振込 | 最も一般的、手数料発生 |
143
- | **手形** | 約束手形 | 支払サイト延長、不渡りリスク |
144
- | **現金** | 現金払い | 即時決済、少額向け |
145
- | **相殺** | 債権債務相殺 | 売掛金との相殺 |
146
- | **電子記録債権** | でんさい | 手形の電子版 |
147
-
148
- ### 支払関連の ER 図
149
-
150
- ```plantuml
151
- @startuml
152
-
153
- title 債務管理のER図
154
-
155
- entity 取引先マスタ {
156
- 取引先コード <<PK>>
157
- --
158
- 取引先名
159
- 取引先区分
160
- 締日
161
- 支払月
162
- 支払日
163
- }
164
-
165
- entity 仕入データ {
166
- 仕入番号 <<PK>>
167
- --
168
- 仕入先コード <<FK>>
169
- 仕入日
170
- 仕入金額
171
- 消費税額
172
- ステータス
173
- }
174
-
175
- entity 支払データ {
176
- ID <<PK>>
177
- --
178
- 支払番号 <<UK>>
179
- 仕入先コード <<FK>>
180
- 支払締日
181
- 支払予定日
182
- 支払方法
183
- 支払金額
184
- 消費税額
185
- 源泉徴収額
186
- 差引支払額
187
- 支払実行日
188
- ステータス
189
- バージョン
190
- }
191
-
192
- entity 支払明細データ {
193
- ID <<PK>>
194
- --
195
- 支払ID <<FK>>
196
- 支払行番号
197
- 仕入番号 <<FK>>
198
- 仕入日
199
- 仕入金額
200
- 消費税額
201
- 支払対象金額
202
- }
203
-
204
- entity 買掛金残高データ {
205
- ID <<PK>>
206
- --
207
- 仕入先コード <<FK>>
208
- 年月
209
- 前月残高
210
- 当月仕入高
211
- 当月支払高
212
- 当月残高
213
- バージョン
214
- }
215
-
216
- entity 支払予定データ {
217
- ID <<PK>>
218
- --
219
- 仕入先コード <<FK>>
220
- 支払予定日
221
- 支払予定額
222
- 支払方法
223
- 支払済フラグ
224
- 支払ID <<FK>>
225
- }
226
-
227
- 取引先マスタ ||--o{ 仕入データ
228
- 取引先マスタ ||--o{ 支払データ
229
- 取引先マスタ ||--o{ 買掛金残高データ
230
- 取引先マスタ ||--o{ 支払予定データ
231
- 支払データ ||--o{ 支払明細データ
232
- 仕入データ ||--o{ 支払明細データ
233
- 支払データ |o--o{ 支払予定データ
234
-
235
- @enduml
236
- ```
237
-
238
- ### マイグレーション:支払関連テーブルの作成
239
-
240
- <details>
241
- <summary>V017__create_payment_tables.sql</summary>
242
-
243
- ```sql
244
- -- src/main/resources/db/migration/V017__create_payment_tables.sql
245
-
246
- -- 支払ステータス
247
- CREATE TYPE 支払ステータス AS ENUM ('作成中', '承認待ち', '承認済', '支払済', '取消');
248
-
249
- -- 支払方法に不足している値を追加(既存のENUMはV001で定義済み)
250
- ALTER TYPE 支払方法 ADD VALUE IF NOT EXISTS '相殺';
251
- ALTER TYPE 支払方法 ADD VALUE IF NOT EXISTS '電子記録債権';
252
-
253
- -- 支払データ(ヘッダ)
254
- CREATE TABLE "支払データ" (
255
- "ID" SERIAL PRIMARY KEY,
256
- "支払番号" VARCHAR(20) UNIQUE NOT NULL,
257
- "仕入先コード" VARCHAR(20) NOT NULL,
258
- "支払締日" DATE NOT NULL,
259
- "支払予定日" DATE NOT NULL,
260
- "支払方法" 支払方法 NOT NULL,
261
- "支払金額" DECIMAL(15, 2) NOT NULL,
262
- "消費税額" DECIMAL(15, 2) NOT NULL,
263
- "源泉徴収額" DECIMAL(15, 2) DEFAULT 0 NOT NULL,
264
- "差引支払額" DECIMAL(15, 2) NOT NULL,
265
- "支払実行日" DATE,
266
- "ステータス" 支払ステータス DEFAULT '作成中' NOT NULL,
267
- "振込先銀行コード" VARCHAR(10),
268
- "振込先支店コード" VARCHAR(10),
269
- "振込先口座種別" VARCHAR(10),
270
- "振込先口座番号" VARCHAR(20),
271
- "振込先口座名義" VARCHAR(100),
272
- "備考" TEXT,
273
- "バージョン" INTEGER DEFAULT 1 NOT NULL,
274
- "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
275
- "作成者" VARCHAR(50),
276
- "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
277
- "更新者" VARCHAR(50),
278
- CONSTRAINT "fk_支払_仕入先"
279
- FOREIGN KEY ("仕入先コード") REFERENCES "取引先マスタ"("取引先コード")
280
- );
281
-
282
- -- 支払明細データ
283
- CREATE TABLE "支払明細データ" (
284
- "ID" SERIAL PRIMARY KEY,
285
- "支払ID" INTEGER NOT NULL,
286
- "支払行番号" INTEGER NOT NULL,
287
- "仕入番号" VARCHAR(20) NOT NULL,
288
- "仕入日" DATE NOT NULL,
289
- "仕入金額" DECIMAL(15, 2) NOT NULL,
290
- "消費税額" DECIMAL(15, 2) NOT NULL,
291
- "支払対象金額" DECIMAL(15, 2) NOT NULL,
292
- "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
293
- "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
294
- CONSTRAINT "fk_支払明細_支払"
295
- FOREIGN KEY ("支払ID") REFERENCES "支払データ"("ID") ON DELETE CASCADE,
296
- CONSTRAINT "fk_支払明細_仕入"
297
- FOREIGN KEY ("仕入番号") REFERENCES "仕入データ"("仕入番号"),
298
- CONSTRAINT "uk_支払明細_支払_行" UNIQUE ("支払ID", "支払行番号")
299
- );
300
-
301
- -- 買掛金残高データ
302
- CREATE TABLE "買掛金残高データ" (
303
- "ID" SERIAL PRIMARY KEY,
304
- "仕入先コード" VARCHAR(20) NOT NULL,
305
- "年月" DATE NOT NULL,
306
- "前月残高" DECIMAL(15, 2) DEFAULT 0 NOT NULL,
307
- "当月仕入高" DECIMAL(15, 2) DEFAULT 0 NOT NULL,
308
- "当月支払高" DECIMAL(15, 2) DEFAULT 0 NOT NULL,
309
- "当月残高" DECIMAL(15, 2) DEFAULT 0 NOT NULL,
310
- "バージョン" INTEGER DEFAULT 1 NOT NULL,
311
- "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
312
- "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
313
- CONSTRAINT "fk_買掛金残高_仕入先"
314
- FOREIGN KEY ("仕入先コード") REFERENCES "取引先マスタ"("取引先コード"),
315
- UNIQUE ("仕入先コード", "年月")
316
- );
317
-
318
- -- 支払予定データ
319
- CREATE TABLE "支払予定データ" (
320
- "ID" SERIAL PRIMARY KEY,
321
- "仕入先コード" VARCHAR(20) NOT NULL,
322
- "支払予定日" DATE NOT NULL,
323
- "支払予定額" DECIMAL(15, 2) NOT NULL,
324
- "支払方法" 支払方法 NOT NULL,
325
- "支払済フラグ" BOOLEAN DEFAULT FALSE NOT NULL,
326
- "支払ID" INTEGER,
327
- "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
328
- "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
329
- CONSTRAINT "fk_支払予定_仕入先"
330
- FOREIGN KEY ("仕入先コード") REFERENCES "取引先マスタ"("取引先コード"),
331
- CONSTRAINT "fk_支払予定_支払"
332
- FOREIGN KEY ("支払ID") REFERENCES "支払データ"("ID")
333
- );
334
-
335
- -- 当月残高を自動計算するトリガー
336
- CREATE OR REPLACE FUNCTION calculate_payable_balance()
337
- RETURNS TRIGGER AS $$
338
- BEGIN
339
- NEW."当月残高" := NEW."前月残高" + NEW."当月仕入高" - NEW."当月支払高";
340
- RETURN NEW;
341
- END;
342
- $$ LANGUAGE plpgsql;
343
-
344
- CREATE TRIGGER trg_買掛金残高_残高計算
345
- BEFORE INSERT OR UPDATE ON "買掛金残高データ"
346
- FOR EACH ROW
347
- EXECUTE FUNCTION calculate_payable_balance();
348
-
349
- -- インデックス
350
- CREATE INDEX "idx_支払_仕入先コード" ON "支払データ"("仕入先コード");
351
- CREATE INDEX "idx_支払_支払締日" ON "支払データ"("支払締日");
352
- CREATE INDEX "idx_支払_支払予定日" ON "支払データ"("支払予定日");
353
- CREATE INDEX "idx_支払_ステータス" ON "支払データ"("ステータス");
354
- CREATE INDEX "idx_支払明細_仕入番号" ON "支払明細データ"("仕入番号");
355
- CREATE INDEX "idx_買掛金残高_年月" ON "買掛金残高データ"("年月");
356
- CREATE INDEX "idx_支払予定_支払予定日" ON "支払予定データ"("支払予定日");
357
- CREATE INDEX "idx_支払予定_支払済" ON "支払予定データ"("支払済フラグ");
358
-
359
- -- テーブルコメント
360
- COMMENT ON TABLE "支払データ" IS '支払締処理で作成される支払ヘッダ情報';
361
- COMMENT ON TABLE "支払明細データ" IS '支払データと仕入データの紐付け';
362
- COMMENT ON TABLE "買掛金残高データ" IS '仕入先別・月次の買掛金残高';
363
- COMMENT ON TABLE "支払予定データ" IS '支払スケジュールの管理';
364
- COMMENT ON COLUMN "支払データ"."バージョン" IS '楽観ロック用バージョン番号';
365
- COMMENT ON COLUMN "買掛金残高データ"."バージョン" IS '楽観ロック用バージョン番号';
366
- ```
367
-
368
- </details>
369
-
370
- ### 支払関連エンティティの実装
371
-
372
- <details>
373
- <summary>支払ステータス ENUM</summary>
374
-
375
- ```java
376
- // src/main/java/com/example/sms/domain/model/payment/PaymentStatus.java
377
- package com.example.sms.domain.model.payment;
378
-
379
- import lombok.Getter;
380
- import lombok.RequiredArgsConstructor;
381
-
382
- @Getter
383
- @RequiredArgsConstructor
384
- public enum PaymentStatus {
385
- DRAFT("作成中"),
386
- PENDING_APPROVAL("承認待ち"),
387
- APPROVED("承認済"),
388
- PAID("支払済"),
389
- CANCELLED("取消");
390
-
391
- private final String displayName;
392
-
393
- public static PaymentStatus fromDisplayName(String displayName) {
394
- for (PaymentStatus status : values()) {
395
- if (status.displayName.equals(displayName)) {
396
- return status;
397
- }
398
- }
399
- throw new IllegalArgumentException("Unknown payment status: " + displayName);
400
- }
401
-
402
- /**
403
- * 承認可能なステータスかどうか.
404
- */
405
- public boolean canApprove() {
406
- return this == PENDING_APPROVAL;
407
- }
408
-
409
- /**
410
- * 支払実行可能なステータスかどうか.
411
- */
412
- public boolean canExecute() {
413
- return this == APPROVED;
414
- }
415
-
416
- /**
417
- * 取消可能なステータスかどうか.
418
- */
419
- public boolean canCancel() {
420
- return this == DRAFT || this == PENDING_APPROVAL;
421
- }
422
- }
423
- ```
424
-
425
- </details>
426
-
427
- <details>
428
- <summary>支払方法 ENUM</summary>
429
-
430
- ```java
431
- // src/main/java/com/example/sms/domain/model/payment/PaymentMethod.java
432
- package com.example.sms.domain.model.payment;
433
-
434
- import lombok.Getter;
435
- import lombok.RequiredArgsConstructor;
436
-
437
- @Getter
438
- @RequiredArgsConstructor
439
- public enum PaymentMethod {
440
- BANK_TRANSFER("振込"),
441
- BILL("手形"),
442
- CASH("現金"),
443
- OFFSET("相殺"),
444
- ELECTRONIC_RECORD("電子記録債権");
445
-
446
- private final String displayName;
447
-
448
- public static PaymentMethod fromDisplayName(String displayName) {
449
- for (PaymentMethod method : values()) {
450
- if (method.displayName.equals(displayName)) {
451
- return method;
452
- }
453
- }
454
- throw new IllegalArgumentException("Unknown payment method: " + displayName);
455
- }
456
-
457
- /**
458
- * 振込先情報が必要かどうか.
459
- */
460
- public boolean requiresBankInfo() {
461
- return this == BANK_TRANSFER;
462
- }
463
- }
464
- ```
465
-
466
- </details>
467
-
468
- <details>
469
- <summary>支払データエンティティ</summary>
470
-
471
- ```java
472
- // src/main/java/com/example/sms/domain/model/payment/Payment.java
473
- package com.example.sms.domain.model.payment;
474
-
475
- import lombok.AllArgsConstructor;
476
- import lombok.Builder;
477
- import lombok.Data;
478
- import lombok.NoArgsConstructor;
479
-
480
- import java.math.BigDecimal;
481
- import java.time.LocalDate;
482
- import java.time.LocalDateTime;
483
- import java.util.ArrayList;
484
- import java.util.List;
485
-
486
- @Data
487
- @Builder
488
- @NoArgsConstructor
489
- @AllArgsConstructor
490
- public class Payment {
491
- private Integer id;
492
- private String paymentNumber;
493
- private String supplierCode;
494
- private LocalDate paymentClosingDate;
495
- private LocalDate paymentDueDate;
496
- private PaymentMethod paymentMethod;
497
- private BigDecimal paymentAmount;
498
- private BigDecimal taxAmount;
499
- private BigDecimal withholdingAmount;
500
- private BigDecimal netPaymentAmount;
501
- private LocalDate paymentExecutionDate;
502
- private PaymentStatus status;
503
- private String bankCode;
504
- private String branchCode;
505
- private String accountType;
506
- private String accountNumber;
507
- private String accountName;
508
- private String remarks;
509
- private LocalDateTime createdAt;
510
- private String createdBy;
511
- private LocalDateTime updatedAt;
512
- private String updatedBy;
513
-
514
- @Builder.Default
515
- private Integer version = 1;
516
-
517
- @Builder.Default
518
- private List<PaymentDetail> details = new ArrayList<>();
519
-
520
- /**
521
- * 明細の合計金額を計算.
522
- */
523
- public BigDecimal calculateTotalAmount() {
524
- if (details == null || details.isEmpty()) {
525
- return BigDecimal.ZERO;
526
- }
527
- return details.stream()
528
- .map(PaymentDetail::getPaymentTargetAmount)
529
- .reduce(BigDecimal.ZERO, BigDecimal::add);
530
- }
531
-
532
- /**
533
- * 差引支払額を計算.
534
- */
535
- public BigDecimal calculateNetAmount() {
536
- BigDecimal wh = withholdingAmount != null ? withholdingAmount : BigDecimal.ZERO;
537
- return paymentAmount.add(taxAmount).subtract(wh);
538
- }
539
-
540
- /**
541
- * 承認申請.
542
- */
543
- public void submitForApproval() {
544
- if (this.status != PaymentStatus.DRAFT) {
545
- throw new IllegalStateException("作成中ステータスのみ承認申請が可能です");
546
- }
547
- this.status = PaymentStatus.PENDING_APPROVAL;
548
- }
549
-
550
- /**
551
- * 承認.
552
- */
553
- public void approve() {
554
- if (!this.status.canApprove()) {
555
- throw new IllegalStateException(
556
- "このステータスでは承認できません: " + this.status.getDisplayName());
557
- }
558
- this.status = PaymentStatus.APPROVED;
559
- }
560
-
561
- /**
562
- * 支払実行.
563
- */
564
- public void execute(LocalDate executionDate) {
565
- if (!this.status.canExecute()) {
566
- throw new IllegalStateException(
567
- "このステータスでは支払実行できません: " + this.status.getDisplayName());
568
- }
569
- this.paymentExecutionDate = executionDate;
570
- this.status = PaymentStatus.PAID;
571
- }
572
-
573
- /**
574
- * 取消.
575
- */
576
- public void cancel() {
577
- if (!this.status.canCancel()) {
578
- throw new IllegalStateException(
579
- "このステータスでは取消できません: " + this.status.getDisplayName());
580
- }
581
- this.status = PaymentStatus.CANCELLED;
582
- }
583
- }
584
- ```
585
-
586
- </details>
587
-
588
- <details>
589
- <summary>支払明細データエンティティ</summary>
590
-
591
- ```java
592
- // src/main/java/com/example/sms/domain/model/payment/PaymentDetail.java
593
- package com.example.sms.domain.model.payment;
594
-
595
- import lombok.AllArgsConstructor;
596
- import lombok.Builder;
597
- import lombok.Data;
598
- import lombok.NoArgsConstructor;
599
-
600
- import java.math.BigDecimal;
601
- import java.time.LocalDate;
602
- import java.time.LocalDateTime;
603
-
604
- @Data
605
- @Builder
606
- @NoArgsConstructor
607
- @AllArgsConstructor
608
- public class PaymentDetail {
609
- private Integer id;
610
- private Integer paymentId;
611
- private Integer lineNumber;
612
- private String purchaseNumber;
613
- private LocalDate purchaseDate;
614
- private BigDecimal purchaseAmount;
615
- private BigDecimal taxAmount;
616
- private BigDecimal paymentTargetAmount;
617
- private LocalDateTime createdAt;
618
- private LocalDateTime updatedAt;
619
- }
620
- ```
621
-
622
- </details>
623
-
624
- <details>
625
- <summary>買掛金残高データエンティティ</summary>
626
-
627
- ```java
628
- // src/main/java/com/example/sms/domain/model/payment/PayableBalance.java
629
- package com.example.sms.domain.model.payment;
630
-
631
- import lombok.AllArgsConstructor;
632
- import lombok.Builder;
633
- import lombok.Data;
634
- import lombok.NoArgsConstructor;
635
-
636
- import java.math.BigDecimal;
637
- import java.time.LocalDate;
638
- import java.time.LocalDateTime;
639
-
640
- @Data
641
- @Builder
642
- @NoArgsConstructor
643
- @AllArgsConstructor
644
- public class PayableBalance {
645
- private Integer id;
646
- private String supplierCode;
647
- private LocalDate yearMonth;
648
- private BigDecimal previousBalance;
649
- private BigDecimal currentPurchaseAmount;
650
- private BigDecimal currentPaymentAmount;
651
- private BigDecimal currentBalance;
652
- private LocalDateTime createdAt;
653
- private LocalDateTime updatedAt;
654
-
655
- @Builder.Default
656
- private Integer version = 1;
657
-
658
- /**
659
- * 残高を計算.
660
- */
661
- public BigDecimal calculateBalance() {
662
- BigDecimal prev = previousBalance != null ? previousBalance : BigDecimal.ZERO;
663
- BigDecimal purchase = currentPurchaseAmount != null ? currentPurchaseAmount : BigDecimal.ZERO;
664
- BigDecimal payment = currentPaymentAmount != null ? currentPaymentAmount : BigDecimal.ZERO;
665
- return prev.add(purchase).subtract(payment);
666
- }
667
-
668
- /**
669
- * 仕入を加算.
670
- */
671
- public void addPurchase(BigDecimal amount) {
672
- if (this.currentPurchaseAmount == null) {
673
- this.currentPurchaseAmount = BigDecimal.ZERO;
674
- }
675
- this.currentPurchaseAmount = this.currentPurchaseAmount.add(amount);
676
- this.currentBalance = calculateBalance();
677
- }
678
-
679
- /**
680
- * 支払を加算.
681
- */
682
- public void addPayment(BigDecimal amount) {
683
- if (this.currentPaymentAmount == null) {
684
- this.currentPaymentAmount = BigDecimal.ZERO;
685
- }
686
- this.currentPaymentAmount = this.currentPaymentAmount.add(amount);
687
- this.currentBalance = calculateBalance();
688
- }
689
- }
690
- ```
691
-
692
- </details>
693
-
694
- <details>
695
- <summary>支払予定データエンティティ</summary>
696
-
697
- ```java
698
- // src/main/java/com/example/sms/domain/model/payment/PaymentSchedule.java
699
- package com.example.sms.domain.model.payment;
700
-
701
- import lombok.AllArgsConstructor;
702
- import lombok.Builder;
703
- import lombok.Data;
704
- import lombok.NoArgsConstructor;
705
-
706
- import java.math.BigDecimal;
707
- import java.time.LocalDate;
708
- import java.time.LocalDateTime;
709
-
710
- @Data
711
- @Builder
712
- @NoArgsConstructor
713
- @AllArgsConstructor
714
- public class PaymentSchedule {
715
- private Integer id;
716
- private String supplierCode;
717
- private LocalDate paymentDueDate;
718
- private BigDecimal scheduledAmount;
719
- private PaymentMethod paymentMethod;
720
- @Builder.Default
721
- private Boolean paidFlag = false;
722
- private Integer paymentId;
723
- private LocalDateTime createdAt;
724
- private LocalDateTime updatedAt;
725
-
726
- /**
727
- * 支払済みとしてマーク.
728
- */
729
- public void markAsPaid(Integer paymentId) {
730
- this.paidFlag = true;
731
- this.paymentId = paymentId;
732
- }
733
-
734
- /**
735
- * 支払期限が過ぎているかどうか.
736
- */
737
- public boolean isOverdue() {
738
- return !paidFlag && LocalDate.now().isAfter(paymentDueDate);
739
- }
740
- }
741
- ```
742
-
743
- </details>
744
-
745
- ### 支払サービスの実装(参考実装例)
746
-
747
- > **Note**: 以下のサービス層のコードは参考実装例です。現在はリポジトリ層(永続化層)が実装済みで、アプリケーションサービス層は別途実装が必要です。
748
-
749
- <details>
750
- <summary>支払サービス</summary>
751
-
752
- ```java
753
- // src/main/java/com/example/sms/application/service/PaymentService.java
754
- package com.example.sms.application.service;
755
-
756
- import com.example.sms.domain.model.partner.Partner;
757
- import com.example.sms.domain.model.payment.*;
758
- import com.example.sms.domain.model.purchase.Purchase;
759
- import com.example.sms.infrastructure.out.persistence.mapper.*;
760
- import lombok.RequiredArgsConstructor;
761
- import org.springframework.stereotype.Service;
762
- import org.springframework.transaction.annotation.Transactional;
763
-
764
- import java.math.BigDecimal;
765
- import java.time.LocalDate;
766
- import java.time.format.DateTimeFormatter;
767
- import java.util.ArrayList;
768
- import java.util.List;
769
-
770
- @Service
771
- @RequiredArgsConstructor
772
- public class PaymentService {
773
-
774
- private final PaymentMapper paymentMapper;
775
- private final PaymentDetailMapper paymentDetailMapper;
776
- private final PayableBalanceMapper payableBalanceMapper;
777
- private final PartnerMapper partnerMapper;
778
- private final PurchaseMapper purchaseMapper;
779
-
780
- /**
781
- * 支払番号を生成
782
- */
783
- private String generatePaymentNumber(LocalDate closingDate) {
784
- String prefix = "PAY-" + closingDate.format(
785
- DateTimeFormatter.ofPattern("yyyyMM")) + "-";
786
- String latestNumber = paymentMapper.findLatestPaymentNumber(prefix + "%");
787
-
788
- int sequence = 1;
789
- if (latestNumber != null) {
790
- int currentSequence = Integer.parseInt(
791
- latestNumber.substring(latestNumber.length() - 4));
792
- sequence = currentSequence + 1;
793
- }
794
-
795
- return prefix + String.format("%04d", sequence);
796
- }
797
-
798
- /**
799
- * 支払予定日を計算
800
- */
801
- private LocalDate calculatePaymentDueDate(LocalDate closingDate,
802
- Partner supplier) {
803
- // 締日の翌月から支払月数分を加算
804
- LocalDate baseDate = closingDate.plusMonths(supplier.getPaymentMonth());
805
-
806
- // 支払日を設定
807
- int paymentDay = supplier.getPaymentDay();
808
- int maxDay = baseDate.lengthOfMonth();
809
- if (paymentDay > maxDay) {
810
- paymentDay = maxDay; // 月末処理
811
- }
812
-
813
- return LocalDate.of(baseDate.getYear(), baseDate.getMonth(), paymentDay);
814
- }
815
-
816
- /**
817
- * 支払締処理を実行
818
- */
819
- @Transactional
820
- public Payment processPaymentClosing(PaymentClosingInput input) {
821
- // 仕入先情報を取得
822
- Partner supplier = partnerMapper.findByPartnerCode(input.getSupplierCode());
823
- if (supplier == null) {
824
- throw new IllegalArgumentException(
825
- "仕入先が見つかりません: " + input.getSupplierCode());
826
- }
827
-
828
- // 対象仕入データを取得(前回締日翌日~今回締日)
829
- List<Purchase> purchases = purchaseMapper.findBySupplierAndDateRange(
830
- input.getSupplierCode(),
831
- input.getFromDate() != null
832
- ? input.getFromDate()
833
- : LocalDate.of(1900, 1, 1),
834
- input.getClosingDate()
835
- );
836
-
837
- if (purchases.isEmpty()) {
838
- throw new IllegalStateException(
839
- "対象となる仕入データがありません");
840
- }
841
-
842
- // 支払データを作成
843
- String paymentNumber = generatePaymentNumber(input.getClosingDate());
844
- LocalDate paymentDueDate = calculatePaymentDueDate(
845
- input.getClosingDate(), supplier);
846
-
847
- BigDecimal totalAmount = BigDecimal.ZERO;
848
- BigDecimal totalTax = BigDecimal.ZERO;
849
-
850
- List<PaymentDetail> details = new ArrayList<>();
851
- int detailNumber = 1;
852
-
853
- for (Purchase purchase : purchases) {
854
- totalAmount = totalAmount.add(purchase.getPurchaseAmount());
855
- totalTax = totalTax.add(purchase.getTaxAmount());
856
-
857
- PaymentDetail detail = PaymentDetail.builder()
858
- .paymentNumber(paymentNumber)
859
- .detailNumber(detailNumber++)
860
- .purchaseNumber(purchase.getPurchaseNumber())
861
- .purchaseDate(purchase.getPurchaseDate())
862
- .purchaseAmount(purchase.getPurchaseAmount())
863
- .taxAmount(purchase.getTaxAmount())
864
- .paymentTargetAmount(purchase.getTotalAmount())
865
- .build();
866
- details.add(detail);
867
- }
868
-
869
- BigDecimal withholdingAmount = input.getWithholdingAmount() != null
870
- ? input.getWithholdingAmount()
871
- : BigDecimal.ZERO;
872
- BigDecimal netAmount = totalAmount.add(totalTax)
873
- .subtract(withholdingAmount);
874
-
875
- Payment payment = Payment.builder()
876
- .paymentNumber(paymentNumber)
877
- .supplierCode(input.getSupplierCode())
878
- .paymentClosingDate(input.getClosingDate())
879
- .paymentDueDate(paymentDueDate)
880
- .paymentMethod(input.getPaymentMethod())
881
- .paymentAmount(totalAmount)
882
- .taxAmount(totalTax)
883
- .withholdingAmount(withholdingAmount)
884
- .netPaymentAmount(netAmount)
885
- .status(PaymentStatus.DRAFT)
886
- .bankCode(supplier.getBankCode())
887
- .branchCode(supplier.getBranchCode())
888
- .accountType(supplier.getAccountType())
889
- .accountNumber(supplier.getAccountNumber())
890
- .accountName(supplier.getAccountName())
891
- .build();
892
-
893
- paymentMapper.insert(payment);
894
-
895
- for (PaymentDetail detail : details) {
896
- paymentDetailMapper.insert(detail);
897
- }
898
-
899
- payment.setDetails(details);
900
- return payment;
901
- }
902
-
903
- /**
904
- * 承認申請
905
- */
906
- @Transactional
907
- public void submitForApproval(String paymentNumber) {
908
- Payment payment = paymentMapper.findByPaymentNumber(paymentNumber);
909
- if (payment == null) {
910
- throw new IllegalArgumentException(
911
- "支払データが見つかりません: " + paymentNumber);
912
- }
913
-
914
- payment.submitForApproval();
915
- paymentMapper.updateStatus(paymentNumber, PaymentStatus.PENDING_APPROVAL);
916
- }
917
-
918
- /**
919
- * 承認
920
- */
921
- @Transactional
922
- public void approve(String paymentNumber) {
923
- Payment payment = paymentMapper.findByPaymentNumber(paymentNumber);
924
- if (payment == null) {
925
- throw new IllegalArgumentException(
926
- "支払データが見つかりません: " + paymentNumber);
927
- }
928
-
929
- payment.approve();
930
- paymentMapper.updateStatus(paymentNumber, PaymentStatus.APPROVED);
931
- }
932
-
933
- /**
934
- * 支払実行
935
- */
936
- @Transactional
937
- public void executePayment(String paymentNumber, LocalDate executionDate) {
938
- Payment payment = paymentMapper.findByPaymentNumber(paymentNumber);
939
- if (payment == null) {
940
- throw new IllegalArgumentException(
941
- "支払データが見つかりません: " + paymentNumber);
942
- }
943
-
944
- payment.execute(executionDate);
945
- paymentMapper.executePayment(paymentNumber, executionDate);
946
-
947
- // 買掛金残高を更新
948
- LocalDate yearMonth = LocalDate.of(
949
- executionDate.getYear(), executionDate.getMonth(), 1);
950
- payableBalanceMapper.addPayment(
951
- payment.getSupplierCode(), yearMonth, payment.getNetPaymentAmount());
952
- }
953
-
954
- /**
955
- * 支払取消
956
- */
957
- @Transactional
958
- public void cancelPayment(String paymentNumber) {
959
- Payment payment = paymentMapper.findByPaymentNumber(paymentNumber);
960
- if (payment == null) {
961
- throw new IllegalArgumentException(
962
- "支払データが見つかりません: " + paymentNumber);
963
- }
964
-
965
- payment.cancel();
966
- paymentMapper.updateStatus(paymentNumber, PaymentStatus.CANCELLED);
967
- }
968
-
969
- /**
970
- * 買掛金残高を取得
971
- */
972
- public PayableBalance getPayableBalance(String supplierCode,
973
- LocalDate yearMonth) {
974
- return payableBalanceMapper.findBySupplierAndYearMonth(
975
- supplierCode, yearMonth);
976
- }
977
-
978
- /**
979
- * 月次の買掛金合計を取得
980
- */
981
- public BigDecimal getTotalPayableByMonth(LocalDate yearMonth) {
982
- return payableBalanceMapper.sumCurrentBalanceByYearMonth(yearMonth);
983
- }
984
-
985
- /**
986
- * 買掛金残高を初期化(月次繰越)
987
- */
988
- @Transactional
989
- public PayableBalance initializePayableBalance(String supplierCode,
990
- LocalDate yearMonth) {
991
- // 前月残高を取得
992
- LocalDate previousMonth = yearMonth.minusMonths(1);
993
- PayableBalance previousBalance = payableBalanceMapper
994
- .findBySupplierAndYearMonth(supplierCode, previousMonth);
995
-
996
- BigDecimal carryOver = BigDecimal.ZERO;
997
- if (previousBalance != null) {
998
- carryOver = previousBalance.getCurrentBalance();
999
- }
1000
-
1001
- PayableBalance newBalance = PayableBalance.builder()
1002
- .supplierCode(supplierCode)
1003
- .yearMonth(yearMonth)
1004
- .previousBalance(carryOver)
1005
- .currentPurchaseAmount(BigDecimal.ZERO)
1006
- .currentPaymentAmount(BigDecimal.ZERO)
1007
- .build();
1008
-
1009
- payableBalanceMapper.insert(newBalance);
1010
- return newBalance;
1011
- }
1012
- }
1013
- ```
1014
-
1015
- </details>
1016
-
1017
- <details>
1018
- <summary>支払締処理の入力クラス</summary>
1019
-
1020
- ```java
1021
- // src/main/java/com/example/sms/application/service/PaymentClosingInput.java
1022
- package com.example.sms.application.service;
1023
-
1024
- import com.example.sms.domain.model.payment.PaymentMethod;
1025
- import lombok.Builder;
1026
- import lombok.Data;
1027
-
1028
- import java.math.BigDecimal;
1029
- import java.time.LocalDate;
1030
-
1031
- @Data
1032
- @Builder
1033
- public class PaymentClosingInput {
1034
- private String supplierCode;
1035
- private LocalDate closingDate;
1036
- private LocalDate fromDate;
1037
- private PaymentMethod paymentMethod;
1038
- private BigDecimal withholdingAmount;
1039
- }
1040
- ```
1041
-
1042
- </details>
1043
-
1044
- ---
1045
-
1046
- ## 10.2 リレーションと楽観ロックの設計
1047
-
1048
- ### MyBatis ネストした ResultMap による関連付け
1049
-
1050
- 支払データは、支払(ヘッダ)→ 支払明細の2層構造を持ちます。N+1 問題を回避するため、JOIN クエリで一括取得します。
1051
-
1052
- ```plantuml
1053
- @startuml
1054
-
1055
- title N+1 問題の回避
1056
-
1057
- rectangle "N+1 問題(非効率)" #FFCCCC {
1058
- card "1. 支払ヘッダ取得\n SELECT * FROM 支払データ\n (1回)" as q1
1059
- card "2. 各支払の明細取得\n SELECT * FROM 支払明細データ\n WHERE 支払番号 = ?\n (N回)" as q2
1060
- }
1061
-
1062
- rectangle "解決策:JOINによる一括取得" #CCFFCC {
1063
- card "1回のJOINクエリで\n全データを取得\n SELECT p.*, pd.*\n FROM 支払データ p\n LEFT JOIN 支払明細データ pd ON ..." as sol
1064
- }
1065
-
1066
- q1 --> q2 : N件
1067
- note right of q2
1068
- 10件の支払があれば
1069
- 合計11回のクエリ発行
1070
- end note
1071
-
1072
- @enduml
1073
- ```
1074
-
1075
- <details>
1076
- <summary>支払データのリレーション設定(PaymentMapper.xml)</summary>
1077
-
1078
- ```xml
1079
- <?xml version="1.0" encoding="UTF-8" ?>
1080
- <!DOCTYPE mapper
1081
- PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
1082
- "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
1083
- <mapper namespace="com.example.sms.infrastructure.out.persistence.mapper.PaymentMapper">
1084
-
1085
- <!-- 支払(ヘッダ)の ResultMap -->
1086
- <resultMap id="PaymentWithDetailsResultMap"
1087
- type="com.example.sms.domain.model.payment.Payment">
1088
- <id property="id" column="p_id"/>
1089
- <result property="paymentNumber" column="p_支払番号"/>
1090
- <result property="supplierCode" column="p_仕入先コード"/>
1091
- <result property="paymentClosingDate" column="p_支払締日"/>
1092
- <result property="paymentDueDate" column="p_支払予定日"/>
1093
- <result property="paymentMethod" column="p_支払方法"
1094
- typeHandler="com.example.sms.infrastructure.out.persistence.typehandler.PaymentMethodTypeHandler"/>
1095
- <result property="paymentAmount" column="p_支払金額"/>
1096
- <result property="taxAmount" column="p_消費税額"/>
1097
- <result property="withholdingAmount" column="p_源泉徴収額"/>
1098
- <result property="netPaymentAmount" column="p_差引支払額"/>
1099
- <result property="paymentExecutionDate" column="p_支払実行日"/>
1100
- <result property="status" column="p_ステータス"
1101
- typeHandler="com.example.sms.infrastructure.out.persistence.typehandler.PaymentStatusTypeHandler"/>
1102
- <result property="version" column="p_バージョン"/>
1103
- <result property="createdAt" column="p_作成日時"/>
1104
- <result property="updatedAt" column="p_更新日時"/>
1105
- <!-- 支払明細との1:N関連 -->
1106
- <collection property="details"
1107
- ofType="com.example.sms.domain.model.payment.PaymentDetail"
1108
- resultMap="PaymentDetailNestedResultMap"/>
1109
- </resultMap>
1110
-
1111
- <!-- 支払明細のネスト ResultMap -->
1112
- <resultMap id="PaymentDetailNestedResultMap"
1113
- type="com.example.sms.domain.model.payment.PaymentDetail">
1114
- <id property="id" column="pd_id"/>
1115
- <result property="paymentId" column="pd_支払ID"/>
1116
- <result property="lineNumber" column="pd_支払行番号"/>
1117
- <result property="purchaseNumber" column="pd_仕入番号"/>
1118
- <result property="purchaseDate" column="pd_仕入日"/>
1119
- <result property="purchaseAmount" column="pd_仕入金額"/>
1120
- <result property="taxAmount" column="pd_消費税額"/>
1121
- <result property="paymentTargetAmount" column="pd_支払対象金額"/>
1122
- </resultMap>
1123
-
1124
- <!-- JOIN による一括取得クエリ -->
1125
- <select id="findWithDetailsByPaymentNumber"
1126
- resultMap="PaymentWithDetailsResultMap">
1127
- SELECT
1128
- p."ID" AS p_id,
1129
- p."支払番号" AS p_支払番号,
1130
- p."仕入先コード" AS p_仕入先コード,
1131
- p."支払締日" AS p_支払締日,
1132
- p."支払予定日" AS p_支払予定日,
1133
- p."支払方法" AS p_支払方法,
1134
- p."支払金額" AS p_支払金額,
1135
- p."消費税額" AS p_消費税額,
1136
- p."源泉徴収額" AS p_源泉徴収額,
1137
- p."差引支払額" AS p_差引支払額,
1138
- p."支払実行日" AS p_支払実行日,
1139
- p."ステータス" AS p_ステータス,
1140
- p."バージョン" AS p_バージョン,
1141
- p."作成日時" AS p_作成日時,
1142
- p."更新日時" AS p_更新日時,
1143
- pd."ID" AS pd_id,
1144
- pd."支払ID" AS pd_支払ID,
1145
- pd."支払行番号" AS pd_支払行番号,
1146
- pd."仕入番号" AS pd_仕入番号,
1147
- pd."仕入日" AS pd_仕入日,
1148
- pd."仕入金額" AS pd_仕入金額,
1149
- pd."消費税額" AS pd_消費税額,
1150
- pd."支払対象金額" AS pd_支払対象金額
1151
- FROM "支払データ" p
1152
- LEFT JOIN "支払明細データ" pd
1153
- ON p."ID" = pd."支払ID"
1154
- WHERE p."支払番号" = #{paymentNumber}
1155
- ORDER BY pd."支払行番号"
1156
- </select>
1157
-
1158
- </mapper>
1159
- ```
1160
-
1161
- </details>
1162
-
1163
- ### リレーション設定のポイント
1164
-
1165
- | 設定項目 | 説明 |
1166
- |---------|------|
1167
- | `<collection>` | 1:N 関連のマッピング |
1168
- | `<id>` | 主キーの識別(MyBatis が重複排除に使用) |
1169
- | `resultMap` | ネストした ResultMap の参照 |
1170
- | エイリアス(AS) | カラム名の重複を避けるためのプレフィックス |
1171
- | `ORDER BY` | コレクションの順序を保証 |
1172
-
1173
- ### 楽観ロック(Optimistic Locking)の実装
1174
-
1175
- 複数ユーザーが同時に支払データを編集する場合、データの整合性を保つために楽観ロックを実装します。
1176
-
1177
- ```plantuml
1178
- @startuml
1179
-
1180
- title 楽観ロックの動作シーケンス
1181
-
1182
- actor "担当者A" as userA
1183
- actor "担当者B" as userB
1184
- database "支払テーブル" as db
1185
-
1186
- userA -> db : SELECT(version=1)
1187
- userB -> db : SELECT(version=1)
1188
- userA -> userA : 支払内容を編集
1189
- userB -> userB : 支払内容を編集
1190
- userA -> db : UPDATE WHERE version=1\n→ version=2 に更新
1191
- db -> userA : 更新成功(1行)
1192
- userB -> db : UPDATE WHERE version=1
1193
- db -> userB : 更新失敗(0行)\n楽観ロック例外
1194
- userB -> userB : 再読み込みを促す
1195
-
1196
- @enduml
1197
- ```
1198
-
1199
- ### バージョンカラムによる同時更新制御
1200
-
1201
- <details>
1202
- <summary>楽観ロック対応の MyBatis Mapper</summary>
1203
-
1204
- ```xml
1205
- <!-- 楽観ロック対応の支払更新 -->
1206
- <update id="updateWithOptimisticLock"
1207
- parameterType="com.example.sms.domain.model.payment.Payment">
1208
- UPDATE "支払データ"
1209
- SET
1210
- "支払締日" = #{paymentClosingDate},
1211
- "支払予定日" = #{paymentDueDate},
1212
- "支払方法" = #{paymentMethod,
1213
- typeHandler=com.example.sms.infrastructure.out.persistence.typehandler.PaymentMethodTypeHandler}::支払方法,
1214
- "支払金額" = #{paymentAmount},
1215
- "消費税額" = #{taxAmount},
1216
- "源泉徴収額" = #{withholdingAmount},
1217
- "差引支払額" = #{netPaymentAmount},
1218
- "ステータス" = #{status,
1219
- typeHandler=com.example.sms.infrastructure.out.persistence.typehandler.PaymentStatusTypeHandler}::支払ステータス,
1220
- "備考" = #{remarks},
1221
- "更新日時" = CURRENT_TIMESTAMP,
1222
- "バージョン" = "バージョン" + 1
1223
- WHERE "ID" = #{id}
1224
- AND "バージョン" = #{version}
1225
- </update>
1226
- ```
1227
-
1228
- </details>
1229
-
1230
- <details>
1231
- <summary>Repository 実装:楽観ロック対応</summary>
1232
-
1233
- ```java
1234
- // src/main/java/com/example/sms/infrastructure/persistence/repository/PaymentRepositoryImpl.java
1235
- package com.example.sms.infrastructure.out.persistence.repository;
1236
-
1237
- import com.example.sms.application.port.out.PaymentRepository;
1238
- import com.example.sms.domain.exception.OptimisticLockException;
1239
- import com.example.sms.domain.model.payment.*;
1240
- import com.example.sms.infrastructure.out.persistence.mapper.PaymentMapper;
1241
- import lombok.RequiredArgsConstructor;
1242
- import org.springframework.stereotype.Repository;
1243
- import org.springframework.transaction.annotation.Transactional;
1244
-
1245
- @Repository
1246
- @RequiredArgsConstructor
1247
- public class PaymentRepositoryImpl implements PaymentRepository {
1248
-
1249
- private final PaymentMapper paymentMapper;
1250
-
1251
- /**
1252
- * 楽観ロック対応の更新
1253
- * @throws OptimisticLockException 他のユーザーによって更新された場合
1254
- */
1255
- @Override
1256
- @Transactional
1257
- public void update(Payment payment) {
1258
- int updatedCount = paymentMapper.updateWithOptimisticLock(payment);
1259
-
1260
- if (updatedCount == 0) {
1261
- Integer currentVersion = paymentMapper.findVersionById(payment.getId());
1262
-
1263
- if (currentVersion == null) {
1264
- throw new OptimisticLockException(
1265
- "支払", payment.getId().toString());
1266
- } else {
1267
- throw new OptimisticLockException(
1268
- "支払",
1269
- payment.getId().toString(),
1270
- payment.getVersion(),
1271
- currentVersion
1272
- );
1273
- }
1274
- }
1275
- }
1276
- }
1277
- ```
1278
-
1279
- </details>
1280
-
1281
- ### 楽観ロックのベストプラクティス
1282
-
1283
- | ポイント | 説明 |
1284
- |---------|------|
1285
- | **バージョンカラム** | INTEGER 型で十分(オーバーフローは現実的に発生しない) |
1286
- | **WHERE 条件** | 必ず `AND "バージョン" = #{version}` を含める |
1287
- | **インクリメント** | `"バージョン" = "バージョン" + 1` でアトミックに更新 |
1288
- | **例外処理** | 更新件数が0の場合は楽観ロック例外をスロー |
1289
- | **エラーメッセージ** | ユーザーにわかりやすいメッセージで再読み込みを促す |
1290
-
1291
- ---
1292
-
1293
- ## 債権管理との対比
1294
-
1295
- 債権管理(第7章)と債務管理(第10章)は、売り手と買い手の立場が逆転した対称的な関係にあります。
1296
-
1297
- | 項目 | 債権管理(第7章) | 債務管理(第10章) |
1298
- |------|------------------|------------------|
1299
- | 取引先 | 顧客 | 仕入先 |
1300
- | 締処理 | 請求締処理 | 支払締処理 |
1301
- | データ | 請求データ | 支払データ |
1302
- | 残高 | 売掛金残高 | 買掛金残高 |
1303
- | 消込 | 入金消込 | 支払消込 |
1304
- | 予定管理 | 回収予定 | 支払予定 |
1305
-
1306
- ```plantuml
1307
- @startuml
1308
-
1309
- title 債権管理と債務管理の関係
1310
-
1311
- rectangle "自社" as company #LightGreen
1312
-
1313
- rectangle "顧客" as customer #LightBlue
1314
- rectangle "仕入先" as supplier #LightCoral
1315
-
1316
- company -right-> customer : 売上・請求
1317
- customer -left-> company : 入金
1318
-
1319
- supplier -right-> company : 仕入・請求
1320
- company -left-> supplier : 支払
1321
-
1322
- note top of company
1323
- 債権管理:
1324
- - 請求締処理
1325
- - 売掛金残高
1326
- - 入金消込
1327
- end note
1328
-
1329
- note bottom of company
1330
- 債務管理:
1331
- - 支払締処理
1332
- - 買掛金残高
1333
- - 支払消込
1334
- end note
1335
-
1336
- @enduml
1337
- ```
1338
-
1339
- ---
1340
-
1341
- ## 第10章のまとめ
1342
-
1343
- 本章では、債務管理(支払締処理・買掛金残高管理)の DB 設計と実装について学びました。
1344
-
1345
- ### 学んだこと
1346
-
1347
- 1. **支払締処理の設計**
1348
- - 仕入データから支払データへの集計
1349
- - 支払予定日の自動計算(支払サイト)
1350
- - 承認ワークフロー(作成→承認待ち→承認済→支払済)
1351
-
1352
- 2. **支払方法の管理**
1353
- - 振込、手形、現金、相殺、電子記録債権
1354
- - 支払方法に応じた必要情報の管理
1355
-
1356
- 3. **買掛金残高管理の設計**
1357
- - 仕入先別の残高管理
1358
- - 月次繰越処理
1359
- - 当月残高の自動計算(トリガー)
1360
-
1361
- 4. **リレーションと楽観ロック**
1362
- - N+1 問題の回避(JOIN による一括取得)
1363
- - 同時更新の競合制御
1364
-
1365
- ### テーブル一覧
1366
-
1367
- | テーブル名(日本語) | Java エンティティ | 説明 |
1368
- |---|---|---|
1369
- | 支払データ | Payment | 支払ヘッダ情報 |
1370
- | 支払明細データ | PaymentDetail | 仕入との紐付け |
1371
- | 買掛金残高データ | PayableBalance | 月次残高管理 |
1372
- | 支払予定データ | PaymentSchedule | 支払スケジュール |
1373
-
1374
- ### ENUM 一覧
1375
-
1376
- | DB ENUM 型(日本語) | Java Enum | 値 |
1377
- |---|---|---|
1378
- | 支払ステータス | PaymentStatus | 作成中→DRAFT, 承認待ち→PENDING_APPROVAL, 承認済→APPROVED, 支払済→PAID, 取消→CANCELLED |
1379
- | 支払方法 | PaymentMethod | 振込→BANK_TRANSFER, 手形→BILL, 現金→CASH, 相殺→OFFSET, 電子記録債権→ELECTRONIC_RECORD |
1380
-
1381
- ### 次章の予告
1382
-
1383
- 第11章では、財務会計システムの全体像に進みます。販売管理システムで発生した取引を財務会計の仕訳としてどのように記録するかを学びます。
1384
-
1385
- ---
1386
-
1387
- [← 第9章:在庫管理の設計](./chapter09.md) | [第11章:財務会計システムの全体像 →](./chapter11.md)
1
+ # 第10章:債務管理の設計
2
+
3
+ 販売管理システムにおいて、債務管理は仕入先への支払を適切に管理するための重要な機能です。本章では、支払締処理、支払方法の管理、買掛金残高管理のデータベース設計と実装を行います。債権管理(第7章)と対をなす概念として、仕入側からの視点でデータ構造を設計します。
4
+
5
+ ## 支払業務の全体像
6
+
7
+ 支払業務は「仕入計上 → 支払締処理 → 支払承認 → 支払実行 → 買掛金消込」という一連のフローで構成されます。
8
+
9
+ ```plantuml
10
+ @startuml
11
+
12
+ title 支払業務の全体フロー
13
+
14
+ |調達部門|
15
+ start
16
+ :発注;
17
+ :入荷・検収;
18
+ :仕入計上;
19
+ note right: 仕入データから\n買掛金が発生
20
+
21
+ |経理部門|
22
+ :支払締処理;
23
+ note right: 締日ごとに\n仕入金額を集計
24
+
25
+ :支払予定作成;
26
+ note right: 支払日・支払方法\nを決定
27
+
28
+ :支払承認;
29
+
30
+ |財務部門|
31
+ if (支払方法?) then (振込)
32
+ :振込データ作成;
33
+ :銀行振込実行;
34
+ elseif (支払方法?) then (手形)
35
+ :手形発行;
36
+ :手形交付;
37
+ else (現金)
38
+ :現金支払;
39
+ endif
40
+
41
+ |経理部門|
42
+ :支払消込;
43
+ note right: 買掛金残高から\n支払額を減少
44
+
45
+ :買掛金残高更新;
46
+
47
+ stop
48
+
49
+ @enduml
50
+ ```
51
+
52
+ ### 支払管理で扱うデータ
53
+
54
+ | データ | 説明 |
55
+ |-------|------|
56
+ | **支払データ** | 支払締処理で作成されるヘッダ情報 |
57
+ | **支払明細データ** | 仕入データとの紐付け |
58
+ | **買掛金残高データ** | 仕入先別・月次の残高管理 |
59
+ | **支払予定データ** | 支払スケジュールの管理 |
60
+
61
+ ---
62
+
63
+ ## 10.1 支払業務の DB 設計
64
+
65
+ ### 支払締処理の仕組み
66
+
67
+ 支払締処理は、債権管理の請求締処理と対になる仕組みです。仕入先マスタに設定された締日に基づいて、仕入データを集計し支払データを作成します。
68
+
69
+ ```plantuml
70
+ @startuml
71
+
72
+ title 支払締処理の流れ
73
+
74
+ |経理部門|
75
+ start
76
+ :締日到来確認;
77
+ note right: 仕入先マスタの\n締日を参照
78
+
79
+ :対象仕入データ抽出;
80
+ note right: 前回締日翌日~今回締日\nの仕入を対象
81
+
82
+ :支払データ作成;
83
+ note right: 支払ヘッダ・明細を\n自動生成
84
+
85
+ :支払金額計算;
86
+ fork
87
+ :仕入金額合計;
88
+ fork again
89
+ :消費税計算;
90
+ fork again
91
+ :源泉徴収計算;
92
+ fork end
93
+
94
+ :支払予定日算出;
95
+ note right: 支払サイト(月数)\nから計算
96
+
97
+ :支払データ確定;
98
+
99
+ stop
100
+
101
+ @enduml
102
+ ```
103
+
104
+ ### 支払ステータスの遷移
105
+
106
+ 支払データは承認ワークフローに従ってステータスが変化します。
107
+
108
+ ```plantuml
109
+ @startuml
110
+
111
+ title 支払ステータスの遷移
112
+
113
+ [*] --> 作成中
114
+
115
+ 作成中 --> 承認待ち : 承認申請
116
+
117
+ 承認待ち --> 承認済 : 承認
118
+
119
+ 承認済 --> 支払済 : 支払実行
120
+
121
+ 作成中 --> 取消 : 取消
122
+ 承認待ち --> 取消 : 取消
123
+
124
+ 取消 --> [*]
125
+ 支払済 --> [*]
126
+
127
+ @enduml
128
+ ```
129
+
130
+ | ステータス | 説明 | 次のアクション |
131
+ |-----------|------|---------------|
132
+ | **作成中** | 支払データ作成直後 | 承認申請または取消 |
133
+ | **承認待ち** | 承認申請済み | 承認または取消 |
134
+ | **承認済** | 管理者が承認 | 支払実行 |
135
+ | **支払済** | 支払処理完了 | 終了状態 |
136
+ | **取消** | 支払を取り消し | 終了状態 |
137
+
138
+ ### 支払方法の種類
139
+
140
+ | 支払方法 | 説明 | 特徴 |
141
+ |---------|------|------|
142
+ | **振込** | 銀行振込 | 最も一般的、手数料発生 |
143
+ | **手形** | 約束手形 | 支払サイト延長、不渡りリスク |
144
+ | **現金** | 現金払い | 即時決済、少額向け |
145
+ | **相殺** | 債権債務相殺 | 売掛金との相殺 |
146
+ | **電子記録債権** | でんさい | 手形の電子版 |
147
+
148
+ ### 支払関連の ER 図
149
+
150
+ ```plantuml
151
+ @startuml
152
+
153
+ title 債務管理のER図
154
+
155
+ entity 取引先マスタ {
156
+ 取引先コード <<PK>>
157
+ --
158
+ 取引先名
159
+ 取引先区分
160
+ 締日
161
+ 支払月
162
+ 支払日
163
+ }
164
+
165
+ entity 仕入データ {
166
+ 仕入番号 <<PK>>
167
+ --
168
+ 仕入先コード <<FK>>
169
+ 仕入日
170
+ 仕入金額
171
+ 消費税額
172
+ ステータス
173
+ }
174
+
175
+ entity 支払データ {
176
+ ID <<PK>>
177
+ --
178
+ 支払番号 <<UK>>
179
+ 仕入先コード <<FK>>
180
+ 支払締日
181
+ 支払予定日
182
+ 支払方法
183
+ 支払金額
184
+ 消費税額
185
+ 源泉徴収額
186
+ 差引支払額
187
+ 支払実行日
188
+ ステータス
189
+ バージョン
190
+ }
191
+
192
+ entity 支払明細データ {
193
+ ID <<PK>>
194
+ --
195
+ 支払ID <<FK>>
196
+ 支払行番号
197
+ 仕入番号 <<FK>>
198
+ 仕入日
199
+ 仕入金額
200
+ 消費税額
201
+ 支払対象金額
202
+ }
203
+
204
+ entity 買掛金残高データ {
205
+ ID <<PK>>
206
+ --
207
+ 仕入先コード <<FK>>
208
+ 年月
209
+ 前月残高
210
+ 当月仕入高
211
+ 当月支払高
212
+ 当月残高
213
+ バージョン
214
+ }
215
+
216
+ entity 支払予定データ {
217
+ ID <<PK>>
218
+ --
219
+ 仕入先コード <<FK>>
220
+ 支払予定日
221
+ 支払予定額
222
+ 支払方法
223
+ 支払済フラグ
224
+ 支払ID <<FK>>
225
+ }
226
+
227
+ 取引先マスタ ||--o{ 仕入データ
228
+ 取引先マスタ ||--o{ 支払データ
229
+ 取引先マスタ ||--o{ 買掛金残高データ
230
+ 取引先マスタ ||--o{ 支払予定データ
231
+ 支払データ ||--o{ 支払明細データ
232
+ 仕入データ ||--o{ 支払明細データ
233
+ 支払データ |o--o{ 支払予定データ
234
+
235
+ @enduml
236
+ ```
237
+
238
+ ### マイグレーション:支払関連テーブルの作成
239
+
240
+ <details>
241
+ <summary>V017__create_payment_tables.sql</summary>
242
+
243
+ ```sql
244
+ -- src/main/resources/db/migration/V017__create_payment_tables.sql
245
+
246
+ -- 支払ステータス
247
+ CREATE TYPE 支払ステータス AS ENUM ('作成中', '承認待ち', '承認済', '支払済', '取消');
248
+
249
+ -- 支払方法に不足している値を追加(既存のENUMはV001で定義済み)
250
+ ALTER TYPE 支払方法 ADD VALUE IF NOT EXISTS '相殺';
251
+ ALTER TYPE 支払方法 ADD VALUE IF NOT EXISTS '電子記録債権';
252
+
253
+ -- 支払データ(ヘッダ)
254
+ CREATE TABLE "支払データ" (
255
+ "ID" SERIAL PRIMARY KEY,
256
+ "支払番号" VARCHAR(20) UNIQUE NOT NULL,
257
+ "仕入先コード" VARCHAR(20) NOT NULL,
258
+ "支払締日" DATE NOT NULL,
259
+ "支払予定日" DATE NOT NULL,
260
+ "支払方法" 支払方法 NOT NULL,
261
+ "支払金額" DECIMAL(15, 2) NOT NULL,
262
+ "消費税額" DECIMAL(15, 2) NOT NULL,
263
+ "源泉徴収額" DECIMAL(15, 2) DEFAULT 0 NOT NULL,
264
+ "差引支払額" DECIMAL(15, 2) NOT NULL,
265
+ "支払実行日" DATE,
266
+ "ステータス" 支払ステータス DEFAULT '作成中' NOT NULL,
267
+ "振込先銀行コード" VARCHAR(10),
268
+ "振込先支店コード" VARCHAR(10),
269
+ "振込先口座種別" VARCHAR(10),
270
+ "振込先口座番号" VARCHAR(20),
271
+ "振込先口座名義" VARCHAR(100),
272
+ "備考" TEXT,
273
+ "バージョン" INTEGER DEFAULT 1 NOT NULL,
274
+ "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
275
+ "作成者" VARCHAR(50),
276
+ "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
277
+ "更新者" VARCHAR(50),
278
+ CONSTRAINT "fk_支払_仕入先"
279
+ FOREIGN KEY ("仕入先コード") REFERENCES "取引先マスタ"("取引先コード")
280
+ );
281
+
282
+ -- 支払明細データ
283
+ CREATE TABLE "支払明細データ" (
284
+ "ID" SERIAL PRIMARY KEY,
285
+ "支払ID" INTEGER NOT NULL,
286
+ "支払行番号" INTEGER NOT NULL,
287
+ "仕入番号" VARCHAR(20) NOT NULL,
288
+ "仕入日" DATE NOT NULL,
289
+ "仕入金額" DECIMAL(15, 2) NOT NULL,
290
+ "消費税額" DECIMAL(15, 2) NOT NULL,
291
+ "支払対象金額" DECIMAL(15, 2) NOT NULL,
292
+ "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
293
+ "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
294
+ CONSTRAINT "fk_支払明細_支払"
295
+ FOREIGN KEY ("支払ID") REFERENCES "支払データ"("ID") ON DELETE CASCADE,
296
+ CONSTRAINT "fk_支払明細_仕入"
297
+ FOREIGN KEY ("仕入番号") REFERENCES "仕入データ"("仕入番号"),
298
+ CONSTRAINT "uk_支払明細_支払_行" UNIQUE ("支払ID", "支払行番号")
299
+ );
300
+
301
+ -- 買掛金残高データ
302
+ CREATE TABLE "買掛金残高データ" (
303
+ "ID" SERIAL PRIMARY KEY,
304
+ "仕入先コード" VARCHAR(20) NOT NULL,
305
+ "年月" DATE NOT NULL,
306
+ "前月残高" DECIMAL(15, 2) DEFAULT 0 NOT NULL,
307
+ "当月仕入高" DECIMAL(15, 2) DEFAULT 0 NOT NULL,
308
+ "当月支払高" DECIMAL(15, 2) DEFAULT 0 NOT NULL,
309
+ "当月残高" DECIMAL(15, 2) DEFAULT 0 NOT NULL,
310
+ "バージョン" INTEGER DEFAULT 1 NOT NULL,
311
+ "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
312
+ "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
313
+ CONSTRAINT "fk_買掛金残高_仕入先"
314
+ FOREIGN KEY ("仕入先コード") REFERENCES "取引先マスタ"("取引先コード"),
315
+ UNIQUE ("仕入先コード", "年月")
316
+ );
317
+
318
+ -- 支払予定データ
319
+ CREATE TABLE "支払予定データ" (
320
+ "ID" SERIAL PRIMARY KEY,
321
+ "仕入先コード" VARCHAR(20) NOT NULL,
322
+ "支払予定日" DATE NOT NULL,
323
+ "支払予定額" DECIMAL(15, 2) NOT NULL,
324
+ "支払方法" 支払方法 NOT NULL,
325
+ "支払済フラグ" BOOLEAN DEFAULT FALSE NOT NULL,
326
+ "支払ID" INTEGER,
327
+ "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
328
+ "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
329
+ CONSTRAINT "fk_支払予定_仕入先"
330
+ FOREIGN KEY ("仕入先コード") REFERENCES "取引先マスタ"("取引先コード"),
331
+ CONSTRAINT "fk_支払予定_支払"
332
+ FOREIGN KEY ("支払ID") REFERENCES "支払データ"("ID")
333
+ );
334
+
335
+ -- 当月残高を自動計算するトリガー
336
+ CREATE OR REPLACE FUNCTION calculate_payable_balance()
337
+ RETURNS TRIGGER AS $$
338
+ BEGIN
339
+ NEW."当月残高" := NEW."前月残高" + NEW."当月仕入高" - NEW."当月支払高";
340
+ RETURN NEW;
341
+ END;
342
+ $$ LANGUAGE plpgsql;
343
+
344
+ CREATE TRIGGER trg_買掛金残高_残高計算
345
+ BEFORE INSERT OR UPDATE ON "買掛金残高データ"
346
+ FOR EACH ROW
347
+ EXECUTE FUNCTION calculate_payable_balance();
348
+
349
+ -- インデックス
350
+ CREATE INDEX "idx_支払_仕入先コード" ON "支払データ"("仕入先コード");
351
+ CREATE INDEX "idx_支払_支払締日" ON "支払データ"("支払締日");
352
+ CREATE INDEX "idx_支払_支払予定日" ON "支払データ"("支払予定日");
353
+ CREATE INDEX "idx_支払_ステータス" ON "支払データ"("ステータス");
354
+ CREATE INDEX "idx_支払明細_仕入番号" ON "支払明細データ"("仕入番号");
355
+ CREATE INDEX "idx_買掛金残高_年月" ON "買掛金残高データ"("年月");
356
+ CREATE INDEX "idx_支払予定_支払予定日" ON "支払予定データ"("支払予定日");
357
+ CREATE INDEX "idx_支払予定_支払済" ON "支払予定データ"("支払済フラグ");
358
+
359
+ -- テーブルコメント
360
+ COMMENT ON TABLE "支払データ" IS '支払締処理で作成される支払ヘッダ情報';
361
+ COMMENT ON TABLE "支払明細データ" IS '支払データと仕入データの紐付け';
362
+ COMMENT ON TABLE "買掛金残高データ" IS '仕入先別・月次の買掛金残高';
363
+ COMMENT ON TABLE "支払予定データ" IS '支払スケジュールの管理';
364
+ COMMENT ON COLUMN "支払データ"."バージョン" IS '楽観ロック用バージョン番号';
365
+ COMMENT ON COLUMN "買掛金残高データ"."バージョン" IS '楽観ロック用バージョン番号';
366
+ ```
367
+
368
+ </details>
369
+
370
+ ### 支払関連エンティティの実装
371
+
372
+ <details>
373
+ <summary>支払ステータス ENUM</summary>
374
+
375
+ ```java
376
+ // src/main/java/com/example/sms/domain/model/payment/PaymentStatus.java
377
+ package com.example.sms.domain.model.payment;
378
+
379
+ import lombok.Getter;
380
+ import lombok.RequiredArgsConstructor;
381
+
382
+ @Getter
383
+ @RequiredArgsConstructor
384
+ public enum PaymentStatus {
385
+ DRAFT("作成中"),
386
+ PENDING_APPROVAL("承認待ち"),
387
+ APPROVED("承認済"),
388
+ PAID("支払済"),
389
+ CANCELLED("取消");
390
+
391
+ private final String displayName;
392
+
393
+ public static PaymentStatus fromDisplayName(String displayName) {
394
+ for (PaymentStatus status : values()) {
395
+ if (status.displayName.equals(displayName)) {
396
+ return status;
397
+ }
398
+ }
399
+ throw new IllegalArgumentException("Unknown payment status: " + displayName);
400
+ }
401
+
402
+ /**
403
+ * 承認可能なステータスかどうか.
404
+ */
405
+ public boolean canApprove() {
406
+ return this == PENDING_APPROVAL;
407
+ }
408
+
409
+ /**
410
+ * 支払実行可能なステータスかどうか.
411
+ */
412
+ public boolean canExecute() {
413
+ return this == APPROVED;
414
+ }
415
+
416
+ /**
417
+ * 取消可能なステータスかどうか.
418
+ */
419
+ public boolean canCancel() {
420
+ return this == DRAFT || this == PENDING_APPROVAL;
421
+ }
422
+ }
423
+ ```
424
+
425
+ </details>
426
+
427
+ <details>
428
+ <summary>支払方法 ENUM</summary>
429
+
430
+ ```java
431
+ // src/main/java/com/example/sms/domain/model/payment/PaymentMethod.java
432
+ package com.example.sms.domain.model.payment;
433
+
434
+ import lombok.Getter;
435
+ import lombok.RequiredArgsConstructor;
436
+
437
+ @Getter
438
+ @RequiredArgsConstructor
439
+ public enum PaymentMethod {
440
+ BANK_TRANSFER("振込"),
441
+ BILL("手形"),
442
+ CASH("現金"),
443
+ OFFSET("相殺"),
444
+ ELECTRONIC_RECORD("電子記録債権");
445
+
446
+ private final String displayName;
447
+
448
+ public static PaymentMethod fromDisplayName(String displayName) {
449
+ for (PaymentMethod method : values()) {
450
+ if (method.displayName.equals(displayName)) {
451
+ return method;
452
+ }
453
+ }
454
+ throw new IllegalArgumentException("Unknown payment method: " + displayName);
455
+ }
456
+
457
+ /**
458
+ * 振込先情報が必要かどうか.
459
+ */
460
+ public boolean requiresBankInfo() {
461
+ return this == BANK_TRANSFER;
462
+ }
463
+ }
464
+ ```
465
+
466
+ </details>
467
+
468
+ <details>
469
+ <summary>支払データエンティティ</summary>
470
+
471
+ ```java
472
+ // src/main/java/com/example/sms/domain/model/payment/Payment.java
473
+ package com.example.sms.domain.model.payment;
474
+
475
+ import lombok.AllArgsConstructor;
476
+ import lombok.Builder;
477
+ import lombok.Data;
478
+ import lombok.NoArgsConstructor;
479
+
480
+ import java.math.BigDecimal;
481
+ import java.time.LocalDate;
482
+ import java.time.LocalDateTime;
483
+ import java.util.ArrayList;
484
+ import java.util.List;
485
+
486
+ @Data
487
+ @Builder
488
+ @NoArgsConstructor
489
+ @AllArgsConstructor
490
+ public class Payment {
491
+ private Integer id;
492
+ private String paymentNumber;
493
+ private String supplierCode;
494
+ private LocalDate paymentClosingDate;
495
+ private LocalDate paymentDueDate;
496
+ private PaymentMethod paymentMethod;
497
+ private BigDecimal paymentAmount;
498
+ private BigDecimal taxAmount;
499
+ private BigDecimal withholdingAmount;
500
+ private BigDecimal netPaymentAmount;
501
+ private LocalDate paymentExecutionDate;
502
+ private PaymentStatus status;
503
+ private String bankCode;
504
+ private String branchCode;
505
+ private String accountType;
506
+ private String accountNumber;
507
+ private String accountName;
508
+ private String remarks;
509
+ private LocalDateTime createdAt;
510
+ private String createdBy;
511
+ private LocalDateTime updatedAt;
512
+ private String updatedBy;
513
+
514
+ @Builder.Default
515
+ private Integer version = 1;
516
+
517
+ @Builder.Default
518
+ private List<PaymentDetail> details = new ArrayList<>();
519
+
520
+ /**
521
+ * 明細の合計金額を計算.
522
+ */
523
+ public BigDecimal calculateTotalAmount() {
524
+ if (details == null || details.isEmpty()) {
525
+ return BigDecimal.ZERO;
526
+ }
527
+ return details.stream()
528
+ .map(PaymentDetail::getPaymentTargetAmount)
529
+ .reduce(BigDecimal.ZERO, BigDecimal::add);
530
+ }
531
+
532
+ /**
533
+ * 差引支払額を計算.
534
+ */
535
+ public BigDecimal calculateNetAmount() {
536
+ BigDecimal wh = withholdingAmount != null ? withholdingAmount : BigDecimal.ZERO;
537
+ return paymentAmount.add(taxAmount).subtract(wh);
538
+ }
539
+
540
+ /**
541
+ * 承認申請.
542
+ */
543
+ public void submitForApproval() {
544
+ if (this.status != PaymentStatus.DRAFT) {
545
+ throw new IllegalStateException("作成中ステータスのみ承認申請が可能です");
546
+ }
547
+ this.status = PaymentStatus.PENDING_APPROVAL;
548
+ }
549
+
550
+ /**
551
+ * 承認.
552
+ */
553
+ public void approve() {
554
+ if (!this.status.canApprove()) {
555
+ throw new IllegalStateException(
556
+ "このステータスでは承認できません: " + this.status.getDisplayName());
557
+ }
558
+ this.status = PaymentStatus.APPROVED;
559
+ }
560
+
561
+ /**
562
+ * 支払実行.
563
+ */
564
+ public void execute(LocalDate executionDate) {
565
+ if (!this.status.canExecute()) {
566
+ throw new IllegalStateException(
567
+ "このステータスでは支払実行できません: " + this.status.getDisplayName());
568
+ }
569
+ this.paymentExecutionDate = executionDate;
570
+ this.status = PaymentStatus.PAID;
571
+ }
572
+
573
+ /**
574
+ * 取消.
575
+ */
576
+ public void cancel() {
577
+ if (!this.status.canCancel()) {
578
+ throw new IllegalStateException(
579
+ "このステータスでは取消できません: " + this.status.getDisplayName());
580
+ }
581
+ this.status = PaymentStatus.CANCELLED;
582
+ }
583
+ }
584
+ ```
585
+
586
+ </details>
587
+
588
+ <details>
589
+ <summary>支払明細データエンティティ</summary>
590
+
591
+ ```java
592
+ // src/main/java/com/example/sms/domain/model/payment/PaymentDetail.java
593
+ package com.example.sms.domain.model.payment;
594
+
595
+ import lombok.AllArgsConstructor;
596
+ import lombok.Builder;
597
+ import lombok.Data;
598
+ import lombok.NoArgsConstructor;
599
+
600
+ import java.math.BigDecimal;
601
+ import java.time.LocalDate;
602
+ import java.time.LocalDateTime;
603
+
604
+ @Data
605
+ @Builder
606
+ @NoArgsConstructor
607
+ @AllArgsConstructor
608
+ public class PaymentDetail {
609
+ private Integer id;
610
+ private Integer paymentId;
611
+ private Integer lineNumber;
612
+ private String purchaseNumber;
613
+ private LocalDate purchaseDate;
614
+ private BigDecimal purchaseAmount;
615
+ private BigDecimal taxAmount;
616
+ private BigDecimal paymentTargetAmount;
617
+ private LocalDateTime createdAt;
618
+ private LocalDateTime updatedAt;
619
+ }
620
+ ```
621
+
622
+ </details>
623
+
624
+ <details>
625
+ <summary>買掛金残高データエンティティ</summary>
626
+
627
+ ```java
628
+ // src/main/java/com/example/sms/domain/model/payment/PayableBalance.java
629
+ package com.example.sms.domain.model.payment;
630
+
631
+ import lombok.AllArgsConstructor;
632
+ import lombok.Builder;
633
+ import lombok.Data;
634
+ import lombok.NoArgsConstructor;
635
+
636
+ import java.math.BigDecimal;
637
+ import java.time.LocalDate;
638
+ import java.time.LocalDateTime;
639
+
640
+ @Data
641
+ @Builder
642
+ @NoArgsConstructor
643
+ @AllArgsConstructor
644
+ public class PayableBalance {
645
+ private Integer id;
646
+ private String supplierCode;
647
+ private LocalDate yearMonth;
648
+ private BigDecimal previousBalance;
649
+ private BigDecimal currentPurchaseAmount;
650
+ private BigDecimal currentPaymentAmount;
651
+ private BigDecimal currentBalance;
652
+ private LocalDateTime createdAt;
653
+ private LocalDateTime updatedAt;
654
+
655
+ @Builder.Default
656
+ private Integer version = 1;
657
+
658
+ /**
659
+ * 残高を計算.
660
+ */
661
+ public BigDecimal calculateBalance() {
662
+ BigDecimal prev = previousBalance != null ? previousBalance : BigDecimal.ZERO;
663
+ BigDecimal purchase = currentPurchaseAmount != null ? currentPurchaseAmount : BigDecimal.ZERO;
664
+ BigDecimal payment = currentPaymentAmount != null ? currentPaymentAmount : BigDecimal.ZERO;
665
+ return prev.add(purchase).subtract(payment);
666
+ }
667
+
668
+ /**
669
+ * 仕入を加算.
670
+ */
671
+ public void addPurchase(BigDecimal amount) {
672
+ if (this.currentPurchaseAmount == null) {
673
+ this.currentPurchaseAmount = BigDecimal.ZERO;
674
+ }
675
+ this.currentPurchaseAmount = this.currentPurchaseAmount.add(amount);
676
+ this.currentBalance = calculateBalance();
677
+ }
678
+
679
+ /**
680
+ * 支払を加算.
681
+ */
682
+ public void addPayment(BigDecimal amount) {
683
+ if (this.currentPaymentAmount == null) {
684
+ this.currentPaymentAmount = BigDecimal.ZERO;
685
+ }
686
+ this.currentPaymentAmount = this.currentPaymentAmount.add(amount);
687
+ this.currentBalance = calculateBalance();
688
+ }
689
+ }
690
+ ```
691
+
692
+ </details>
693
+
694
+ <details>
695
+ <summary>支払予定データエンティティ</summary>
696
+
697
+ ```java
698
+ // src/main/java/com/example/sms/domain/model/payment/PaymentSchedule.java
699
+ package com.example.sms.domain.model.payment;
700
+
701
+ import lombok.AllArgsConstructor;
702
+ import lombok.Builder;
703
+ import lombok.Data;
704
+ import lombok.NoArgsConstructor;
705
+
706
+ import java.math.BigDecimal;
707
+ import java.time.LocalDate;
708
+ import java.time.LocalDateTime;
709
+
710
+ @Data
711
+ @Builder
712
+ @NoArgsConstructor
713
+ @AllArgsConstructor
714
+ public class PaymentSchedule {
715
+ private Integer id;
716
+ private String supplierCode;
717
+ private LocalDate paymentDueDate;
718
+ private BigDecimal scheduledAmount;
719
+ private PaymentMethod paymentMethod;
720
+ @Builder.Default
721
+ private Boolean paidFlag = false;
722
+ private Integer paymentId;
723
+ private LocalDateTime createdAt;
724
+ private LocalDateTime updatedAt;
725
+
726
+ /**
727
+ * 支払済みとしてマーク.
728
+ */
729
+ public void markAsPaid(Integer paymentId) {
730
+ this.paidFlag = true;
731
+ this.paymentId = paymentId;
732
+ }
733
+
734
+ /**
735
+ * 支払期限が過ぎているかどうか.
736
+ */
737
+ public boolean isOverdue() {
738
+ return !paidFlag && LocalDate.now().isAfter(paymentDueDate);
739
+ }
740
+ }
741
+ ```
742
+
743
+ </details>
744
+
745
+ ### 支払サービスの実装(参考実装例)
746
+
747
+ > **Note**: 以下のサービス層のコードは参考実装例です。現在はリポジトリ層(永続化層)が実装済みで、アプリケーションサービス層は別途実装が必要です。
748
+
749
+ <details>
750
+ <summary>支払サービス</summary>
751
+
752
+ ```java
753
+ // src/main/java/com/example/sms/application/service/PaymentService.java
754
+ package com.example.sms.application.service;
755
+
756
+ import com.example.sms.domain.model.partner.Partner;
757
+ import com.example.sms.domain.model.payment.*;
758
+ import com.example.sms.domain.model.purchase.Purchase;
759
+ import com.example.sms.infrastructure.out.persistence.mapper.*;
760
+ import lombok.RequiredArgsConstructor;
761
+ import org.springframework.stereotype.Service;
762
+ import org.springframework.transaction.annotation.Transactional;
763
+
764
+ import java.math.BigDecimal;
765
+ import java.time.LocalDate;
766
+ import java.time.format.DateTimeFormatter;
767
+ import java.util.ArrayList;
768
+ import java.util.List;
769
+
770
+ @Service
771
+ @RequiredArgsConstructor
772
+ public class PaymentService {
773
+
774
+ private final PaymentMapper paymentMapper;
775
+ private final PaymentDetailMapper paymentDetailMapper;
776
+ private final PayableBalanceMapper payableBalanceMapper;
777
+ private final PartnerMapper partnerMapper;
778
+ private final PurchaseMapper purchaseMapper;
779
+
780
+ /**
781
+ * 支払番号を生成
782
+ */
783
+ private String generatePaymentNumber(LocalDate closingDate) {
784
+ String prefix = "PAY-" + closingDate.format(
785
+ DateTimeFormatter.ofPattern("yyyyMM")) + "-";
786
+ String latestNumber = paymentMapper.findLatestPaymentNumber(prefix + "%");
787
+
788
+ int sequence = 1;
789
+ if (latestNumber != null) {
790
+ int currentSequence = Integer.parseInt(
791
+ latestNumber.substring(latestNumber.length() - 4));
792
+ sequence = currentSequence + 1;
793
+ }
794
+
795
+ return prefix + String.format("%04d", sequence);
796
+ }
797
+
798
+ /**
799
+ * 支払予定日を計算
800
+ */
801
+ private LocalDate calculatePaymentDueDate(LocalDate closingDate,
802
+ Partner supplier) {
803
+ // 締日の翌月から支払月数分を加算
804
+ LocalDate baseDate = closingDate.plusMonths(supplier.getPaymentMonth());
805
+
806
+ // 支払日を設定
807
+ int paymentDay = supplier.getPaymentDay();
808
+ int maxDay = baseDate.lengthOfMonth();
809
+ if (paymentDay > maxDay) {
810
+ paymentDay = maxDay; // 月末処理
811
+ }
812
+
813
+ return LocalDate.of(baseDate.getYear(), baseDate.getMonth(), paymentDay);
814
+ }
815
+
816
+ /**
817
+ * 支払締処理を実行
818
+ */
819
+ @Transactional
820
+ public Payment processPaymentClosing(PaymentClosingInput input) {
821
+ // 仕入先情報を取得
822
+ Partner supplier = partnerMapper.findByPartnerCode(input.getSupplierCode());
823
+ if (supplier == null) {
824
+ throw new IllegalArgumentException(
825
+ "仕入先が見つかりません: " + input.getSupplierCode());
826
+ }
827
+
828
+ // 対象仕入データを取得(前回締日翌日~今回締日)
829
+ List<Purchase> purchases = purchaseMapper.findBySupplierAndDateRange(
830
+ input.getSupplierCode(),
831
+ input.getFromDate() != null
832
+ ? input.getFromDate()
833
+ : LocalDate.of(1900, 1, 1),
834
+ input.getClosingDate()
835
+ );
836
+
837
+ if (purchases.isEmpty()) {
838
+ throw new IllegalStateException(
839
+ "対象となる仕入データがありません");
840
+ }
841
+
842
+ // 支払データを作成
843
+ String paymentNumber = generatePaymentNumber(input.getClosingDate());
844
+ LocalDate paymentDueDate = calculatePaymentDueDate(
845
+ input.getClosingDate(), supplier);
846
+
847
+ BigDecimal totalAmount = BigDecimal.ZERO;
848
+ BigDecimal totalTax = BigDecimal.ZERO;
849
+
850
+ List<PaymentDetail> details = new ArrayList<>();
851
+ int detailNumber = 1;
852
+
853
+ for (Purchase purchase : purchases) {
854
+ totalAmount = totalAmount.add(purchase.getPurchaseAmount());
855
+ totalTax = totalTax.add(purchase.getTaxAmount());
856
+
857
+ PaymentDetail detail = PaymentDetail.builder()
858
+ .paymentNumber(paymentNumber)
859
+ .detailNumber(detailNumber++)
860
+ .purchaseNumber(purchase.getPurchaseNumber())
861
+ .purchaseDate(purchase.getPurchaseDate())
862
+ .purchaseAmount(purchase.getPurchaseAmount())
863
+ .taxAmount(purchase.getTaxAmount())
864
+ .paymentTargetAmount(purchase.getTotalAmount())
865
+ .build();
866
+ details.add(detail);
867
+ }
868
+
869
+ BigDecimal withholdingAmount = input.getWithholdingAmount() != null
870
+ ? input.getWithholdingAmount()
871
+ : BigDecimal.ZERO;
872
+ BigDecimal netAmount = totalAmount.add(totalTax)
873
+ .subtract(withholdingAmount);
874
+
875
+ Payment payment = Payment.builder()
876
+ .paymentNumber(paymentNumber)
877
+ .supplierCode(input.getSupplierCode())
878
+ .paymentClosingDate(input.getClosingDate())
879
+ .paymentDueDate(paymentDueDate)
880
+ .paymentMethod(input.getPaymentMethod())
881
+ .paymentAmount(totalAmount)
882
+ .taxAmount(totalTax)
883
+ .withholdingAmount(withholdingAmount)
884
+ .netPaymentAmount(netAmount)
885
+ .status(PaymentStatus.DRAFT)
886
+ .bankCode(supplier.getBankCode())
887
+ .branchCode(supplier.getBranchCode())
888
+ .accountType(supplier.getAccountType())
889
+ .accountNumber(supplier.getAccountNumber())
890
+ .accountName(supplier.getAccountName())
891
+ .build();
892
+
893
+ paymentMapper.insert(payment);
894
+
895
+ for (PaymentDetail detail : details) {
896
+ paymentDetailMapper.insert(detail);
897
+ }
898
+
899
+ payment.setDetails(details);
900
+ return payment;
901
+ }
902
+
903
+ /**
904
+ * 承認申請
905
+ */
906
+ @Transactional
907
+ public void submitForApproval(String paymentNumber) {
908
+ Payment payment = paymentMapper.findByPaymentNumber(paymentNumber);
909
+ if (payment == null) {
910
+ throw new IllegalArgumentException(
911
+ "支払データが見つかりません: " + paymentNumber);
912
+ }
913
+
914
+ payment.submitForApproval();
915
+ paymentMapper.updateStatus(paymentNumber, PaymentStatus.PENDING_APPROVAL);
916
+ }
917
+
918
+ /**
919
+ * 承認
920
+ */
921
+ @Transactional
922
+ public void approve(String paymentNumber) {
923
+ Payment payment = paymentMapper.findByPaymentNumber(paymentNumber);
924
+ if (payment == null) {
925
+ throw new IllegalArgumentException(
926
+ "支払データが見つかりません: " + paymentNumber);
927
+ }
928
+
929
+ payment.approve();
930
+ paymentMapper.updateStatus(paymentNumber, PaymentStatus.APPROVED);
931
+ }
932
+
933
+ /**
934
+ * 支払実行
935
+ */
936
+ @Transactional
937
+ public void executePayment(String paymentNumber, LocalDate executionDate) {
938
+ Payment payment = paymentMapper.findByPaymentNumber(paymentNumber);
939
+ if (payment == null) {
940
+ throw new IllegalArgumentException(
941
+ "支払データが見つかりません: " + paymentNumber);
942
+ }
943
+
944
+ payment.execute(executionDate);
945
+ paymentMapper.executePayment(paymentNumber, executionDate);
946
+
947
+ // 買掛金残高を更新
948
+ LocalDate yearMonth = LocalDate.of(
949
+ executionDate.getYear(), executionDate.getMonth(), 1);
950
+ payableBalanceMapper.addPayment(
951
+ payment.getSupplierCode(), yearMonth, payment.getNetPaymentAmount());
952
+ }
953
+
954
+ /**
955
+ * 支払取消
956
+ */
957
+ @Transactional
958
+ public void cancelPayment(String paymentNumber) {
959
+ Payment payment = paymentMapper.findByPaymentNumber(paymentNumber);
960
+ if (payment == null) {
961
+ throw new IllegalArgumentException(
962
+ "支払データが見つかりません: " + paymentNumber);
963
+ }
964
+
965
+ payment.cancel();
966
+ paymentMapper.updateStatus(paymentNumber, PaymentStatus.CANCELLED);
967
+ }
968
+
969
+ /**
970
+ * 買掛金残高を取得
971
+ */
972
+ public PayableBalance getPayableBalance(String supplierCode,
973
+ LocalDate yearMonth) {
974
+ return payableBalanceMapper.findBySupplierAndYearMonth(
975
+ supplierCode, yearMonth);
976
+ }
977
+
978
+ /**
979
+ * 月次の買掛金合計を取得
980
+ */
981
+ public BigDecimal getTotalPayableByMonth(LocalDate yearMonth) {
982
+ return payableBalanceMapper.sumCurrentBalanceByYearMonth(yearMonth);
983
+ }
984
+
985
+ /**
986
+ * 買掛金残高を初期化(月次繰越)
987
+ */
988
+ @Transactional
989
+ public PayableBalance initializePayableBalance(String supplierCode,
990
+ LocalDate yearMonth) {
991
+ // 前月残高を取得
992
+ LocalDate previousMonth = yearMonth.minusMonths(1);
993
+ PayableBalance previousBalance = payableBalanceMapper
994
+ .findBySupplierAndYearMonth(supplierCode, previousMonth);
995
+
996
+ BigDecimal carryOver = BigDecimal.ZERO;
997
+ if (previousBalance != null) {
998
+ carryOver = previousBalance.getCurrentBalance();
999
+ }
1000
+
1001
+ PayableBalance newBalance = PayableBalance.builder()
1002
+ .supplierCode(supplierCode)
1003
+ .yearMonth(yearMonth)
1004
+ .previousBalance(carryOver)
1005
+ .currentPurchaseAmount(BigDecimal.ZERO)
1006
+ .currentPaymentAmount(BigDecimal.ZERO)
1007
+ .build();
1008
+
1009
+ payableBalanceMapper.insert(newBalance);
1010
+ return newBalance;
1011
+ }
1012
+ }
1013
+ ```
1014
+
1015
+ </details>
1016
+
1017
+ <details>
1018
+ <summary>支払締処理の入力クラス</summary>
1019
+
1020
+ ```java
1021
+ // src/main/java/com/example/sms/application/service/PaymentClosingInput.java
1022
+ package com.example.sms.application.service;
1023
+
1024
+ import com.example.sms.domain.model.payment.PaymentMethod;
1025
+ import lombok.Builder;
1026
+ import lombok.Data;
1027
+
1028
+ import java.math.BigDecimal;
1029
+ import java.time.LocalDate;
1030
+
1031
+ @Data
1032
+ @Builder
1033
+ public class PaymentClosingInput {
1034
+ private String supplierCode;
1035
+ private LocalDate closingDate;
1036
+ private LocalDate fromDate;
1037
+ private PaymentMethod paymentMethod;
1038
+ private BigDecimal withholdingAmount;
1039
+ }
1040
+ ```
1041
+
1042
+ </details>
1043
+
1044
+ ---
1045
+
1046
+ ## 10.2 リレーションと楽観ロックの設計
1047
+
1048
+ ### MyBatis ネストした ResultMap による関連付け
1049
+
1050
+ 支払データは、支払(ヘッダ)→ 支払明細の2層構造を持ちます。N+1 問題を回避するため、JOIN クエリで一括取得します。
1051
+
1052
+ ```plantuml
1053
+ @startuml
1054
+
1055
+ title N+1 問題の回避
1056
+
1057
+ rectangle "N+1 問題(非効率)" #FFCCCC {
1058
+ card "1. 支払ヘッダ取得\n SELECT * FROM 支払データ\n (1回)" as q1
1059
+ card "2. 各支払の明細取得\n SELECT * FROM 支払明細データ\n WHERE 支払番号 = ?\n (N回)" as q2
1060
+ }
1061
+
1062
+ rectangle "解決策:JOINによる一括取得" #CCFFCC {
1063
+ card "1回のJOINクエリで\n全データを取得\n SELECT p.*, pd.*\n FROM 支払データ p\n LEFT JOIN 支払明細データ pd ON ..." as sol
1064
+ }
1065
+
1066
+ q1 --> q2 : N件
1067
+ note right of q2
1068
+ 10件の支払があれば
1069
+ 合計11回のクエリ発行
1070
+ end note
1071
+
1072
+ @enduml
1073
+ ```
1074
+
1075
+ <details>
1076
+ <summary>支払データのリレーション設定(PaymentMapper.xml)</summary>
1077
+
1078
+ ```xml
1079
+ <?xml version="1.0" encoding="UTF-8" ?>
1080
+ <!DOCTYPE mapper
1081
+ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
1082
+ "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
1083
+ <mapper namespace="com.example.sms.infrastructure.out.persistence.mapper.PaymentMapper">
1084
+
1085
+ <!-- 支払(ヘッダ)の ResultMap -->
1086
+ <resultMap id="PaymentWithDetailsResultMap"
1087
+ type="com.example.sms.domain.model.payment.Payment">
1088
+ <id property="id" column="p_id"/>
1089
+ <result property="paymentNumber" column="p_支払番号"/>
1090
+ <result property="supplierCode" column="p_仕入先コード"/>
1091
+ <result property="paymentClosingDate" column="p_支払締日"/>
1092
+ <result property="paymentDueDate" column="p_支払予定日"/>
1093
+ <result property="paymentMethod" column="p_支払方法"
1094
+ typeHandler="com.example.sms.infrastructure.out.persistence.typehandler.PaymentMethodTypeHandler"/>
1095
+ <result property="paymentAmount" column="p_支払金額"/>
1096
+ <result property="taxAmount" column="p_消費税額"/>
1097
+ <result property="withholdingAmount" column="p_源泉徴収額"/>
1098
+ <result property="netPaymentAmount" column="p_差引支払額"/>
1099
+ <result property="paymentExecutionDate" column="p_支払実行日"/>
1100
+ <result property="status" column="p_ステータス"
1101
+ typeHandler="com.example.sms.infrastructure.out.persistence.typehandler.PaymentStatusTypeHandler"/>
1102
+ <result property="version" column="p_バージョン"/>
1103
+ <result property="createdAt" column="p_作成日時"/>
1104
+ <result property="updatedAt" column="p_更新日時"/>
1105
+ <!-- 支払明細との1:N関連 -->
1106
+ <collection property="details"
1107
+ ofType="com.example.sms.domain.model.payment.PaymentDetail"
1108
+ resultMap="PaymentDetailNestedResultMap"/>
1109
+ </resultMap>
1110
+
1111
+ <!-- 支払明細のネスト ResultMap -->
1112
+ <resultMap id="PaymentDetailNestedResultMap"
1113
+ type="com.example.sms.domain.model.payment.PaymentDetail">
1114
+ <id property="id" column="pd_id"/>
1115
+ <result property="paymentId" column="pd_支払ID"/>
1116
+ <result property="lineNumber" column="pd_支払行番号"/>
1117
+ <result property="purchaseNumber" column="pd_仕入番号"/>
1118
+ <result property="purchaseDate" column="pd_仕入日"/>
1119
+ <result property="purchaseAmount" column="pd_仕入金額"/>
1120
+ <result property="taxAmount" column="pd_消費税額"/>
1121
+ <result property="paymentTargetAmount" column="pd_支払対象金額"/>
1122
+ </resultMap>
1123
+
1124
+ <!-- JOIN による一括取得クエリ -->
1125
+ <select id="findWithDetailsByPaymentNumber"
1126
+ resultMap="PaymentWithDetailsResultMap">
1127
+ SELECT
1128
+ p."ID" AS p_id,
1129
+ p."支払番号" AS p_支払番号,
1130
+ p."仕入先コード" AS p_仕入先コード,
1131
+ p."支払締日" AS p_支払締日,
1132
+ p."支払予定日" AS p_支払予定日,
1133
+ p."支払方法" AS p_支払方法,
1134
+ p."支払金額" AS p_支払金額,
1135
+ p."消費税額" AS p_消費税額,
1136
+ p."源泉徴収額" AS p_源泉徴収額,
1137
+ p."差引支払額" AS p_差引支払額,
1138
+ p."支払実行日" AS p_支払実行日,
1139
+ p."ステータス" AS p_ステータス,
1140
+ p."バージョン" AS p_バージョン,
1141
+ p."作成日時" AS p_作成日時,
1142
+ p."更新日時" AS p_更新日時,
1143
+ pd."ID" AS pd_id,
1144
+ pd."支払ID" AS pd_支払ID,
1145
+ pd."支払行番号" AS pd_支払行番号,
1146
+ pd."仕入番号" AS pd_仕入番号,
1147
+ pd."仕入日" AS pd_仕入日,
1148
+ pd."仕入金額" AS pd_仕入金額,
1149
+ pd."消費税額" AS pd_消費税額,
1150
+ pd."支払対象金額" AS pd_支払対象金額
1151
+ FROM "支払データ" p
1152
+ LEFT JOIN "支払明細データ" pd
1153
+ ON p."ID" = pd."支払ID"
1154
+ WHERE p."支払番号" = #{paymentNumber}
1155
+ ORDER BY pd."支払行番号"
1156
+ </select>
1157
+
1158
+ </mapper>
1159
+ ```
1160
+
1161
+ </details>
1162
+
1163
+ ### リレーション設定のポイント
1164
+
1165
+ | 設定項目 | 説明 |
1166
+ |---------|------|
1167
+ | `<collection>` | 1:N 関連のマッピング |
1168
+ | `<id>` | 主キーの識別(MyBatis が重複排除に使用) |
1169
+ | `resultMap` | ネストした ResultMap の参照 |
1170
+ | エイリアス(AS) | カラム名の重複を避けるためのプレフィックス |
1171
+ | `ORDER BY` | コレクションの順序を保証 |
1172
+
1173
+ ### 楽観ロック(Optimistic Locking)の実装
1174
+
1175
+ 複数ユーザーが同時に支払データを編集する場合、データの整合性を保つために楽観ロックを実装します。
1176
+
1177
+ ```plantuml
1178
+ @startuml
1179
+
1180
+ title 楽観ロックの動作シーケンス
1181
+
1182
+ actor "担当者A" as userA
1183
+ actor "担当者B" as userB
1184
+ database "支払テーブル" as db
1185
+
1186
+ userA -> db : SELECT(version=1)
1187
+ userB -> db : SELECT(version=1)
1188
+ userA -> userA : 支払内容を編集
1189
+ userB -> userB : 支払内容を編集
1190
+ userA -> db : UPDATE WHERE version=1\n→ version=2 に更新
1191
+ db -> userA : 更新成功(1行)
1192
+ userB -> db : UPDATE WHERE version=1
1193
+ db -> userB : 更新失敗(0行)\n楽観ロック例外
1194
+ userB -> userB : 再読み込みを促す
1195
+
1196
+ @enduml
1197
+ ```
1198
+
1199
+ ### バージョンカラムによる同時更新制御
1200
+
1201
+ <details>
1202
+ <summary>楽観ロック対応の MyBatis Mapper</summary>
1203
+
1204
+ ```xml
1205
+ <!-- 楽観ロック対応の支払更新 -->
1206
+ <update id="updateWithOptimisticLock"
1207
+ parameterType="com.example.sms.domain.model.payment.Payment">
1208
+ UPDATE "支払データ"
1209
+ SET
1210
+ "支払締日" = #{paymentClosingDate},
1211
+ "支払予定日" = #{paymentDueDate},
1212
+ "支払方法" = #{paymentMethod,
1213
+ typeHandler=com.example.sms.infrastructure.out.persistence.typehandler.PaymentMethodTypeHandler}::支払方法,
1214
+ "支払金額" = #{paymentAmount},
1215
+ "消費税額" = #{taxAmount},
1216
+ "源泉徴収額" = #{withholdingAmount},
1217
+ "差引支払額" = #{netPaymentAmount},
1218
+ "ステータス" = #{status,
1219
+ typeHandler=com.example.sms.infrastructure.out.persistence.typehandler.PaymentStatusTypeHandler}::支払ステータス,
1220
+ "備考" = #{remarks},
1221
+ "更新日時" = CURRENT_TIMESTAMP,
1222
+ "バージョン" = "バージョン" + 1
1223
+ WHERE "ID" = #{id}
1224
+ AND "バージョン" = #{version}
1225
+ </update>
1226
+ ```
1227
+
1228
+ </details>
1229
+
1230
+ <details>
1231
+ <summary>Repository 実装:楽観ロック対応</summary>
1232
+
1233
+ ```java
1234
+ // src/main/java/com/example/sms/infrastructure/persistence/repository/PaymentRepositoryImpl.java
1235
+ package com.example.sms.infrastructure.out.persistence.repository;
1236
+
1237
+ import com.example.sms.application.port.out.PaymentRepository;
1238
+ import com.example.sms.domain.exception.OptimisticLockException;
1239
+ import com.example.sms.domain.model.payment.*;
1240
+ import com.example.sms.infrastructure.out.persistence.mapper.PaymentMapper;
1241
+ import lombok.RequiredArgsConstructor;
1242
+ import org.springframework.stereotype.Repository;
1243
+ import org.springframework.transaction.annotation.Transactional;
1244
+
1245
+ @Repository
1246
+ @RequiredArgsConstructor
1247
+ public class PaymentRepositoryImpl implements PaymentRepository {
1248
+
1249
+ private final PaymentMapper paymentMapper;
1250
+
1251
+ /**
1252
+ * 楽観ロック対応の更新
1253
+ * @throws OptimisticLockException 他のユーザーによって更新された場合
1254
+ */
1255
+ @Override
1256
+ @Transactional
1257
+ public void update(Payment payment) {
1258
+ int updatedCount = paymentMapper.updateWithOptimisticLock(payment);
1259
+
1260
+ if (updatedCount == 0) {
1261
+ Integer currentVersion = paymentMapper.findVersionById(payment.getId());
1262
+
1263
+ if (currentVersion == null) {
1264
+ throw new OptimisticLockException(
1265
+ "支払", payment.getId().toString());
1266
+ } else {
1267
+ throw new OptimisticLockException(
1268
+ "支払",
1269
+ payment.getId().toString(),
1270
+ payment.getVersion(),
1271
+ currentVersion
1272
+ );
1273
+ }
1274
+ }
1275
+ }
1276
+ }
1277
+ ```
1278
+
1279
+ </details>
1280
+
1281
+ ### 楽観ロックのベストプラクティス
1282
+
1283
+ | ポイント | 説明 |
1284
+ |---------|------|
1285
+ | **バージョンカラム** | INTEGER 型で十分(オーバーフローは現実的に発生しない) |
1286
+ | **WHERE 条件** | 必ず `AND "バージョン" = #{version}` を含める |
1287
+ | **インクリメント** | `"バージョン" = "バージョン" + 1` でアトミックに更新 |
1288
+ | **例外処理** | 更新件数が0の場合は楽観ロック例外をスロー |
1289
+ | **エラーメッセージ** | ユーザーにわかりやすいメッセージで再読み込みを促す |
1290
+
1291
+ ---
1292
+
1293
+ ## 債権管理との対比
1294
+
1295
+ 債権管理(第7章)と債務管理(第10章)は、売り手と買い手の立場が逆転した対称的な関係にあります。
1296
+
1297
+ | 項目 | 債権管理(第7章) | 債務管理(第10章) |
1298
+ |------|------------------|------------------|
1299
+ | 取引先 | 顧客 | 仕入先 |
1300
+ | 締処理 | 請求締処理 | 支払締処理 |
1301
+ | データ | 請求データ | 支払データ |
1302
+ | 残高 | 売掛金残高 | 買掛金残高 |
1303
+ | 消込 | 入金消込 | 支払消込 |
1304
+ | 予定管理 | 回収予定 | 支払予定 |
1305
+
1306
+ ```plantuml
1307
+ @startuml
1308
+
1309
+ title 債権管理と債務管理の関係
1310
+
1311
+ rectangle "自社" as company #LightGreen
1312
+
1313
+ rectangle "顧客" as customer #LightBlue
1314
+ rectangle "仕入先" as supplier #LightCoral
1315
+
1316
+ company -right-> customer : 売上・請求
1317
+ customer -left-> company : 入金
1318
+
1319
+ supplier -right-> company : 仕入・請求
1320
+ company -left-> supplier : 支払
1321
+
1322
+ note top of company
1323
+ 債権管理:
1324
+ - 請求締処理
1325
+ - 売掛金残高
1326
+ - 入金消込
1327
+ end note
1328
+
1329
+ note bottom of company
1330
+ 債務管理:
1331
+ - 支払締処理
1332
+ - 買掛金残高
1333
+ - 支払消込
1334
+ end note
1335
+
1336
+ @enduml
1337
+ ```
1338
+
1339
+ ---
1340
+
1341
+ ## 第10章のまとめ
1342
+
1343
+ 本章では、債務管理(支払締処理・買掛金残高管理)の DB 設計と実装について学びました。
1344
+
1345
+ ### 学んだこと
1346
+
1347
+ 1. **支払締処理の設計**
1348
+ - 仕入データから支払データへの集計
1349
+ - 支払予定日の自動計算(支払サイト)
1350
+ - 承認ワークフロー(作成→承認待ち→承認済→支払済)
1351
+
1352
+ 2. **支払方法の管理**
1353
+ - 振込、手形、現金、相殺、電子記録債権
1354
+ - 支払方法に応じた必要情報の管理
1355
+
1356
+ 3. **買掛金残高管理の設計**
1357
+ - 仕入先別の残高管理
1358
+ - 月次繰越処理
1359
+ - 当月残高の自動計算(トリガー)
1360
+
1361
+ 4. **リレーションと楽観ロック**
1362
+ - N+1 問題の回避(JOIN による一括取得)
1363
+ - 同時更新の競合制御
1364
+
1365
+ ### テーブル一覧
1366
+
1367
+ | テーブル名(日本語) | Java エンティティ | 説明 |
1368
+ |---|---|---|
1369
+ | 支払データ | Payment | 支払ヘッダ情報 |
1370
+ | 支払明細データ | PaymentDetail | 仕入との紐付け |
1371
+ | 買掛金残高データ | PayableBalance | 月次残高管理 |
1372
+ | 支払予定データ | PaymentSchedule | 支払スケジュール |
1373
+
1374
+ ### ENUM 一覧
1375
+
1376
+ | DB ENUM 型(日本語) | Java Enum | 値 |
1377
+ |---|---|---|
1378
+ | 支払ステータス | PaymentStatus | 作成中→DRAFT, 承認待ち→PENDING_APPROVAL, 承認済→APPROVED, 支払済→PAID, 取消→CANCELLED |
1379
+ | 支払方法 | PaymentMethod | 振込→BANK_TRANSFER, 手形→BILL, 現金→CASH, 相殺→OFFSET, 電子記録債権→ELECTRONIC_RECORD |
1380
+
1381
+ ### 次章の予告
1382
+
1383
+ 第11章では、財務会計システムの全体像に進みます。販売管理システムで発生した取引を財務会計の仕訳としてどのように記録するかを学びます。
1384
+
1385
+ ---
1386
+
1387
+ [← 第9章:在庫管理の設計](./chapter09.md) | [第11章:財務会計システムの全体像 →](./chapter11.md)