@k2works/claude-code-booster 3.5.0 → 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 +215 -215
  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,2031 +1,2031 @@
1
- # 第9章:在庫管理の設計
2
-
3
- 販売管理システムにおいて、在庫管理は商品の物理的な状態を追跡し、適正在庫を維持するための重要な機能です。本章では、倉庫別・商品別の在庫管理、引当処理、入出庫履歴、棚卸業務のデータベース設計と実装を行います。
4
-
5
- ## 在庫管理の全体像
6
-
7
- 在庫管理は「入庫 → 在庫保管 → 引当 → 出庫」という一連のフローで構成されます。
8
-
9
- ```plantuml
10
- @startuml
11
-
12
- title 在庫管理の業務フロー
13
-
14
- |倉庫部門|
15
- start
16
- :入荷・入庫処理;
17
- note right: 仕入・倉庫移動などで\n在庫が増加
18
-
19
- :在庫情報更新;
20
- fork
21
- :現在庫数 増加;
22
- fork again
23
- :入出庫履歴 記録;
24
- fork end
25
-
26
- |営業部門|
27
- :受注登録;
28
- :在庫引当;
29
- note right: 受注時に引当数を\n計上して確保
30
-
31
- |倉庫部門|
32
- :出荷指示受領;
33
- :ピッキング;
34
- :出荷処理;
35
-
36
- :在庫情報更新;
37
- fork
38
- :現在庫数 減少;
39
- fork again
40
- :引当数 減少;
41
- fork again
42
- :入出庫履歴 記録;
43
- fork end
44
-
45
- |経理部門|
46
- if (棚卸時期?) then (yes)
47
- :棚卸指示;
48
- |倉庫部門|
49
- :棚卸表発行;
50
- :実棚カウント;
51
- :差異入力;
52
- if (差異あり?) then (yes)
53
- :在庫調整処理;
54
- else (no)
55
- endif
56
- :棚卸完了;
57
- endif
58
-
59
- stop
60
-
61
- @enduml
62
- ```
63
-
64
- ### 在庫管理で扱うデータ
65
-
66
- | データ | 説明 |
67
- |-------|------|
68
- | **在庫データ** | 倉庫×商品の現在の在庫状態 |
69
- | **入出庫履歴** | 在庫の増減を時系列で記録 |
70
- | **棚卸データ** | 棚卸作業の管理情報 |
71
- | **棚卸明細** | 商品ごとの帳簿在庫と実棚数 |
72
-
73
- ---
74
-
75
- ## 9.1 在庫情報の DB 設計
76
-
77
- ### 在庫数量の種類
78
-
79
- 在庫管理では「倉庫コード × 商品コード」の組み合わせで在庫を管理します。
80
-
81
- ```plantuml
82
- @startuml
83
-
84
- title 在庫数量の関係
85
-
86
- rectangle "在庫数量の種類" {
87
- card "現在庫数\n(物理的な数量)" as current
88
- card "引当数\n(確保済み数量)" as allocated
89
- card "有効在庫数\n(利用可能数量)" as available
90
- card "発注残数\n(未入荷数量)" as ordered
91
- card "予定在庫数\n(将来の見込み)" as projected
92
- }
93
-
94
- current --> available : "- 引当数"
95
- available --> projected : "+ 発注残数"
96
-
97
- note bottom of current
98
- 入庫数 - 出庫数
99
- の累計
100
- end note
101
-
102
- note bottom of allocated
103
- 受注時に確保
104
- 出荷で解消
105
- end note
106
-
107
- note bottom of available
108
- 現在庫数 - 引当数
109
- 実際に出荷可能な数量
110
- end note
111
-
112
- note bottom of projected
113
- 有効在庫数 + 発注残数
114
- 将来的に利用可能な見込み
115
- end note
116
-
117
- @enduml
118
- ```
119
-
120
- | 数量種別 | 説明 | 計算式 |
121
- |---------|------|--------|
122
- | **現在庫数** | 物理的に倉庫にある数量 | 入庫数 - 出庫数 |
123
- | **引当数** | 受注等で確保済みの数量 | 受注引当の合計 |
124
- | **有効在庫数** | 実際に利用可能な数量 | 現在庫数 - 引当数 |
125
- | **発注残数** | 発注済みで未入荷の数量 | 発注数 - 入荷数 |
126
- | **予定在庫数** | 将来的に利用可能な見込み | 有効在庫数 + 発注残数 |
127
-
128
- ### 倉庫マスタの設計
129
-
130
- ```plantuml
131
- @startuml
132
-
133
- title 倉庫・ロケーション関連テーブル
134
-
135
- entity 倉庫マスタ {
136
- 倉庫コード <<PK>>
137
- --
138
- 倉庫名
139
- 倉庫区分
140
- 郵便番号
141
- 住所
142
- 電話番号
143
- 有効フラグ
144
- 作成日時
145
- 更新日時
146
- }
147
-
148
- entity ロケーションマスタ {
149
- ロケーションコード <<PK>>
150
- --
151
- 倉庫コード <<FK>>
152
- 棚番
153
- ゾーン
154
- 通路
155
- ラック
156
-
157
- 間口
158
- 有効フラグ
159
- 作成日時
160
- 更新日時
161
- }
162
-
163
- entity 在庫データ {
164
- ID <<PK>>
165
- --
166
- 倉庫コード <<FK>>
167
- 商品コード <<FK>>
168
- ロケーションコード <<FK>>
169
- 現在庫数
170
- 引当数
171
- 発注残数
172
- 最終入庫日
173
- 最終出庫日
174
- ロット番号
175
- シリアル番号
176
- 有効期限
177
- バージョン
178
- 作成日時
179
- 更新日時
180
- }
181
-
182
- entity 商品マスタ {
183
- 商品コード <<PK>>
184
- --
185
- 商品名
186
- 単価
187
- }
188
-
189
- 倉庫マスタ ||--o{ ロケーションマスタ : 倉庫コード
190
- 倉庫マスタ ||--o{ 在庫データ : 倉庫コード
191
- 商品マスタ ||--o{ 在庫データ : 商品コード
192
- ロケーションマスタ ||--o{ 在庫データ : ロケーションコード
193
-
194
- @enduml
195
- ```
196
-
197
- ### 倉庫区分の定義
198
-
199
- ```plantuml
200
- @startuml
201
-
202
- title 倉庫区分
203
-
204
- rectangle "自社倉庫" #CCFFCC {
205
- card "本社倉庫" as own1
206
- card "支店倉庫" as own2
207
- }
208
-
209
- rectangle "外部倉庫" #CCCCFF {
210
- card "営業倉庫\n(3PL)" as ext1
211
- card "保管倉庫" as ext2
212
- }
213
-
214
- rectangle "仮想倉庫" #FFFFCC {
215
- card "預け在庫" as virt1
216
- card "輸送中在庫" as virt2
217
- }
218
-
219
- note bottom of own1
220
- 自社で管理
221
- 入出庫を直接制御
222
- end note
223
-
224
- note bottom of ext1
225
- 外部委託
226
- 入出庫依頼が必要
227
- end note
228
-
229
- note bottom of virt1
230
- 物理的に存在しない
231
- 管理上の区分
232
- end note
233
-
234
- @enduml
235
- ```
236
-
237
- | 倉庫区分 | 説明 |
238
- |---------|------|
239
- | **自社** | 自社所有・管理の倉庫 |
240
- | **外部** | 外部委託(3PL)の倉庫 |
241
- | **仮想** | 輸送中、預け在庫など |
242
-
243
- ### マイグレーション:倉庫・在庫関連テーブルの作成
244
-
245
- <details>
246
- <summary>V015__create_inventory_tables.sql</summary>
247
-
248
- ```sql
249
- -- src/main/resources/db/migration/V015__create_inventory_tables.sql
250
-
251
- -- 倉庫区分
252
- CREATE TYPE 倉庫区分 AS ENUM ('自社', '外部', '仮想');
253
-
254
- -- 倉庫マスタに倉庫区分カラムを追加
255
- -- (V013で作成済みの倉庫マスタにカラム追加)
256
- ALTER TABLE "倉庫マスタ" ADD COLUMN "倉庫区分" 倉庫区分 DEFAULT '自社' NOT NULL;
257
-
258
- -- ロケーションマスタ
259
- CREATE TABLE "ロケーションマスタ" (
260
- "ロケーションコード" VARCHAR(20) PRIMARY KEY,
261
- "倉庫コード" VARCHAR(20) NOT NULL,
262
- "棚番" VARCHAR(20) NOT NULL,
263
- "ゾーン" VARCHAR(10),
264
- "通路" VARCHAR(10),
265
- "ラック" VARCHAR(10),
266
- "段" VARCHAR(10),
267
- "間口" VARCHAR(10),
268
- "有効フラグ" BOOLEAN DEFAULT true NOT NULL,
269
- "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
270
- "作成者" VARCHAR(50),
271
- "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
272
- "更新者" VARCHAR(50),
273
- CONSTRAINT "fk_ロケーション_倉庫"
274
- FOREIGN KEY ("倉庫コード") REFERENCES "倉庫マスタ"("倉庫コード"),
275
- UNIQUE ("倉庫コード", "棚番")
276
- );
277
-
278
- -- 在庫データ(倉庫 × 商品)
279
- CREATE TABLE "在庫データ" (
280
- "ID" SERIAL PRIMARY KEY,
281
- "倉庫コード" VARCHAR(20) NOT NULL,
282
- "商品コード" VARCHAR(20) NOT NULL,
283
- "ロケーションコード" VARCHAR(20),
284
- "現在庫数" DECIMAL(15, 2) DEFAULT 0 NOT NULL,
285
- "引当数" DECIMAL(15, 2) DEFAULT 0 NOT NULL,
286
- "発注残数" DECIMAL(15, 2) DEFAULT 0 NOT NULL,
287
- "最終入庫日" DATE,
288
- "最終出庫日" DATE,
289
- "ロット番号" VARCHAR(50),
290
- "シリアル番号" VARCHAR(50),
291
- "有効期限" DATE,
292
- "バージョン" INTEGER DEFAULT 1 NOT NULL,
293
- "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
294
- "作成者" VARCHAR(50),
295
- "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
296
- "更新者" VARCHAR(50),
297
- CONSTRAINT "fk_在庫_倉庫"
298
- FOREIGN KEY ("倉庫コード") REFERENCES "倉庫マスタ"("倉庫コード"),
299
- CONSTRAINT "fk_在庫_商品"
300
- FOREIGN KEY ("商品コード") REFERENCES "商品マスタ"("商品コード"),
301
- CONSTRAINT "fk_在庫_ロケーション"
302
- FOREIGN KEY ("ロケーションコード") REFERENCES "ロケーションマスタ"("ロケーションコード"),
303
- UNIQUE ("倉庫コード", "商品コード", "ロケーションコード", "ロット番号")
304
- );
305
-
306
- -- 有効在庫数を計算するビュー
307
- CREATE VIEW "有効在庫ビュー" AS
308
- SELECT
309
- "倉庫コード",
310
- "商品コード",
311
- "ロケーションコード",
312
- "現在庫数",
313
- "引当数",
314
- "現在庫数" - "引当数" AS "有効在庫数",
315
- "発注残数",
316
- "現在庫数" - "引当数" + "発注残数" AS "予定在庫数"
317
- FROM "在庫データ";
318
-
319
- -- インデックス
320
- CREATE INDEX "idx_ロケーション_倉庫" ON "ロケーションマスタ"("倉庫コード");
321
- CREATE INDEX "idx_在庫_倉庫コード" ON "在庫データ"("倉庫コード");
322
- CREATE INDEX "idx_在庫_商品コード" ON "在庫データ"("商品コード");
323
- CREATE INDEX "idx_在庫_ロケーション" ON "在庫データ"("ロケーションコード");
324
- CREATE INDEX "idx_在庫_ロット番号" ON "在庫データ"("ロット番号");
325
- CREATE INDEX "idx_在庫_有効期限" ON "在庫データ"("有効期限");
326
-
327
- -- テーブルコメント
328
- COMMENT ON TABLE "倉庫マスタ" IS '倉庫情報を管理するマスタテーブル';
329
- COMMENT ON TABLE "ロケーションマスタ" IS '倉庫内の棚・ロケーション情報を管理するテーブル';
330
- COMMENT ON TABLE "在庫データ" IS '倉庫×商品の在庫情報を管理するテーブル';
331
- COMMENT ON COLUMN "在庫データ"."バージョン" IS '楽観ロック用バージョン番号';
332
- ```
333
-
334
- </details>
335
-
336
- ### 倉庫・在庫エンティティの実装
337
-
338
- <details>
339
- <summary>倉庫区分 ENUM</summary>
340
-
341
- ```java
342
- // src/main/java/com/example/sales/domain/model/inventory/WarehouseType.java
343
- package com.example.sms.domain.model.inventory;
344
-
345
- import lombok.Getter;
346
- import lombok.RequiredArgsConstructor;
347
-
348
- @Getter
349
- @RequiredArgsConstructor
350
- public enum WarehouseType {
351
- OWN("自社"),
352
- EXTERNAL("外部"),
353
- VIRTUAL("仮想");
354
-
355
- private final String displayName;
356
-
357
- public static WarehouseType fromDisplayName(String displayName) {
358
- for (WarehouseType type : values()) {
359
- if (type.displayName.equals(displayName)) {
360
- return type;
361
- }
362
- }
363
- throw new IllegalArgumentException("Unknown warehouse type: " + displayName);
364
- }
365
- }
366
- ```
367
-
368
- </details>
369
-
370
- <details>
371
- <summary>倉庫マスタエンティティ</summary>
372
-
373
- ```java
374
- // src/main/java/com/example/sales/domain/model/inventory/Warehouse.java
375
- package com.example.sms.domain.model.inventory;
376
-
377
- import lombok.Builder;
378
- import lombok.Data;
379
-
380
- import java.time.LocalDateTime;
381
-
382
- @Data
383
- @Builder
384
- public class Warehouse {
385
- private String warehouseCode;
386
- private String warehouseName;
387
- private WarehouseType warehouseType;
388
- private String postalCode;
389
- private String address;
390
- private String phoneNumber;
391
- private Boolean activeFlag;
392
- private LocalDateTime createdAt;
393
- private String createdBy;
394
- private LocalDateTime updatedAt;
395
- private String updatedBy;
396
-
397
- /**
398
- * 自社倉庫かどうか
399
- */
400
- public boolean isOwnWarehouse() {
401
- return this.warehouseType == WarehouseType.OWN;
402
- }
403
-
404
- /**
405
- * 仮想倉庫かどうか
406
- */
407
- public boolean isVirtualWarehouse() {
408
- return this.warehouseType == WarehouseType.VIRTUAL;
409
- }
410
- }
411
- ```
412
-
413
- </details>
414
-
415
- <details>
416
- <summary>ロケーションマスタエンティティ</summary>
417
-
418
- ```java
419
- // src/main/java/com/example/sales/domain/model/inventory/Location.java
420
- package com.example.sms.domain.model.inventory;
421
-
422
- import lombok.Builder;
423
- import lombok.Data;
424
-
425
- import java.time.LocalDateTime;
426
-
427
- @Data
428
- @Builder
429
- public class Location {
430
- private String locationCode;
431
- private String warehouseCode;
432
- private String shelfNumber;
433
- private String zone;
434
- private String aisle;
435
- private String rack;
436
- private String level;
437
- private String bay;
438
- private Boolean activeFlag;
439
- private LocalDateTime createdAt;
440
- private String createdBy;
441
- private LocalDateTime updatedAt;
442
- private String updatedBy;
443
-
444
- // リレーション
445
- private Warehouse warehouse;
446
-
447
- /**
448
- * ロケーションの完全表記を取得
449
- * 例: A-01-02-03-04
450
- */
451
- public String getFullAddress() {
452
- StringBuilder sb = new StringBuilder();
453
- if (zone != null) sb.append(zone).append("-");
454
- if (aisle != null) sb.append(aisle).append("-");
455
- if (rack != null) sb.append(rack).append("-");
456
- if (level != null) sb.append(level).append("-");
457
- if (bay != null) sb.append(bay);
458
- return sb.toString().replaceAll("-$", "");
459
- }
460
- }
461
- ```
462
-
463
- </details>
464
-
465
- <details>
466
- <summary>在庫エンティティ</summary>
467
-
468
- ```java
469
- // src/main/java/com/example/sales/domain/model/inventory/Inventory.java
470
- package com.example.sms.domain.model.inventory;
471
-
472
- import com.example.sms.domain.model.product.Product;
473
- import lombok.Builder;
474
- import lombok.Data;
475
-
476
- import java.math.BigDecimal;
477
- import java.time.LocalDate;
478
- import java.time.LocalDateTime;
479
-
480
- @Data
481
- @Builder
482
- public class Inventory {
483
- private Integer id;
484
- private String warehouseCode;
485
- private String productCode;
486
- private String locationCode;
487
- private BigDecimal currentQuantity;
488
- private BigDecimal allocatedQuantity;
489
- private BigDecimal orderedQuantity;
490
- private LocalDate lastReceiptDate;
491
- private LocalDate lastShipmentDate;
492
- private String lotNumber;
493
- private String serialNumber;
494
- private LocalDate expirationDate;
495
- private LocalDateTime createdAt;
496
- private String createdBy;
497
- private LocalDateTime updatedAt;
498
- private String updatedBy;
499
-
500
- // 楽観ロック用バージョン
501
- @Builder.Default
502
- private Integer version = 1;
503
-
504
- // リレーション
505
- private Warehouse warehouse;
506
- private Product product;
507
- private Location location;
508
-
509
- /**
510
- * 有効在庫数を計算
511
- */
512
- public BigDecimal getAvailableQuantity() {
513
- return currentQuantity.subtract(allocatedQuantity);
514
- }
515
-
516
- /**
517
- * 予定在庫数を計算
518
- */
519
- public BigDecimal getProjectedQuantity() {
520
- return currentQuantity.subtract(allocatedQuantity).add(orderedQuantity);
521
- }
522
-
523
- /**
524
- * 引当可能かどうかを判定
525
- */
526
- public boolean canAllocate(BigDecimal quantity) {
527
- return getAvailableQuantity().compareTo(quantity) >= 0;
528
- }
529
-
530
- /**
531
- * 在庫を引当
532
- */
533
- public void allocate(BigDecimal quantity) {
534
- if (!canAllocate(quantity)) {
535
- throw new IllegalStateException(
536
- "有効在庫数が不足しています。有効在庫: " + getAvailableQuantity() +
537
- ", 引当要求: " + quantity);
538
- }
539
- this.allocatedQuantity = this.allocatedQuantity.add(quantity);
540
- }
541
-
542
- /**
543
- * 引当を解除
544
- */
545
- public void deallocate(BigDecimal quantity) {
546
- if (this.allocatedQuantity.compareTo(quantity) < 0) {
547
- throw new IllegalStateException(
548
- "引当数を超える解除はできません。引当数: " + this.allocatedQuantity +
549
- ", 解除要求: " + quantity);
550
- }
551
- this.allocatedQuantity = this.allocatedQuantity.subtract(quantity);
552
- }
553
-
554
- /**
555
- * 入庫処理
556
- */
557
- public void receive(BigDecimal quantity) {
558
- this.currentQuantity = this.currentQuantity.add(quantity);
559
- this.lastReceiptDate = LocalDate.now();
560
- }
561
-
562
- /**
563
- * 出庫処理
564
- */
565
- public void ship(BigDecimal quantity) {
566
- if (this.currentQuantity.compareTo(quantity) < 0) {
567
- throw new IllegalStateException(
568
- "現在庫数が不足しています。現在庫: " + this.currentQuantity +
569
- ", 出庫要求: " + quantity);
570
- }
571
- this.currentQuantity = this.currentQuantity.subtract(quantity);
572
- this.lastShipmentDate = LocalDate.now();
573
- }
574
-
575
- /**
576
- * 有効期限切れかどうか
577
- */
578
- public boolean isExpired() {
579
- if (expirationDate == null) {
580
- return false;
581
- }
582
- return expirationDate.isBefore(LocalDate.now());
583
- }
584
- }
585
- ```
586
-
587
- </details>
588
-
589
- ### 在庫サービスの実装
590
-
591
- <details>
592
- <summary>在庫サービス</summary>
593
-
594
- ```java
595
- // src/main/java/com/example/sales/application/service/InventoryService.java
596
- package com.example.sms.application.service;
597
-
598
- import com.example.sms.application.port.out.InventoryRepository;
599
- import com.example.sms.domain.model.inventory.*;
600
- import lombok.RequiredArgsConstructor;
601
- import org.springframework.stereotype.Service;
602
- import org.springframework.transaction.annotation.Transactional;
603
-
604
- import java.math.BigDecimal;
605
- import java.time.LocalDate;
606
- import java.util.List;
607
- import java.util.Optional;
608
-
609
- @Service
610
- @RequiredArgsConstructor
611
- public class InventoryService {
612
-
613
- private final InventoryRepository inventoryRepository;
614
-
615
- /**
616
- * 商品別の在庫サマリを取得
617
- */
618
- public InventorySummary getProductInventorySummary(String productCode) {
619
- List<Inventory> inventories = inventoryRepository.findByProductCode(productCode);
620
-
621
- BigDecimal totalCurrent = BigDecimal.ZERO;
622
- BigDecimal totalAllocated = BigDecimal.ZERO;
623
- BigDecimal totalOrdered = BigDecimal.ZERO;
624
-
625
- for (Inventory inv : inventories) {
626
- totalCurrent = totalCurrent.add(inv.getCurrentQuantity());
627
- totalAllocated = totalAllocated.add(inv.getAllocatedQuantity());
628
- totalOrdered = totalOrdered.add(inv.getOrderedQuantity());
629
- }
630
-
631
- return InventorySummary.builder()
632
- .productCode(productCode)
633
- .totalCurrentQuantity(totalCurrent)
634
- .totalAllocatedQuantity(totalAllocated)
635
- .totalAvailableQuantity(totalCurrent.subtract(totalAllocated))
636
- .totalOrderedQuantity(totalOrdered)
637
- .warehouseDetails(inventories)
638
- .build();
639
- }
640
-
641
- /**
642
- * 倉庫別の在庫一覧を取得
643
- */
644
- public List<Inventory> getInventoryByWarehouse(String warehouseCode) {
645
- return inventoryRepository.findByWarehouseCode(warehouseCode);
646
- }
647
-
648
- /**
649
- * 在庫を引当
650
- */
651
- @Transactional
652
- public AllocationResult allocate(String warehouseCode, String productCode,
653
- BigDecimal quantity) {
654
- Optional<Inventory> inventoryOpt = inventoryRepository
655
- .findByWarehouseAndProduct(warehouseCode, productCode);
656
-
657
- if (inventoryOpt.isEmpty()) {
658
- return AllocationResult.failure("在庫が見つかりません");
659
- }
660
-
661
- Inventory inventory = inventoryOpt.get();
662
- BigDecimal available = inventory.getAvailableQuantity();
663
-
664
- if (available.compareTo(quantity) < 0) {
665
- return AllocationResult.failure(
666
- "有効在庫数が不足しています。有効在庫: " + available);
667
- }
668
-
669
- inventoryRepository.allocate(inventory.getId(), quantity);
670
-
671
- return AllocationResult.success(quantity);
672
- }
673
-
674
- /**
675
- * 引当を解除
676
- */
677
- @Transactional
678
- public void deallocate(String warehouseCode, String productCode,
679
- BigDecimal quantity) {
680
- Optional<Inventory> inventoryOpt = inventoryRepository
681
- .findByWarehouseAndProduct(warehouseCode, productCode);
682
-
683
- if (inventoryOpt.isEmpty()) {
684
- throw new IllegalArgumentException("在庫が見つかりません");
685
- }
686
-
687
- Inventory inventory = inventoryOpt.get();
688
- inventoryRepository.deallocate(inventory.getId(), quantity);
689
- }
690
-
691
- /**
692
- * 入庫処理
693
- */
694
- @Transactional
695
- public void receive(String warehouseCode, String productCode,
696
- BigDecimal quantity) {
697
- Optional<Inventory> inventoryOpt = inventoryRepository
698
- .findByWarehouseAndProduct(warehouseCode, productCode);
699
-
700
- if (inventoryOpt.isEmpty()) {
701
- // 新規在庫レコード作成
702
- Inventory newInventory = Inventory.builder()
703
- .warehouseCode(warehouseCode)
704
- .productCode(productCode)
705
- .currentQuantity(quantity)
706
- .allocatedQuantity(BigDecimal.ZERO)
707
- .orderedQuantity(BigDecimal.ZERO)
708
- .lastReceiptDate(LocalDate.now())
709
- .build();
710
- inventoryRepository.save(newInventory);
711
- } else {
712
- Inventory inventory = inventoryOpt.get();
713
- inventoryRepository.receive(inventory.getId(), quantity, LocalDate.now());
714
- }
715
- }
716
-
717
- /**
718
- * 出庫処理
719
- */
720
- @Transactional
721
- public void ship(String warehouseCode, String productCode,
722
- BigDecimal quantity) {
723
- Optional<Inventory> inventoryOpt = inventoryRepository
724
- .findByWarehouseAndProduct(warehouseCode, productCode);
725
-
726
- if (inventoryOpt.isEmpty()) {
727
- throw new IllegalArgumentException("在庫が見つかりません");
728
- }
729
-
730
- Inventory inventory = inventoryOpt.get();
731
- inventoryRepository.ship(inventory.getId(), quantity, LocalDate.now());
732
- }
733
-
734
- /**
735
- * 在庫数量を直接更新(棚卸調整用)
736
- */
737
- @Transactional
738
- public void adjustQuantity(String warehouseCode, String productCode,
739
- BigDecimal newQuantity, String reason) {
740
- Optional<Inventory> inventoryOpt = inventoryRepository
741
- .findByWarehouseAndProduct(warehouseCode, productCode);
742
-
743
- if (inventoryOpt.isEmpty()) {
744
- throw new IllegalArgumentException("在庫が見つかりません");
745
- }
746
-
747
- Inventory inventory = inventoryOpt.get();
748
- inventoryRepository.updateCurrentQuantity(
749
- inventory.getId(), newQuantity, reason);
750
- }
751
- }
752
- ```
753
-
754
- </details>
755
-
756
- <details>
757
- <summary>在庫サマリ・引当結果クラス</summary>
758
-
759
- ```java
760
- // src/main/java/com/example/sales/application/service/InventorySummary.java
761
- package com.example.sms.application.service;
762
-
763
- import com.example.sms.domain.model.inventory.Inventory;
764
- import lombok.Builder;
765
- import lombok.Data;
766
-
767
- import java.math.BigDecimal;
768
- import java.util.List;
769
-
770
- @Data
771
- @Builder
772
- public class InventorySummary {
773
- private String productCode;
774
- private BigDecimal totalCurrentQuantity;
775
- private BigDecimal totalAllocatedQuantity;
776
- private BigDecimal totalAvailableQuantity;
777
- private BigDecimal totalOrderedQuantity;
778
- private List<Inventory> warehouseDetails;
779
- }
780
- ```
781
-
782
- ```java
783
- // src/main/java/com/example/sales/application/service/AllocationResult.java
784
- package com.example.sms.application.service;
785
-
786
- import lombok.Builder;
787
- import lombok.Data;
788
-
789
- import java.math.BigDecimal;
790
-
791
- @Data
792
- @Builder
793
- public class AllocationResult {
794
- private boolean success;
795
- private BigDecimal allocatedQuantity;
796
- private String reason;
797
-
798
- public static AllocationResult success(BigDecimal quantity) {
799
- return AllocationResult.builder()
800
- .success(true)
801
- .allocatedQuantity(quantity)
802
- .build();
803
- }
804
-
805
- public static AllocationResult failure(String reason) {
806
- return AllocationResult.builder()
807
- .success(false)
808
- .allocatedQuantity(BigDecimal.ZERO)
809
- .reason(reason)
810
- .build();
811
- }
812
- }
813
- ```
814
-
815
- </details>
816
-
817
- ---
818
-
819
- ## 9.2 在庫移動・棚卸の DB 設計
820
-
821
- ### 入出庫履歴の管理
822
-
823
- 入出庫履歴は在庫の増減を時系列で記録し、トレーサビリティを確保します。
824
-
825
- ```plantuml
826
- @startuml
827
-
828
- title 入出庫履歴の構造
829
-
830
- entity "入出庫履歴データ" as movement {
831
- * 履歴ID <<PK>>
832
- --
833
- * 倉庫コード <<FK>>
834
- * 商品コード <<FK>>
835
- * 移動日時
836
- * 移動区分
837
- * 移動数量
838
- * 移動前在庫数
839
- * 移動後在庫数
840
- 伝票番号
841
- 伝票種別
842
- 移動理由
843
- ロケーションコード
844
- ロット番号
845
- }
846
-
847
- note right of movement
848
- 移動区分:
849
- - 入荷入庫
850
- - 出荷出庫
851
- - 倉庫間移動(出)
852
- - 倉庫間移動(入)
853
- - 棚卸調整(増)
854
- - 棚卸調整(減)
855
- - 返品入庫
856
- - 廃棄
857
- end note
858
-
859
- @enduml
860
- ```
861
-
862
- ### 移動区分の定義
863
-
864
- | 移動区分 | 在庫影響 | 説明 |
865
- |---------|---------|------|
866
- | **入荷入庫** | 増加 | 仕入先からの入荷 |
867
- | **出荷出庫** | 減少 | 顧客への出荷 |
868
- | **倉庫間移動(出)** | 減少 | 他倉庫への移動(出庫側) |
869
- | **倉庫間移動(入)** | 増加 | 他倉庫からの移動(入庫側) |
870
- | **棚卸調整(増)** | 増加 | 棚卸による増加調整 |
871
- | **棚卸調整(減)** | 減少 | 棚卸による減少調整 |
872
- | **返品入庫** | 増加 | 顧客からの返品 |
873
- | **廃棄** | 減少 | 廃棄処理 |
874
-
875
- ### 棚卸業務の設計
876
-
877
- 棚卸業務は「棚卸表発行 → 実棚カウント → 差異確認 → 在庫調整」の流れで進みます。
878
-
879
- ```plantuml
880
- @startuml
881
-
882
- title 棚卸業務フロー
883
-
884
- [*] --> 作成中
885
-
886
- 作成中 --> 実施中 : 棚卸開始
887
-
888
- 実施中 --> 確定 : 棚卸完了
889
-
890
- 確定 --> [*]
891
-
892
- 作成中 --> 取消 : 取消
893
- 実施中 --> 取消 : 取消
894
-
895
- 取消 --> [*]
896
-
897
- @enduml
898
- ```
899
-
900
- ### 棚卸データの ER 図
901
-
902
- ```plantuml
903
- @startuml
904
-
905
- title 棚卸関連テーブル
906
-
907
- entity 棚卸データ {
908
- ID <<PK>>
909
- --
910
- 棚卸番号 <<UK>>
911
- 倉庫コード <<FK>>
912
- 棚卸日
913
- 棚卸開始日時
914
- 棚卸終了日時
915
- ステータス
916
- 備考
917
- 作成日時
918
- 更新日時
919
- }
920
-
921
- entity 棚卸明細データ {
922
- ID <<PK>>
923
- --
924
- 棚卸ID <<FK>>
925
- 棚卸行番号
926
- 商品コード <<FK>>
927
- ロケーションコード
928
- ロット番号
929
- 帳簿在庫数
930
- 実棚数
931
- 差異数
932
- 差異理由
933
- 調整済フラグ
934
- 作成日時
935
- 更新日時
936
- }
937
-
938
- entity 倉庫マスタ {
939
- 倉庫コード <<PK>>
940
- --
941
- 倉庫名
942
- }
943
-
944
- entity 商品マスタ {
945
- 商品コード <<PK>>
946
- --
947
- 商品名
948
- }
949
-
950
- 倉庫マスタ ||--o{ 棚卸データ : 倉庫コード
951
- 棚卸データ ||--o{ 棚卸明細データ : 棚卸ID
952
- 商品マスタ ||--o{ 棚卸明細データ : 商品コード
953
-
954
- @enduml
955
- ```
956
-
957
- ### マイグレーション:入出庫履歴・棚卸テーブルの作成
958
-
959
- <details>
960
- <summary>V016__create_stock_movement_tables.sql</summary>
961
-
962
- ```sql
963
- -- src/main/resources/db/migration/V016__create_stock_movement_tables.sql
964
-
965
- -- 移動区分
966
- CREATE TYPE 移動区分 AS ENUM (
967
- '入荷入庫',
968
- '出荷出庫',
969
- '倉庫間移動出',
970
- '倉庫間移動入',
971
- '棚卸調整増',
972
- '棚卸調整減',
973
- '返品入庫',
974
- '廃棄'
975
- );
976
-
977
- -- 棚卸ステータス
978
- CREATE TYPE 棚卸ステータス AS ENUM ('作成中', '実施中', '確定', '取消');
979
-
980
- -- 入出庫履歴データ
981
- CREATE TABLE "入出庫履歴データ" (
982
- "ID" SERIAL PRIMARY KEY,
983
- "倉庫コード" VARCHAR(20) NOT NULL,
984
- "商品コード" VARCHAR(20) NOT NULL,
985
- "移動日時" TIMESTAMP NOT NULL,
986
- "移動区分" 移動区分 NOT NULL,
987
- "移動数量" DECIMAL(15, 2) NOT NULL,
988
- "移動前在庫数" DECIMAL(15, 2) NOT NULL,
989
- "移動後在庫数" DECIMAL(15, 2) NOT NULL,
990
- "伝票番号" VARCHAR(20),
991
- "伝票種別" VARCHAR(20),
992
- "移動理由" TEXT,
993
- "ロケーションコード" VARCHAR(20),
994
- "ロット番号" VARCHAR(50),
995
- "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
996
- "作成者" VARCHAR(50),
997
- CONSTRAINT "fk_入出庫履歴_倉庫"
998
- FOREIGN KEY ("倉庫コード") REFERENCES "倉庫マスタ"("倉庫コード"),
999
- CONSTRAINT "fk_入出庫履歴_商品"
1000
- FOREIGN KEY ("商品コード") REFERENCES "商品マスタ"("商品コード")
1001
- );
1002
-
1003
- -- 棚卸データ(ヘッダ)
1004
- CREATE TABLE "棚卸データ" (
1005
- "ID" SERIAL PRIMARY KEY,
1006
- "棚卸番号" VARCHAR(20) UNIQUE NOT NULL,
1007
- "倉庫コード" VARCHAR(20) NOT NULL,
1008
- "棚卸日" DATE NOT NULL,
1009
- "棚卸開始日時" TIMESTAMP,
1010
- "棚卸終了日時" TIMESTAMP,
1011
- "ステータス" 棚卸ステータス DEFAULT '作成中' NOT NULL,
1012
- "備考" TEXT,
1013
- "バージョン" INTEGER DEFAULT 1 NOT NULL,
1014
- "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
1015
- "作成者" VARCHAR(50),
1016
- "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
1017
- "更新者" VARCHAR(50),
1018
- CONSTRAINT "fk_棚卸_倉庫"
1019
- FOREIGN KEY ("倉庫コード") REFERENCES "倉庫マスタ"("倉庫コード")
1020
- );
1021
-
1022
- -- 棚卸明細データ
1023
- CREATE TABLE "棚卸明細データ" (
1024
- "ID" SERIAL PRIMARY KEY,
1025
- "棚卸ID" INTEGER NOT NULL,
1026
- "棚卸行番号" INTEGER NOT NULL,
1027
- "商品コード" VARCHAR(20) NOT NULL,
1028
- "ロケーションコード" VARCHAR(20),
1029
- "ロット番号" VARCHAR(50),
1030
- "帳簿在庫数" DECIMAL(15, 2) NOT NULL,
1031
- "実棚数" DECIMAL(15, 2),
1032
- "差異数" DECIMAL(15, 2),
1033
- "差異理由" TEXT,
1034
- "調整済フラグ" BOOLEAN DEFAULT FALSE NOT NULL,
1035
- "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
1036
- "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
1037
- CONSTRAINT "fk_棚卸明細_棚卸"
1038
- FOREIGN KEY ("棚卸ID") REFERENCES "棚卸データ"("ID") ON DELETE CASCADE,
1039
- CONSTRAINT "fk_棚卸明細_商品"
1040
- FOREIGN KEY ("商品コード") REFERENCES "商品マスタ"("商品コード"),
1041
- CONSTRAINT "uk_棚卸明細_棚卸_行" UNIQUE ("棚卸ID", "棚卸行番号")
1042
- );
1043
-
1044
- -- 差異数を自動計算するトリガー
1045
- CREATE OR REPLACE FUNCTION calculate_stocktaking_difference()
1046
- RETURNS TRIGGER AS $$
1047
- BEGIN
1048
- IF NEW."実棚数" IS NOT NULL THEN
1049
- NEW."差異数" := NEW."実棚数" - NEW."帳簿在庫数";
1050
- END IF;
1051
- RETURN NEW;
1052
- END;
1053
- $$ LANGUAGE plpgsql;
1054
-
1055
- CREATE TRIGGER trg_棚卸明細_差異計算
1056
- BEFORE INSERT OR UPDATE ON "棚卸明細データ"
1057
- FOR EACH ROW
1058
- EXECUTE FUNCTION calculate_stocktaking_difference();
1059
-
1060
- -- インデックス
1061
- CREATE INDEX "idx_入出庫履歴_倉庫コード" ON "入出庫履歴データ"("倉庫コード");
1062
- CREATE INDEX "idx_入出庫履歴_商品コード" ON "入出庫履歴データ"("商品コード");
1063
- CREATE INDEX "idx_入出庫履歴_移動日時" ON "入出庫履歴データ"("移動日時");
1064
- CREATE INDEX "idx_入出庫履歴_伝票番号" ON "入出庫履歴データ"("伝票番号");
1065
- CREATE INDEX "idx_棚卸_倉庫コード" ON "棚卸データ"("倉庫コード");
1066
- CREATE INDEX "idx_棚卸_棚卸日" ON "棚卸データ"("棚卸日");
1067
- CREATE INDEX "idx_棚卸_ステータス" ON "棚卸データ"("ステータス");
1068
- CREATE INDEX "idx_棚卸明細_商品コード" ON "棚卸明細データ"("商品コード");
1069
-
1070
- -- テーブルコメント
1071
- COMMENT ON TABLE "入出庫履歴データ" IS '在庫の入出庫履歴を記録するテーブル';
1072
- COMMENT ON TABLE "棚卸データ" IS '棚卸作業のヘッダ情報を管理するテーブル';
1073
- COMMENT ON TABLE "棚卸明細データ" IS '棚卸明細・差異情報を管理するテーブル';
1074
- COMMENT ON COLUMN "棚卸データ"."バージョン" IS '楽観ロック用バージョン番号';
1075
- ```
1076
-
1077
- </details>
1078
-
1079
- ### 入出庫履歴・棚卸エンティティの実装
1080
-
1081
- <details>
1082
- <summary>移動区分 ENUM</summary>
1083
-
1084
- ```java
1085
- // src/main/java/com/example/sales/domain/model/inventory/MovementType.java
1086
- package com.example.sms.domain.model.inventory;
1087
-
1088
- import lombok.Getter;
1089
- import lombok.RequiredArgsConstructor;
1090
-
1091
- @Getter
1092
- @RequiredArgsConstructor
1093
- public enum MovementType {
1094
- RECEIPT("入荷入庫"),
1095
- SHIPMENT("出荷出庫"),
1096
- TRANSFER_OUT("倉庫間移動出"),
1097
- TRANSFER_IN("倉庫間移動入"),
1098
- ADJUSTMENT_PLUS("棚卸調整増"),
1099
- ADJUSTMENT_MINUS("棚卸調整減"),
1100
- RETURN_RECEIPT("返品入庫"),
1101
- DISPOSAL("廃棄");
1102
-
1103
- private final String displayName;
1104
-
1105
- public static MovementType fromDisplayName(String displayName) {
1106
- for (MovementType type : values()) {
1107
- if (type.displayName.equals(displayName)) {
1108
- return type;
1109
- }
1110
- }
1111
- throw new IllegalArgumentException("Unknown movement type: " + displayName);
1112
- }
1113
-
1114
- /**
1115
- * 入庫系の移動かどうか
1116
- */
1117
- public boolean isInbound() {
1118
- return this == RECEIPT || this == TRANSFER_IN ||
1119
- this == ADJUSTMENT_PLUS || this == RETURN_RECEIPT;
1120
- }
1121
-
1122
- /**
1123
- * 出庫系の移動かどうか
1124
- */
1125
- public boolean isOutbound() {
1126
- return this == SHIPMENT || this == TRANSFER_OUT ||
1127
- this == ADJUSTMENT_MINUS || this == DISPOSAL;
1128
- }
1129
- }
1130
- ```
1131
-
1132
- </details>
1133
-
1134
- <details>
1135
- <summary>棚卸ステータス ENUM</summary>
1136
-
1137
- ```java
1138
- // src/main/java/com/example/sales/domain/model/inventory/StocktakingStatus.java
1139
- package com.example.sms.domain.model.inventory;
1140
-
1141
- import lombok.Getter;
1142
- import lombok.RequiredArgsConstructor;
1143
-
1144
- @Getter
1145
- @RequiredArgsConstructor
1146
- public enum StocktakingStatus {
1147
- DRAFT("作成中"),
1148
- IN_PROGRESS("実施中"),
1149
- CONFIRMED("確定"),
1150
- CANCELLED("取消");
1151
-
1152
- private final String displayName;
1153
-
1154
- public static StocktakingStatus fromDisplayName(String displayName) {
1155
- for (StocktakingStatus status : values()) {
1156
- if (status.displayName.equals(displayName)) {
1157
- return status;
1158
- }
1159
- }
1160
- throw new IllegalArgumentException("Unknown stocktaking status: " + displayName);
1161
- }
1162
-
1163
- /**
1164
- * 実施開始可能かどうか
1165
- */
1166
- public boolean canStart() {
1167
- return this == DRAFT;
1168
- }
1169
-
1170
- /**
1171
- * 確定可能かどうか
1172
- */
1173
- public boolean canConfirm() {
1174
- return this == IN_PROGRESS;
1175
- }
1176
-
1177
- /**
1178
- * 取消可能かどうか
1179
- */
1180
- public boolean canCancel() {
1181
- return this == DRAFT || this == IN_PROGRESS;
1182
- }
1183
- }
1184
- ```
1185
-
1186
- </details>
1187
-
1188
- <details>
1189
- <summary>入出庫履歴エンティティ</summary>
1190
-
1191
- ```java
1192
- // src/main/java/com/example/sales/domain/model/inventory/StockMovement.java
1193
- package com.example.sms.domain.model.inventory;
1194
-
1195
- import lombok.Builder;
1196
- import lombok.Data;
1197
-
1198
- import java.math.BigDecimal;
1199
- import java.time.LocalDateTime;
1200
-
1201
- @Data
1202
- @Builder
1203
- public class StockMovement {
1204
- private Integer id;
1205
- private String warehouseCode;
1206
- private String productCode;
1207
- private LocalDateTime movementDateTime;
1208
- private MovementType movementType;
1209
- private BigDecimal movementQuantity;
1210
- private BigDecimal beforeQuantity;
1211
- private BigDecimal afterQuantity;
1212
- private String documentNumber;
1213
- private String documentType;
1214
- private String movementReason;
1215
- private String locationCode;
1216
- private String lotNumber;
1217
- private LocalDateTime createdAt;
1218
- private String createdBy;
1219
-
1220
- // リレーション
1221
- private Warehouse warehouse;
1222
- private Product product;
1223
- }
1224
- ```
1225
-
1226
- </details>
1227
-
1228
- <details>
1229
- <summary>棚卸データ・棚卸明細エンティティ</summary>
1230
-
1231
- ```java
1232
- // src/main/java/com/example/sales/domain/model/inventory/Stocktaking.java
1233
- package com.example.sms.domain.model.inventory;
1234
-
1235
- import lombok.AllArgsConstructor;
1236
- import lombok.Builder;
1237
- import lombok.Data;
1238
- import lombok.NoArgsConstructor;
1239
-
1240
- import java.time.LocalDate;
1241
- import java.time.LocalDateTime;
1242
- import java.util.ArrayList;
1243
- import java.util.List;
1244
-
1245
- @Data
1246
- @Builder
1247
- @NoArgsConstructor
1248
- @AllArgsConstructor
1249
- public class Stocktaking {
1250
- private Integer id;
1251
- private String stocktakingNumber;
1252
- private String warehouseCode;
1253
- private LocalDate stocktakingDate;
1254
- private LocalDateTime startDateTime;
1255
- private LocalDateTime endDateTime;
1256
- private StocktakingStatus status;
1257
- private String remarks;
1258
- private LocalDateTime createdAt;
1259
- private String createdBy;
1260
- private LocalDateTime updatedAt;
1261
- private String updatedBy;
1262
-
1263
- // 楽観ロック用バージョン
1264
- @Builder.Default
1265
- private Integer version = 1;
1266
-
1267
- // リレーション
1268
- @Builder.Default
1269
- private List<StocktakingDetail> details = new ArrayList<>();
1270
-
1271
- /**
1272
- * 差異がある明細があるかどうか
1273
- */
1274
- public boolean hasDifference() {
1275
- return details.stream()
1276
- .anyMatch(d -> d.getDifferenceQuantity() != null &&
1277
- d.getDifferenceQuantity().signum() != 0);
1278
- }
1279
-
1280
- /**
1281
- * 全明細が調整済みかどうか
1282
- */
1283
- public boolean isAllAdjusted() {
1284
- return details.stream()
1285
- .allMatch(StocktakingDetail::getAdjustedFlag);
1286
- }
1287
- }
1288
- ```
1289
-
1290
- ```java
1291
- // src/main/java/com/example/sms/domain/model/inventory/StocktakingDetail.java
1292
- package com.example.sms.domain.model.inventory;
1293
-
1294
- import lombok.AllArgsConstructor;
1295
- import lombok.Builder;
1296
- import lombok.Data;
1297
- import lombok.NoArgsConstructor;
1298
-
1299
- import java.math.BigDecimal;
1300
- import java.time.LocalDateTime;
1301
-
1302
- @Data
1303
- @Builder
1304
- @NoArgsConstructor
1305
- @AllArgsConstructor
1306
- public class StocktakingDetail {
1307
- private Integer id;
1308
- private Integer stocktakingId;
1309
- private Integer lineNumber;
1310
- private String productCode;
1311
- private String locationCode;
1312
- private String lotNumber;
1313
- private BigDecimal bookQuantity;
1314
- private BigDecimal actualQuantity;
1315
- private BigDecimal differenceQuantity;
1316
- private String differenceReason;
1317
- @Builder.Default
1318
- private Boolean adjustedFlag = false;
1319
- private LocalDateTime createdAt;
1320
- private LocalDateTime updatedAt;
1321
-
1322
- /**
1323
- * 実棚数を入力
1324
- */
1325
- public void setActualQuantityAndCalculateDifference(BigDecimal actualQty) {
1326
- this.actualQuantity = actualQty;
1327
- this.differenceQuantity = actualQty.subtract(this.bookQuantity);
1328
- }
1329
-
1330
- /**
1331
- * 差異があるかどうか
1332
- */
1333
- public boolean hasDifference() {
1334
- return differenceQuantity != null && differenceQuantity.signum() != 0;
1335
- }
1336
- }
1337
- ```
1338
-
1339
- </details>
1340
-
1341
- ### 棚卸サービスの実装
1342
-
1343
- <details>
1344
- <summary>棚卸サービス</summary>
1345
-
1346
- ```java
1347
- // src/main/java/com/example/sales/application/service/StocktakingService.java
1348
- package com.example.sms.application.service;
1349
-
1350
- import com.example.sms.application.port.out.*;
1351
- import com.example.sms.domain.model.inventory.*;
1352
- import lombok.RequiredArgsConstructor;
1353
- import org.springframework.stereotype.Service;
1354
- import org.springframework.transaction.annotation.Transactional;
1355
-
1356
- import java.math.BigDecimal;
1357
- import java.time.LocalDate;
1358
- import java.time.LocalDateTime;
1359
- import java.time.format.DateTimeFormatter;
1360
- import java.util.ArrayList;
1361
- import java.util.List;
1362
- import java.util.Optional;
1363
-
1364
- @Service
1365
- @RequiredArgsConstructor
1366
- public class StocktakingService {
1367
-
1368
- private final StocktakingRepository stocktakingRepository;
1369
- private final InventoryRepository inventoryRepository;
1370
- private final StockMovementRepository stockMovementRepository;
1371
-
1372
- /**
1373
- * 棚卸を作成する
1374
- */
1375
- @Transactional
1376
- public Stocktaking createStocktaking(String warehouseCode, LocalDate stocktakingDate) {
1377
- String stocktakingNumber = generateStocktakingNumber(stocktakingDate);
1378
-
1379
- // 倉庫の在庫から棚卸明細を作成
1380
- List<Inventory> inventories = inventoryRepository.findByWarehouseCode(warehouseCode);
1381
-
1382
- List<StocktakingDetail> details = new ArrayList<>();
1383
- for (Inventory inv : inventories) {
1384
- StocktakingDetail detail = StocktakingDetail.builder()
1385
- .stocktakingNumber(stocktakingNumber)
1386
- .productCode(inv.getProductCode())
1387
- .locationCode(inv.getLocationCode())
1388
- .lotNumber(inv.getLotNumber())
1389
- .bookQuantity(inv.getCurrentQuantity())
1390
- .adjustedFlag(false)
1391
- .build();
1392
- details.add(detail);
1393
- }
1394
-
1395
- Stocktaking stocktaking = Stocktaking.builder()
1396
- .stocktakingNumber(stocktakingNumber)
1397
- .warehouseCode(warehouseCode)
1398
- .stocktakingDate(stocktakingDate)
1399
- .status(StocktakingStatus.DRAFT)
1400
- .details(details)
1401
- .build();
1402
-
1403
- stocktakingRepository.save(stocktaking);
1404
-
1405
- return stocktaking;
1406
- }
1407
-
1408
- /**
1409
- * 棚卸を開始する
1410
- */
1411
- @Transactional
1412
- public void startStocktaking(String stocktakingNumber) {
1413
- Stocktaking stocktaking = findByStocktakingNumber(stocktakingNumber)
1414
- .orElseThrow(() -> new IllegalArgumentException(
1415
- "棚卸が見つかりません: " + stocktakingNumber));
1416
-
1417
- if (!stocktaking.getStatus().canStart()) {
1418
- throw new IllegalStateException(
1419
- "この棚卸は開始できません。現在のステータス: " +
1420
- stocktaking.getStatus().getDisplayName());
1421
- }
1422
-
1423
- stocktakingRepository.updateStatus(
1424
- stocktakingNumber, StocktakingStatus.IN_PROGRESS);
1425
- stocktakingRepository.updateStartDateTime(
1426
- stocktakingNumber, LocalDateTime.now());
1427
- }
1428
-
1429
- /**
1430
- * 実棚数を入力する
1431
- */
1432
- @Transactional
1433
- public void inputActualQuantity(String stocktakingNumber, String productCode,
1434
- BigDecimal actualQuantity, String differenceReason) {
1435
- stocktakingRepository.updateActualQuantity(
1436
- stocktakingNumber, productCode, actualQuantity, differenceReason);
1437
- }
1438
-
1439
- /**
1440
- * 棚卸を確定する(在庫調整を実行)
1441
- */
1442
- @Transactional
1443
- public void confirmStocktaking(String stocktakingNumber) {
1444
- Stocktaking stocktaking = findByStocktakingNumber(stocktakingNumber)
1445
- .orElseThrow(() -> new IllegalArgumentException(
1446
- "棚卸が見つかりません: " + stocktakingNumber));
1447
-
1448
- if (!stocktaking.getStatus().canConfirm()) {
1449
- throw new IllegalStateException(
1450
- "この棚卸は確定できません。現在のステータス: " +
1451
- stocktaking.getStatus().getDisplayName());
1452
- }
1453
-
1454
- // 差異がある明細の在庫を調整
1455
- for (StocktakingDetail detail : stocktaking.getDetails()) {
1456
- if (detail.hasDifference()) {
1457
- adjustInventory(stocktaking.getWarehouseCode(), detail);
1458
- }
1459
- }
1460
-
1461
- stocktakingRepository.updateStatus(
1462
- stocktakingNumber, StocktakingStatus.CONFIRMED);
1463
- stocktakingRepository.updateEndDateTime(
1464
- stocktakingNumber, LocalDateTime.now());
1465
- }
1466
-
1467
- /**
1468
- * 在庫を調整する
1469
- */
1470
- private void adjustInventory(String warehouseCode, StocktakingDetail detail) {
1471
- Optional<Inventory> inventoryOpt = inventoryRepository
1472
- .findByWarehouseAndProduct(warehouseCode, detail.getProductCode());
1473
-
1474
- if (inventoryOpt.isEmpty()) {
1475
- return;
1476
- }
1477
-
1478
- Inventory inventory = inventoryOpt.get();
1479
- BigDecimal difference = detail.getDifferenceQuantity();
1480
- BigDecimal beforeQuantity = inventory.getCurrentQuantity();
1481
- BigDecimal afterQuantity = beforeQuantity.add(difference);
1482
-
1483
- // 在庫を更新
1484
- inventoryRepository.updateCurrentQuantity(
1485
- inventory.getId(), afterQuantity, detail.getDifferenceReason());
1486
-
1487
- // 入出庫履歴を記録
1488
- MovementType movementType = difference.signum() > 0
1489
- ? MovementType.ADJUSTMENT_PLUS
1490
- : MovementType.ADJUSTMENT_MINUS;
1491
-
1492
- StockMovement movement = StockMovement.builder()
1493
- .warehouseCode(warehouseCode)
1494
- .productCode(detail.getProductCode())
1495
- .movementDateTime(LocalDateTime.now())
1496
- .movementType(movementType)
1497
- .movementQuantity(difference.abs())
1498
- .beforeQuantity(beforeQuantity)
1499
- .afterQuantity(afterQuantity)
1500
- .movementReason(detail.getDifferenceReason())
1501
- .locationCode(detail.getLocationCode())
1502
- .lotNumber(detail.getLotNumber())
1503
- .build();
1504
-
1505
- stockMovementRepository.save(movement);
1506
-
1507
- // 調整済みフラグを更新
1508
- stocktakingRepository.markAsAdjusted(
1509
- detail.getStocktakingNumber(), detail.getProductCode());
1510
- }
1511
-
1512
- public Optional<Stocktaking> findByStocktakingNumber(String stocktakingNumber) {
1513
- return stocktakingRepository.findByStocktakingNumber(stocktakingNumber);
1514
- }
1515
-
1516
- private String generateStocktakingNumber(LocalDate stocktakingDate) {
1517
- String prefix = "ST-" + stocktakingDate.format(
1518
- DateTimeFormatter.ofPattern("yyyyMM")) + "-";
1519
- Optional<String> latestNumber = stocktakingRepository
1520
- .findLatestStocktakingNumber(prefix + "%");
1521
-
1522
- int sequence = 1;
1523
- if (latestNumber.isPresent()) {
1524
- String latest = latestNumber.get();
1525
- int currentSequence = Integer.parseInt(
1526
- latest.substring(latest.length() - 4));
1527
- sequence = currentSequence + 1;
1528
- }
1529
-
1530
- return prefix + String.format("%04d", sequence);
1531
- }
1532
- }
1533
- ```
1534
-
1535
- </details>
1536
-
1537
- ---
1538
-
1539
- ## 9.3 リレーションと楽観ロックの設計
1540
-
1541
- ### MyBatis ネストした ResultMap による関連付け
1542
-
1543
- 棚卸データは、棚卸(ヘッダ)→ 棚卸明細の2層構造を持ちます。N+1 問題を回避するため、JOIN クエリで一括取得します。
1544
-
1545
- ```plantuml
1546
- @startuml
1547
-
1548
- title N+1 問題の回避
1549
-
1550
- rectangle "N+1 問題(非効率)" #FFCCCC {
1551
- card "1. 棚卸ヘッダ取得\n SELECT * FROM 棚卸データ\n (1回)" as q1
1552
- card "2. 各棚卸の明細取得\n SELECT * FROM 棚卸明細データ\n WHERE 棚卸番号 = ?\n (N回)" as q2
1553
- }
1554
-
1555
- rectangle "解決策:JOINによる一括取得" #CCFFCC {
1556
- card "1回のJOINクエリで\n全データを取得\n SELECT st.*, std.*\n FROM 棚卸データ st\n LEFT JOIN 棚卸明細データ std ON ..." as sol
1557
- }
1558
-
1559
- q1 --> q2 : N件
1560
- note right of q2
1561
- 10件の棚卸があれば
1562
- 合計11回のクエリ発行
1563
- end note
1564
-
1565
- @enduml
1566
- ```
1567
-
1568
- <details>
1569
- <summary>棚卸データのリレーション設定(StocktakingMapper.xml)</summary>
1570
-
1571
- ```xml
1572
- <?xml version="1.0" encoding="UTF-8" ?>
1573
- <!DOCTYPE mapper
1574
- PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
1575
- "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
1576
- <mapper namespace="com.example.sms.infrastructure.persistence.mapper.StocktakingMapper">
1577
-
1578
- <!-- 棚卸(ヘッダ)の ResultMap -->
1579
- <resultMap id="StocktakingWithDetailsResultMap"
1580
- type="com.example.sms.domain.model.inventory.Stocktaking">
1581
- <id property="id" column="st_id"/>
1582
- <result property="stocktakingNumber" column="st_棚卸番号"/>
1583
- <result property="warehouseCode" column="st_倉庫コード"/>
1584
- <result property="stocktakingDate" column="st_棚卸日"/>
1585
- <result property="startDateTime" column="st_棚卸開始日時"/>
1586
- <result property="endDateTime" column="st_棚卸終了日時"/>
1587
- <result property="status" column="st_ステータス"
1588
- typeHandler="com.example.sms.infrastructure.persistence.typehandler.StocktakingStatusTypeHandler"/>
1589
- <result property="remarks" column="st_備考"/>
1590
- <result property="version" column="st_バージョン"/>
1591
- <result property="createdAt" column="st_作成日時"/>
1592
- <result property="updatedAt" column="st_更新日時"/>
1593
- <!-- 棚卸明細との1:N関連 -->
1594
- <collection property="details"
1595
- ofType="com.example.sms.domain.model.inventory.StocktakingDetail"
1596
- resultMap="StocktakingDetailNestedResultMap"/>
1597
- </resultMap>
1598
-
1599
- <!-- 棚卸明細のネスト ResultMap -->
1600
- <resultMap id="StocktakingDetailNestedResultMap"
1601
- type="com.example.sms.domain.model.inventory.StocktakingDetail">
1602
- <id property="id" column="std_id"/>
1603
- <result property="stocktakingId" column="std_棚卸ID"/>
1604
- <result property="lineNumber" column="std_棚卸行番号"/>
1605
- <result property="productCode" column="std_商品コード"/>
1606
- <result property="locationCode" column="std_ロケーションコード"/>
1607
- <result property="lotNumber" column="std_ロット番号"/>
1608
- <result property="bookQuantity" column="std_帳簿在庫数"/>
1609
- <result property="actualQuantity" column="std_実棚数"/>
1610
- <result property="differenceQuantity" column="std_差異数"/>
1611
- <result property="differenceReason" column="std_差異理由"/>
1612
- <result property="adjustedFlag" column="std_調整済フラグ"/>
1613
- </resultMap>
1614
-
1615
- <!-- JOIN による一括取得クエリ -->
1616
- <select id="findWithDetailsByStocktakingNumber"
1617
- resultMap="StocktakingWithDetailsResultMap">
1618
- SELECT
1619
- st."ID" AS st_id,
1620
- st."棚卸番号" AS st_棚卸番号,
1621
- st."倉庫コード" AS st_倉庫コード,
1622
- st."棚卸日" AS st_棚卸日,
1623
- st."棚卸開始日時" AS st_棚卸開始日時,
1624
- st."棚卸終了日時" AS st_棚卸終了日時,
1625
- st."ステータス" AS st_ステータス,
1626
- st."備考" AS st_備考,
1627
- st."バージョン" AS st_バージョン,
1628
- st."作成日時" AS st_作成日時,
1629
- st."更新日時" AS st_更新日時,
1630
- std."ID" AS std_id,
1631
- std."棚卸ID" AS std_棚卸ID,
1632
- std."棚卸行番号" AS std_棚卸行番号,
1633
- std."商品コード" AS std_商品コード,
1634
- std."ロケーションコード" AS std_ロケーションコード,
1635
- std."ロット番号" AS std_ロット番号,
1636
- std."帳簿在庫数" AS std_帳簿在庫数,
1637
- std."実棚数" AS std_実棚数,
1638
- std."差異数" AS std_差異数,
1639
- std."差異理由" AS std_差異理由,
1640
- std."調整済フラグ" AS std_調整済フラグ
1641
- FROM "棚卸データ" st
1642
- LEFT JOIN "棚卸明細データ" std
1643
- ON st."ID" = std."棚卸ID"
1644
- WHERE st."棚卸番号" = #{stocktakingNumber}
1645
- ORDER BY std."棚卸行番号"
1646
- </select>
1647
-
1648
- </mapper>
1649
- ```
1650
-
1651
- </details>
1652
-
1653
- ### リレーション設定のポイント
1654
-
1655
- | 設定項目 | 説明 |
1656
- |---------|------|
1657
- | `<collection>` | 1:N 関連のマッピング |
1658
- | `<id>` | 主キーの識別(MyBatis が重複排除に使用) |
1659
- | `resultMap` | ネストした ResultMap の参照 |
1660
- | エイリアス(AS) | カラム名の重複を避けるためのプレフィックス |
1661
- | `ORDER BY` | コレクションの順序を保証 |
1662
-
1663
- ### 楽観ロック(Optimistic Locking)の実装
1664
-
1665
- 複数ユーザーが同時に在庫データや棚卸データを編集する場合、データの整合性を保つために楽観ロックを実装します。
1666
-
1667
- ```plantuml
1668
- @startuml
1669
-
1670
- title 楽観ロックの動作シーケンス
1671
-
1672
- actor "担当者A" as userA
1673
- actor "担当者B" as userB
1674
- database "在庫テーブル" as db
1675
-
1676
- userA -> db : SELECT(version=1)
1677
- userB -> db : SELECT(version=1)
1678
- userA -> userA : 在庫数を編集
1679
- userB -> userB : 在庫数を編集
1680
- userA -> db : UPDATE WHERE version=1\n→ version=2 に更新
1681
- db -> userA : 更新成功(1行)
1682
- userB -> db : UPDATE WHERE version=1
1683
- db -> userB : 更新失敗(0行)\n楽観ロック例外
1684
- userB -> userB : 再読み込みを促す
1685
-
1686
- @enduml
1687
- ```
1688
-
1689
- ### バージョンカラムによる同時更新制御
1690
-
1691
- <details>
1692
- <summary>楽観ロック対応の MyBatis Mapper</summary>
1693
-
1694
- ```xml
1695
- <!-- 楽観ロック対応の在庫更新 -->
1696
- <update id="updateWithOptimisticLock"
1697
- parameterType="com.example.sms.domain.model.inventory.Inventory">
1698
- UPDATE "在庫データ"
1699
- SET
1700
- "現在庫数" = #{currentQuantity},
1701
- "引当数" = #{allocatedQuantity},
1702
- "発注残数" = #{orderedQuantity},
1703
- "最終入庫日" = #{lastReceiptDate},
1704
- "最終出庫日" = #{lastShipmentDate},
1705
- "更新日時" = CURRENT_TIMESTAMP,
1706
- "バージョン" = "バージョン" + 1
1707
- WHERE "ID" = #{id}
1708
- AND "バージョン" = #{version}
1709
- </update>
1710
-
1711
- <!-- 楽観ロック対応の棚卸更新 -->
1712
- <update id="updateStocktakingWithOptimisticLock"
1713
- parameterType="com.example.sms.domain.model.inventory.Stocktaking">
1714
- UPDATE "棚卸データ"
1715
- SET
1716
- "ステータス" = #{status,
1717
- typeHandler=com.example.sms.infrastructure.persistence.typehandler.StocktakingStatusTypeHandler}::棚卸ステータス,
1718
- "棚卸終了日時" = #{endDateTime},
1719
- "備考" = #{remarks},
1720
- "更新日時" = CURRENT_TIMESTAMP,
1721
- "バージョン" = "バージョン" + 1
1722
- WHERE "棚卸番号" = #{stocktakingNumber}
1723
- AND "バージョン" = #{version}
1724
- </update>
1725
- ```
1726
-
1727
- </details>
1728
-
1729
- <details>
1730
- <summary>Repository 実装:楽観ロック対応</summary>
1731
-
1732
- ```java
1733
- // src/main/java/com/example/sales/infrastructure/persistence/repository/InventoryRepositoryImpl.java
1734
- package com.example.sms.infrastructure.persistence.repository;
1735
-
1736
- import com.example.sms.application.port.out.InventoryRepository;
1737
- import com.example.sms.domain.exception.OptimisticLockException;
1738
- import com.example.sms.domain.model.inventory.*;
1739
- import com.example.sms.infrastructure.persistence.mapper.InventoryMapper;
1740
- import lombok.RequiredArgsConstructor;
1741
- import org.springframework.stereotype.Repository;
1742
- import org.springframework.transaction.annotation.Transactional;
1743
-
1744
- import java.math.BigDecimal;
1745
- import java.time.LocalDate;
1746
- import java.util.List;
1747
- import java.util.Optional;
1748
-
1749
- @Repository
1750
- @RequiredArgsConstructor
1751
- public class InventoryRepositoryImpl implements InventoryRepository {
1752
-
1753
- private final InventoryMapper inventoryMapper;
1754
-
1755
- @Override
1756
- public void save(Inventory inventory) {
1757
- inventoryMapper.insert(inventory);
1758
- }
1759
-
1760
- @Override
1761
- public Optional<Inventory> findByWarehouseAndProduct(String warehouseCode,
1762
- String productCode) {
1763
- List<Inventory> inventories = inventoryMapper
1764
- .findByWarehouseAndProduct(warehouseCode, productCode);
1765
- return inventories.isEmpty() ? Optional.empty() : Optional.of(inventories.get(0));
1766
- }
1767
-
1768
- @Override
1769
- public List<Inventory> findByWarehouseCode(String warehouseCode) {
1770
- return inventoryMapper.findByWarehouseCode(warehouseCode);
1771
- }
1772
-
1773
- @Override
1774
- public List<Inventory> findByProductCode(String productCode) {
1775
- return inventoryMapper.findByProductCode(productCode);
1776
- }
1777
-
1778
- /**
1779
- * 楽観ロック対応の更新
1780
- * @throws OptimisticLockException 他のユーザーによって更新された場合
1781
- */
1782
- @Override
1783
- @Transactional
1784
- public void update(Inventory inventory) {
1785
- int updatedCount = inventoryMapper.updateWithOptimisticLock(inventory);
1786
-
1787
- if (updatedCount == 0) {
1788
- Integer currentVersion = inventoryMapper.findVersionById(inventory.getId());
1789
-
1790
- if (currentVersion == null) {
1791
- throw new OptimisticLockException("在庫", inventory.getId().toString());
1792
- } else {
1793
- throw new OptimisticLockException(
1794
- "在庫",
1795
- inventory.getId().toString(),
1796
- inventory.getVersion(),
1797
- currentVersion
1798
- );
1799
- }
1800
- }
1801
- }
1802
-
1803
- @Override
1804
- public void allocate(Integer id, BigDecimal quantity) {
1805
- int updated = inventoryMapper.allocate(id, quantity);
1806
- if (updated == 0) {
1807
- throw new IllegalStateException("引当に失敗しました。有効在庫不足の可能性があります。");
1808
- }
1809
- }
1810
-
1811
- @Override
1812
- public void deallocate(Integer id, BigDecimal quantity) {
1813
- int updated = inventoryMapper.deallocate(id, quantity);
1814
- if (updated == 0) {
1815
- throw new IllegalStateException("引当解除に失敗しました。");
1816
- }
1817
- }
1818
-
1819
- @Override
1820
- public void receive(Integer id, BigDecimal quantity, LocalDate receiptDate) {
1821
- inventoryMapper.receive(id, quantity, receiptDate);
1822
- }
1823
-
1824
- @Override
1825
- public void ship(Integer id, BigDecimal quantity, LocalDate shipmentDate) {
1826
- int updated = inventoryMapper.ship(id, quantity, shipmentDate);
1827
- if (updated == 0) {
1828
- throw new IllegalStateException("出庫に失敗しました。在庫不足の可能性があります。");
1829
- }
1830
- }
1831
-
1832
- @Override
1833
- public void updateCurrentQuantity(Integer id, BigDecimal newQuantity, String reason) {
1834
- inventoryMapper.updateCurrentQuantity(id, newQuantity);
1835
- }
1836
- }
1837
- ```
1838
-
1839
- </details>
1840
-
1841
- ### 楽観ロックのテスト
1842
-
1843
- <details>
1844
- <summary>楽観ロックのテストコード</summary>
1845
-
1846
- ```java
1847
- @Nested
1848
- @DisplayName("楽観ロックの在庫更新テスト")
1849
- class OptimisticLockInventoryTest {
1850
-
1851
- @Test
1852
- @DisplayName("正しいバージョンで更新できる")
1853
- void shouldUpdateWithCorrectVersion() {
1854
- // Given: 在庫を登録
1855
- var inventory = createTestInventory("WH001", "PROD-001");
1856
- inventoryRepository.save(inventory);
1857
-
1858
- // When: バージョン1で更新
1859
- var saved = inventoryRepository
1860
- .findByWarehouseAndProduct("WH001", "PROD-001").orElseThrow();
1861
- assertThat(saved.getVersion()).isEqualTo(1);
1862
- saved.setCurrentQuantity(new BigDecimal("150"));
1863
- inventoryRepository.update(saved);
1864
-
1865
- // Then: バージョンが2に増加
1866
- var updated = inventoryRepository
1867
- .findByWarehouseAndProduct("WH001", "PROD-001").orElseThrow();
1868
- assertThat(updated.getVersion()).isEqualTo(2);
1869
- assertThat(updated.getCurrentQuantity()).isEqualByComparingTo(new BigDecimal("150"));
1870
- }
1871
-
1872
- @Test
1873
- @DisplayName("古いバージョンで更新すると楽観ロック例外がスローされる")
1874
- void shouldThrowOptimisticLockExceptionWithOldVersion() {
1875
- // Given: 在庫を登録して、別のセッションで更新
1876
- var inventory = createTestInventory("WH001", "PROD-002");
1877
- inventoryRepository.save(inventory);
1878
-
1879
- var sessionA = inventoryRepository
1880
- .findByWarehouseAndProduct("WH001", "PROD-002").orElseThrow();
1881
- var sessionB = inventoryRepository
1882
- .findByWarehouseAndProduct("WH001", "PROD-002").orElseThrow();
1883
-
1884
- // セッションAで更新
1885
- sessionA.setCurrentQuantity(new BigDecimal("200"));
1886
- inventoryRepository.update(sessionA);
1887
-
1888
- // When/Then: セッションBで更新すると例外
1889
- sessionB.setCurrentQuantity(new BigDecimal("300"));
1890
- assertThatThrownBy(() -> inventoryRepository.update(sessionB))
1891
- .isInstanceOf(OptimisticLockException.class)
1892
- .hasMessageContaining("他のユーザーによって更新されました");
1893
- }
1894
- }
1895
- ```
1896
-
1897
- </details>
1898
-
1899
- ### 楽観ロックのベストプラクティス
1900
-
1901
- | ポイント | 説明 |
1902
- |---------|------|
1903
- | **バージョンカラム** | INTEGER 型で十分(オーバーフローは現実的に発生しない) |
1904
- | **WHERE 条件** | 必ず `AND "バージョン" = #{version}` を含める |
1905
- | **インクリメント** | `"バージョン" = "バージョン" + 1` でアトミックに更新 |
1906
- | **例外処理** | 更新件数が0の場合は楽観ロック例外をスロー |
1907
- | **エラーメッセージ** | ユーザーにわかりやすいメッセージで再読み込みを促す |
1908
-
1909
- ---
1910
-
1911
- ## 第9章のまとめ
1912
-
1913
- 本章では、在庫管理の核心部分である在庫情報管理、入出庫履歴、棚卸処理について学びました。
1914
-
1915
- ### 学んだこと
1916
-
1917
- 1. **在庫数量の種類**
1918
- - 現在庫数、引当数、有効在庫数、発注残数の関係
1919
-
1920
- 2. **倉庫・ロケーション管理**
1921
- - 倉庫マスタの設計
1922
- - ロケーションによる細かな在庫管理
1923
-
1924
- 3. **入出庫履歴の設計**
1925
- - 移動区分による分類
1926
- - トレーサビリティの確保
1927
-
1928
- 4. **棚卸業務の設計**
1929
- - 帳簿在庫と実棚の差異管理
1930
- - 在庫調整処理
1931
-
1932
- 5. **リレーションと楽観ロック**
1933
- - N+1 問題の回避
1934
- - 同時更新の競合制御
1935
-
1936
- ### 在庫管理の ER 図(全体像)
1937
-
1938
- ```plantuml
1939
- @startuml
1940
-
1941
- title 在庫管理のER図
1942
-
1943
- entity 倉庫マスタ {
1944
- 倉庫コード <<PK>>
1945
- --
1946
- 倉庫名
1947
- 倉庫区分
1948
- }
1949
-
1950
- entity ロケーションマスタ {
1951
- ロケーションコード <<PK>>
1952
- --
1953
- 倉庫コード <<FK>>
1954
- 棚番
1955
- ゾーン
1956
- }
1957
-
1958
- entity 商品マスタ {
1959
- 商品コード <<PK>>
1960
- --
1961
- 商品名
1962
- }
1963
-
1964
- entity 在庫データ {
1965
- ID <<PK>>
1966
- --
1967
- 倉庫コード <<FK>>
1968
- 商品コード <<FK>>
1969
- ロケーションコード <<FK>>
1970
- 現在庫数
1971
- 引当数
1972
- 発注残数
1973
- ロット番号
1974
- バージョン
1975
- }
1976
-
1977
- entity 入出庫履歴データ {
1978
- ID <<PK>>
1979
- --
1980
- 倉庫コード <<FK>>
1981
- 商品コード <<FK>>
1982
- 移動日時
1983
- 移動区分
1984
- 移動数量
1985
- 移動前在庫数
1986
- 移動後在庫数
1987
- 伝票番号
1988
- }
1989
-
1990
- entity 棚卸データ {
1991
- ID <<PK>>
1992
- --
1993
- 棚卸番号
1994
- 倉庫コード <<FK>>
1995
- 棚卸日
1996
- ステータス
1997
- バージョン
1998
- }
1999
-
2000
- entity 棚卸明細データ {
2001
- ID <<PK>>
2002
- --
2003
- 棚卸ID <<FK>>
2004
- 棚卸行番号
2005
- 商品コード <<FK>>
2006
- 帳簿在庫数
2007
- 実棚数
2008
- 差異数
2009
- 調整済フラグ
2010
- }
2011
-
2012
- 倉庫マスタ ||--o{ ロケーションマスタ
2013
- 倉庫マスタ ||--o{ 在庫データ
2014
- 商品マスタ ||--o{ 在庫データ
2015
- ロケーションマスタ ||--o{ 在庫データ
2016
- 倉庫マスタ ||--o{ 入出庫履歴データ
2017
- 商品マスタ ||--o{ 入出庫履歴データ
2018
- 倉庫マスタ ||--o{ 棚卸データ
2019
- 棚卸データ ||--o{ 棚卸明細データ
2020
- 商品マスタ ||--o{ 棚卸明細データ
2021
-
2022
- @enduml
2023
- ```
2024
-
2025
- ### 次章の予告
2026
-
2027
- 第10章では、債務管理の設計に進みます。買掛金残高の管理、支払処理のデータベース設計と実装を行います。
2028
-
2029
- ---
2030
-
2031
- [← 第8章:調達管理の設計](./chapter08.md) | [第10章:債務管理の設計 →](./chapter10.md)
1
+ # 第9章:在庫管理の設計
2
+
3
+ 販売管理システムにおいて、在庫管理は商品の物理的な状態を追跡し、適正在庫を維持するための重要な機能です。本章では、倉庫別・商品別の在庫管理、引当処理、入出庫履歴、棚卸業務のデータベース設計と実装を行います。
4
+
5
+ ## 在庫管理の全体像
6
+
7
+ 在庫管理は「入庫 → 在庫保管 → 引当 → 出庫」という一連のフローで構成されます。
8
+
9
+ ```plantuml
10
+ @startuml
11
+
12
+ title 在庫管理の業務フロー
13
+
14
+ |倉庫部門|
15
+ start
16
+ :入荷・入庫処理;
17
+ note right: 仕入・倉庫移動などで\n在庫が増加
18
+
19
+ :在庫情報更新;
20
+ fork
21
+ :現在庫数 増加;
22
+ fork again
23
+ :入出庫履歴 記録;
24
+ fork end
25
+
26
+ |営業部門|
27
+ :受注登録;
28
+ :在庫引当;
29
+ note right: 受注時に引当数を\n計上して確保
30
+
31
+ |倉庫部門|
32
+ :出荷指示受領;
33
+ :ピッキング;
34
+ :出荷処理;
35
+
36
+ :在庫情報更新;
37
+ fork
38
+ :現在庫数 減少;
39
+ fork again
40
+ :引当数 減少;
41
+ fork again
42
+ :入出庫履歴 記録;
43
+ fork end
44
+
45
+ |経理部門|
46
+ if (棚卸時期?) then (yes)
47
+ :棚卸指示;
48
+ |倉庫部門|
49
+ :棚卸表発行;
50
+ :実棚カウント;
51
+ :差異入力;
52
+ if (差異あり?) then (yes)
53
+ :在庫調整処理;
54
+ else (no)
55
+ endif
56
+ :棚卸完了;
57
+ endif
58
+
59
+ stop
60
+
61
+ @enduml
62
+ ```
63
+
64
+ ### 在庫管理で扱うデータ
65
+
66
+ | データ | 説明 |
67
+ |-------|------|
68
+ | **在庫データ** | 倉庫×商品の現在の在庫状態 |
69
+ | **入出庫履歴** | 在庫の増減を時系列で記録 |
70
+ | **棚卸データ** | 棚卸作業の管理情報 |
71
+ | **棚卸明細** | 商品ごとの帳簿在庫と実棚数 |
72
+
73
+ ---
74
+
75
+ ## 9.1 在庫情報の DB 設計
76
+
77
+ ### 在庫数量の種類
78
+
79
+ 在庫管理では「倉庫コード × 商品コード」の組み合わせで在庫を管理します。
80
+
81
+ ```plantuml
82
+ @startuml
83
+
84
+ title 在庫数量の関係
85
+
86
+ rectangle "在庫数量の種類" {
87
+ card "現在庫数\n(物理的な数量)" as current
88
+ card "引当数\n(確保済み数量)" as allocated
89
+ card "有効在庫数\n(利用可能数量)" as available
90
+ card "発注残数\n(未入荷数量)" as ordered
91
+ card "予定在庫数\n(将来の見込み)" as projected
92
+ }
93
+
94
+ current --> available : "- 引当数"
95
+ available --> projected : "+ 発注残数"
96
+
97
+ note bottom of current
98
+ 入庫数 - 出庫数
99
+ の累計
100
+ end note
101
+
102
+ note bottom of allocated
103
+ 受注時に確保
104
+ 出荷で解消
105
+ end note
106
+
107
+ note bottom of available
108
+ 現在庫数 - 引当数
109
+ 実際に出荷可能な数量
110
+ end note
111
+
112
+ note bottom of projected
113
+ 有効在庫数 + 発注残数
114
+ 将来的に利用可能な見込み
115
+ end note
116
+
117
+ @enduml
118
+ ```
119
+
120
+ | 数量種別 | 説明 | 計算式 |
121
+ |---------|------|--------|
122
+ | **現在庫数** | 物理的に倉庫にある数量 | 入庫数 - 出庫数 |
123
+ | **引当数** | 受注等で確保済みの数量 | 受注引当の合計 |
124
+ | **有効在庫数** | 実際に利用可能な数量 | 現在庫数 - 引当数 |
125
+ | **発注残数** | 発注済みで未入荷の数量 | 発注数 - 入荷数 |
126
+ | **予定在庫数** | 将来的に利用可能な見込み | 有効在庫数 + 発注残数 |
127
+
128
+ ### 倉庫マスタの設計
129
+
130
+ ```plantuml
131
+ @startuml
132
+
133
+ title 倉庫・ロケーション関連テーブル
134
+
135
+ entity 倉庫マスタ {
136
+ 倉庫コード <<PK>>
137
+ --
138
+ 倉庫名
139
+ 倉庫区分
140
+ 郵便番号
141
+ 住所
142
+ 電話番号
143
+ 有効フラグ
144
+ 作成日時
145
+ 更新日時
146
+ }
147
+
148
+ entity ロケーションマスタ {
149
+ ロケーションコード <<PK>>
150
+ --
151
+ 倉庫コード <<FK>>
152
+ 棚番
153
+ ゾーン
154
+ 通路
155
+ ラック
156
+
157
+ 間口
158
+ 有効フラグ
159
+ 作成日時
160
+ 更新日時
161
+ }
162
+
163
+ entity 在庫データ {
164
+ ID <<PK>>
165
+ --
166
+ 倉庫コード <<FK>>
167
+ 商品コード <<FK>>
168
+ ロケーションコード <<FK>>
169
+ 現在庫数
170
+ 引当数
171
+ 発注残数
172
+ 最終入庫日
173
+ 最終出庫日
174
+ ロット番号
175
+ シリアル番号
176
+ 有効期限
177
+ バージョン
178
+ 作成日時
179
+ 更新日時
180
+ }
181
+
182
+ entity 商品マスタ {
183
+ 商品コード <<PK>>
184
+ --
185
+ 商品名
186
+ 単価
187
+ }
188
+
189
+ 倉庫マスタ ||--o{ ロケーションマスタ : 倉庫コード
190
+ 倉庫マスタ ||--o{ 在庫データ : 倉庫コード
191
+ 商品マスタ ||--o{ 在庫データ : 商品コード
192
+ ロケーションマスタ ||--o{ 在庫データ : ロケーションコード
193
+
194
+ @enduml
195
+ ```
196
+
197
+ ### 倉庫区分の定義
198
+
199
+ ```plantuml
200
+ @startuml
201
+
202
+ title 倉庫区分
203
+
204
+ rectangle "自社倉庫" #CCFFCC {
205
+ card "本社倉庫" as own1
206
+ card "支店倉庫" as own2
207
+ }
208
+
209
+ rectangle "外部倉庫" #CCCCFF {
210
+ card "営業倉庫\n(3PL)" as ext1
211
+ card "保管倉庫" as ext2
212
+ }
213
+
214
+ rectangle "仮想倉庫" #FFFFCC {
215
+ card "預け在庫" as virt1
216
+ card "輸送中在庫" as virt2
217
+ }
218
+
219
+ note bottom of own1
220
+ 自社で管理
221
+ 入出庫を直接制御
222
+ end note
223
+
224
+ note bottom of ext1
225
+ 外部委託
226
+ 入出庫依頼が必要
227
+ end note
228
+
229
+ note bottom of virt1
230
+ 物理的に存在しない
231
+ 管理上の区分
232
+ end note
233
+
234
+ @enduml
235
+ ```
236
+
237
+ | 倉庫区分 | 説明 |
238
+ |---------|------|
239
+ | **自社** | 自社所有・管理の倉庫 |
240
+ | **外部** | 外部委託(3PL)の倉庫 |
241
+ | **仮想** | 輸送中、預け在庫など |
242
+
243
+ ### マイグレーション:倉庫・在庫関連テーブルの作成
244
+
245
+ <details>
246
+ <summary>V015__create_inventory_tables.sql</summary>
247
+
248
+ ```sql
249
+ -- src/main/resources/db/migration/V015__create_inventory_tables.sql
250
+
251
+ -- 倉庫区分
252
+ CREATE TYPE 倉庫区分 AS ENUM ('自社', '外部', '仮想');
253
+
254
+ -- 倉庫マスタに倉庫区分カラムを追加
255
+ -- (V013で作成済みの倉庫マスタにカラム追加)
256
+ ALTER TABLE "倉庫マスタ" ADD COLUMN "倉庫区分" 倉庫区分 DEFAULT '自社' NOT NULL;
257
+
258
+ -- ロケーションマスタ
259
+ CREATE TABLE "ロケーションマスタ" (
260
+ "ロケーションコード" VARCHAR(20) PRIMARY KEY,
261
+ "倉庫コード" VARCHAR(20) NOT NULL,
262
+ "棚番" VARCHAR(20) NOT NULL,
263
+ "ゾーン" VARCHAR(10),
264
+ "通路" VARCHAR(10),
265
+ "ラック" VARCHAR(10),
266
+ "段" VARCHAR(10),
267
+ "間口" VARCHAR(10),
268
+ "有効フラグ" BOOLEAN DEFAULT true NOT NULL,
269
+ "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
270
+ "作成者" VARCHAR(50),
271
+ "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
272
+ "更新者" VARCHAR(50),
273
+ CONSTRAINT "fk_ロケーション_倉庫"
274
+ FOREIGN KEY ("倉庫コード") REFERENCES "倉庫マスタ"("倉庫コード"),
275
+ UNIQUE ("倉庫コード", "棚番")
276
+ );
277
+
278
+ -- 在庫データ(倉庫 × 商品)
279
+ CREATE TABLE "在庫データ" (
280
+ "ID" SERIAL PRIMARY KEY,
281
+ "倉庫コード" VARCHAR(20) NOT NULL,
282
+ "商品コード" VARCHAR(20) NOT NULL,
283
+ "ロケーションコード" VARCHAR(20),
284
+ "現在庫数" DECIMAL(15, 2) DEFAULT 0 NOT NULL,
285
+ "引当数" DECIMAL(15, 2) DEFAULT 0 NOT NULL,
286
+ "発注残数" DECIMAL(15, 2) DEFAULT 0 NOT NULL,
287
+ "最終入庫日" DATE,
288
+ "最終出庫日" DATE,
289
+ "ロット番号" VARCHAR(50),
290
+ "シリアル番号" VARCHAR(50),
291
+ "有効期限" DATE,
292
+ "バージョン" INTEGER DEFAULT 1 NOT NULL,
293
+ "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
294
+ "作成者" VARCHAR(50),
295
+ "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
296
+ "更新者" VARCHAR(50),
297
+ CONSTRAINT "fk_在庫_倉庫"
298
+ FOREIGN KEY ("倉庫コード") REFERENCES "倉庫マスタ"("倉庫コード"),
299
+ CONSTRAINT "fk_在庫_商品"
300
+ FOREIGN KEY ("商品コード") REFERENCES "商品マスタ"("商品コード"),
301
+ CONSTRAINT "fk_在庫_ロケーション"
302
+ FOREIGN KEY ("ロケーションコード") REFERENCES "ロケーションマスタ"("ロケーションコード"),
303
+ UNIQUE ("倉庫コード", "商品コード", "ロケーションコード", "ロット番号")
304
+ );
305
+
306
+ -- 有効在庫数を計算するビュー
307
+ CREATE VIEW "有効在庫ビュー" AS
308
+ SELECT
309
+ "倉庫コード",
310
+ "商品コード",
311
+ "ロケーションコード",
312
+ "現在庫数",
313
+ "引当数",
314
+ "現在庫数" - "引当数" AS "有効在庫数",
315
+ "発注残数",
316
+ "現在庫数" - "引当数" + "発注残数" AS "予定在庫数"
317
+ FROM "在庫データ";
318
+
319
+ -- インデックス
320
+ CREATE INDEX "idx_ロケーション_倉庫" ON "ロケーションマスタ"("倉庫コード");
321
+ CREATE INDEX "idx_在庫_倉庫コード" ON "在庫データ"("倉庫コード");
322
+ CREATE INDEX "idx_在庫_商品コード" ON "在庫データ"("商品コード");
323
+ CREATE INDEX "idx_在庫_ロケーション" ON "在庫データ"("ロケーションコード");
324
+ CREATE INDEX "idx_在庫_ロット番号" ON "在庫データ"("ロット番号");
325
+ CREATE INDEX "idx_在庫_有効期限" ON "在庫データ"("有効期限");
326
+
327
+ -- テーブルコメント
328
+ COMMENT ON TABLE "倉庫マスタ" IS '倉庫情報を管理するマスタテーブル';
329
+ COMMENT ON TABLE "ロケーションマスタ" IS '倉庫内の棚・ロケーション情報を管理するテーブル';
330
+ COMMENT ON TABLE "在庫データ" IS '倉庫×商品の在庫情報を管理するテーブル';
331
+ COMMENT ON COLUMN "在庫データ"."バージョン" IS '楽観ロック用バージョン番号';
332
+ ```
333
+
334
+ </details>
335
+
336
+ ### 倉庫・在庫エンティティの実装
337
+
338
+ <details>
339
+ <summary>倉庫区分 ENUM</summary>
340
+
341
+ ```java
342
+ // src/main/java/com/example/sales/domain/model/inventory/WarehouseType.java
343
+ package com.example.sms.domain.model.inventory;
344
+
345
+ import lombok.Getter;
346
+ import lombok.RequiredArgsConstructor;
347
+
348
+ @Getter
349
+ @RequiredArgsConstructor
350
+ public enum WarehouseType {
351
+ OWN("自社"),
352
+ EXTERNAL("外部"),
353
+ VIRTUAL("仮想");
354
+
355
+ private final String displayName;
356
+
357
+ public static WarehouseType fromDisplayName(String displayName) {
358
+ for (WarehouseType type : values()) {
359
+ if (type.displayName.equals(displayName)) {
360
+ return type;
361
+ }
362
+ }
363
+ throw new IllegalArgumentException("Unknown warehouse type: " + displayName);
364
+ }
365
+ }
366
+ ```
367
+
368
+ </details>
369
+
370
+ <details>
371
+ <summary>倉庫マスタエンティティ</summary>
372
+
373
+ ```java
374
+ // src/main/java/com/example/sales/domain/model/inventory/Warehouse.java
375
+ package com.example.sms.domain.model.inventory;
376
+
377
+ import lombok.Builder;
378
+ import lombok.Data;
379
+
380
+ import java.time.LocalDateTime;
381
+
382
+ @Data
383
+ @Builder
384
+ public class Warehouse {
385
+ private String warehouseCode;
386
+ private String warehouseName;
387
+ private WarehouseType warehouseType;
388
+ private String postalCode;
389
+ private String address;
390
+ private String phoneNumber;
391
+ private Boolean activeFlag;
392
+ private LocalDateTime createdAt;
393
+ private String createdBy;
394
+ private LocalDateTime updatedAt;
395
+ private String updatedBy;
396
+
397
+ /**
398
+ * 自社倉庫かどうか
399
+ */
400
+ public boolean isOwnWarehouse() {
401
+ return this.warehouseType == WarehouseType.OWN;
402
+ }
403
+
404
+ /**
405
+ * 仮想倉庫かどうか
406
+ */
407
+ public boolean isVirtualWarehouse() {
408
+ return this.warehouseType == WarehouseType.VIRTUAL;
409
+ }
410
+ }
411
+ ```
412
+
413
+ </details>
414
+
415
+ <details>
416
+ <summary>ロケーションマスタエンティティ</summary>
417
+
418
+ ```java
419
+ // src/main/java/com/example/sales/domain/model/inventory/Location.java
420
+ package com.example.sms.domain.model.inventory;
421
+
422
+ import lombok.Builder;
423
+ import lombok.Data;
424
+
425
+ import java.time.LocalDateTime;
426
+
427
+ @Data
428
+ @Builder
429
+ public class Location {
430
+ private String locationCode;
431
+ private String warehouseCode;
432
+ private String shelfNumber;
433
+ private String zone;
434
+ private String aisle;
435
+ private String rack;
436
+ private String level;
437
+ private String bay;
438
+ private Boolean activeFlag;
439
+ private LocalDateTime createdAt;
440
+ private String createdBy;
441
+ private LocalDateTime updatedAt;
442
+ private String updatedBy;
443
+
444
+ // リレーション
445
+ private Warehouse warehouse;
446
+
447
+ /**
448
+ * ロケーションの完全表記を取得
449
+ * 例: A-01-02-03-04
450
+ */
451
+ public String getFullAddress() {
452
+ StringBuilder sb = new StringBuilder();
453
+ if (zone != null) sb.append(zone).append("-");
454
+ if (aisle != null) sb.append(aisle).append("-");
455
+ if (rack != null) sb.append(rack).append("-");
456
+ if (level != null) sb.append(level).append("-");
457
+ if (bay != null) sb.append(bay);
458
+ return sb.toString().replaceAll("-$", "");
459
+ }
460
+ }
461
+ ```
462
+
463
+ </details>
464
+
465
+ <details>
466
+ <summary>在庫エンティティ</summary>
467
+
468
+ ```java
469
+ // src/main/java/com/example/sales/domain/model/inventory/Inventory.java
470
+ package com.example.sms.domain.model.inventory;
471
+
472
+ import com.example.sms.domain.model.product.Product;
473
+ import lombok.Builder;
474
+ import lombok.Data;
475
+
476
+ import java.math.BigDecimal;
477
+ import java.time.LocalDate;
478
+ import java.time.LocalDateTime;
479
+
480
+ @Data
481
+ @Builder
482
+ public class Inventory {
483
+ private Integer id;
484
+ private String warehouseCode;
485
+ private String productCode;
486
+ private String locationCode;
487
+ private BigDecimal currentQuantity;
488
+ private BigDecimal allocatedQuantity;
489
+ private BigDecimal orderedQuantity;
490
+ private LocalDate lastReceiptDate;
491
+ private LocalDate lastShipmentDate;
492
+ private String lotNumber;
493
+ private String serialNumber;
494
+ private LocalDate expirationDate;
495
+ private LocalDateTime createdAt;
496
+ private String createdBy;
497
+ private LocalDateTime updatedAt;
498
+ private String updatedBy;
499
+
500
+ // 楽観ロック用バージョン
501
+ @Builder.Default
502
+ private Integer version = 1;
503
+
504
+ // リレーション
505
+ private Warehouse warehouse;
506
+ private Product product;
507
+ private Location location;
508
+
509
+ /**
510
+ * 有効在庫数を計算
511
+ */
512
+ public BigDecimal getAvailableQuantity() {
513
+ return currentQuantity.subtract(allocatedQuantity);
514
+ }
515
+
516
+ /**
517
+ * 予定在庫数を計算
518
+ */
519
+ public BigDecimal getProjectedQuantity() {
520
+ return currentQuantity.subtract(allocatedQuantity).add(orderedQuantity);
521
+ }
522
+
523
+ /**
524
+ * 引当可能かどうかを判定
525
+ */
526
+ public boolean canAllocate(BigDecimal quantity) {
527
+ return getAvailableQuantity().compareTo(quantity) >= 0;
528
+ }
529
+
530
+ /**
531
+ * 在庫を引当
532
+ */
533
+ public void allocate(BigDecimal quantity) {
534
+ if (!canAllocate(quantity)) {
535
+ throw new IllegalStateException(
536
+ "有効在庫数が不足しています。有効在庫: " + getAvailableQuantity() +
537
+ ", 引当要求: " + quantity);
538
+ }
539
+ this.allocatedQuantity = this.allocatedQuantity.add(quantity);
540
+ }
541
+
542
+ /**
543
+ * 引当を解除
544
+ */
545
+ public void deallocate(BigDecimal quantity) {
546
+ if (this.allocatedQuantity.compareTo(quantity) < 0) {
547
+ throw new IllegalStateException(
548
+ "引当数を超える解除はできません。引当数: " + this.allocatedQuantity +
549
+ ", 解除要求: " + quantity);
550
+ }
551
+ this.allocatedQuantity = this.allocatedQuantity.subtract(quantity);
552
+ }
553
+
554
+ /**
555
+ * 入庫処理
556
+ */
557
+ public void receive(BigDecimal quantity) {
558
+ this.currentQuantity = this.currentQuantity.add(quantity);
559
+ this.lastReceiptDate = LocalDate.now();
560
+ }
561
+
562
+ /**
563
+ * 出庫処理
564
+ */
565
+ public void ship(BigDecimal quantity) {
566
+ if (this.currentQuantity.compareTo(quantity) < 0) {
567
+ throw new IllegalStateException(
568
+ "現在庫数が不足しています。現在庫: " + this.currentQuantity +
569
+ ", 出庫要求: " + quantity);
570
+ }
571
+ this.currentQuantity = this.currentQuantity.subtract(quantity);
572
+ this.lastShipmentDate = LocalDate.now();
573
+ }
574
+
575
+ /**
576
+ * 有効期限切れかどうか
577
+ */
578
+ public boolean isExpired() {
579
+ if (expirationDate == null) {
580
+ return false;
581
+ }
582
+ return expirationDate.isBefore(LocalDate.now());
583
+ }
584
+ }
585
+ ```
586
+
587
+ </details>
588
+
589
+ ### 在庫サービスの実装
590
+
591
+ <details>
592
+ <summary>在庫サービス</summary>
593
+
594
+ ```java
595
+ // src/main/java/com/example/sales/application/service/InventoryService.java
596
+ package com.example.sms.application.service;
597
+
598
+ import com.example.sms.application.port.out.InventoryRepository;
599
+ import com.example.sms.domain.model.inventory.*;
600
+ import lombok.RequiredArgsConstructor;
601
+ import org.springframework.stereotype.Service;
602
+ import org.springframework.transaction.annotation.Transactional;
603
+
604
+ import java.math.BigDecimal;
605
+ import java.time.LocalDate;
606
+ import java.util.List;
607
+ import java.util.Optional;
608
+
609
+ @Service
610
+ @RequiredArgsConstructor
611
+ public class InventoryService {
612
+
613
+ private final InventoryRepository inventoryRepository;
614
+
615
+ /**
616
+ * 商品別の在庫サマリを取得
617
+ */
618
+ public InventorySummary getProductInventorySummary(String productCode) {
619
+ List<Inventory> inventories = inventoryRepository.findByProductCode(productCode);
620
+
621
+ BigDecimal totalCurrent = BigDecimal.ZERO;
622
+ BigDecimal totalAllocated = BigDecimal.ZERO;
623
+ BigDecimal totalOrdered = BigDecimal.ZERO;
624
+
625
+ for (Inventory inv : inventories) {
626
+ totalCurrent = totalCurrent.add(inv.getCurrentQuantity());
627
+ totalAllocated = totalAllocated.add(inv.getAllocatedQuantity());
628
+ totalOrdered = totalOrdered.add(inv.getOrderedQuantity());
629
+ }
630
+
631
+ return InventorySummary.builder()
632
+ .productCode(productCode)
633
+ .totalCurrentQuantity(totalCurrent)
634
+ .totalAllocatedQuantity(totalAllocated)
635
+ .totalAvailableQuantity(totalCurrent.subtract(totalAllocated))
636
+ .totalOrderedQuantity(totalOrdered)
637
+ .warehouseDetails(inventories)
638
+ .build();
639
+ }
640
+
641
+ /**
642
+ * 倉庫別の在庫一覧を取得
643
+ */
644
+ public List<Inventory> getInventoryByWarehouse(String warehouseCode) {
645
+ return inventoryRepository.findByWarehouseCode(warehouseCode);
646
+ }
647
+
648
+ /**
649
+ * 在庫を引当
650
+ */
651
+ @Transactional
652
+ public AllocationResult allocate(String warehouseCode, String productCode,
653
+ BigDecimal quantity) {
654
+ Optional<Inventory> inventoryOpt = inventoryRepository
655
+ .findByWarehouseAndProduct(warehouseCode, productCode);
656
+
657
+ if (inventoryOpt.isEmpty()) {
658
+ return AllocationResult.failure("在庫が見つかりません");
659
+ }
660
+
661
+ Inventory inventory = inventoryOpt.get();
662
+ BigDecimal available = inventory.getAvailableQuantity();
663
+
664
+ if (available.compareTo(quantity) < 0) {
665
+ return AllocationResult.failure(
666
+ "有効在庫数が不足しています。有効在庫: " + available);
667
+ }
668
+
669
+ inventoryRepository.allocate(inventory.getId(), quantity);
670
+
671
+ return AllocationResult.success(quantity);
672
+ }
673
+
674
+ /**
675
+ * 引当を解除
676
+ */
677
+ @Transactional
678
+ public void deallocate(String warehouseCode, String productCode,
679
+ BigDecimal quantity) {
680
+ Optional<Inventory> inventoryOpt = inventoryRepository
681
+ .findByWarehouseAndProduct(warehouseCode, productCode);
682
+
683
+ if (inventoryOpt.isEmpty()) {
684
+ throw new IllegalArgumentException("在庫が見つかりません");
685
+ }
686
+
687
+ Inventory inventory = inventoryOpt.get();
688
+ inventoryRepository.deallocate(inventory.getId(), quantity);
689
+ }
690
+
691
+ /**
692
+ * 入庫処理
693
+ */
694
+ @Transactional
695
+ public void receive(String warehouseCode, String productCode,
696
+ BigDecimal quantity) {
697
+ Optional<Inventory> inventoryOpt = inventoryRepository
698
+ .findByWarehouseAndProduct(warehouseCode, productCode);
699
+
700
+ if (inventoryOpt.isEmpty()) {
701
+ // 新規在庫レコード作成
702
+ Inventory newInventory = Inventory.builder()
703
+ .warehouseCode(warehouseCode)
704
+ .productCode(productCode)
705
+ .currentQuantity(quantity)
706
+ .allocatedQuantity(BigDecimal.ZERO)
707
+ .orderedQuantity(BigDecimal.ZERO)
708
+ .lastReceiptDate(LocalDate.now())
709
+ .build();
710
+ inventoryRepository.save(newInventory);
711
+ } else {
712
+ Inventory inventory = inventoryOpt.get();
713
+ inventoryRepository.receive(inventory.getId(), quantity, LocalDate.now());
714
+ }
715
+ }
716
+
717
+ /**
718
+ * 出庫処理
719
+ */
720
+ @Transactional
721
+ public void ship(String warehouseCode, String productCode,
722
+ BigDecimal quantity) {
723
+ Optional<Inventory> inventoryOpt = inventoryRepository
724
+ .findByWarehouseAndProduct(warehouseCode, productCode);
725
+
726
+ if (inventoryOpt.isEmpty()) {
727
+ throw new IllegalArgumentException("在庫が見つかりません");
728
+ }
729
+
730
+ Inventory inventory = inventoryOpt.get();
731
+ inventoryRepository.ship(inventory.getId(), quantity, LocalDate.now());
732
+ }
733
+
734
+ /**
735
+ * 在庫数量を直接更新(棚卸調整用)
736
+ */
737
+ @Transactional
738
+ public void adjustQuantity(String warehouseCode, String productCode,
739
+ BigDecimal newQuantity, String reason) {
740
+ Optional<Inventory> inventoryOpt = inventoryRepository
741
+ .findByWarehouseAndProduct(warehouseCode, productCode);
742
+
743
+ if (inventoryOpt.isEmpty()) {
744
+ throw new IllegalArgumentException("在庫が見つかりません");
745
+ }
746
+
747
+ Inventory inventory = inventoryOpt.get();
748
+ inventoryRepository.updateCurrentQuantity(
749
+ inventory.getId(), newQuantity, reason);
750
+ }
751
+ }
752
+ ```
753
+
754
+ </details>
755
+
756
+ <details>
757
+ <summary>在庫サマリ・引当結果クラス</summary>
758
+
759
+ ```java
760
+ // src/main/java/com/example/sales/application/service/InventorySummary.java
761
+ package com.example.sms.application.service;
762
+
763
+ import com.example.sms.domain.model.inventory.Inventory;
764
+ import lombok.Builder;
765
+ import lombok.Data;
766
+
767
+ import java.math.BigDecimal;
768
+ import java.util.List;
769
+
770
+ @Data
771
+ @Builder
772
+ public class InventorySummary {
773
+ private String productCode;
774
+ private BigDecimal totalCurrentQuantity;
775
+ private BigDecimal totalAllocatedQuantity;
776
+ private BigDecimal totalAvailableQuantity;
777
+ private BigDecimal totalOrderedQuantity;
778
+ private List<Inventory> warehouseDetails;
779
+ }
780
+ ```
781
+
782
+ ```java
783
+ // src/main/java/com/example/sales/application/service/AllocationResult.java
784
+ package com.example.sms.application.service;
785
+
786
+ import lombok.Builder;
787
+ import lombok.Data;
788
+
789
+ import java.math.BigDecimal;
790
+
791
+ @Data
792
+ @Builder
793
+ public class AllocationResult {
794
+ private boolean success;
795
+ private BigDecimal allocatedQuantity;
796
+ private String reason;
797
+
798
+ public static AllocationResult success(BigDecimal quantity) {
799
+ return AllocationResult.builder()
800
+ .success(true)
801
+ .allocatedQuantity(quantity)
802
+ .build();
803
+ }
804
+
805
+ public static AllocationResult failure(String reason) {
806
+ return AllocationResult.builder()
807
+ .success(false)
808
+ .allocatedQuantity(BigDecimal.ZERO)
809
+ .reason(reason)
810
+ .build();
811
+ }
812
+ }
813
+ ```
814
+
815
+ </details>
816
+
817
+ ---
818
+
819
+ ## 9.2 在庫移動・棚卸の DB 設計
820
+
821
+ ### 入出庫履歴の管理
822
+
823
+ 入出庫履歴は在庫の増減を時系列で記録し、トレーサビリティを確保します。
824
+
825
+ ```plantuml
826
+ @startuml
827
+
828
+ title 入出庫履歴の構造
829
+
830
+ entity "入出庫履歴データ" as movement {
831
+ * 履歴ID <<PK>>
832
+ --
833
+ * 倉庫コード <<FK>>
834
+ * 商品コード <<FK>>
835
+ * 移動日時
836
+ * 移動区分
837
+ * 移動数量
838
+ * 移動前在庫数
839
+ * 移動後在庫数
840
+ 伝票番号
841
+ 伝票種別
842
+ 移動理由
843
+ ロケーションコード
844
+ ロット番号
845
+ }
846
+
847
+ note right of movement
848
+ 移動区分:
849
+ - 入荷入庫
850
+ - 出荷出庫
851
+ - 倉庫間移動(出)
852
+ - 倉庫間移動(入)
853
+ - 棚卸調整(増)
854
+ - 棚卸調整(減)
855
+ - 返品入庫
856
+ - 廃棄
857
+ end note
858
+
859
+ @enduml
860
+ ```
861
+
862
+ ### 移動区分の定義
863
+
864
+ | 移動区分 | 在庫影響 | 説明 |
865
+ |---------|---------|------|
866
+ | **入荷入庫** | 増加 | 仕入先からの入荷 |
867
+ | **出荷出庫** | 減少 | 顧客への出荷 |
868
+ | **倉庫間移動(出)** | 減少 | 他倉庫への移動(出庫側) |
869
+ | **倉庫間移動(入)** | 増加 | 他倉庫からの移動(入庫側) |
870
+ | **棚卸調整(増)** | 増加 | 棚卸による増加調整 |
871
+ | **棚卸調整(減)** | 減少 | 棚卸による減少調整 |
872
+ | **返品入庫** | 増加 | 顧客からの返品 |
873
+ | **廃棄** | 減少 | 廃棄処理 |
874
+
875
+ ### 棚卸業務の設計
876
+
877
+ 棚卸業務は「棚卸表発行 → 実棚カウント → 差異確認 → 在庫調整」の流れで進みます。
878
+
879
+ ```plantuml
880
+ @startuml
881
+
882
+ title 棚卸業務フロー
883
+
884
+ [*] --> 作成中
885
+
886
+ 作成中 --> 実施中 : 棚卸開始
887
+
888
+ 実施中 --> 確定 : 棚卸完了
889
+
890
+ 確定 --> [*]
891
+
892
+ 作成中 --> 取消 : 取消
893
+ 実施中 --> 取消 : 取消
894
+
895
+ 取消 --> [*]
896
+
897
+ @enduml
898
+ ```
899
+
900
+ ### 棚卸データの ER 図
901
+
902
+ ```plantuml
903
+ @startuml
904
+
905
+ title 棚卸関連テーブル
906
+
907
+ entity 棚卸データ {
908
+ ID <<PK>>
909
+ --
910
+ 棚卸番号 <<UK>>
911
+ 倉庫コード <<FK>>
912
+ 棚卸日
913
+ 棚卸開始日時
914
+ 棚卸終了日時
915
+ ステータス
916
+ 備考
917
+ 作成日時
918
+ 更新日時
919
+ }
920
+
921
+ entity 棚卸明細データ {
922
+ ID <<PK>>
923
+ --
924
+ 棚卸ID <<FK>>
925
+ 棚卸行番号
926
+ 商品コード <<FK>>
927
+ ロケーションコード
928
+ ロット番号
929
+ 帳簿在庫数
930
+ 実棚数
931
+ 差異数
932
+ 差異理由
933
+ 調整済フラグ
934
+ 作成日時
935
+ 更新日時
936
+ }
937
+
938
+ entity 倉庫マスタ {
939
+ 倉庫コード <<PK>>
940
+ --
941
+ 倉庫名
942
+ }
943
+
944
+ entity 商品マスタ {
945
+ 商品コード <<PK>>
946
+ --
947
+ 商品名
948
+ }
949
+
950
+ 倉庫マスタ ||--o{ 棚卸データ : 倉庫コード
951
+ 棚卸データ ||--o{ 棚卸明細データ : 棚卸ID
952
+ 商品マスタ ||--o{ 棚卸明細データ : 商品コード
953
+
954
+ @enduml
955
+ ```
956
+
957
+ ### マイグレーション:入出庫履歴・棚卸テーブルの作成
958
+
959
+ <details>
960
+ <summary>V016__create_stock_movement_tables.sql</summary>
961
+
962
+ ```sql
963
+ -- src/main/resources/db/migration/V016__create_stock_movement_tables.sql
964
+
965
+ -- 移動区分
966
+ CREATE TYPE 移動区分 AS ENUM (
967
+ '入荷入庫',
968
+ '出荷出庫',
969
+ '倉庫間移動出',
970
+ '倉庫間移動入',
971
+ '棚卸調整増',
972
+ '棚卸調整減',
973
+ '返品入庫',
974
+ '廃棄'
975
+ );
976
+
977
+ -- 棚卸ステータス
978
+ CREATE TYPE 棚卸ステータス AS ENUM ('作成中', '実施中', '確定', '取消');
979
+
980
+ -- 入出庫履歴データ
981
+ CREATE TABLE "入出庫履歴データ" (
982
+ "ID" SERIAL PRIMARY KEY,
983
+ "倉庫コード" VARCHAR(20) NOT NULL,
984
+ "商品コード" VARCHAR(20) NOT NULL,
985
+ "移動日時" TIMESTAMP NOT NULL,
986
+ "移動区分" 移動区分 NOT NULL,
987
+ "移動数量" DECIMAL(15, 2) NOT NULL,
988
+ "移動前在庫数" DECIMAL(15, 2) NOT NULL,
989
+ "移動後在庫数" DECIMAL(15, 2) NOT NULL,
990
+ "伝票番号" VARCHAR(20),
991
+ "伝票種別" VARCHAR(20),
992
+ "移動理由" TEXT,
993
+ "ロケーションコード" VARCHAR(20),
994
+ "ロット番号" VARCHAR(50),
995
+ "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
996
+ "作成者" VARCHAR(50),
997
+ CONSTRAINT "fk_入出庫履歴_倉庫"
998
+ FOREIGN KEY ("倉庫コード") REFERENCES "倉庫マスタ"("倉庫コード"),
999
+ CONSTRAINT "fk_入出庫履歴_商品"
1000
+ FOREIGN KEY ("商品コード") REFERENCES "商品マスタ"("商品コード")
1001
+ );
1002
+
1003
+ -- 棚卸データ(ヘッダ)
1004
+ CREATE TABLE "棚卸データ" (
1005
+ "ID" SERIAL PRIMARY KEY,
1006
+ "棚卸番号" VARCHAR(20) UNIQUE NOT NULL,
1007
+ "倉庫コード" VARCHAR(20) NOT NULL,
1008
+ "棚卸日" DATE NOT NULL,
1009
+ "棚卸開始日時" TIMESTAMP,
1010
+ "棚卸終了日時" TIMESTAMP,
1011
+ "ステータス" 棚卸ステータス DEFAULT '作成中' NOT NULL,
1012
+ "備考" TEXT,
1013
+ "バージョン" INTEGER DEFAULT 1 NOT NULL,
1014
+ "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
1015
+ "作成者" VARCHAR(50),
1016
+ "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
1017
+ "更新者" VARCHAR(50),
1018
+ CONSTRAINT "fk_棚卸_倉庫"
1019
+ FOREIGN KEY ("倉庫コード") REFERENCES "倉庫マスタ"("倉庫コード")
1020
+ );
1021
+
1022
+ -- 棚卸明細データ
1023
+ CREATE TABLE "棚卸明細データ" (
1024
+ "ID" SERIAL PRIMARY KEY,
1025
+ "棚卸ID" INTEGER NOT NULL,
1026
+ "棚卸行番号" INTEGER NOT NULL,
1027
+ "商品コード" VARCHAR(20) NOT NULL,
1028
+ "ロケーションコード" VARCHAR(20),
1029
+ "ロット番号" VARCHAR(50),
1030
+ "帳簿在庫数" DECIMAL(15, 2) NOT NULL,
1031
+ "実棚数" DECIMAL(15, 2),
1032
+ "差異数" DECIMAL(15, 2),
1033
+ "差異理由" TEXT,
1034
+ "調整済フラグ" BOOLEAN DEFAULT FALSE NOT NULL,
1035
+ "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
1036
+ "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
1037
+ CONSTRAINT "fk_棚卸明細_棚卸"
1038
+ FOREIGN KEY ("棚卸ID") REFERENCES "棚卸データ"("ID") ON DELETE CASCADE,
1039
+ CONSTRAINT "fk_棚卸明細_商品"
1040
+ FOREIGN KEY ("商品コード") REFERENCES "商品マスタ"("商品コード"),
1041
+ CONSTRAINT "uk_棚卸明細_棚卸_行" UNIQUE ("棚卸ID", "棚卸行番号")
1042
+ );
1043
+
1044
+ -- 差異数を自動計算するトリガー
1045
+ CREATE OR REPLACE FUNCTION calculate_stocktaking_difference()
1046
+ RETURNS TRIGGER AS $$
1047
+ BEGIN
1048
+ IF NEW."実棚数" IS NOT NULL THEN
1049
+ NEW."差異数" := NEW."実棚数" - NEW."帳簿在庫数";
1050
+ END IF;
1051
+ RETURN NEW;
1052
+ END;
1053
+ $$ LANGUAGE plpgsql;
1054
+
1055
+ CREATE TRIGGER trg_棚卸明細_差異計算
1056
+ BEFORE INSERT OR UPDATE ON "棚卸明細データ"
1057
+ FOR EACH ROW
1058
+ EXECUTE FUNCTION calculate_stocktaking_difference();
1059
+
1060
+ -- インデックス
1061
+ CREATE INDEX "idx_入出庫履歴_倉庫コード" ON "入出庫履歴データ"("倉庫コード");
1062
+ CREATE INDEX "idx_入出庫履歴_商品コード" ON "入出庫履歴データ"("商品コード");
1063
+ CREATE INDEX "idx_入出庫履歴_移動日時" ON "入出庫履歴データ"("移動日時");
1064
+ CREATE INDEX "idx_入出庫履歴_伝票番号" ON "入出庫履歴データ"("伝票番号");
1065
+ CREATE INDEX "idx_棚卸_倉庫コード" ON "棚卸データ"("倉庫コード");
1066
+ CREATE INDEX "idx_棚卸_棚卸日" ON "棚卸データ"("棚卸日");
1067
+ CREATE INDEX "idx_棚卸_ステータス" ON "棚卸データ"("ステータス");
1068
+ CREATE INDEX "idx_棚卸明細_商品コード" ON "棚卸明細データ"("商品コード");
1069
+
1070
+ -- テーブルコメント
1071
+ COMMENT ON TABLE "入出庫履歴データ" IS '在庫の入出庫履歴を記録するテーブル';
1072
+ COMMENT ON TABLE "棚卸データ" IS '棚卸作業のヘッダ情報を管理するテーブル';
1073
+ COMMENT ON TABLE "棚卸明細データ" IS '棚卸明細・差異情報を管理するテーブル';
1074
+ COMMENT ON COLUMN "棚卸データ"."バージョン" IS '楽観ロック用バージョン番号';
1075
+ ```
1076
+
1077
+ </details>
1078
+
1079
+ ### 入出庫履歴・棚卸エンティティの実装
1080
+
1081
+ <details>
1082
+ <summary>移動区分 ENUM</summary>
1083
+
1084
+ ```java
1085
+ // src/main/java/com/example/sales/domain/model/inventory/MovementType.java
1086
+ package com.example.sms.domain.model.inventory;
1087
+
1088
+ import lombok.Getter;
1089
+ import lombok.RequiredArgsConstructor;
1090
+
1091
+ @Getter
1092
+ @RequiredArgsConstructor
1093
+ public enum MovementType {
1094
+ RECEIPT("入荷入庫"),
1095
+ SHIPMENT("出荷出庫"),
1096
+ TRANSFER_OUT("倉庫間移動出"),
1097
+ TRANSFER_IN("倉庫間移動入"),
1098
+ ADJUSTMENT_PLUS("棚卸調整増"),
1099
+ ADJUSTMENT_MINUS("棚卸調整減"),
1100
+ RETURN_RECEIPT("返品入庫"),
1101
+ DISPOSAL("廃棄");
1102
+
1103
+ private final String displayName;
1104
+
1105
+ public static MovementType fromDisplayName(String displayName) {
1106
+ for (MovementType type : values()) {
1107
+ if (type.displayName.equals(displayName)) {
1108
+ return type;
1109
+ }
1110
+ }
1111
+ throw new IllegalArgumentException("Unknown movement type: " + displayName);
1112
+ }
1113
+
1114
+ /**
1115
+ * 入庫系の移動かどうか
1116
+ */
1117
+ public boolean isInbound() {
1118
+ return this == RECEIPT || this == TRANSFER_IN ||
1119
+ this == ADJUSTMENT_PLUS || this == RETURN_RECEIPT;
1120
+ }
1121
+
1122
+ /**
1123
+ * 出庫系の移動かどうか
1124
+ */
1125
+ public boolean isOutbound() {
1126
+ return this == SHIPMENT || this == TRANSFER_OUT ||
1127
+ this == ADJUSTMENT_MINUS || this == DISPOSAL;
1128
+ }
1129
+ }
1130
+ ```
1131
+
1132
+ </details>
1133
+
1134
+ <details>
1135
+ <summary>棚卸ステータス ENUM</summary>
1136
+
1137
+ ```java
1138
+ // src/main/java/com/example/sales/domain/model/inventory/StocktakingStatus.java
1139
+ package com.example.sms.domain.model.inventory;
1140
+
1141
+ import lombok.Getter;
1142
+ import lombok.RequiredArgsConstructor;
1143
+
1144
+ @Getter
1145
+ @RequiredArgsConstructor
1146
+ public enum StocktakingStatus {
1147
+ DRAFT("作成中"),
1148
+ IN_PROGRESS("実施中"),
1149
+ CONFIRMED("確定"),
1150
+ CANCELLED("取消");
1151
+
1152
+ private final String displayName;
1153
+
1154
+ public static StocktakingStatus fromDisplayName(String displayName) {
1155
+ for (StocktakingStatus status : values()) {
1156
+ if (status.displayName.equals(displayName)) {
1157
+ return status;
1158
+ }
1159
+ }
1160
+ throw new IllegalArgumentException("Unknown stocktaking status: " + displayName);
1161
+ }
1162
+
1163
+ /**
1164
+ * 実施開始可能かどうか
1165
+ */
1166
+ public boolean canStart() {
1167
+ return this == DRAFT;
1168
+ }
1169
+
1170
+ /**
1171
+ * 確定可能かどうか
1172
+ */
1173
+ public boolean canConfirm() {
1174
+ return this == IN_PROGRESS;
1175
+ }
1176
+
1177
+ /**
1178
+ * 取消可能かどうか
1179
+ */
1180
+ public boolean canCancel() {
1181
+ return this == DRAFT || this == IN_PROGRESS;
1182
+ }
1183
+ }
1184
+ ```
1185
+
1186
+ </details>
1187
+
1188
+ <details>
1189
+ <summary>入出庫履歴エンティティ</summary>
1190
+
1191
+ ```java
1192
+ // src/main/java/com/example/sales/domain/model/inventory/StockMovement.java
1193
+ package com.example.sms.domain.model.inventory;
1194
+
1195
+ import lombok.Builder;
1196
+ import lombok.Data;
1197
+
1198
+ import java.math.BigDecimal;
1199
+ import java.time.LocalDateTime;
1200
+
1201
+ @Data
1202
+ @Builder
1203
+ public class StockMovement {
1204
+ private Integer id;
1205
+ private String warehouseCode;
1206
+ private String productCode;
1207
+ private LocalDateTime movementDateTime;
1208
+ private MovementType movementType;
1209
+ private BigDecimal movementQuantity;
1210
+ private BigDecimal beforeQuantity;
1211
+ private BigDecimal afterQuantity;
1212
+ private String documentNumber;
1213
+ private String documentType;
1214
+ private String movementReason;
1215
+ private String locationCode;
1216
+ private String lotNumber;
1217
+ private LocalDateTime createdAt;
1218
+ private String createdBy;
1219
+
1220
+ // リレーション
1221
+ private Warehouse warehouse;
1222
+ private Product product;
1223
+ }
1224
+ ```
1225
+
1226
+ </details>
1227
+
1228
+ <details>
1229
+ <summary>棚卸データ・棚卸明細エンティティ</summary>
1230
+
1231
+ ```java
1232
+ // src/main/java/com/example/sales/domain/model/inventory/Stocktaking.java
1233
+ package com.example.sms.domain.model.inventory;
1234
+
1235
+ import lombok.AllArgsConstructor;
1236
+ import lombok.Builder;
1237
+ import lombok.Data;
1238
+ import lombok.NoArgsConstructor;
1239
+
1240
+ import java.time.LocalDate;
1241
+ import java.time.LocalDateTime;
1242
+ import java.util.ArrayList;
1243
+ import java.util.List;
1244
+
1245
+ @Data
1246
+ @Builder
1247
+ @NoArgsConstructor
1248
+ @AllArgsConstructor
1249
+ public class Stocktaking {
1250
+ private Integer id;
1251
+ private String stocktakingNumber;
1252
+ private String warehouseCode;
1253
+ private LocalDate stocktakingDate;
1254
+ private LocalDateTime startDateTime;
1255
+ private LocalDateTime endDateTime;
1256
+ private StocktakingStatus status;
1257
+ private String remarks;
1258
+ private LocalDateTime createdAt;
1259
+ private String createdBy;
1260
+ private LocalDateTime updatedAt;
1261
+ private String updatedBy;
1262
+
1263
+ // 楽観ロック用バージョン
1264
+ @Builder.Default
1265
+ private Integer version = 1;
1266
+
1267
+ // リレーション
1268
+ @Builder.Default
1269
+ private List<StocktakingDetail> details = new ArrayList<>();
1270
+
1271
+ /**
1272
+ * 差異がある明細があるかどうか
1273
+ */
1274
+ public boolean hasDifference() {
1275
+ return details.stream()
1276
+ .anyMatch(d -> d.getDifferenceQuantity() != null &&
1277
+ d.getDifferenceQuantity().signum() != 0);
1278
+ }
1279
+
1280
+ /**
1281
+ * 全明細が調整済みかどうか
1282
+ */
1283
+ public boolean isAllAdjusted() {
1284
+ return details.stream()
1285
+ .allMatch(StocktakingDetail::getAdjustedFlag);
1286
+ }
1287
+ }
1288
+ ```
1289
+
1290
+ ```java
1291
+ // src/main/java/com/example/sms/domain/model/inventory/StocktakingDetail.java
1292
+ package com.example.sms.domain.model.inventory;
1293
+
1294
+ import lombok.AllArgsConstructor;
1295
+ import lombok.Builder;
1296
+ import lombok.Data;
1297
+ import lombok.NoArgsConstructor;
1298
+
1299
+ import java.math.BigDecimal;
1300
+ import java.time.LocalDateTime;
1301
+
1302
+ @Data
1303
+ @Builder
1304
+ @NoArgsConstructor
1305
+ @AllArgsConstructor
1306
+ public class StocktakingDetail {
1307
+ private Integer id;
1308
+ private Integer stocktakingId;
1309
+ private Integer lineNumber;
1310
+ private String productCode;
1311
+ private String locationCode;
1312
+ private String lotNumber;
1313
+ private BigDecimal bookQuantity;
1314
+ private BigDecimal actualQuantity;
1315
+ private BigDecimal differenceQuantity;
1316
+ private String differenceReason;
1317
+ @Builder.Default
1318
+ private Boolean adjustedFlag = false;
1319
+ private LocalDateTime createdAt;
1320
+ private LocalDateTime updatedAt;
1321
+
1322
+ /**
1323
+ * 実棚数を入力
1324
+ */
1325
+ public void setActualQuantityAndCalculateDifference(BigDecimal actualQty) {
1326
+ this.actualQuantity = actualQty;
1327
+ this.differenceQuantity = actualQty.subtract(this.bookQuantity);
1328
+ }
1329
+
1330
+ /**
1331
+ * 差異があるかどうか
1332
+ */
1333
+ public boolean hasDifference() {
1334
+ return differenceQuantity != null && differenceQuantity.signum() != 0;
1335
+ }
1336
+ }
1337
+ ```
1338
+
1339
+ </details>
1340
+
1341
+ ### 棚卸サービスの実装
1342
+
1343
+ <details>
1344
+ <summary>棚卸サービス</summary>
1345
+
1346
+ ```java
1347
+ // src/main/java/com/example/sales/application/service/StocktakingService.java
1348
+ package com.example.sms.application.service;
1349
+
1350
+ import com.example.sms.application.port.out.*;
1351
+ import com.example.sms.domain.model.inventory.*;
1352
+ import lombok.RequiredArgsConstructor;
1353
+ import org.springframework.stereotype.Service;
1354
+ import org.springframework.transaction.annotation.Transactional;
1355
+
1356
+ import java.math.BigDecimal;
1357
+ import java.time.LocalDate;
1358
+ import java.time.LocalDateTime;
1359
+ import java.time.format.DateTimeFormatter;
1360
+ import java.util.ArrayList;
1361
+ import java.util.List;
1362
+ import java.util.Optional;
1363
+
1364
+ @Service
1365
+ @RequiredArgsConstructor
1366
+ public class StocktakingService {
1367
+
1368
+ private final StocktakingRepository stocktakingRepository;
1369
+ private final InventoryRepository inventoryRepository;
1370
+ private final StockMovementRepository stockMovementRepository;
1371
+
1372
+ /**
1373
+ * 棚卸を作成する
1374
+ */
1375
+ @Transactional
1376
+ public Stocktaking createStocktaking(String warehouseCode, LocalDate stocktakingDate) {
1377
+ String stocktakingNumber = generateStocktakingNumber(stocktakingDate);
1378
+
1379
+ // 倉庫の在庫から棚卸明細を作成
1380
+ List<Inventory> inventories = inventoryRepository.findByWarehouseCode(warehouseCode);
1381
+
1382
+ List<StocktakingDetail> details = new ArrayList<>();
1383
+ for (Inventory inv : inventories) {
1384
+ StocktakingDetail detail = StocktakingDetail.builder()
1385
+ .stocktakingNumber(stocktakingNumber)
1386
+ .productCode(inv.getProductCode())
1387
+ .locationCode(inv.getLocationCode())
1388
+ .lotNumber(inv.getLotNumber())
1389
+ .bookQuantity(inv.getCurrentQuantity())
1390
+ .adjustedFlag(false)
1391
+ .build();
1392
+ details.add(detail);
1393
+ }
1394
+
1395
+ Stocktaking stocktaking = Stocktaking.builder()
1396
+ .stocktakingNumber(stocktakingNumber)
1397
+ .warehouseCode(warehouseCode)
1398
+ .stocktakingDate(stocktakingDate)
1399
+ .status(StocktakingStatus.DRAFT)
1400
+ .details(details)
1401
+ .build();
1402
+
1403
+ stocktakingRepository.save(stocktaking);
1404
+
1405
+ return stocktaking;
1406
+ }
1407
+
1408
+ /**
1409
+ * 棚卸を開始する
1410
+ */
1411
+ @Transactional
1412
+ public void startStocktaking(String stocktakingNumber) {
1413
+ Stocktaking stocktaking = findByStocktakingNumber(stocktakingNumber)
1414
+ .orElseThrow(() -> new IllegalArgumentException(
1415
+ "棚卸が見つかりません: " + stocktakingNumber));
1416
+
1417
+ if (!stocktaking.getStatus().canStart()) {
1418
+ throw new IllegalStateException(
1419
+ "この棚卸は開始できません。現在のステータス: " +
1420
+ stocktaking.getStatus().getDisplayName());
1421
+ }
1422
+
1423
+ stocktakingRepository.updateStatus(
1424
+ stocktakingNumber, StocktakingStatus.IN_PROGRESS);
1425
+ stocktakingRepository.updateStartDateTime(
1426
+ stocktakingNumber, LocalDateTime.now());
1427
+ }
1428
+
1429
+ /**
1430
+ * 実棚数を入力する
1431
+ */
1432
+ @Transactional
1433
+ public void inputActualQuantity(String stocktakingNumber, String productCode,
1434
+ BigDecimal actualQuantity, String differenceReason) {
1435
+ stocktakingRepository.updateActualQuantity(
1436
+ stocktakingNumber, productCode, actualQuantity, differenceReason);
1437
+ }
1438
+
1439
+ /**
1440
+ * 棚卸を確定する(在庫調整を実行)
1441
+ */
1442
+ @Transactional
1443
+ public void confirmStocktaking(String stocktakingNumber) {
1444
+ Stocktaking stocktaking = findByStocktakingNumber(stocktakingNumber)
1445
+ .orElseThrow(() -> new IllegalArgumentException(
1446
+ "棚卸が見つかりません: " + stocktakingNumber));
1447
+
1448
+ if (!stocktaking.getStatus().canConfirm()) {
1449
+ throw new IllegalStateException(
1450
+ "この棚卸は確定できません。現在のステータス: " +
1451
+ stocktaking.getStatus().getDisplayName());
1452
+ }
1453
+
1454
+ // 差異がある明細の在庫を調整
1455
+ for (StocktakingDetail detail : stocktaking.getDetails()) {
1456
+ if (detail.hasDifference()) {
1457
+ adjustInventory(stocktaking.getWarehouseCode(), detail);
1458
+ }
1459
+ }
1460
+
1461
+ stocktakingRepository.updateStatus(
1462
+ stocktakingNumber, StocktakingStatus.CONFIRMED);
1463
+ stocktakingRepository.updateEndDateTime(
1464
+ stocktakingNumber, LocalDateTime.now());
1465
+ }
1466
+
1467
+ /**
1468
+ * 在庫を調整する
1469
+ */
1470
+ private void adjustInventory(String warehouseCode, StocktakingDetail detail) {
1471
+ Optional<Inventory> inventoryOpt = inventoryRepository
1472
+ .findByWarehouseAndProduct(warehouseCode, detail.getProductCode());
1473
+
1474
+ if (inventoryOpt.isEmpty()) {
1475
+ return;
1476
+ }
1477
+
1478
+ Inventory inventory = inventoryOpt.get();
1479
+ BigDecimal difference = detail.getDifferenceQuantity();
1480
+ BigDecimal beforeQuantity = inventory.getCurrentQuantity();
1481
+ BigDecimal afterQuantity = beforeQuantity.add(difference);
1482
+
1483
+ // 在庫を更新
1484
+ inventoryRepository.updateCurrentQuantity(
1485
+ inventory.getId(), afterQuantity, detail.getDifferenceReason());
1486
+
1487
+ // 入出庫履歴を記録
1488
+ MovementType movementType = difference.signum() > 0
1489
+ ? MovementType.ADJUSTMENT_PLUS
1490
+ : MovementType.ADJUSTMENT_MINUS;
1491
+
1492
+ StockMovement movement = StockMovement.builder()
1493
+ .warehouseCode(warehouseCode)
1494
+ .productCode(detail.getProductCode())
1495
+ .movementDateTime(LocalDateTime.now())
1496
+ .movementType(movementType)
1497
+ .movementQuantity(difference.abs())
1498
+ .beforeQuantity(beforeQuantity)
1499
+ .afterQuantity(afterQuantity)
1500
+ .movementReason(detail.getDifferenceReason())
1501
+ .locationCode(detail.getLocationCode())
1502
+ .lotNumber(detail.getLotNumber())
1503
+ .build();
1504
+
1505
+ stockMovementRepository.save(movement);
1506
+
1507
+ // 調整済みフラグを更新
1508
+ stocktakingRepository.markAsAdjusted(
1509
+ detail.getStocktakingNumber(), detail.getProductCode());
1510
+ }
1511
+
1512
+ public Optional<Stocktaking> findByStocktakingNumber(String stocktakingNumber) {
1513
+ return stocktakingRepository.findByStocktakingNumber(stocktakingNumber);
1514
+ }
1515
+
1516
+ private String generateStocktakingNumber(LocalDate stocktakingDate) {
1517
+ String prefix = "ST-" + stocktakingDate.format(
1518
+ DateTimeFormatter.ofPattern("yyyyMM")) + "-";
1519
+ Optional<String> latestNumber = stocktakingRepository
1520
+ .findLatestStocktakingNumber(prefix + "%");
1521
+
1522
+ int sequence = 1;
1523
+ if (latestNumber.isPresent()) {
1524
+ String latest = latestNumber.get();
1525
+ int currentSequence = Integer.parseInt(
1526
+ latest.substring(latest.length() - 4));
1527
+ sequence = currentSequence + 1;
1528
+ }
1529
+
1530
+ return prefix + String.format("%04d", sequence);
1531
+ }
1532
+ }
1533
+ ```
1534
+
1535
+ </details>
1536
+
1537
+ ---
1538
+
1539
+ ## 9.3 リレーションと楽観ロックの設計
1540
+
1541
+ ### MyBatis ネストした ResultMap による関連付け
1542
+
1543
+ 棚卸データは、棚卸(ヘッダ)→ 棚卸明細の2層構造を持ちます。N+1 問題を回避するため、JOIN クエリで一括取得します。
1544
+
1545
+ ```plantuml
1546
+ @startuml
1547
+
1548
+ title N+1 問題の回避
1549
+
1550
+ rectangle "N+1 問題(非効率)" #FFCCCC {
1551
+ card "1. 棚卸ヘッダ取得\n SELECT * FROM 棚卸データ\n (1回)" as q1
1552
+ card "2. 各棚卸の明細取得\n SELECT * FROM 棚卸明細データ\n WHERE 棚卸番号 = ?\n (N回)" as q2
1553
+ }
1554
+
1555
+ rectangle "解決策:JOINによる一括取得" #CCFFCC {
1556
+ card "1回のJOINクエリで\n全データを取得\n SELECT st.*, std.*\n FROM 棚卸データ st\n LEFT JOIN 棚卸明細データ std ON ..." as sol
1557
+ }
1558
+
1559
+ q1 --> q2 : N件
1560
+ note right of q2
1561
+ 10件の棚卸があれば
1562
+ 合計11回のクエリ発行
1563
+ end note
1564
+
1565
+ @enduml
1566
+ ```
1567
+
1568
+ <details>
1569
+ <summary>棚卸データのリレーション設定(StocktakingMapper.xml)</summary>
1570
+
1571
+ ```xml
1572
+ <?xml version="1.0" encoding="UTF-8" ?>
1573
+ <!DOCTYPE mapper
1574
+ PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
1575
+ "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
1576
+ <mapper namespace="com.example.sms.infrastructure.persistence.mapper.StocktakingMapper">
1577
+
1578
+ <!-- 棚卸(ヘッダ)の ResultMap -->
1579
+ <resultMap id="StocktakingWithDetailsResultMap"
1580
+ type="com.example.sms.domain.model.inventory.Stocktaking">
1581
+ <id property="id" column="st_id"/>
1582
+ <result property="stocktakingNumber" column="st_棚卸番号"/>
1583
+ <result property="warehouseCode" column="st_倉庫コード"/>
1584
+ <result property="stocktakingDate" column="st_棚卸日"/>
1585
+ <result property="startDateTime" column="st_棚卸開始日時"/>
1586
+ <result property="endDateTime" column="st_棚卸終了日時"/>
1587
+ <result property="status" column="st_ステータス"
1588
+ typeHandler="com.example.sms.infrastructure.persistence.typehandler.StocktakingStatusTypeHandler"/>
1589
+ <result property="remarks" column="st_備考"/>
1590
+ <result property="version" column="st_バージョン"/>
1591
+ <result property="createdAt" column="st_作成日時"/>
1592
+ <result property="updatedAt" column="st_更新日時"/>
1593
+ <!-- 棚卸明細との1:N関連 -->
1594
+ <collection property="details"
1595
+ ofType="com.example.sms.domain.model.inventory.StocktakingDetail"
1596
+ resultMap="StocktakingDetailNestedResultMap"/>
1597
+ </resultMap>
1598
+
1599
+ <!-- 棚卸明細のネスト ResultMap -->
1600
+ <resultMap id="StocktakingDetailNestedResultMap"
1601
+ type="com.example.sms.domain.model.inventory.StocktakingDetail">
1602
+ <id property="id" column="std_id"/>
1603
+ <result property="stocktakingId" column="std_棚卸ID"/>
1604
+ <result property="lineNumber" column="std_棚卸行番号"/>
1605
+ <result property="productCode" column="std_商品コード"/>
1606
+ <result property="locationCode" column="std_ロケーションコード"/>
1607
+ <result property="lotNumber" column="std_ロット番号"/>
1608
+ <result property="bookQuantity" column="std_帳簿在庫数"/>
1609
+ <result property="actualQuantity" column="std_実棚数"/>
1610
+ <result property="differenceQuantity" column="std_差異数"/>
1611
+ <result property="differenceReason" column="std_差異理由"/>
1612
+ <result property="adjustedFlag" column="std_調整済フラグ"/>
1613
+ </resultMap>
1614
+
1615
+ <!-- JOIN による一括取得クエリ -->
1616
+ <select id="findWithDetailsByStocktakingNumber"
1617
+ resultMap="StocktakingWithDetailsResultMap">
1618
+ SELECT
1619
+ st."ID" AS st_id,
1620
+ st."棚卸番号" AS st_棚卸番号,
1621
+ st."倉庫コード" AS st_倉庫コード,
1622
+ st."棚卸日" AS st_棚卸日,
1623
+ st."棚卸開始日時" AS st_棚卸開始日時,
1624
+ st."棚卸終了日時" AS st_棚卸終了日時,
1625
+ st."ステータス" AS st_ステータス,
1626
+ st."備考" AS st_備考,
1627
+ st."バージョン" AS st_バージョン,
1628
+ st."作成日時" AS st_作成日時,
1629
+ st."更新日時" AS st_更新日時,
1630
+ std."ID" AS std_id,
1631
+ std."棚卸ID" AS std_棚卸ID,
1632
+ std."棚卸行番号" AS std_棚卸行番号,
1633
+ std."商品コード" AS std_商品コード,
1634
+ std."ロケーションコード" AS std_ロケーションコード,
1635
+ std."ロット番号" AS std_ロット番号,
1636
+ std."帳簿在庫数" AS std_帳簿在庫数,
1637
+ std."実棚数" AS std_実棚数,
1638
+ std."差異数" AS std_差異数,
1639
+ std."差異理由" AS std_差異理由,
1640
+ std."調整済フラグ" AS std_調整済フラグ
1641
+ FROM "棚卸データ" st
1642
+ LEFT JOIN "棚卸明細データ" std
1643
+ ON st."ID" = std."棚卸ID"
1644
+ WHERE st."棚卸番号" = #{stocktakingNumber}
1645
+ ORDER BY std."棚卸行番号"
1646
+ </select>
1647
+
1648
+ </mapper>
1649
+ ```
1650
+
1651
+ </details>
1652
+
1653
+ ### リレーション設定のポイント
1654
+
1655
+ | 設定項目 | 説明 |
1656
+ |---------|------|
1657
+ | `<collection>` | 1:N 関連のマッピング |
1658
+ | `<id>` | 主キーの識別(MyBatis が重複排除に使用) |
1659
+ | `resultMap` | ネストした ResultMap の参照 |
1660
+ | エイリアス(AS) | カラム名の重複を避けるためのプレフィックス |
1661
+ | `ORDER BY` | コレクションの順序を保証 |
1662
+
1663
+ ### 楽観ロック(Optimistic Locking)の実装
1664
+
1665
+ 複数ユーザーが同時に在庫データや棚卸データを編集する場合、データの整合性を保つために楽観ロックを実装します。
1666
+
1667
+ ```plantuml
1668
+ @startuml
1669
+
1670
+ title 楽観ロックの動作シーケンス
1671
+
1672
+ actor "担当者A" as userA
1673
+ actor "担当者B" as userB
1674
+ database "在庫テーブル" as db
1675
+
1676
+ userA -> db : SELECT(version=1)
1677
+ userB -> db : SELECT(version=1)
1678
+ userA -> userA : 在庫数を編集
1679
+ userB -> userB : 在庫数を編集
1680
+ userA -> db : UPDATE WHERE version=1\n→ version=2 に更新
1681
+ db -> userA : 更新成功(1行)
1682
+ userB -> db : UPDATE WHERE version=1
1683
+ db -> userB : 更新失敗(0行)\n楽観ロック例外
1684
+ userB -> userB : 再読み込みを促す
1685
+
1686
+ @enduml
1687
+ ```
1688
+
1689
+ ### バージョンカラムによる同時更新制御
1690
+
1691
+ <details>
1692
+ <summary>楽観ロック対応の MyBatis Mapper</summary>
1693
+
1694
+ ```xml
1695
+ <!-- 楽観ロック対応の在庫更新 -->
1696
+ <update id="updateWithOptimisticLock"
1697
+ parameterType="com.example.sms.domain.model.inventory.Inventory">
1698
+ UPDATE "在庫データ"
1699
+ SET
1700
+ "現在庫数" = #{currentQuantity},
1701
+ "引当数" = #{allocatedQuantity},
1702
+ "発注残数" = #{orderedQuantity},
1703
+ "最終入庫日" = #{lastReceiptDate},
1704
+ "最終出庫日" = #{lastShipmentDate},
1705
+ "更新日時" = CURRENT_TIMESTAMP,
1706
+ "バージョン" = "バージョン" + 1
1707
+ WHERE "ID" = #{id}
1708
+ AND "バージョン" = #{version}
1709
+ </update>
1710
+
1711
+ <!-- 楽観ロック対応の棚卸更新 -->
1712
+ <update id="updateStocktakingWithOptimisticLock"
1713
+ parameterType="com.example.sms.domain.model.inventory.Stocktaking">
1714
+ UPDATE "棚卸データ"
1715
+ SET
1716
+ "ステータス" = #{status,
1717
+ typeHandler=com.example.sms.infrastructure.persistence.typehandler.StocktakingStatusTypeHandler}::棚卸ステータス,
1718
+ "棚卸終了日時" = #{endDateTime},
1719
+ "備考" = #{remarks},
1720
+ "更新日時" = CURRENT_TIMESTAMP,
1721
+ "バージョン" = "バージョン" + 1
1722
+ WHERE "棚卸番号" = #{stocktakingNumber}
1723
+ AND "バージョン" = #{version}
1724
+ </update>
1725
+ ```
1726
+
1727
+ </details>
1728
+
1729
+ <details>
1730
+ <summary>Repository 実装:楽観ロック対応</summary>
1731
+
1732
+ ```java
1733
+ // src/main/java/com/example/sales/infrastructure/persistence/repository/InventoryRepositoryImpl.java
1734
+ package com.example.sms.infrastructure.persistence.repository;
1735
+
1736
+ import com.example.sms.application.port.out.InventoryRepository;
1737
+ import com.example.sms.domain.exception.OptimisticLockException;
1738
+ import com.example.sms.domain.model.inventory.*;
1739
+ import com.example.sms.infrastructure.persistence.mapper.InventoryMapper;
1740
+ import lombok.RequiredArgsConstructor;
1741
+ import org.springframework.stereotype.Repository;
1742
+ import org.springframework.transaction.annotation.Transactional;
1743
+
1744
+ import java.math.BigDecimal;
1745
+ import java.time.LocalDate;
1746
+ import java.util.List;
1747
+ import java.util.Optional;
1748
+
1749
+ @Repository
1750
+ @RequiredArgsConstructor
1751
+ public class InventoryRepositoryImpl implements InventoryRepository {
1752
+
1753
+ private final InventoryMapper inventoryMapper;
1754
+
1755
+ @Override
1756
+ public void save(Inventory inventory) {
1757
+ inventoryMapper.insert(inventory);
1758
+ }
1759
+
1760
+ @Override
1761
+ public Optional<Inventory> findByWarehouseAndProduct(String warehouseCode,
1762
+ String productCode) {
1763
+ List<Inventory> inventories = inventoryMapper
1764
+ .findByWarehouseAndProduct(warehouseCode, productCode);
1765
+ return inventories.isEmpty() ? Optional.empty() : Optional.of(inventories.get(0));
1766
+ }
1767
+
1768
+ @Override
1769
+ public List<Inventory> findByWarehouseCode(String warehouseCode) {
1770
+ return inventoryMapper.findByWarehouseCode(warehouseCode);
1771
+ }
1772
+
1773
+ @Override
1774
+ public List<Inventory> findByProductCode(String productCode) {
1775
+ return inventoryMapper.findByProductCode(productCode);
1776
+ }
1777
+
1778
+ /**
1779
+ * 楽観ロック対応の更新
1780
+ * @throws OptimisticLockException 他のユーザーによって更新された場合
1781
+ */
1782
+ @Override
1783
+ @Transactional
1784
+ public void update(Inventory inventory) {
1785
+ int updatedCount = inventoryMapper.updateWithOptimisticLock(inventory);
1786
+
1787
+ if (updatedCount == 0) {
1788
+ Integer currentVersion = inventoryMapper.findVersionById(inventory.getId());
1789
+
1790
+ if (currentVersion == null) {
1791
+ throw new OptimisticLockException("在庫", inventory.getId().toString());
1792
+ } else {
1793
+ throw new OptimisticLockException(
1794
+ "在庫",
1795
+ inventory.getId().toString(),
1796
+ inventory.getVersion(),
1797
+ currentVersion
1798
+ );
1799
+ }
1800
+ }
1801
+ }
1802
+
1803
+ @Override
1804
+ public void allocate(Integer id, BigDecimal quantity) {
1805
+ int updated = inventoryMapper.allocate(id, quantity);
1806
+ if (updated == 0) {
1807
+ throw new IllegalStateException("引当に失敗しました。有効在庫不足の可能性があります。");
1808
+ }
1809
+ }
1810
+
1811
+ @Override
1812
+ public void deallocate(Integer id, BigDecimal quantity) {
1813
+ int updated = inventoryMapper.deallocate(id, quantity);
1814
+ if (updated == 0) {
1815
+ throw new IllegalStateException("引当解除に失敗しました。");
1816
+ }
1817
+ }
1818
+
1819
+ @Override
1820
+ public void receive(Integer id, BigDecimal quantity, LocalDate receiptDate) {
1821
+ inventoryMapper.receive(id, quantity, receiptDate);
1822
+ }
1823
+
1824
+ @Override
1825
+ public void ship(Integer id, BigDecimal quantity, LocalDate shipmentDate) {
1826
+ int updated = inventoryMapper.ship(id, quantity, shipmentDate);
1827
+ if (updated == 0) {
1828
+ throw new IllegalStateException("出庫に失敗しました。在庫不足の可能性があります。");
1829
+ }
1830
+ }
1831
+
1832
+ @Override
1833
+ public void updateCurrentQuantity(Integer id, BigDecimal newQuantity, String reason) {
1834
+ inventoryMapper.updateCurrentQuantity(id, newQuantity);
1835
+ }
1836
+ }
1837
+ ```
1838
+
1839
+ </details>
1840
+
1841
+ ### 楽観ロックのテスト
1842
+
1843
+ <details>
1844
+ <summary>楽観ロックのテストコード</summary>
1845
+
1846
+ ```java
1847
+ @Nested
1848
+ @DisplayName("楽観ロックの在庫更新テスト")
1849
+ class OptimisticLockInventoryTest {
1850
+
1851
+ @Test
1852
+ @DisplayName("正しいバージョンで更新できる")
1853
+ void shouldUpdateWithCorrectVersion() {
1854
+ // Given: 在庫を登録
1855
+ var inventory = createTestInventory("WH001", "PROD-001");
1856
+ inventoryRepository.save(inventory);
1857
+
1858
+ // When: バージョン1で更新
1859
+ var saved = inventoryRepository
1860
+ .findByWarehouseAndProduct("WH001", "PROD-001").orElseThrow();
1861
+ assertThat(saved.getVersion()).isEqualTo(1);
1862
+ saved.setCurrentQuantity(new BigDecimal("150"));
1863
+ inventoryRepository.update(saved);
1864
+
1865
+ // Then: バージョンが2に増加
1866
+ var updated = inventoryRepository
1867
+ .findByWarehouseAndProduct("WH001", "PROD-001").orElseThrow();
1868
+ assertThat(updated.getVersion()).isEqualTo(2);
1869
+ assertThat(updated.getCurrentQuantity()).isEqualByComparingTo(new BigDecimal("150"));
1870
+ }
1871
+
1872
+ @Test
1873
+ @DisplayName("古いバージョンで更新すると楽観ロック例外がスローされる")
1874
+ void shouldThrowOptimisticLockExceptionWithOldVersion() {
1875
+ // Given: 在庫を登録して、別のセッションで更新
1876
+ var inventory = createTestInventory("WH001", "PROD-002");
1877
+ inventoryRepository.save(inventory);
1878
+
1879
+ var sessionA = inventoryRepository
1880
+ .findByWarehouseAndProduct("WH001", "PROD-002").orElseThrow();
1881
+ var sessionB = inventoryRepository
1882
+ .findByWarehouseAndProduct("WH001", "PROD-002").orElseThrow();
1883
+
1884
+ // セッションAで更新
1885
+ sessionA.setCurrentQuantity(new BigDecimal("200"));
1886
+ inventoryRepository.update(sessionA);
1887
+
1888
+ // When/Then: セッションBで更新すると例外
1889
+ sessionB.setCurrentQuantity(new BigDecimal("300"));
1890
+ assertThatThrownBy(() -> inventoryRepository.update(sessionB))
1891
+ .isInstanceOf(OptimisticLockException.class)
1892
+ .hasMessageContaining("他のユーザーによって更新されました");
1893
+ }
1894
+ }
1895
+ ```
1896
+
1897
+ </details>
1898
+
1899
+ ### 楽観ロックのベストプラクティス
1900
+
1901
+ | ポイント | 説明 |
1902
+ |---------|------|
1903
+ | **バージョンカラム** | INTEGER 型で十分(オーバーフローは現実的に発生しない) |
1904
+ | **WHERE 条件** | 必ず `AND "バージョン" = #{version}` を含める |
1905
+ | **インクリメント** | `"バージョン" = "バージョン" + 1` でアトミックに更新 |
1906
+ | **例外処理** | 更新件数が0の場合は楽観ロック例外をスロー |
1907
+ | **エラーメッセージ** | ユーザーにわかりやすいメッセージで再読み込みを促す |
1908
+
1909
+ ---
1910
+
1911
+ ## 第9章のまとめ
1912
+
1913
+ 本章では、在庫管理の核心部分である在庫情報管理、入出庫履歴、棚卸処理について学びました。
1914
+
1915
+ ### 学んだこと
1916
+
1917
+ 1. **在庫数量の種類**
1918
+ - 現在庫数、引当数、有効在庫数、発注残数の関係
1919
+
1920
+ 2. **倉庫・ロケーション管理**
1921
+ - 倉庫マスタの設計
1922
+ - ロケーションによる細かな在庫管理
1923
+
1924
+ 3. **入出庫履歴の設計**
1925
+ - 移動区分による分類
1926
+ - トレーサビリティの確保
1927
+
1928
+ 4. **棚卸業務の設計**
1929
+ - 帳簿在庫と実棚の差異管理
1930
+ - 在庫調整処理
1931
+
1932
+ 5. **リレーションと楽観ロック**
1933
+ - N+1 問題の回避
1934
+ - 同時更新の競合制御
1935
+
1936
+ ### 在庫管理の ER 図(全体像)
1937
+
1938
+ ```plantuml
1939
+ @startuml
1940
+
1941
+ title 在庫管理のER図
1942
+
1943
+ entity 倉庫マスタ {
1944
+ 倉庫コード <<PK>>
1945
+ --
1946
+ 倉庫名
1947
+ 倉庫区分
1948
+ }
1949
+
1950
+ entity ロケーションマスタ {
1951
+ ロケーションコード <<PK>>
1952
+ --
1953
+ 倉庫コード <<FK>>
1954
+ 棚番
1955
+ ゾーン
1956
+ }
1957
+
1958
+ entity 商品マスタ {
1959
+ 商品コード <<PK>>
1960
+ --
1961
+ 商品名
1962
+ }
1963
+
1964
+ entity 在庫データ {
1965
+ ID <<PK>>
1966
+ --
1967
+ 倉庫コード <<FK>>
1968
+ 商品コード <<FK>>
1969
+ ロケーションコード <<FK>>
1970
+ 現在庫数
1971
+ 引当数
1972
+ 発注残数
1973
+ ロット番号
1974
+ バージョン
1975
+ }
1976
+
1977
+ entity 入出庫履歴データ {
1978
+ ID <<PK>>
1979
+ --
1980
+ 倉庫コード <<FK>>
1981
+ 商品コード <<FK>>
1982
+ 移動日時
1983
+ 移動区分
1984
+ 移動数量
1985
+ 移動前在庫数
1986
+ 移動後在庫数
1987
+ 伝票番号
1988
+ }
1989
+
1990
+ entity 棚卸データ {
1991
+ ID <<PK>>
1992
+ --
1993
+ 棚卸番号
1994
+ 倉庫コード <<FK>>
1995
+ 棚卸日
1996
+ ステータス
1997
+ バージョン
1998
+ }
1999
+
2000
+ entity 棚卸明細データ {
2001
+ ID <<PK>>
2002
+ --
2003
+ 棚卸ID <<FK>>
2004
+ 棚卸行番号
2005
+ 商品コード <<FK>>
2006
+ 帳簿在庫数
2007
+ 実棚数
2008
+ 差異数
2009
+ 調整済フラグ
2010
+ }
2011
+
2012
+ 倉庫マスタ ||--o{ ロケーションマスタ
2013
+ 倉庫マスタ ||--o{ 在庫データ
2014
+ 商品マスタ ||--o{ 在庫データ
2015
+ ロケーションマスタ ||--o{ 在庫データ
2016
+ 倉庫マスタ ||--o{ 入出庫履歴データ
2017
+ 商品マスタ ||--o{ 入出庫履歴データ
2018
+ 倉庫マスタ ||--o{ 棚卸データ
2019
+ 棚卸データ ||--o{ 棚卸明細データ
2020
+ 商品マスタ ||--o{ 棚卸明細データ
2021
+
2022
+ @enduml
2023
+ ```
2024
+
2025
+ ### 次章の予告
2026
+
2027
+ 第10章では、債務管理の設計に進みます。買掛金残高の管理、支払処理のデータベース設計と実装を行います。
2028
+
2029
+ ---
2030
+
2031
+ [← 第8章:調達管理の設計](./chapter08.md) | [第10章:債務管理の設計 →](./chapter10.md)