@k2works/claude-code-booster 3.4.1 → 3.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (712) 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 +239 -239
  5. package/lib/assets/.claude/scripts/generate-inception-deck.mjs +911 -911
  6. package/lib/assets/.claude/settings.json +11 -11
  7. package/lib/assets/.claude/skills/ai-agent-guidelines/SKILL.md +111 -111
  8. package/lib/assets/.claude/skills/analyzing-architecture/SKILL.md +83 -83
  9. package/lib/assets/.claude/skills/analyzing-business/SKILL.md +95 -95
  10. package/lib/assets/.claude/skills/analyzing-data-model/SKILL.md +77 -77
  11. package/lib/assets/.claude/skills/analyzing-domain-model/SKILL.md +117 -88
  12. package/lib/assets/.claude/skills/analyzing-inception-deck/SKILL.md +84 -84
  13. package/lib/assets/.claude/skills/analyzing-non-functional/SKILL.md +95 -95
  14. package/lib/assets/.claude/skills/analyzing-operation/SKILL.md +95 -95
  15. package/lib/assets/.claude/skills/analyzing-requirements/SKILL.md +91 -91
  16. package/lib/assets/.claude/skills/analyzing-tech-stack/SKILL.md +101 -101
  17. package/lib/assets/.claude/skills/analyzing-test-strategy/SKILL.md +89 -89
  18. package/lib/assets/.claude/skills/analyzing-ui-design/SKILL.md +80 -80
  19. package/lib/assets/.claude/skills/analyzing-usecases/SKILL.md +72 -72
  20. package/lib/assets/.claude/skills/creating-adr/SKILL.md +113 -113
  21. package/lib/assets/.claude/skills/developing-backend/SKILL.md +100 -100
  22. package/lib/assets/.claude/skills/developing-frontend/SKILL.md +93 -93
  23. package/lib/assets/.claude/skills/developing-release/SKILL.md +120 -120
  24. package/lib/assets/.claude/skills/generating-slides/SKILL.md +94 -94
  25. package/lib/assets/.claude/skills/git-commit/SKILL.md +81 -81
  26. package/lib/assets/.claude/skills/killing-processes/SKILL.md +44 -44
  27. package/lib/assets/.claude/skills/operating-backup/SKILL.md +59 -59
  28. package/lib/assets/.claude/skills/operating-cicd/SKILL.md +54 -54
  29. package/lib/assets/.claude/skills/operating-deploy/SKILL.md +67 -67
  30. package/lib/assets/.claude/skills/operating-docs/SKILL.md +219 -219
  31. package/lib/assets/.claude/skills/operating-provision/SKILL.md +77 -77
  32. package/lib/assets/.claude/skills/operating-setup/SKILL.md +63 -63
  33. package/lib/assets/.claude/skills/orchestrating-analysis/SKILL.md +104 -104
  34. package/lib/assets/.claude/skills/orchestrating-development/SKILL.md +162 -161
  35. package/lib/assets/.claude/skills/orchestrating-operation/SKILL.md +158 -158
  36. package/lib/assets/.claude/skills/orchestrating-project/SKILL.md +144 -144
  37. package/lib/assets/.claude/skills/planning-releases/SKILL.md +119 -119
  38. package/lib/assets/.claude/skills/syncing-github-project/SKILL.md +151 -151
  39. package/lib/assets/.claude/skills/tracking-progress/SKILL.md +91 -91
  40. package/lib/assets/.claude/skills/validating-iteration-plan/SKILL.md +29 -1
  41. package/lib/assets/.devcontainer/devcontainer.json +34 -34
  42. package/lib/assets/.env.example +17 -17
  43. package/lib/assets/.gitattributes +4 -4
  44. package/lib/assets/.github/workflows/docker-publish.yml +77 -77
  45. package/lib/assets/.github/workflows/mkdocs.yml +39 -39
  46. package/lib/assets/AGENTS.md +94 -94
  47. package/lib/assets/CLAUDE.md +183 -183
  48. package/lib/assets/README.md +254 -254
  49. package/lib/assets/docker-compose.yml +33 -33
  50. package/lib/assets/docs/adr/index.md +10 -10
  51. package/lib/assets/docs/article/functional-desgin-ppp/all/01-immutability-and-data-transformation.md +475 -475
  52. package/lib/assets/docs/article/functional-desgin-ppp/all/02-function-composition.md +519 -519
  53. package/lib/assets/docs/article/functional-desgin-ppp/all/03-polymorphism.md +537 -537
  54. package/lib/assets/docs/article/functional-desgin-ppp/all/04-data-validation.md +300 -300
  55. package/lib/assets/docs/article/functional-desgin-ppp/all/05-property-based-testing.md +320 -320
  56. package/lib/assets/docs/article/functional-desgin-ppp/all/06-tdd-and-functional.md +498 -498
  57. package/lib/assets/docs/article/functional-desgin-ppp/all/07-composite-pattern.md +298 -298
  58. package/lib/assets/docs/article/functional-desgin-ppp/all/08-decorator-pattern.md +291 -291
  59. package/lib/assets/docs/article/functional-desgin-ppp/all/09-adapter-pattern.md +336 -336
  60. package/lib/assets/docs/article/functional-desgin-ppp/all/10-strategy-pattern.md +303 -303
  61. package/lib/assets/docs/article/functional-desgin-ppp/all/11-command-pattern.md +286 -286
  62. package/lib/assets/docs/article/functional-desgin-ppp/all/12-visitor-pattern.md +322 -322
  63. package/lib/assets/docs/article/functional-desgin-ppp/all/13-abstract-factory-pattern.md +319 -319
  64. package/lib/assets/docs/article/functional-desgin-ppp/all/14-abstract-server-pattern.md +365 -365
  65. package/lib/assets/docs/article/functional-desgin-ppp/all/15-gossiping-bus-drivers.md +156 -156
  66. package/lib/assets/docs/article/functional-desgin-ppp/all/16-payroll-system.md +178 -178
  67. package/lib/assets/docs/article/functional-desgin-ppp/all/17-video-rental-system.md +312 -312
  68. package/lib/assets/docs/article/functional-desgin-ppp/all/18-concurrency-system.md +287 -287
  69. package/lib/assets/docs/article/functional-desgin-ppp/all/19-wa-tor-simulation.md +286 -286
  70. package/lib/assets/docs/article/functional-desgin-ppp/all/20-pattern-interactions.md +274 -274
  71. package/lib/assets/docs/article/functional-desgin-ppp/all/21-best-practices.md +294 -294
  72. package/lib/assets/docs/article/functional-desgin-ppp/all/22-oo-to-fp-migration.md +337 -337
  73. package/lib/assets/docs/article/functional-desgin-ppp/all/index.md +388 -388
  74. package/lib/assets/docs/article/functional-desgin-ppp/clojure/01-immutability-and-data-transformation.md +273 -273
  75. package/lib/assets/docs/article/functional-desgin-ppp/clojure/02-function-composition.md +380 -380
  76. package/lib/assets/docs/article/functional-desgin-ppp/clojure/03-polymorphism.md +384 -384
  77. package/lib/assets/docs/article/functional-desgin-ppp/clojure/04-clojure-spec.md +350 -350
  78. package/lib/assets/docs/article/functional-desgin-ppp/clojure/05-property-based-testing.md +352 -352
  79. package/lib/assets/docs/article/functional-desgin-ppp/clojure/06-tdd-in-functional.md +383 -383
  80. package/lib/assets/docs/article/functional-desgin-ppp/clojure/07-composite-pattern.md +529 -529
  81. package/lib/assets/docs/article/functional-desgin-ppp/clojure/08-decorator-pattern.md +395 -395
  82. package/lib/assets/docs/article/functional-desgin-ppp/clojure/09-adapter-pattern.md +399 -399
  83. package/lib/assets/docs/article/functional-desgin-ppp/clojure/10-strategy-pattern.md +485 -485
  84. package/lib/assets/docs/article/functional-desgin-ppp/clojure/11-command-pattern.md +566 -566
  85. package/lib/assets/docs/article/functional-desgin-ppp/clojure/12-visitor-pattern.md +567 -567
  86. package/lib/assets/docs/article/functional-desgin-ppp/clojure/13-abstract-factory-pattern.md +475 -475
  87. package/lib/assets/docs/article/functional-desgin-ppp/clojure/14-abstract-server-pattern.md +462 -462
  88. package/lib/assets/docs/article/functional-desgin-ppp/clojure/15-gossiping-bus-drivers.md +325 -325
  89. package/lib/assets/docs/article/functional-desgin-ppp/clojure/16-payroll-system.md +401 -401
  90. package/lib/assets/docs/article/functional-desgin-ppp/clojure/17-video-rental-system.md +450 -450
  91. package/lib/assets/docs/article/functional-desgin-ppp/clojure/18-concurrency-system.md +475 -475
  92. package/lib/assets/docs/article/functional-desgin-ppp/clojure/19-wator-simulation.md +739 -739
  93. package/lib/assets/docs/article/functional-desgin-ppp/clojure/20-pattern-interactions.md +567 -567
  94. package/lib/assets/docs/article/functional-desgin-ppp/clojure/21-best-practices.md +518 -518
  95. package/lib/assets/docs/article/functional-desgin-ppp/clojure/22-oo-to-fp-migration.md +532 -532
  96. package/lib/assets/docs/article/functional-desgin-ppp/clojure/index.md +241 -241
  97. package/lib/assets/docs/article/functional-desgin-ppp/elixir/01-immutability-and-data-transformation.md +383 -383
  98. package/lib/assets/docs/article/functional-desgin-ppp/elixir/02-function-composition.md +374 -374
  99. package/lib/assets/docs/article/functional-desgin-ppp/elixir/03-polymorphism.md +375 -375
  100. package/lib/assets/docs/article/functional-desgin-ppp/elixir/04-data-validation.md +195 -195
  101. package/lib/assets/docs/article/functional-desgin-ppp/elixir/05-property-based-testing.md +268 -268
  102. package/lib/assets/docs/article/functional-desgin-ppp/elixir/06-tdd-and-fp.md +294 -294
  103. package/lib/assets/docs/article/functional-desgin-ppp/elixir/07-effects-and-pure-functions.md +164 -164
  104. package/lib/assets/docs/article/functional-desgin-ppp/elixir/08-error-handling-strategies.md +168 -168
  105. package/lib/assets/docs/article/functional-desgin-ppp/elixir/09-io-and-external-systems.md +254 -254
  106. package/lib/assets/docs/article/functional-desgin-ppp/elixir/10-concurrency-patterns.md +269 -269
  107. package/lib/assets/docs/article/functional-desgin-ppp/elixir/11-command-pattern.md +148 -148
  108. package/lib/assets/docs/article/functional-desgin-ppp/elixir/12-visitor-pattern.md +176 -176
  109. package/lib/assets/docs/article/functional-desgin-ppp/elixir/13-abstract-factory-pattern.md +604 -604
  110. package/lib/assets/docs/article/functional-desgin-ppp/elixir/14-abstract-server-pattern.md +729 -729
  111. package/lib/assets/docs/article/functional-desgin-ppp/elixir/15-gossiping-bus-drivers.md +291 -291
  112. package/lib/assets/docs/article/functional-desgin-ppp/elixir/16-payroll-system.md +420 -420
  113. package/lib/assets/docs/article/functional-desgin-ppp/elixir/17-video-rental-system.md +319 -319
  114. package/lib/assets/docs/article/functional-desgin-ppp/elixir/18-concurrency-system.md +466 -466
  115. package/lib/assets/docs/article/functional-desgin-ppp/elixir/19-wator-simulation.md +523 -523
  116. package/lib/assets/docs/article/functional-desgin-ppp/elixir/20-pattern-interactions.md +287 -287
  117. package/lib/assets/docs/article/functional-desgin-ppp/elixir/21-best-practices.md +340 -340
  118. package/lib/assets/docs/article/functional-desgin-ppp/elixir/22-oo-to-fp-migration.md +395 -395
  119. package/lib/assets/docs/article/functional-desgin-ppp/elixir/index.md +248 -248
  120. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/01-immutability-and-data-transformation.md +384 -384
  121. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/02-function-composition.md +452 -452
  122. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/03-polymorphism.md +495 -495
  123. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/04-data-validation.md +416 -416
  124. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/05-property-based-testing.md +382 -382
  125. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/06-tdd-functional.md +687 -687
  126. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/07-composite-pattern.md +442 -442
  127. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/08-decorator-pattern.md +479 -479
  128. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/09-adapter-pattern.md +479 -479
  129. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/10-strategy-pattern.md +427 -427
  130. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/11-command-pattern.md +428 -428
  131. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/12-visitor-pattern.md +339 -339
  132. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/13-abstract-factory-pattern.md +309 -309
  133. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/14-abstract-server-pattern.md +596 -596
  134. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/15-gossiping-bus-drivers.md +355 -355
  135. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/16-payroll-system.md +350 -350
  136. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/17-video-rental-system.md +414 -414
  137. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/18-concurrency-system.md +367 -367
  138. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/19-wator-simulation.md +403 -403
  139. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/20-pattern-interactions.md +291 -291
  140. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/21-best-practices.md +324 -324
  141. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/22-oo-to-fp-migration.md +332 -332
  142. package/lib/assets/docs/article/functional-desgin-ppp/fsharp/index.md +274 -274
  143. package/lib/assets/docs/article/functional-desgin-ppp/haskell/01-immutability-and-data-transformation.md +298 -298
  144. package/lib/assets/docs/article/functional-desgin-ppp/haskell/02-function-composition.md +304 -304
  145. package/lib/assets/docs/article/functional-desgin-ppp/haskell/03-polymorphism.md +362 -362
  146. package/lib/assets/docs/article/functional-desgin-ppp/haskell/04-data-validation.md +257 -257
  147. package/lib/assets/docs/article/functional-desgin-ppp/haskell/05-property-based-testing.md +254 -254
  148. package/lib/assets/docs/article/functional-desgin-ppp/haskell/06-tdd-functional.md +283 -283
  149. package/lib/assets/docs/article/functional-desgin-ppp/haskell/07-composite-pattern.md +395 -395
  150. package/lib/assets/docs/article/functional-desgin-ppp/haskell/08-decorator-pattern.md +319 -319
  151. package/lib/assets/docs/article/functional-desgin-ppp/haskell/09-adapter-pattern.md +382 -382
  152. package/lib/assets/docs/article/functional-desgin-ppp/haskell/10-strategy-pattern.md +287 -287
  153. package/lib/assets/docs/article/functional-desgin-ppp/haskell/11-command-pattern.md +303 -303
  154. package/lib/assets/docs/article/functional-desgin-ppp/haskell/12-visitor-pattern.md +326 -326
  155. package/lib/assets/docs/article/functional-desgin-ppp/haskell/13-abstract-factory-pattern.md +332 -332
  156. package/lib/assets/docs/article/functional-desgin-ppp/haskell/14-abstract-server-pattern.md +379 -379
  157. package/lib/assets/docs/article/functional-desgin-ppp/haskell/15-gossiping-bus-drivers.md +177 -177
  158. package/lib/assets/docs/article/functional-desgin-ppp/haskell/16-payroll-system.md +219 -219
  159. package/lib/assets/docs/article/functional-desgin-ppp/haskell/17-video-rental-system.md +244 -244
  160. package/lib/assets/docs/article/functional-desgin-ppp/haskell/18-concurrency-system.md +363 -363
  161. package/lib/assets/docs/article/functional-desgin-ppp/haskell/19-wator-simulation.md +438 -438
  162. package/lib/assets/docs/article/functional-desgin-ppp/haskell/20-pattern-interactions.md +325 -325
  163. package/lib/assets/docs/article/functional-desgin-ppp/haskell/21-best-practices.md +403 -403
  164. package/lib/assets/docs/article/functional-desgin-ppp/haskell/22-oo-to-fp-migration.md +469 -469
  165. package/lib/assets/docs/article/functional-desgin-ppp/haskell/index.md +174 -174
  166. package/lib/assets/docs/article/functional-desgin-ppp/index.md +90 -90
  167. package/lib/assets/docs/article/functional-desgin-ppp/rust/01-immutability-and-data-transformation.md +450 -450
  168. package/lib/assets/docs/article/functional-desgin-ppp/rust/02-function-composition.md +463 -463
  169. package/lib/assets/docs/article/functional-desgin-ppp/rust/03-polymorphism.md +425 -425
  170. package/lib/assets/docs/article/functional-desgin-ppp/rust/04-data-validation.md +273 -273
  171. package/lib/assets/docs/article/functional-desgin-ppp/rust/05-property-based-testing.md +247 -247
  172. package/lib/assets/docs/article/functional-desgin-ppp/rust/06-tdd-and-functional.md +841 -841
  173. package/lib/assets/docs/article/functional-desgin-ppp/rust/07-composite-pattern.md +384 -384
  174. package/lib/assets/docs/article/functional-desgin-ppp/rust/08-decorator-pattern.md +383 -383
  175. package/lib/assets/docs/article/functional-desgin-ppp/rust/09-adapter-pattern.md +339 -339
  176. package/lib/assets/docs/article/functional-desgin-ppp/rust/10-strategy-pattern.md +331 -331
  177. package/lib/assets/docs/article/functional-desgin-ppp/rust/11-command-pattern.md +356 -356
  178. package/lib/assets/docs/article/functional-desgin-ppp/rust/12-visitor-pattern.md +379 -379
  179. package/lib/assets/docs/article/functional-desgin-ppp/rust/13-abstract-factory-pattern.md +361 -361
  180. package/lib/assets/docs/article/functional-desgin-ppp/rust/14-abstract-server-pattern.md +392 -392
  181. package/lib/assets/docs/article/functional-desgin-ppp/rust/15-gossiping-bus-drivers.md +300 -300
  182. package/lib/assets/docs/article/functional-desgin-ppp/rust/16-payroll-system.md +297 -297
  183. package/lib/assets/docs/article/functional-desgin-ppp/rust/17-video-rental-system.md +304 -304
  184. package/lib/assets/docs/article/functional-desgin-ppp/rust/18-concurrency-system.md +315 -315
  185. package/lib/assets/docs/article/functional-desgin-ppp/rust/19-wator-simulation.md +311 -311
  186. package/lib/assets/docs/article/functional-desgin-ppp/rust/20-pattern-interactions.md +304 -304
  187. package/lib/assets/docs/article/functional-desgin-ppp/rust/21-best-practices.md +336 -336
  188. package/lib/assets/docs/article/functional-desgin-ppp/rust/22-oo-to-fp-migration.md +349 -349
  189. package/lib/assets/docs/article/functional-desgin-ppp/rust/index.md +243 -243
  190. package/lib/assets/docs/article/functional-desgin-ppp/scala/01-immutability-and-data-transformation.md +328 -328
  191. package/lib/assets/docs/article/functional-desgin-ppp/scala/02-function-composition.md +348 -348
  192. package/lib/assets/docs/article/functional-desgin-ppp/scala/03-polymorphism.md +357 -357
  193. package/lib/assets/docs/article/functional-desgin-ppp/scala/04-data-validation.md +364 -364
  194. package/lib/assets/docs/article/functional-desgin-ppp/scala/05-property-based-testing.md +515 -515
  195. package/lib/assets/docs/article/functional-desgin-ppp/scala/06-tdd-functional.md +557 -557
  196. package/lib/assets/docs/article/functional-desgin-ppp/scala/07-composite-pattern.md +363 -363
  197. package/lib/assets/docs/article/functional-desgin-ppp/scala/08-decorator-pattern.md +327 -327
  198. package/lib/assets/docs/article/functional-desgin-ppp/scala/09-adapter-pattern.md +517 -517
  199. package/lib/assets/docs/article/functional-desgin-ppp/scala/10-strategy-pattern.md +441 -441
  200. package/lib/assets/docs/article/functional-desgin-ppp/scala/11-command-pattern.md +407 -407
  201. package/lib/assets/docs/article/functional-desgin-ppp/scala/12-visitor-pattern.md +379 -379
  202. package/lib/assets/docs/article/functional-desgin-ppp/scala/13-abstract-factory-pattern.md +398 -398
  203. package/lib/assets/docs/article/functional-desgin-ppp/scala/14-abstract-server-pattern.md +476 -476
  204. package/lib/assets/docs/article/functional-desgin-ppp/scala/15-gossiping-bus-drivers.md +391 -391
  205. package/lib/assets/docs/article/functional-desgin-ppp/scala/16-payroll-system.md +342 -342
  206. package/lib/assets/docs/article/functional-desgin-ppp/scala/17-video-rental-system.md +324 -324
  207. package/lib/assets/docs/article/functional-desgin-ppp/scala/18-concurrency-system.md +730 -730
  208. package/lib/assets/docs/article/functional-desgin-ppp/scala/19-wator-simulation.md +624 -624
  209. package/lib/assets/docs/article/functional-desgin-ppp/scala/20-pattern-interactions.md +512 -512
  210. package/lib/assets/docs/article/functional-desgin-ppp/scala/21-best-practices.md +433 -433
  211. package/lib/assets/docs/article/functional-desgin-ppp/scala/22-oo-to-fp-migration.md +688 -688
  212. package/lib/assets/docs/article/functional-desgin-ppp/scala/index.md +243 -243
  213. package/lib/assets/docs/article/getting-start-tdd/clojure/01-todo-list-and-first-test.md +166 -166
  214. package/lib/assets/docs/article/getting-start-tdd/clojure/02-fake-it-and-triangulation.md +162 -162
  215. package/lib/assets/docs/article/getting-start-tdd/clojure/03-obvious-implementation-and-refactoring.md +135 -135
  216. package/lib/assets/docs/article/getting-start-tdd/clojure/04-version-control-and-conventional-commits.md +88 -88
  217. package/lib/assets/docs/article/getting-start-tdd/clojure/05-package-management-and-static-analysis.md +299 -299
  218. package/lib/assets/docs/article/getting-start-tdd/clojure/06-task-runner-and-ci-cd.md +241 -241
  219. package/lib/assets/docs/article/getting-start-tdd/clojure/07-protocols-and-records.md +131 -131
  220. package/lib/assets/docs/article/getting-start-tdd/clojure/08-multimethods-and-design-patterns.md +130 -130
  221. package/lib/assets/docs/article/getting-start-tdd/clojure/09-namespaces-and-module-design.md +127 -127
  222. package/lib/assets/docs/article/getting-start-tdd/clojure/10-higher-order-functions-and-composition.md +114 -114
  223. package/lib/assets/docs/article/getting-start-tdd/clojure/11-persistent-data-and-pipeline.md +138 -138
  224. package/lib/assets/docs/article/getting-start-tdd/clojure/12-error-handling-and-spec.md +161 -161
  225. package/lib/assets/docs/article/getting-start-tdd/clojure/index.md +65 -65
  226. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter01.md +232 -232
  227. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter02.md +244 -244
  228. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter03.md +202 -202
  229. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter04.md +92 -92
  230. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter05.md +256 -256
  231. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter06.md +195 -195
  232. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter07.md +214 -214
  233. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter08.md +249 -249
  234. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter09.md +174 -174
  235. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter10.md +166 -166
  236. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter11.md +192 -192
  237. package/lib/assets/docs/article/getting-start-tdd/csharp/chapter12.md +211 -211
  238. package/lib/assets/docs/article/getting-start-tdd/csharp/index.md +83 -83
  239. package/lib/assets/docs/article/getting-start-tdd/elixir/01-todo-list-and-first-test.md +87 -87
  240. package/lib/assets/docs/article/getting-start-tdd/elixir/02-fake-it-and-triangulation.md +95 -95
  241. package/lib/assets/docs/article/getting-start-tdd/elixir/03-obvious-implementation-and-refactoring.md +109 -109
  242. package/lib/assets/docs/article/getting-start-tdd/elixir/04-version-control-and-conventional-commits.md +96 -96
  243. package/lib/assets/docs/article/getting-start-tdd/elixir/05-package-management-and-static-analysis.md +88 -88
  244. package/lib/assets/docs/article/getting-start-tdd/elixir/06-task-runner-and-ci-cd.md +71 -71
  245. package/lib/assets/docs/article/getting-start-tdd/elixir/07-structs-and-protocols.md +110 -110
  246. package/lib/assets/docs/article/getting-start-tdd/elixir/08-pattern-matching-and-guards.md +108 -108
  247. package/lib/assets/docs/article/getting-start-tdd/elixir/09-module-design-and-behaviours.md +104 -104
  248. package/lib/assets/docs/article/getting-start-tdd/elixir/10-higher-order-functions-and-pipeline.md +178 -178
  249. package/lib/assets/docs/article/getting-start-tdd/elixir/11-stream-and-lazy-evaluation.md +142 -142
  250. package/lib/assets/docs/article/getting-start-tdd/elixir/12-error-handling-and-with.md +145 -145
  251. package/lib/assets/docs/article/getting-start-tdd/elixir/index.md +35 -35
  252. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter01.md +202 -202
  253. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter02.md +246 -246
  254. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter03.md +218 -218
  255. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter04.md +179 -179
  256. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter05.md +267 -267
  257. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter06.md +190 -190
  258. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter07.md +161 -161
  259. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter08.md +175 -175
  260. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter09.md +222 -222
  261. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter10.md +189 -189
  262. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter11.md +212 -212
  263. package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter12.md +215 -215
  264. package/lib/assets/docs/article/getting-start-tdd/fsharp/index.md +71 -71
  265. package/lib/assets/docs/article/getting-start-tdd/go/01-todo-list-and-first-test.md +213 -213
  266. package/lib/assets/docs/article/getting-start-tdd/go/02-fake-it-and-triangulation.md +302 -302
  267. package/lib/assets/docs/article/getting-start-tdd/go/03-obvious-implementation-and-refactoring.md +339 -339
  268. package/lib/assets/docs/article/getting-start-tdd/go/04-version-control-and-conventional-commits.md +112 -112
  269. package/lib/assets/docs/article/getting-start-tdd/go/05-package-management-and-static-analysis.md +272 -272
  270. package/lib/assets/docs/article/getting-start-tdd/go/06-task-runner-and-ci-cd.md +233 -233
  271. package/lib/assets/docs/article/getting-start-tdd/go/07-encapsulation-and-polymorphism.md +394 -394
  272. package/lib/assets/docs/article/getting-start-tdd/go/08-design-patterns.md +422 -422
  273. package/lib/assets/docs/article/getting-start-tdd/go/09-solid-principles-and-module-design.md +400 -400
  274. package/lib/assets/docs/article/getting-start-tdd/go/10-higher-order-functions-and-composition.md +226 -226
  275. package/lib/assets/docs/article/getting-start-tdd/go/11-immutable-data-and-pipeline.md +296 -296
  276. package/lib/assets/docs/article/getting-start-tdd/go/12-error-handling-and-type-safety.md +411 -411
  277. package/lib/assets/docs/article/getting-start-tdd/go/index.md +83 -83
  278. package/lib/assets/docs/article/getting-start-tdd/haskell/01-todo-list-and-first-test.md +279 -279
  279. package/lib/assets/docs/article/getting-start-tdd/haskell/02-fake-it-and-triangulation.md +337 -337
  280. package/lib/assets/docs/article/getting-start-tdd/haskell/03-obvious-implementation-and-refactoring.md +257 -257
  281. package/lib/assets/docs/article/getting-start-tdd/haskell/04-version-control-and-conventional-commits.md +182 -182
  282. package/lib/assets/docs/article/getting-start-tdd/haskell/05-package-management-and-static-analysis.md +313 -313
  283. package/lib/assets/docs/article/getting-start-tdd/haskell/06-task-runner-and-ci-cd.md +309 -309
  284. package/lib/assets/docs/article/getting-start-tdd/haskell/07-algebraic-data-types-and-type-classes.md +412 -412
  285. package/lib/assets/docs/article/getting-start-tdd/haskell/08-pattern-matching-and-guards.md +390 -390
  286. package/lib/assets/docs/article/getting-start-tdd/haskell/09-module-design-and-smart-constructors.md +461 -461
  287. package/lib/assets/docs/article/getting-start-tdd/haskell/10-higher-order-functions-and-currying.md +434 -434
  288. package/lib/assets/docs/article/getting-start-tdd/haskell/11-function-composition-and-point-free.md +392 -392
  289. package/lib/assets/docs/article/getting-start-tdd/haskell/12-monad-and-error-handling.md +631 -631
  290. package/lib/assets/docs/article/getting-start-tdd/haskell/index.md +49 -49
  291. package/lib/assets/docs/article/getting-start-tdd/index.md +93 -93
  292. package/lib/assets/docs/article/getting-start-tdd/integration/01-language-overview.md +375 -375
  293. package/lib/assets/docs/article/getting-start-tdd/integration/02-test-framework-comparison.md +349 -349
  294. package/lib/assets/docs/article/getting-start-tdd/integration/03-tdd-pattern-comparison.md +445 -445
  295. package/lib/assets/docs/article/getting-start-tdd/integration/04-type-system-comparison.md +409 -409
  296. package/lib/assets/docs/article/getting-start-tdd/integration/05-dev-environment-comparison.md +330 -330
  297. package/lib/assets/docs/article/getting-start-tdd/integration/06-learning-roadmap.md +290 -290
  298. package/lib/assets/docs/article/getting-start-tdd/integration/index.md +69 -69
  299. package/lib/assets/docs/article/getting-start-tdd/java/01-todo-list-and-first-test.md +234 -234
  300. package/lib/assets/docs/article/getting-start-tdd/java/02-fake-it-and-triangulation.md +261 -261
  301. package/lib/assets/docs/article/getting-start-tdd/java/03-obvious-implementation-and-refactoring.md +185 -185
  302. package/lib/assets/docs/article/getting-start-tdd/java/04-version-control-and-conventional-commits.md +115 -115
  303. package/lib/assets/docs/article/getting-start-tdd/java/05-package-management-and-static-analysis.md +382 -382
  304. package/lib/assets/docs/article/getting-start-tdd/java/06-task-runner-and-ci-cd.md +272 -272
  305. package/lib/assets/docs/article/getting-start-tdd/java/07-encapsulation-and-polymorphism.md +626 -626
  306. package/lib/assets/docs/article/getting-start-tdd/java/08-design-patterns.md +393 -393
  307. package/lib/assets/docs/article/getting-start-tdd/java/09-solid-principles-and-module-design.md +310 -310
  308. package/lib/assets/docs/article/getting-start-tdd/java/10-higher-order-functions-and-composition.md +188 -188
  309. package/lib/assets/docs/article/getting-start-tdd/java/11-immutable-data-and-pipeline.md +167 -167
  310. package/lib/assets/docs/article/getting-start-tdd/java/12-error-handling-and-type-safety.md +205 -205
  311. package/lib/assets/docs/article/getting-start-tdd/java/index.md +61 -61
  312. package/lib/assets/docs/article/getting-start-tdd/node/01-todo-list-and-first-test.md +244 -244
  313. package/lib/assets/docs/article/getting-start-tdd/node/02-fake-it-and-triangulation.md +262 -262
  314. package/lib/assets/docs/article/getting-start-tdd/node/03-obvious-implementation-and-refactoring.md +169 -169
  315. package/lib/assets/docs/article/getting-start-tdd/node/04-version-control-and-conventional-commits.md +112 -112
  316. package/lib/assets/docs/article/getting-start-tdd/node/05-package-management-and-static-analysis.md +314 -314
  317. package/lib/assets/docs/article/getting-start-tdd/node/06-task-runner-and-ci-cd.md +235 -235
  318. package/lib/assets/docs/article/getting-start-tdd/node/07-encapsulation-and-polymorphism.md +327 -327
  319. package/lib/assets/docs/article/getting-start-tdd/node/08-design-patterns.md +322 -322
  320. package/lib/assets/docs/article/getting-start-tdd/node/09-solid-principles-and-module-design.md +285 -285
  321. package/lib/assets/docs/article/getting-start-tdd/node/10-higher-order-functions-and-composition.md +199 -199
  322. package/lib/assets/docs/article/getting-start-tdd/node/11-immutable-data-and-pipeline.md +207 -207
  323. package/lib/assets/docs/article/getting-start-tdd/node/12-error-handling-and-type-safety.md +295 -295
  324. package/lib/assets/docs/article/getting-start-tdd/node/index.md +56 -56
  325. package/lib/assets/docs/article/getting-start-tdd/php/01-todo-list-and-first-test.md +259 -259
  326. package/lib/assets/docs/article/getting-start-tdd/php/02-fake-it-and-triangulation.md +200 -200
  327. package/lib/assets/docs/article/getting-start-tdd/php/03-obvious-implementation-and-refactoring.md +248 -248
  328. package/lib/assets/docs/article/getting-start-tdd/php/04-version-control-and-conventional-commits.md +141 -141
  329. package/lib/assets/docs/article/getting-start-tdd/php/05-package-management-and-static-analysis.md +410 -410
  330. package/lib/assets/docs/article/getting-start-tdd/php/06-task-runner-and-ci-cd.md +321 -321
  331. package/lib/assets/docs/article/getting-start-tdd/php/07-encapsulation-and-polymorphism.md +372 -372
  332. package/lib/assets/docs/article/getting-start-tdd/php/08-design-patterns.md +453 -453
  333. package/lib/assets/docs/article/getting-start-tdd/php/09-solid-principles-and-module-design.md +460 -460
  334. package/lib/assets/docs/article/getting-start-tdd/php/10-higher-order-functions-and-composition.md +182 -182
  335. package/lib/assets/docs/article/getting-start-tdd/php/11-immutable-data-and-pipeline.md +266 -266
  336. package/lib/assets/docs/article/getting-start-tdd/php/12-error-handling-and-type-safety.md +308 -308
  337. package/lib/assets/docs/article/getting-start-tdd/php/index.md +84 -84
  338. package/lib/assets/docs/article/getting-start-tdd/python/01-todo-list-and-first-test.md +201 -201
  339. package/lib/assets/docs/article/getting-start-tdd/python/02-fake-it-and-triangulation.md +247 -247
  340. package/lib/assets/docs/article/getting-start-tdd/python/03-obvious-implementation-and-refactoring.md +199 -199
  341. package/lib/assets/docs/article/getting-start-tdd/python/04-version-control-and-conventional-commits.md +87 -87
  342. package/lib/assets/docs/article/getting-start-tdd/python/05-package-management-and-static-analysis.md +274 -274
  343. package/lib/assets/docs/article/getting-start-tdd/python/06-task-runner-and-ci-cd.md +190 -190
  344. package/lib/assets/docs/article/getting-start-tdd/python/07-encapsulation-and-polymorphism.md +208 -208
  345. package/lib/assets/docs/article/getting-start-tdd/python/08-design-patterns.md +172 -172
  346. package/lib/assets/docs/article/getting-start-tdd/python/09-solid-principles-and-module-design.md +130 -130
  347. package/lib/assets/docs/article/getting-start-tdd/python/10-higher-order-functions-and-composition.md +122 -122
  348. package/lib/assets/docs/article/getting-start-tdd/python/11-immutable-data-and-pipeline.md +116 -116
  349. package/lib/assets/docs/article/getting-start-tdd/python/12-error-handling-and-type-safety.md +126 -126
  350. package/lib/assets/docs/article/getting-start-tdd/python/index.md +55 -55
  351. package/lib/assets/docs/article/getting-start-tdd/ruby/01-todo-list-and-first-test.md +231 -231
  352. package/lib/assets/docs/article/getting-start-tdd/ruby/02-fake-it-and-triangulation.md +238 -238
  353. package/lib/assets/docs/article/getting-start-tdd/ruby/03-obvious-implementation-and-refactoring.md +228 -228
  354. package/lib/assets/docs/article/getting-start-tdd/ruby/04-version-control-and-conventional-commits.md +112 -112
  355. package/lib/assets/docs/article/getting-start-tdd/ruby/05-package-management-and-static-analysis.md +287 -287
  356. package/lib/assets/docs/article/getting-start-tdd/ruby/06-task-runner-and-ci-cd.md +248 -248
  357. package/lib/assets/docs/article/getting-start-tdd/ruby/07-encapsulation-and-polymorphism.md +279 -279
  358. package/lib/assets/docs/article/getting-start-tdd/ruby/08-design-patterns.md +329 -329
  359. package/lib/assets/docs/article/getting-start-tdd/ruby/09-solid-principles-and-module-design.md +196 -196
  360. package/lib/assets/docs/article/getting-start-tdd/ruby/10-higher-order-functions-and-composition.md +175 -175
  361. package/lib/assets/docs/article/getting-start-tdd/ruby/11-immutable-data-and-pipeline.md +237 -237
  362. package/lib/assets/docs/article/getting-start-tdd/ruby/12-error-handling-and-type-safety.md +398 -398
  363. package/lib/assets/docs/article/getting-start-tdd/ruby/index.md +83 -83
  364. package/lib/assets/docs/article/getting-start-tdd/rust/01-todo-list-and-first-test.md +211 -211
  365. package/lib/assets/docs/article/getting-start-tdd/rust/02-fake-it-and-triangulation.md +264 -264
  366. package/lib/assets/docs/article/getting-start-tdd/rust/03-obvious-implementation-and-refactoring.md +233 -233
  367. package/lib/assets/docs/article/getting-start-tdd/rust/04-version-control-and-conventional-commits.md +92 -92
  368. package/lib/assets/docs/article/getting-start-tdd/rust/05-package-management-and-static-analysis.md +212 -212
  369. package/lib/assets/docs/article/getting-start-tdd/rust/06-task-runner-and-ci-cd.md +164 -164
  370. package/lib/assets/docs/article/getting-start-tdd/rust/07-encapsulation-and-polymorphism.md +142 -142
  371. package/lib/assets/docs/article/getting-start-tdd/rust/08-design-patterns.md +145 -145
  372. package/lib/assets/docs/article/getting-start-tdd/rust/09-solid-principles-and-module-design.md +110 -110
  373. package/lib/assets/docs/article/getting-start-tdd/rust/10-higher-order-functions-and-composition.md +94 -94
  374. package/lib/assets/docs/article/getting-start-tdd/rust/11-immutable-data-and-pipeline.md +105 -105
  375. package/lib/assets/docs/article/getting-start-tdd/rust/12-error-handling-and-type-safety.md +112 -112
  376. package/lib/assets/docs/article/getting-start-tdd/rust/index.md +83 -83
  377. package/lib/assets/docs/article/getting-start-tdd/scala/01-todo-list-and-first-test.md +111 -111
  378. package/lib/assets/docs/article/getting-start-tdd/scala/02-fake-it-and-triangulation.md +107 -107
  379. package/lib/assets/docs/article/getting-start-tdd/scala/03-obvious-implementation-and-refactoring.md +99 -99
  380. package/lib/assets/docs/article/getting-start-tdd/scala/04-version-control-and-conventional-commits.md +123 -123
  381. package/lib/assets/docs/article/getting-start-tdd/scala/05-package-management-and-static-analysis.md +196 -196
  382. package/lib/assets/docs/article/getting-start-tdd/scala/06-task-runner-and-ci-cd.md +186 -186
  383. package/lib/assets/docs/article/getting-start-tdd/scala/07-case-classes-and-traits.md +139 -139
  384. package/lib/assets/docs/article/getting-start-tdd/scala/08-pattern-matching-and-sealed-traits.md +106 -106
  385. package/lib/assets/docs/article/getting-start-tdd/scala/09-packages-and-module-design.md +75 -75
  386. package/lib/assets/docs/article/getting-start-tdd/scala/10-higher-order-functions-and-composition.md +104 -104
  387. package/lib/assets/docs/article/getting-start-tdd/scala/11-collections-and-lazy-evaluation.md +94 -94
  388. package/lib/assets/docs/article/getting-start-tdd/scala/12-error-handling-and-type-safety.md +92 -92
  389. package/lib/assets/docs/article/getting-start-tdd/scala/index.md +65 -65
  390. package/lib/assets/docs/article/grokking-concurrency/all/index.md +404 -404
  391. package/lib/assets/docs/article/grokking-concurrency/all/part-1-ch02-sequential.md +554 -554
  392. package/lib/assets/docs/article/grokking-concurrency/all/part-2-ch04-05-threads.md +469 -469
  393. package/lib/assets/docs/article/grokking-concurrency/all/part-3-ch06-multitasking.md +520 -520
  394. package/lib/assets/docs/article/grokking-concurrency/all/part-4-ch07-parallel-patterns.md +420 -420
  395. package/lib/assets/docs/article/grokking-concurrency/all/part-5-ch08-09-synchronization.md +510 -510
  396. package/lib/assets/docs/article/grokking-concurrency/all/part-6-ch10-11-nonblocking-io.md +435 -435
  397. package/lib/assets/docs/article/grokking-concurrency/all/part-7-ch12-async.md +465 -465
  398. package/lib/assets/docs/article/grokking-concurrency/all/part-8-ch13-mapreduce.md +377 -377
  399. package/lib/assets/docs/article/grokking-concurrency/clojure/index.md +116 -116
  400. package/lib/assets/docs/article/grokking-concurrency/clojure/part-1.md +108 -108
  401. package/lib/assets/docs/article/grokking-concurrency/clojure/part-2.md +101 -101
  402. package/lib/assets/docs/article/grokking-concurrency/clojure/part-3.md +122 -122
  403. package/lib/assets/docs/article/grokking-concurrency/clojure/part-4.md +123 -123
  404. package/lib/assets/docs/article/grokking-concurrency/clojure/part-5.md +118 -118
  405. package/lib/assets/docs/article/grokking-concurrency/clojure/part-6.md +89 -89
  406. package/lib/assets/docs/article/grokking-concurrency/clojure/part-7.md +100 -100
  407. package/lib/assets/docs/article/grokking-concurrency/clojure/part-8.md +120 -120
  408. package/lib/assets/docs/article/grokking-concurrency/csharp/index.md +101 -101
  409. package/lib/assets/docs/article/grokking-concurrency/csharp/part-1.md +97 -97
  410. package/lib/assets/docs/article/grokking-concurrency/csharp/part-2.md +123 -123
  411. package/lib/assets/docs/article/grokking-concurrency/csharp/part-3.md +101 -101
  412. package/lib/assets/docs/article/grokking-concurrency/csharp/part-4.md +112 -112
  413. package/lib/assets/docs/article/grokking-concurrency/csharp/part-5.md +99 -99
  414. package/lib/assets/docs/article/grokking-concurrency/csharp/part-6.md +61 -61
  415. package/lib/assets/docs/article/grokking-concurrency/csharp/part-7.md +84 -84
  416. package/lib/assets/docs/article/grokking-concurrency/csharp/part-8.md +92 -92
  417. package/lib/assets/docs/article/grokking-concurrency/fsharp/index.md +65 -65
  418. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-1.md +80 -80
  419. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-2.md +103 -103
  420. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-3.md +94 -94
  421. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-4.md +110 -110
  422. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-5.md +104 -104
  423. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-6.md +93 -93
  424. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-7.md +121 -121
  425. package/lib/assets/docs/article/grokking-concurrency/fsharp/part-8.md +107 -107
  426. package/lib/assets/docs/article/grokking-concurrency/haskell/index.md +248 -248
  427. package/lib/assets/docs/article/grokking-concurrency/haskell/part-1.md +96 -96
  428. package/lib/assets/docs/article/grokking-concurrency/haskell/part-2.md +96 -96
  429. package/lib/assets/docs/article/grokking-concurrency/haskell/part-3.md +91 -91
  430. package/lib/assets/docs/article/grokking-concurrency/haskell/part-4.md +106 -106
  431. package/lib/assets/docs/article/grokking-concurrency/haskell/part-5.md +99 -99
  432. package/lib/assets/docs/article/grokking-concurrency/haskell/part-6.md +95 -95
  433. package/lib/assets/docs/article/grokking-concurrency/haskell/part-7.md +111 -111
  434. package/lib/assets/docs/article/grokking-concurrency/haskell/part-8.md +118 -118
  435. package/lib/assets/docs/article/grokking-concurrency/index.md +66 -66
  436. package/lib/assets/docs/article/grokking-concurrency/java/index.md +102 -102
  437. package/lib/assets/docs/article/grokking-concurrency/java/part-1.md +308 -308
  438. package/lib/assets/docs/article/grokking-concurrency/java/part-2.md +334 -334
  439. package/lib/assets/docs/article/grokking-concurrency/java/part-3.md +221 -221
  440. package/lib/assets/docs/article/grokking-concurrency/java/part-4.md +213 -213
  441. package/lib/assets/docs/article/grokking-concurrency/java/part-5.md +112 -112
  442. package/lib/assets/docs/article/grokking-concurrency/java/part-6.md +69 -69
  443. package/lib/assets/docs/article/grokking-concurrency/java/part-7.md +101 -101
  444. package/lib/assets/docs/article/grokking-concurrency/java/part-8.md +101 -101
  445. package/lib/assets/docs/article/grokking-concurrency/python/index.md +313 -313
  446. package/lib/assets/docs/article/grokking-concurrency/python/part-1.md +239 -239
  447. package/lib/assets/docs/article/grokking-concurrency/python/part-2.md +418 -418
  448. package/lib/assets/docs/article/grokking-concurrency/python/part-3.md +227 -227
  449. package/lib/assets/docs/article/grokking-concurrency/python/part-4.md +299 -299
  450. package/lib/assets/docs/article/grokking-concurrency/python/part-5.md +315 -315
  451. package/lib/assets/docs/article/grokking-concurrency/python/part-6.md +297 -297
  452. package/lib/assets/docs/article/grokking-concurrency/python/part-7.md +314 -314
  453. package/lib/assets/docs/article/grokking-concurrency/python/part-8.md +360 -360
  454. package/lib/assets/docs/article/grokking-concurrency/rust/index.md +270 -270
  455. package/lib/assets/docs/article/grokking-concurrency/rust/part-1.md +108 -108
  456. package/lib/assets/docs/article/grokking-concurrency/rust/part-2.md +120 -120
  457. package/lib/assets/docs/article/grokking-concurrency/rust/part-3.md +126 -126
  458. package/lib/assets/docs/article/grokking-concurrency/rust/part-4.md +175 -175
  459. package/lib/assets/docs/article/grokking-concurrency/rust/part-5.md +158 -158
  460. package/lib/assets/docs/article/grokking-concurrency/rust/part-6.md +94 -94
  461. package/lib/assets/docs/article/grokking-concurrency/rust/part-7.md +133 -133
  462. package/lib/assets/docs/article/grokking-concurrency/rust/part-8.md +155 -155
  463. package/lib/assets/docs/article/grokking-concurrency/scala/index.md +69 -69
  464. package/lib/assets/docs/article/grokking-concurrency/scala/part-1.md +78 -78
  465. package/lib/assets/docs/article/grokking-concurrency/scala/part-2.md +112 -112
  466. package/lib/assets/docs/article/grokking-concurrency/scala/part-3.md +93 -93
  467. package/lib/assets/docs/article/grokking-concurrency/scala/part-4.md +110 -110
  468. package/lib/assets/docs/article/grokking-concurrency/scala/part-5.md +119 -119
  469. package/lib/assets/docs/article/grokking-concurrency/scala/part-6.md +83 -83
  470. package/lib/assets/docs/article/grokking-concurrency/scala/part-7.md +131 -131
  471. package/lib/assets/docs/article/grokking-concurrency/scala/part-8.md +129 -129
  472. package/lib/assets/docs/article/grokkingfp/all/index.md +368 -368
  473. package/lib/assets/docs/article/grokkingfp/all/part-1-ch01-fp-introduction.md +530 -530
  474. package/lib/assets/docs/article/grokkingfp/all/part-1-ch02-pure-functions.md +923 -923
  475. package/lib/assets/docs/article/grokkingfp/all/part-2-ch03-immutable-data.md +1128 -1128
  476. package/lib/assets/docs/article/grokkingfp/all/part-2-ch04-higher-order-functions.md +1104 -1104
  477. package/lib/assets/docs/article/grokkingfp/all/part-2-ch05-flatmap.md +1026 -1026
  478. package/lib/assets/docs/article/grokkingfp/all/part-3-ch06-option.md +785 -785
  479. package/lib/assets/docs/article/grokkingfp/all/part-3-ch07-either-adt.md +871 -871
  480. package/lib/assets/docs/article/grokkingfp/all/part-4-ch08-io-monad.md +972 -972
  481. package/lib/assets/docs/article/grokkingfp/all/part-4-ch09-streams.md +926 -926
  482. package/lib/assets/docs/article/grokkingfp/all/part-5-ch10-concurrency.md +870 -870
  483. package/lib/assets/docs/article/grokkingfp/all/part-6-ch11-application.md +715 -715
  484. package/lib/assets/docs/article/grokkingfp/all/part-6-ch12-testing.md +626 -626
  485. package/lib/assets/docs/article/grokkingfp/all/writing-plan.md +712 -712
  486. package/lib/assets/docs/article/grokkingfp/clojure/index.md +276 -276
  487. package/lib/assets/docs/article/grokkingfp/clojure/part-1.md +667 -667
  488. package/lib/assets/docs/article/grokkingfp/clojure/part-2.md +643 -643
  489. package/lib/assets/docs/article/grokkingfp/clojure/part-3.md +620 -620
  490. package/lib/assets/docs/article/grokkingfp/clojure/part-4.md +697 -697
  491. package/lib/assets/docs/article/grokkingfp/clojure/part-5.md +751 -751
  492. package/lib/assets/docs/article/grokkingfp/clojure/part-6.md +721 -721
  493. package/lib/assets/docs/article/grokkingfp/csharp/index.md +246 -246
  494. package/lib/assets/docs/article/grokkingfp/csharp/part-1.md +811 -811
  495. package/lib/assets/docs/article/grokkingfp/csharp/part-2.md +971 -971
  496. package/lib/assets/docs/article/grokkingfp/csharp/part-3.md +981 -981
  497. package/lib/assets/docs/article/grokkingfp/csharp/part-4.md +949 -949
  498. package/lib/assets/docs/article/grokkingfp/csharp/part-5.md +947 -947
  499. package/lib/assets/docs/article/grokkingfp/csharp/part-6.md +739 -739
  500. package/lib/assets/docs/article/grokkingfp/elixir/index.md +203 -203
  501. package/lib/assets/docs/article/grokkingfp/elixir/part-1.md +712 -712
  502. package/lib/assets/docs/article/grokkingfp/elixir/part-2.md +838 -838
  503. package/lib/assets/docs/article/grokkingfp/elixir/part-3.md +985 -985
  504. package/lib/assets/docs/article/grokkingfp/elixir/part-4.md +974 -974
  505. package/lib/assets/docs/article/grokkingfp/elixir/part-5.md +1286 -1286
  506. package/lib/assets/docs/article/grokkingfp/elixir/part-6.md +1049 -1049
  507. package/lib/assets/docs/article/grokkingfp/fsharp/index.md +210 -210
  508. package/lib/assets/docs/article/grokkingfp/fsharp/part-1.md +714 -714
  509. package/lib/assets/docs/article/grokkingfp/fsharp/part-2.md +961 -961
  510. package/lib/assets/docs/article/grokkingfp/fsharp/part-3.md +972 -972
  511. package/lib/assets/docs/article/grokkingfp/fsharp/part-4.md +832 -832
  512. package/lib/assets/docs/article/grokkingfp/fsharp/part-5.md +911 -911
  513. package/lib/assets/docs/article/grokkingfp/fsharp/part-6.md +922 -922
  514. package/lib/assets/docs/article/grokkingfp/haskell/index.md +234 -234
  515. package/lib/assets/docs/article/grokkingfp/haskell/part-1.md +591 -591
  516. package/lib/assets/docs/article/grokkingfp/haskell/part-2.md +866 -866
  517. package/lib/assets/docs/article/grokkingfp/haskell/part-3.md +915 -915
  518. package/lib/assets/docs/article/grokkingfp/haskell/part-4.md +878 -878
  519. package/lib/assets/docs/article/grokkingfp/haskell/part-5.md +845 -845
  520. package/lib/assets/docs/article/grokkingfp/haskell/part-6.md +844 -844
  521. package/lib/assets/docs/article/grokkingfp/index.md +143 -143
  522. package/lib/assets/docs/article/grokkingfp/java/index.md +211 -211
  523. package/lib/assets/docs/article/grokkingfp/java/part-1.md +648 -648
  524. package/lib/assets/docs/article/grokkingfp/java/part-2.md +675 -675
  525. package/lib/assets/docs/article/grokkingfp/java/part-3.md +672 -672
  526. package/lib/assets/docs/article/grokkingfp/java/part-4.md +771 -771
  527. package/lib/assets/docs/article/grokkingfp/java/part-5.md +959 -959
  528. package/lib/assets/docs/article/grokkingfp/java/part-6.md +1328 -1328
  529. package/lib/assets/docs/article/grokkingfp/python/index.md +258 -258
  530. package/lib/assets/docs/article/grokkingfp/python/part-1.md +443 -443
  531. package/lib/assets/docs/article/grokkingfp/python/part-2.md +958 -958
  532. package/lib/assets/docs/article/grokkingfp/python/part-3.md +1004 -1004
  533. package/lib/assets/docs/article/grokkingfp/python/part-4.md +765 -765
  534. package/lib/assets/docs/article/grokkingfp/python/part-5.md +747 -747
  535. package/lib/assets/docs/article/grokkingfp/python/part-6.md +861 -861
  536. package/lib/assets/docs/article/grokkingfp/ruby/index.md +330 -330
  537. package/lib/assets/docs/article/grokkingfp/ruby/part-1.md +755 -755
  538. package/lib/assets/docs/article/grokkingfp/ruby/part-2.md +938 -938
  539. package/lib/assets/docs/article/grokkingfp/ruby/part-3.md +946 -946
  540. package/lib/assets/docs/article/grokkingfp/ruby/part-4.md +921 -921
  541. package/lib/assets/docs/article/grokkingfp/ruby/part-5.md +908 -908
  542. package/lib/assets/docs/article/grokkingfp/ruby/part-6.md +1412 -1412
  543. package/lib/assets/docs/article/grokkingfp/rust/index.md +242 -242
  544. package/lib/assets/docs/article/grokkingfp/rust/part-1.md +634 -634
  545. package/lib/assets/docs/article/grokkingfp/rust/part-2.md +1060 -1060
  546. package/lib/assets/docs/article/grokkingfp/rust/part-3.md +994 -994
  547. package/lib/assets/docs/article/grokkingfp/rust/part-4.md +573 -573
  548. package/lib/assets/docs/article/grokkingfp/rust/part-5.md +705 -705
  549. package/lib/assets/docs/article/grokkingfp/rust/part-6.md +508 -508
  550. package/lib/assets/docs/article/grokkingfp/scala/index.md +171 -171
  551. package/lib/assets/docs/article/grokkingfp/scala/part-1.md +543 -543
  552. package/lib/assets/docs/article/grokkingfp/scala/part-2.md +946 -946
  553. package/lib/assets/docs/article/grokkingfp/scala/part-3.md +919 -919
  554. package/lib/assets/docs/article/grokkingfp/scala/part-4.md +742 -742
  555. package/lib/assets/docs/article/grokkingfp/scala/part-5.md +722 -722
  556. package/lib/assets/docs/article/grokkingfp/scala/part-6.md +867 -867
  557. package/lib/assets/docs/article/grokkingfp/typescript/index.md +273 -273
  558. package/lib/assets/docs/article/grokkingfp/typescript/part-1.md +561 -561
  559. package/lib/assets/docs/article/grokkingfp/typescript/part-2.md +1129 -1129
  560. package/lib/assets/docs/article/grokkingfp/typescript/part-3.md +842 -842
  561. package/lib/assets/docs/article/grokkingfp/typescript/part-4.md +1087 -1087
  562. package/lib/assets/docs/article/grokkingfp/typescript/part-5.md +717 -717
  563. package/lib/assets/docs/article/grokkingfp/typescript/part-6.md +982 -982
  564. package/lib/assets/docs/article/practical-database-design/index.md +121 -121
  565. package/lib/assets/docs/article/practical-database-design/part1/chapter01.md +288 -288
  566. package/lib/assets/docs/article/practical-database-design/part1/chapter02.md +518 -518
  567. package/lib/assets/docs/article/practical-database-design/part1/chapter03.md +557 -557
  568. package/lib/assets/docs/article/practical-database-design/part2/chapter04.md +924 -924
  569. package/lib/assets/docs/article/practical-database-design/part2/chapter05.md +1627 -1627
  570. package/lib/assets/docs/article/practical-database-design/part2/chapter06.md +2716 -2716
  571. package/lib/assets/docs/article/practical-database-design/part2/chapter07.md +2082 -2082
  572. package/lib/assets/docs/article/practical-database-design/part2/chapter08.md +2105 -2105
  573. package/lib/assets/docs/article/practical-database-design/part2/chapter09.md +2031 -2031
  574. package/lib/assets/docs/article/practical-database-design/part2/chapter10.md +1387 -1387
  575. package/lib/assets/docs/article/practical-database-design/part2/chapter11.md +1677 -1677
  576. package/lib/assets/docs/article/practical-database-design/part2/chapter12.md +1417 -1417
  577. package/lib/assets/docs/article/practical-database-design/part2/chapter13.md +1434 -1434
  578. package/lib/assets/docs/article/practical-database-design/part3/chapter14.md +667 -667
  579. package/lib/assets/docs/article/practical-database-design/part3/chapter15.md +1625 -1625
  580. package/lib/assets/docs/article/practical-database-design/part3/chapter16.md +1915 -1915
  581. package/lib/assets/docs/article/practical-database-design/part3/chapter17.md +1708 -1708
  582. package/lib/assets/docs/article/practical-database-design/part3/chapter18.md +2095 -2095
  583. package/lib/assets/docs/article/practical-database-design/part3/chapter19.md +1123 -1123
  584. package/lib/assets/docs/article/practical-database-design/part3/chapter20.md +1031 -1031
  585. package/lib/assets/docs/article/practical-database-design/part3/chapter21.md +1382 -1382
  586. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter14-orm.md +991 -991
  587. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter15-orm.md +1300 -1300
  588. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter16-orm.md +1166 -1166
  589. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter17-orm.md +1584 -1584
  590. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter18-orm.md +1183 -1183
  591. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter19-orm.md +1016 -1016
  592. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter20-orm.md +1753 -1753
  593. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter21-orm.md +1447 -1447
  594. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter22-orm.md +1878 -1878
  595. package/lib/assets/docs/article/practical-database-design/part4/chapter22.md +965 -965
  596. package/lib/assets/docs/article/practical-database-design/part4/chapter23.md +2069 -2069
  597. package/lib/assets/docs/article/practical-database-design/part4/chapter24.md +2439 -2439
  598. package/lib/assets/docs/article/practical-database-design/part4/chapter25.md +3661 -3661
  599. package/lib/assets/docs/article/practical-database-design/part4/chapter26.md +2916 -2916
  600. package/lib/assets/docs/article/practical-database-design/part4/chapter27.md +3105 -3105
  601. package/lib/assets/docs/article/practical-database-design/part4/chapter28.md +2697 -2697
  602. package/lib/assets/docs/article/practical-database-design/part4/chapter29.md +2544 -2544
  603. package/lib/assets/docs/article/practical-database-design/part4/chapter30.md +2180 -2180
  604. package/lib/assets/docs/article/practical-database-design/part4/chapter31.md +1192 -1192
  605. package/lib/assets/docs/article/practical-database-design/part4/chapter32.md +2101 -2101
  606. package/lib/assets/docs/article/practical-database-design/part5/chapter33.md +1032 -1032
  607. package/lib/assets/docs/article/practical-database-design/part5/chapter34.md +1609 -1609
  608. package/lib/assets/docs/article/practical-database-design/part5/chapter35.md +1453 -1453
  609. package/lib/assets/docs/article/practical-database-design/part5/chapter36.md +1292 -1292
  610. package/lib/assets/docs/article/practical-database-design/part5/chapter37.md +1470 -1470
  611. package/lib/assets/docs/article/practical-database-design/part5/chapter38.md +1698 -1698
  612. package/lib/assets/docs/article/practical-database-design/part5/chapter39.md +2334 -2334
  613. package/lib/assets/docs/article/practical-database-design/study/study2-1.md +1693 -1693
  614. package/lib/assets/docs/article/practical-database-design/study/study2-2.md +1347 -1347
  615. package/lib/assets/docs/article/practical-database-design/study/study2-3.md +2044 -2044
  616. package/lib/assets/docs/article/practical-database-design/study/study2-4.md +2229 -2229
  617. package/lib/assets/docs/article/practical-database-design/study/study2-5.md +2418 -2418
  618. package/lib/assets/docs/article/practical-database-design/study/study3-1.md +2205 -2205
  619. package/lib/assets/docs/article/practical-database-design/study/study3-2.md +2221 -2221
  620. package/lib/assets/docs/article/practical-database-design/study/study3-3.md +2253 -2253
  621. package/lib/assets/docs/article/practical-database-design/study/study3-4.md +2106 -2106
  622. package/lib/assets/docs/article/practical-database-design/study/study3-5.md +2507 -2507
  623. package/lib/assets/docs/article/practical-database-design/study/study4-1.md +2587 -2587
  624. package/lib/assets/docs/article/practical-database-design/study/study4-2.md +2075 -2075
  625. package/lib/assets/docs/article/practical-database-design/study/study4-3.md +1805 -1805
  626. package/lib/assets/docs/article/practical-database-design/study/study4-4.md +1895 -1895
  627. package/lib/assets/docs/article/practical-database-design/study/study4-5.md +2878 -2878
  628. package/lib/assets/docs/assets/css/extra.css +29 -29
  629. package/lib/assets/docs/assets/js/extra.js +44 -44
  630. package/lib/assets/docs/development/index.md +39 -39
  631. package/lib/assets/docs/operation/index.md +11 -11
  632. package/lib/assets/docs/reference/CodexCLIMCP/343/202/242/343/203/227/343/203/252/343/202/261/343/203/274/343/202/267/343/203/247/343/203/263/351/226/213/347/231/272/343/203/225/343/203/255/343/203/274.md +532 -532
  633. 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
  634. package/lib/assets/docs/reference/Java/343/202/242/343/203/227/343/203/252/343/202/261/343/203/274/343/202/267/343/203/247/343/203/263/347/222/260/345/242/203/346/247/213/347/257/211/343/202/254/343/202/244/343/203/211.md +581 -580
  635. 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
  636. 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
  637. package/lib/assets/docs/reference/UI/350/250/255/350/250/210/343/202/254/343/202/244/343/203/211.md +450 -450
  638. package/lib/assets/docs/reference/images/Ansoff.drawio.svg +3 -3
  639. package/lib/assets/docs/reference/images/BrandBasicStrategy.drawio.svg +3 -3
  640. package/lib/assets/docs/reference/images/BrandCategorization.drawio.svg +3 -3
  641. package/lib/assets/docs/reference/images/BrandRecurutementStrategy.drawio.svg +3 -3
  642. package/lib/assets/docs/reference/images/BrandValue.drawio.svg +3 -3
  643. package/lib/assets/docs/reference/images/BusinessActivitiy.svg +3 -3
  644. package/lib/assets/docs/reference/images/HRM.drawio.svg +3 -3
  645. package/lib/assets/docs/reference/images/MarketingStructure.drawio.svg +3 -3
  646. package/lib/assets/docs/reference/images/OrganizationElemnts.svg +3 -3
  647. package/lib/assets/docs/reference/images/PPM.drawio.svg +3 -3
  648. package/lib/assets/docs/reference/images/PositioningMap.drawio.svg +3 -3
  649. package/lib/assets/docs/reference/images/ProductLayer.drawio.svg +3 -3
  650. package/lib/assets/docs/reference/images/ProductMix.drawio.svg +3 -3
  651. package/lib/assets/docs/reference/images/SWOT.drawio.svg +3 -3
  652. package/lib/assets/docs/reference/images/TargetMarket.drawio.svg +3 -3
  653. package/lib/assets/docs/reference/images/ThreeGenericStrategies.drawio.svg +3 -3
  654. package/lib/assets/docs/reference/images/VRIO.drawio.svg +3 -3
  655. package/lib/assets/docs/reference/images/ValueChain.drawio.svg +3 -3
  656. package/lib/assets/docs/reference/index.md +52 -52
  657. package/lib/assets/docs/reference//343/202/210/343/201/204/343/202/275/343/203/225/343/203/210/343/202/246/343/202/247/343/202/242/343/201/250/343/201/257.md +250 -242
  658. 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
  659. 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
  660. package/lib/assets/docs/reference//343/202/250/343/202/257/343/202/271/343/203/210/343/203/252/343/203/274/343/203/240/343/203/227/343/203/255/343/202/260/343/203/251/343/203/237/343/203/263/343/202/260.md +550 -544
  661. 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
  662. 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
  663. 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
  664. 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
  665. 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
  666. package/lib/assets/docs/reference//343/203/246/343/203/274/343/202/271/343/202/261/343/203/274/343/202/271/344/275/234/346/210/220/343/202/254/343/202/244/343/203/211.md +689 -682
  667. 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
  668. package/lib/assets/docs/reference//343/203/252/343/203/252/343/203/274/343/202/271/343/203/273/343/202/244/343/203/206/343/203/254/343/203/274/343/202/267/343/203/247/343/203/263/350/250/210/347/224/273/343/202/254/343/202/244/343/203/211.md +580 -560
  669. 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
  670. package/lib/assets/docs/reference//344/274/201/346/245/255/347/265/214/345/226/266/350/253/226.md +2637 -2636
  671. package/lib/assets/docs/reference//347/222/260/345/242/203/345/244/211/346/225/260/347/256/241/347/220/206/343/202/254/343/202/244/343/203/211.md +665 -663
  672. 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
  673. package/lib/assets/docs/reference//350/250/200/350/252/236/345/210/245/351/226/213/347/231/272/343/202/254/343/202/244/343/203/211.md +28 -0
  674. package/lib/assets/docs/reference//351/201/213/345/226/266/347/256/241/347/220/206.md +1482 -1482
  675. 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
  676. 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
  677. package/lib/assets/docs/reference//351/226/213/347/231/272/343/202/254/343/202/244/343/203/211.md +299 -299
  678. 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
  679. package/lib/assets/docs/review/index.md +5 -5
  680. package/lib/assets/docs/strategy/index.md +1 -1
  681. package/lib/assets/docs/template/ADR.md +30 -30
  682. 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
  683. 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
  684. package/lib/assets/docs/template/README.md +50 -50
  685. package/lib/assets/docs/template/index.md +23 -23
  686. 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
  687. 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
  688. 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
  689. 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
  690. 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
  691. package/lib/assets/docs/template//344/274/201/346/245/255/345/210/206/346/236/220.md +573 -573
  692. package/lib/assets/docs/template//345/256/214/345/205/250/345/275/242/345/274/217/343/201/256/343/203/246/343/203/274/343/202/271/343/202/261/343/203/274/343/202/271.md +69 -68
  693. package/lib/assets/docs/template//350/246/201/344/273/266/345/256/232/347/276/251.md +669 -669
  694. package/lib/assets/docs/template//350/250/255/350/250/210.md +173 -173
  695. 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
  696. package/lib/assets/gulpfile.js +25 -25
  697. package/lib/assets/mkdocs.yml +136 -135
  698. package/lib/assets/ops/docker/mkdoc/Dockerfile +19 -19
  699. package/lib/assets/ops/scripts/journal.js +180 -180
  700. package/lib/assets/ops/scripts/mkdocs.js +82 -82
  701. package/lib/assets/ops/scripts/release.js +431 -431
  702. package/lib/assets/ops/scripts/sonar_local.js +726 -726
  703. package/lib/assets/ops/scripts/ssh.js +190 -190
  704. package/lib/assets/ops/scripts/vault.js +299 -299
  705. package/lib/assets/package-lock.json +1653 -1653
  706. package/lib/assets/package.json +40 -40
  707. package/lib/gulpfile.js +37 -37
  708. package/package.json +41 -41
  709. package/lib/assets/.claude/agent-memory/xp-programmer/MEMORY.md +0 -6
  710. package/lib/assets/.claude/agent-memory/xp-programmer/project_cargo_tracker.md +0 -11
  711. package/lib/assets/.claude/agent-memory/xp-programmer/project_ddd_patterns.md +0 -27
  712. package/lib/assets/.claude/agent-memory/xp-programmer/project_us07_route_assignment.md +0 -19
@@ -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)