@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,1584 +1,1584 @@
1
- # 第17章:自動仕訳の設計(ORM版)
2
-
3
- 販売管理システムなどの業務システムから会計システムへの自動仕訳処理を TDD で設計していきます。売上データから仕訳データへの自動変換ルールと、効率的なバッチ処理の設計を行います。
4
-
5
- JPA 版では、@Entity によるリレーション定義、@EntityGraph による N+1 問題対策、Spring Data JPA Specification による動的クエリを活用します。
6
-
7
- ---
8
-
9
- ## 17.1 自動仕訳の概要
10
-
11
- ### 自動仕訳とは
12
-
13
- 自動仕訳は、業務システムのトランザクションデータを会計システムの仕訳データへ自動的に変換する機能です。
14
-
15
- #### 従来のアナログ連携(手入力による連携)
16
-
17
- 従来の方式では、販売管理システムで発行した売上伝票を紙で経理部門に回送し、経理担当者が仕訳入力を行っていました。
18
-
19
- ```plantuml
20
- @startuml
21
- title 販売管理システムと会計システムの非連携(アナログ連携)
22
-
23
- skinparam rectangle {
24
- BackgroundColor White
25
- BorderColor Black
26
- }
27
- skinparam database {
28
- BackgroundColor #E8F5E9
29
- BorderColor Black
30
- }
31
-
32
- package "営業部門" {
33
- rectangle "売上入力" as SalesInput <<Screen>>
34
- database "売上データ" as SalesDB
35
- rectangle "売上伝票" as SalesVoucher <<Document>>
36
- }
37
-
38
- package "経理部門" {
39
- rectangle "仕訳入力" as JournalInput <<Screen>>
40
- database "仕訳データ" as JournalDB
41
- rectangle "損益計算書\n貸借対照表" as FinancialReports <<Document>>
42
- }
43
-
44
- SalesInput -down-> SalesDB
45
- SalesDB -down-> SalesVoucher
46
-
47
- SalesVoucher -[#red,bold]right-> JournalInput : **紙伝票の回送**\n**(手入力による再登録)**
48
-
49
- JournalInput -down-> JournalDB
50
- JournalDB -down-> FinancialReports
51
-
52
- @enduml
53
- ```
54
-
55
- **アナログ連携の問題点:**
56
-
57
- | 問題 | 説明 |
58
- |-----|------|
59
- | 二重入力 | 営業部門と経理部門で同じ情報を入力する作業負荷 |
60
- | 入力ミス | 手入力によるデータ不整合 |
61
- | タイムラグ | 紙伝票の回送による情報遅延 |
62
- | 消費税計算ミス | 手計算による誤り |
63
- | 勘定科目選択ミス | 担当者の判断ばらつき |
64
-
65
- #### 自動仕訳によるデジタル連携
66
-
67
- 自動仕訳処理を導入することで、売上データから仕訳データへの変換を自動化します。
68
-
69
- ```plantuml
70
- @startuml
71
- title 自動仕訳処理のデータの流れ
72
-
73
- skinparam rectangle {
74
- BackgroundColor White
75
- BorderColor Black
76
- }
77
- skinparam database {
78
- BackgroundColor #E8F5E9
79
- BorderColor Black
80
- }
81
-
82
- package "営業部門" {
83
- rectangle "売上入力" as SalesInput <<Screen>>
84
- database "売上データ" as SalesDB
85
- rectangle "売上\nチェックリスト" as SalesList <<Document>>
86
- }
87
-
88
- package "自動仕訳機能" #F1F8E9 {
89
- database "自動仕訳\nパターンマスタ" as PatternMaster
90
- rectangle "自動仕訳処理" as AutoJournalProcess <<Process>> #4CAF50
91
- database "自動仕訳データ" as AutoJournalDB
92
- database "エラーデータ" as ErrorDB
93
- rectangle "エラーリスト" as ErrorList <<Document>>
94
- rectangle "自動仕訳\nチェックリスト" as AutoJournalList <<Document>>
95
-
96
- rectangle "転記処理" as PostingProcess <<Process>> #4CAF50
97
- circle " " as Switch
98
- }
99
-
100
- package "経理部門" {
101
- rectangle "仕訳入力" as JournalInput <<Screen>>
102
- database "仕訳データ" as JournalDB
103
- }
104
-
105
- SalesInput -down-> SalesDB
106
- SalesDB -down-> SalesList
107
-
108
- SalesDB -right-> AutoJournalProcess
109
- PatternMaster -down-> AutoJournalProcess
110
- AutoJournalProcess -right-> AutoJournalDB
111
- AutoJournalProcess -down-> ErrorDB
112
- ErrorDB -down-> ErrorList
113
- AutoJournalDB -down-> AutoJournalList
114
-
115
- AutoJournalDB -right-> Switch
116
- note top of Switch : (転記指示)
117
- Switch -right-> PostingProcess
118
- PostingProcess -right-> JournalDB
119
-
120
- JournalInput -down-> JournalDB
121
-
122
- @enduml
123
- ```
124
-
125
- ### 自動仕訳処理の流れ
126
-
127
- ```plantuml
128
- @startuml
129
-
130
- title 自動仕訳処理フロー
131
-
132
- |営業部門|
133
- start
134
- :売上登録;
135
-
136
- |自動仕訳処理|
137
- :売上データ抽出;
138
- note right
139
- 未処理の売上データを
140
- 抽出条件に基づいて取得
141
- end note
142
-
143
- :パターンマッチング;
144
- note right
145
- 商品区分・顧客区分から
146
- 適用する仕訳パターンを決定
147
- end note
148
-
149
- if (パターン適合?) then (はい)
150
- :仕訳データ生成;
151
- :自動仕訳データに保存;
152
- else (いいえ)
153
- :エラーデータに保存;
154
- :エラーリスト出力;
155
- stop
156
- endif
157
-
158
- :売上データに処理済フラグ設定;
159
-
160
- |経理部門|
161
- :自動仕訳チェックリスト確認;
162
-
163
- if (確認OK?) then (はい)
164
- :転記処理実行;
165
- :仕訳データに登録;
166
- else (いいえ)
167
- :修正・再処理;
168
- stop
169
- endif
170
-
171
- :仕訳完了;
172
- stop
173
-
174
- @enduml
175
- ```
176
-
177
- ### 売上伝票から仕訳伝票への変換例
178
-
179
- #### 売上伝票(ビジネス上の事実)
180
-
181
- | 項目 | 値 |
182
- |-----|-----|
183
- | 伝票日付 | 2024/04/01 |
184
- | 顧客 | DBMフード新宿 |
185
- | 明細1 | いちご蒸缶 1,000個 × 1,000円 = 1,000,000円 |
186
- | 明細2 | アスパラ 200個 × 10,000円 = 2,000,000円 |
187
- | 明細3 | さざえのエスカルゴ 1,500個 × 100円 = 150,000円 |
188
- | 合計 | 3,150,000円(税抜)+ 消費税 315,000円 = 3,465,000円(税込) |
189
-
190
- #### 仕訳伝票(会計上の記録)
191
-
192
- | 借方科目 | 借方金額 | 貸方科目 | 貸方金額 | 摘要 |
193
- |---------|---------|---------|---------|-----|
194
- | 売掛金/DBMフード本社 | 3,465,000 | | | 売上計上 |
195
- | | | 売上加工品/DBMフード新宿 | 1,150,000 | いちご蒸缶・さざえ |
196
- | | | 売上生鮮品/DBMフード新宿 | 2,000,000 | アスパラ |
197
- | | | 仮受消費税 | 315,000 | 消費税10% |
198
-
199
- ---
200
-
201
- ## 17.2 自動仕訳テーブルの設計
202
-
203
- ### フラグ管理方式と日付管理方式
204
-
205
- 売上データの処理状態を管理する方式には、2つのアプローチがあります。
206
-
207
- #### フラグ管理方式
208
-
209
- - **メリット**:シンプルな実装、処理状態が明確
210
- - **デメリット**:再処理時にフラグリセットが必要、大量データでの更新負荷
211
-
212
- #### 日付管理方式
213
-
214
- - **メリット**:差分処理が容易、再処理時の柔軟性
215
- - **デメリット**:管理テーブルが必要、更新日時の整合性管理が必要
216
-
217
- ### セット中心のアプリケーション設計
218
-
219
- 大量データを効率的に処理するには、ループ処理よりもセット中心処理が有効です。
220
-
221
- | 方式 | 処理方法 | DB操作回数 |
222
- |-----|---------|-----------|
223
- | ループ処理 | 売上データを1件ずつ読み込み、各売上に対してパターンマスタを検索、1件ずつ仕訳データを挿入 | N×M回 |
224
- | セット中心処理 | パターンマスタを1件読み込み、該当パターンの売上データを一括INSERT | M回(パターン数) |
225
-
226
- ### 自動仕訳関連テーブルの ER 図
227
-
228
- ```plantuml
229
- @startuml
230
-
231
- entity "自動仕訳パターンマスタ" as AutoJournalPattern {
232
- * **パターンコード**: VARCHAR(10) <<PK>>
233
- --
234
- * **パターン名**: VARCHAR(50)
235
- * **商品グループ**: VARCHAR(10)
236
- * **顧客グループ**: VARCHAR(10)
237
- * **売上区分**: VARCHAR(2)
238
- * **借方勘定科目コード**: VARCHAR(5) <<FK>>
239
- * **借方補助科目設定**: VARCHAR(20)
240
- * **貸方勘定科目コード**: VARCHAR(5) <<FK>>
241
- * **貸方補助科目設定**: VARCHAR(20)
242
- * **返品時借方科目コード**: VARCHAR(5)
243
- * **返品時貸方科目コード**: VARCHAR(5)
244
- * **消費税処理区分**: VARCHAR(2)
245
- * **有効開始日**: DATE
246
- * **有効終了日**: DATE
247
- * **優先順位**: INTEGER
248
- }
249
-
250
- entity "自動仕訳データ" as AutoJournal {
251
- * **自動仕訳番号**: VARCHAR(15) <<PK>>
252
- --
253
- * **売上番号**: VARCHAR(10) <<FK>>
254
- * **売上行番号**: SMALLINT
255
- * **パターンコード**: VARCHAR(10) <<FK>>
256
- * **起票日**: DATE
257
- * **仕訳行貸借区分**: VARCHAR(1)
258
- * **勘定科目コード**: VARCHAR(5) <<FK>>
259
- * **補助科目コード**: VARCHAR(10)
260
- * **部門コード**: VARCHAR(5)
261
- * **仕訳金額**: DECIMAL(14,2)
262
- * **消費税額**: DECIMAL(14,2)
263
- * **処理ステータス**: 自動仕訳ステータス
264
- * **転記済フラグ**: SMALLINT
265
- }
266
-
267
- entity "自動仕訳処理履歴" as AutoJournalHistory {
268
- * **処理番号**: VARCHAR(15) <<PK>>
269
- --
270
- * **処理日時**: TIMESTAMP
271
- * **処理対象開始日**: DATE
272
- * **処理対象終了日**: DATE
273
- * **処理件数**: INTEGER
274
- * **成功件数**: INTEGER
275
- * **エラー件数**: INTEGER
276
- * **処理金額合計**: DECIMAL(15,2)
277
- }
278
-
279
- entity "勘定科目マスタ" as Account {
280
- * **勘定科目コード** <<PK>>
281
- --
282
- 勘定科目名
283
- ...
284
- }
285
-
286
- AutoJournalPattern }o--|| Account : 借方科目
287
- AutoJournalPattern }o--|| Account : 貸方科目
288
- AutoJournal }o--|| AutoJournalPattern
289
- AutoJournal }o--|| Account
290
-
291
- @enduml
292
- ```
293
-
294
- ---
295
-
296
- ## 17.3 MyBatis 版との実装比較
297
-
298
- ### データアクセス層の比較
299
-
300
- | 機能 | MyBatis 版 | JPA 版 |
301
- |-----|-----------|--------|
302
- | ENUM 型変換 | TypeHandler | AttributeConverter |
303
- | 関連エンティティ取得 | resultMap の association | @ManyToOne |
304
- | N+1 問題対策 | JOIN を含む SQL | @EntityGraph |
305
- | 動的クエリ | XML の `<if>` タグ | Specification パターン |
306
- | バッチ INSERT | foreach 句 | saveAll() |
307
- | 有効期間検索 | WHERE 句で条件指定 | JPQL で条件指定 |
308
-
309
- ### Flyway マイグレーション
310
-
311
- <details>
312
- <summary>V007__create_auto_journal_tables.sql</summary>
313
-
314
- ```sql
315
- -- 自動仕訳処理ステータス
316
- CREATE TYPE 自動仕訳ステータス AS ENUM ('処理待ち', '処理中', '処理完了', '転記済', 'エラー');
317
-
318
- -- 自動仕訳パターンマスタ
319
- CREATE TABLE "自動仕訳パターンマスタ" (
320
- "パターンコード" VARCHAR(10) PRIMARY KEY,
321
- "パターン名" VARCHAR(50) NOT NULL,
322
- "商品グループ" VARCHAR(10) DEFAULT 'ALL',
323
- "顧客グループ" VARCHAR(10) DEFAULT 'ALL',
324
- "売上区分" VARCHAR(2) DEFAULT '01',
325
- "借方勘定科目コード" VARCHAR(5) NOT NULL,
326
- "借方補助科目設定" VARCHAR(20),
327
- "貸方勘定科目コード" VARCHAR(5) NOT NULL,
328
- "貸方補助科目設定" VARCHAR(20),
329
- "返品時借方科目コード" VARCHAR(5),
330
- "返品時貸方科目コード" VARCHAR(5),
331
- "消費税処理区分" VARCHAR(2) DEFAULT '01',
332
- "有効開始日" DATE DEFAULT CURRENT_DATE,
333
- "有効終了日" DATE DEFAULT '9999-12-31',
334
- "優先順位" INTEGER DEFAULT 100,
335
- "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
336
- "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
337
- CONSTRAINT "fk_自動仕訳パターン_借方科目"
338
- FOREIGN KEY ("借方勘定科目コード") REFERENCES "勘定科目マスタ"("勘定科目コード"),
339
- CONSTRAINT "fk_自動仕訳パターン_貸方科目"
340
- FOREIGN KEY ("貸方勘定科目コード") REFERENCES "勘定科目マスタ"("勘定科目コード")
341
- );
342
-
343
- -- 自動仕訳データ
344
- CREATE TABLE "自動仕訳データ" (
345
- "自動仕訳番号" VARCHAR(15) PRIMARY KEY,
346
- "売上番号" VARCHAR(10) NOT NULL,
347
- "売上行番号" SMALLINT NOT NULL,
348
- "パターンコード" VARCHAR(10) NOT NULL,
349
- "起票日" DATE NOT NULL,
350
- "仕訳行貸借区分" 仕訳行貸借区分 NOT NULL,
351
- "勘定科目コード" VARCHAR(5) NOT NULL,
352
- "補助科目コード" VARCHAR(10),
353
- "部門コード" VARCHAR(5),
354
- "仕訳金額" DECIMAL(14,2) NOT NULL,
355
- "消費税額" DECIMAL(14,2) DEFAULT 0,
356
- "処理ステータス" 自動仕訳ステータス DEFAULT '処理待ち' NOT NULL,
357
- "転記済フラグ" SMALLINT DEFAULT 0,
358
- "転記日" DATE,
359
- "仕訳伝票番号" VARCHAR(10),
360
- "エラーコード" VARCHAR(10),
361
- "エラーメッセージ" VARCHAR(200),
362
- "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
363
- "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL
364
- );
365
-
366
- -- 自動仕訳処理履歴
367
- CREATE TABLE "自動仕訳処理履歴" (
368
- "処理番号" VARCHAR(15) PRIMARY KEY,
369
- "処理日時" TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
370
- "処理対象開始日" DATE NOT NULL,
371
- "処理対象終了日" DATE NOT NULL,
372
- "処理件数" INTEGER DEFAULT 0,
373
- "成功件数" INTEGER DEFAULT 0,
374
- "エラー件数" INTEGER DEFAULT 0,
375
- "処理金額合計" DECIMAL(15,2) DEFAULT 0,
376
- "処理者" VARCHAR(50),
377
- "備考" TEXT,
378
- "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL
379
- );
380
-
381
- -- インデックス
382
- CREATE INDEX "idx_自動仕訳パターン_優先順位" ON "自動仕訳パターンマスタ"("優先順位");
383
- CREATE INDEX "idx_自動仕訳_売上番号" ON "自動仕訳データ"("売上番号");
384
- CREATE INDEX "idx_自動仕訳_処理ステータス" ON "自動仕訳データ"("処理ステータス");
385
- CREATE INDEX "idx_自動仕訳_転記済フラグ" ON "自動仕訳データ"("転記済フラグ");
386
- ```
387
-
388
- </details>
389
-
390
- ---
391
-
392
- ## 17.4 JPA エンティティの実装
393
-
394
- ### 自動仕訳ステータス Enum と AttributeConverter
395
-
396
- <details>
397
- <summary>AutoJournalStatus.java</summary>
398
-
399
- ```java
400
- package com.example.accounting.domain.model.autojournal;
401
-
402
- import lombok.Getter;
403
- import lombok.RequiredArgsConstructor;
404
-
405
- @Getter
406
- @RequiredArgsConstructor
407
- public enum AutoJournalStatus {
408
- PENDING("処理待ち"),
409
- PROCESSING("処理中"),
410
- COMPLETED("処理完了"),
411
- POSTED("転記済"),
412
- ERROR("エラー");
413
-
414
- private final String displayName;
415
-
416
- public static AutoJournalStatus fromDisplayName(String displayName) {
417
- for (AutoJournalStatus status : values()) {
418
- if (status.displayName.equals(displayName)) {
419
- return status;
420
- }
421
- }
422
- throw new IllegalArgumentException("Unknown auto journal status: " + displayName);
423
- }
424
- }
425
- ```
426
-
427
- </details>
428
-
429
- <details>
430
- <summary>AutoJournalStatusConverter.java</summary>
431
-
432
- ```java
433
- package com.example.accounting.infrastructure.persistence.converter;
434
-
435
- import com.example.accounting.domain.model.autojournal.AutoJournalStatus;
436
- import jakarta.persistence.AttributeConverter;
437
- import jakarta.persistence.Converter;
438
-
439
- /**
440
- * 自動仕訳ステータス ↔ PostgreSQL ENUM 変換
441
- * JPA では AttributeConverter を使用して ENUM 型を変換する
442
- */
443
- @Converter(autoApply = true)
444
- public class AutoJournalStatusConverter implements AttributeConverter<AutoJournalStatus, String> {
445
-
446
- @Override
447
- public String convertToDatabaseColumn(AutoJournalStatus attribute) {
448
- return attribute != null ? attribute.getDisplayName() : null;
449
- }
450
-
451
- @Override
452
- public AutoJournalStatus convertToEntityAttribute(String dbData) {
453
- return dbData != null ? AutoJournalStatus.fromDisplayName(dbData) : null;
454
- }
455
- }
456
- ```
457
-
458
- </details>
459
-
460
- ### 自動仕訳パターンマスタ Entity
461
-
462
- <details>
463
- <summary>AutoJournalPattern.java</summary>
464
-
465
- ```java
466
- package com.example.accounting.domain.model.autojournal;
467
-
468
- import com.example.accounting.domain.model.account.Account;
469
- import jakarta.persistence.*;
470
- import lombok.*;
471
-
472
- import java.time.LocalDate;
473
- import java.time.LocalDateTime;
474
-
475
- /**
476
- * 自動仕訳パターンマスタ Entity
477
- *
478
- * JPA 版では @ManyToOne で勘定科目との関連を定義
479
- * MyBatis 版では resultMap で手動マッピングしていたが、
480
- * JPA では @JoinColumn で外部キー関連を自動的に解決
481
- */
482
- @Entity
483
- @Table(name = "自動仕訳パターンマスタ")
484
- @Data
485
- @Builder
486
- @NoArgsConstructor
487
- @AllArgsConstructor
488
- public class AutoJournalPattern {
489
-
490
- @Id
491
- @Column(name = "パターンコード", length = 10)
492
- private String patternCode;
493
-
494
- @Column(name = "パターン名", length = 50, nullable = false)
495
- private String patternName;
496
-
497
- @Column(name = "商品グループ", length = 10)
498
- private String productGroup;
499
-
500
- @Column(name = "顧客グループ", length = 10)
501
- private String customerGroup;
502
-
503
- @Column(name = "売上区分", length = 2)
504
- private String salesType;
505
-
506
- /**
507
- * 借方勘定科目との関連
508
- * @ManyToOne で遅延ロード(LAZY)を指定
509
- * N+1 問題が発生する場合は @EntityGraph で対策
510
- */
511
- @ManyToOne(fetch = FetchType.LAZY)
512
- @JoinColumn(name = "借方勘定科目コード", referencedColumnName = "勘定科目コード",
513
- insertable = false, updatable = false)
514
- private Account debitAccount;
515
-
516
- @Column(name = "借方勘定科目コード", length = 5, nullable = false)
517
- private String debitAccountCode;
518
-
519
- @Column(name = "借方補助科目設定", length = 20)
520
- private String debitSubAccountSetting;
521
-
522
- /**
523
- * 貸方勘定科目との関連
524
- */
525
- @ManyToOne(fetch = FetchType.LAZY)
526
- @JoinColumn(name = "貸方勘定科目コード", referencedColumnName = "勘定科目コード",
527
- insertable = false, updatable = false)
528
- private Account creditAccount;
529
-
530
- @Column(name = "貸方勘定科目コード", length = 5, nullable = false)
531
- private String creditAccountCode;
532
-
533
- @Column(name = "貸方補助科目設定", length = 20)
534
- private String creditSubAccountSetting;
535
-
536
- @Column(name = "返品時借方科目コード", length = 5)
537
- private String returnDebitAccountCode;
538
-
539
- @Column(name = "返品時貸方科目コード", length = 5)
540
- private String returnCreditAccountCode;
541
-
542
- @Column(name = "消費税処理区分", length = 2)
543
- private String taxProcessingType;
544
-
545
- @Column(name = "有効開始日")
546
- private LocalDate validFrom;
547
-
548
- @Column(name = "有効終了日")
549
- private LocalDate validTo;
550
-
551
- @Column(name = "優先順位")
552
- private Integer priority;
553
-
554
- @Column(name = "作成日時", nullable = false, updatable = false)
555
- private LocalDateTime createdAt;
556
-
557
- @Column(name = "更新日時", nullable = false)
558
- private LocalDateTime updatedAt;
559
-
560
- /**
561
- * 指定日付に有効かどうかを判定
562
- */
563
- public boolean isValidAt(LocalDate date) {
564
- return !date.isBefore(validFrom) && !date.isAfter(validTo);
565
- }
566
-
567
- /**
568
- * 商品グループと顧客グループにマッチするか判定
569
- */
570
- public boolean matches(String productGroup, String customerGroup) {
571
- boolean productMatch = "ALL".equals(this.productGroup) ||
572
- this.productGroup.equals(productGroup);
573
- boolean customerMatch = "ALL".equals(this.customerGroup) ||
574
- this.customerGroup.equals(customerGroup);
575
- return productMatch && customerMatch;
576
- }
577
-
578
- @PrePersist
579
- protected void onCreate() {
580
- LocalDateTime now = LocalDateTime.now();
581
- this.createdAt = now;
582
- this.updatedAt = now;
583
- if (this.validFrom == null) {
584
- this.validFrom = LocalDate.now();
585
- }
586
- if (this.validTo == null) {
587
- this.validTo = LocalDate.of(9999, 12, 31);
588
- }
589
- if (this.priority == null) {
590
- this.priority = 100;
591
- }
592
- }
593
-
594
- @PreUpdate
595
- protected void onUpdate() {
596
- this.updatedAt = LocalDateTime.now();
597
- }
598
- }
599
- ```
600
-
601
- </details>
602
-
603
- ### 自動仕訳データ Entity
604
-
605
- <details>
606
- <summary>AutoJournalEntry.java</summary>
607
-
608
- ```java
609
- package com.example.accounting.domain.model.autojournal;
610
-
611
- import com.example.accounting.domain.model.account.Account;
612
- import com.example.accounting.domain.model.department.Department;
613
- import com.example.accounting.domain.model.journal.DebitCreditType;
614
- import com.example.accounting.infrastructure.persistence.converter.AutoJournalStatusConverter;
615
- import com.example.accounting.infrastructure.persistence.converter.DebitCreditTypeConverter;
616
- import jakarta.persistence.*;
617
- import lombok.*;
618
-
619
- import java.math.BigDecimal;
620
- import java.time.LocalDate;
621
- import java.time.LocalDateTime;
622
-
623
- /**
624
- * 自動仕訳データ Entity
625
- *
626
- * JPA 版では @ManyToOne で関連エンティティを定義
627
- * 状態管理には Enum + AttributeConverter を使用
628
- */
629
- @Entity
630
- @Table(name = "自動仕訳データ")
631
- @Data
632
- @Builder
633
- @NoArgsConstructor
634
- @AllArgsConstructor
635
- public class AutoJournalEntry {
636
-
637
- @Id
638
- @Column(name = "自動仕訳番号", length = 15)
639
- private String autoJournalNumber;
640
-
641
- @Column(name = "売上番号", length = 10, nullable = false)
642
- private String salesNumber;
643
-
644
- @Column(name = "売上行番号", nullable = false)
645
- private Integer salesLineNumber;
646
-
647
- /**
648
- * パターンマスタとの関連
649
- */
650
- @ManyToOne(fetch = FetchType.LAZY)
651
- @JoinColumn(name = "パターンコード", referencedColumnName = "パターンコード",
652
- insertable = false, updatable = false)
653
- private AutoJournalPattern pattern;
654
-
655
- @Column(name = "パターンコード", length = 10, nullable = false)
656
- private String patternCode;
657
-
658
- @Column(name = "起票日", nullable = false)
659
- private LocalDate postingDate;
660
-
661
- /**
662
- * 仕訳行貸借区分
663
- * AttributeConverter で PostgreSQL ENUM と Java Enum を変換
664
- */
665
- @Convert(converter = DebitCreditTypeConverter.class)
666
- @Column(name = "仕訳行貸借区分", nullable = false, columnDefinition = "仕訳行貸借区分")
667
- private DebitCreditType debitCreditType;
668
-
669
- /**
670
- * 勘定科目との関連
671
- */
672
- @ManyToOne(fetch = FetchType.LAZY)
673
- @JoinColumn(name = "勘定科目コード", referencedColumnName = "勘定科目コード",
674
- insertable = false, updatable = false)
675
- private Account account;
676
-
677
- @Column(name = "勘定科目コード", length = 5, nullable = false)
678
- private String accountCode;
679
-
680
- @Column(name = "補助科目コード", length = 10)
681
- private String subAccountCode;
682
-
683
- /**
684
- * 部門との関連
685
- */
686
- @ManyToOne(fetch = FetchType.LAZY)
687
- @JoinColumn(name = "部門コード", referencedColumnName = "部門コード",
688
- insertable = false, updatable = false)
689
- private Department department;
690
-
691
- @Column(name = "部門コード", length = 5)
692
- private String departmentCode;
693
-
694
- @Column(name = "仕訳金額", nullable = false, precision = 14, scale = 2)
695
- private BigDecimal amount;
696
-
697
- @Column(name = "消費税額", precision = 14, scale = 2)
698
- private BigDecimal taxAmount;
699
-
700
- /**
701
- * 処理ステータス
702
- * AttributeConverter で PostgreSQL ENUM と Java Enum を変換
703
- */
704
- @Convert(converter = AutoJournalStatusConverter.class)
705
- @Column(name = "処理ステータス", nullable = false, columnDefinition = "自動仕訳ステータス")
706
- private AutoJournalStatus status;
707
-
708
- @Column(name = "転記済フラグ")
709
- private Boolean postedFlag;
710
-
711
- @Column(name = "転記日")
712
- private LocalDate postedDate;
713
-
714
- @Column(name = "仕訳伝票番号", length = 10)
715
- private String journalVoucherNumber;
716
-
717
- @Column(name = "エラーコード", length = 10)
718
- private String errorCode;
719
-
720
- @Column(name = "エラーメッセージ", length = 200)
721
- private String errorMessage;
722
-
723
- @Column(name = "作成日時", nullable = false, updatable = false)
724
- private LocalDateTime createdAt;
725
-
726
- @Column(name = "更新日時", nullable = false)
727
- private LocalDateTime updatedAt;
728
-
729
- @PrePersist
730
- protected void onCreate() {
731
- LocalDateTime now = LocalDateTime.now();
732
- this.createdAt = now;
733
- this.updatedAt = now;
734
- if (this.status == null) {
735
- this.status = AutoJournalStatus.PENDING;
736
- }
737
- if (this.postedFlag == null) {
738
- this.postedFlag = false;
739
- }
740
- if (this.taxAmount == null) {
741
- this.taxAmount = BigDecimal.ZERO;
742
- }
743
- }
744
-
745
- @PreUpdate
746
- protected void onUpdate() {
747
- this.updatedAt = LocalDateTime.now();
748
- }
749
- }
750
- ```
751
-
752
- </details>
753
-
754
- ### 自動仕訳処理履歴 Entity
755
-
756
- <details>
757
- <summary>AutoJournalHistory.java</summary>
758
-
759
- ```java
760
- package com.example.accounting.domain.model.autojournal;
761
-
762
- import jakarta.persistence.*;
763
- import lombok.*;
764
-
765
- import java.math.BigDecimal;
766
- import java.time.LocalDate;
767
- import java.time.LocalDateTime;
768
-
769
- /**
770
- * 自動仕訳処理履歴 Entity
771
- */
772
- @Entity
773
- @Table(name = "自動仕訳処理履歴")
774
- @Data
775
- @Builder
776
- @NoArgsConstructor
777
- @AllArgsConstructor
778
- public class AutoJournalHistory {
779
-
780
- @Id
781
- @Column(name = "処理番号", length = 15)
782
- private String processNumber;
783
-
784
- @Column(name = "処理日時", nullable = false)
785
- private LocalDateTime processDateTime;
786
-
787
- @Column(name = "処理対象開始日", nullable = false)
788
- private LocalDate targetFromDate;
789
-
790
- @Column(name = "処理対象終了日", nullable = false)
791
- private LocalDate targetToDate;
792
-
793
- @Column(name = "処理件数")
794
- private Integer totalCount;
795
-
796
- @Column(name = "成功件数")
797
- private Integer successCount;
798
-
799
- @Column(name = "エラー件数")
800
- private Integer errorCount;
801
-
802
- @Column(name = "処理金額合計", precision = 15, scale = 2)
803
- private BigDecimal totalAmount;
804
-
805
- @Column(name = "処理者", length = 50)
806
- private String processedBy;
807
-
808
- @Column(name = "備考", columnDefinition = "TEXT")
809
- private String remarks;
810
-
811
- @Column(name = "作成日時", nullable = false, updatable = false)
812
- private LocalDateTime createdAt;
813
-
814
- @PrePersist
815
- protected void onCreate() {
816
- this.createdAt = LocalDateTime.now();
817
- if (this.totalCount == null) {
818
- this.totalCount = 0;
819
- }
820
- if (this.successCount == null) {
821
- this.successCount = 0;
822
- }
823
- if (this.errorCount == null) {
824
- this.errorCount = 0;
825
- }
826
- if (this.totalAmount == null) {
827
- this.totalAmount = BigDecimal.ZERO;
828
- }
829
- }
830
- }
831
- ```
832
-
833
- </details>
834
-
835
- ---
836
-
837
- ## 17.5 TDD によるパターンマッチングの実装
838
-
839
- ### パターンマッチングのテスト
840
-
841
- <details>
842
- <summary>AutoJournalPatternTest.java</summary>
843
-
844
- ```java
845
- package com.example.accounting.domain.model.autojournal;
846
-
847
- import org.junit.jupiter.api.DisplayName;
848
- import org.junit.jupiter.api.Nested;
849
- import org.junit.jupiter.api.Test;
850
-
851
- import java.time.LocalDate;
852
-
853
- import static org.assertj.core.api.Assertions.*;
854
-
855
- @DisplayName("自動仕訳パターンのテスト")
856
- class AutoJournalPatternTest {
857
-
858
- @Nested
859
- @DisplayName("パターンマッチング")
860
- class PatternMatchingTest {
861
-
862
- @Test
863
- @DisplayName("商品グループALLは全ての商品グループにマッチする")
864
- void shouldMatchAllProductGroups() {
865
- // Given: 商品グループALLのパターン
866
- var pattern = AutoJournalPattern.builder()
867
- .patternCode("P001")
868
- .productGroup("ALL")
869
- .customerGroup("ALL")
870
- .build();
871
-
872
- // When & Then
873
- assertThat(pattern.matches("加工品", "一般")).isTrue();
874
- assertThat(pattern.matches("生鮮品", "一般")).isTrue();
875
- assertThat(pattern.matches("雑貨", "特約店")).isTrue();
876
- }
877
-
878
- @Test
879
- @DisplayName("特定の商品グループのみにマッチする")
880
- void shouldMatchSpecificProductGroup() {
881
- // Given: 加工品専用パターン
882
- var pattern = AutoJournalPattern.builder()
883
- .patternCode("P002")
884
- .productGroup("加工品")
885
- .customerGroup("ALL")
886
- .build();
887
-
888
- // When & Then
889
- assertThat(pattern.matches("加工品", "一般")).isTrue();
890
- assertThat(pattern.matches("生鮮品", "一般")).isFalse();
891
- }
892
-
893
- @Test
894
- @DisplayName("商品グループと顧客グループの両方でマッチングする")
895
- void shouldMatchBothProductAndCustomerGroup() {
896
- // Given: 特定の組み合わせパターン
897
- var pattern = AutoJournalPattern.builder()
898
- .patternCode("P003")
899
- .productGroup("加工品")
900
- .customerGroup("特約店")
901
- .build();
902
-
903
- // When & Then
904
- assertThat(pattern.matches("加工品", "特約店")).isTrue();
905
- assertThat(pattern.matches("加工品", "一般")).isFalse();
906
- assertThat(pattern.matches("生鮮品", "特約店")).isFalse();
907
- }
908
- }
909
-
910
- @Nested
911
- @DisplayName("有効期間チェック")
912
- class ValidityCheckTest {
913
-
914
- @Test
915
- @DisplayName("有効期間内の日付でtrueを返す")
916
- void shouldReturnTrueForValidDate() {
917
- // Given
918
- var pattern = AutoJournalPattern.builder()
919
- .patternCode("P001")
920
- .validFrom(LocalDate.of(2024, 1, 1))
921
- .validTo(LocalDate.of(2024, 12, 31))
922
- .build();
923
-
924
- // When & Then
925
- assertThat(pattern.isValidAt(LocalDate.of(2024, 6, 15))).isTrue();
926
- assertThat(pattern.isValidAt(LocalDate.of(2024, 1, 1))).isTrue();
927
- assertThat(pattern.isValidAt(LocalDate.of(2024, 12, 31))).isTrue();
928
- }
929
-
930
- @Test
931
- @DisplayName("有効期間外の日付でfalseを返す")
932
- void shouldReturnFalseForInvalidDate() {
933
- // Given
934
- var pattern = AutoJournalPattern.builder()
935
- .patternCode("P001")
936
- .validFrom(LocalDate.of(2024, 1, 1))
937
- .validTo(LocalDate.of(2024, 12, 31))
938
- .build();
939
-
940
- // When & Then
941
- assertThat(pattern.isValidAt(LocalDate.of(2023, 12, 31))).isFalse();
942
- assertThat(pattern.isValidAt(LocalDate.of(2025, 1, 1))).isFalse();
943
- }
944
- }
945
- }
946
- ```
947
-
948
- </details>
949
-
950
- ### リポジトリのテスト
951
-
952
- <details>
953
- <summary>AutoJournalPatternRepositoryTest.java</summary>
954
-
955
- ```java
956
- package com.example.accounting.infrastructure.persistence;
957
-
958
- import com.example.accounting.domain.model.autojournal.*;
959
- import com.example.accounting.infrastructure.persistence.repository.AutoJournalPatternJpaRepository;
960
- import org.junit.jupiter.api.DisplayName;
961
- import org.junit.jupiter.api.Nested;
962
- import org.junit.jupiter.api.Test;
963
- import org.springframework.beans.factory.annotation.Autowired;
964
- import org.springframework.boot.test.autoconfigure.jdbc.AutoConfigureTestDatabase;
965
- import org.springframework.boot.test.autoconfigure.orm.jpa.DataJpaTest;
966
- import org.springframework.test.context.DynamicPropertyRegistry;
967
- import org.springframework.test.context.DynamicPropertySource;
968
- import org.testcontainers.containers.PostgreSQLContainer;
969
- import org.testcontainers.junit.jupiter.Container;
970
- import org.testcontainers.junit.jupiter.Testcontainers;
971
-
972
- import java.time.LocalDate;
973
-
974
- import static org.assertj.core.api.Assertions.*;
975
-
976
- /**
977
- * 自動仕訳リポジトリのテスト
978
- *
979
- * JPA 版では @DataJpaTest を使用(MyBatis 版では @MybatisTest)
980
- * TestContainers で PostgreSQL コンテナを起動し、実際のDBでテスト
981
- */
982
- @DataJpaTest
983
- @AutoConfigureTestDatabase(replace = AutoConfigureTestDatabase.Replace.NONE)
984
- @Testcontainers
985
- @DisplayName("自動仕訳パターンリポジトリのテスト")
986
- class AutoJournalPatternRepositoryTest {
987
-
988
- @Container
989
- static PostgreSQLContainer<?> postgres = new PostgreSQLContainer<>("postgres:16")
990
- .withDatabaseName("testdb")
991
- .withUsername("test")
992
- .withPassword("test");
993
-
994
- @DynamicPropertySource
995
- static void configureProperties(DynamicPropertyRegistry registry) {
996
- registry.add("spring.datasource.url", postgres::getJdbcUrl);
997
- registry.add("spring.datasource.username", postgres::getUsername);
998
- registry.add("spring.datasource.password", postgres::getPassword);
999
- }
1000
-
1001
- @Autowired
1002
- private AutoJournalPatternJpaRepository patternRepository;
1003
-
1004
- @Nested
1005
- @DisplayName("パターンマスタの操作")
1006
- class PatternMasterTest {
1007
-
1008
- @Test
1009
- @DisplayName("パターンマスタを登録・取得できる")
1010
- void shouldSaveAndFindPattern() {
1011
- // Given
1012
- var pattern = AutoJournalPattern.builder()
1013
- .patternCode("P001")
1014
- .patternName("加工品売上")
1015
- .productGroup("加工品")
1016
- .customerGroup("ALL")
1017
- .debitAccountCode("11300") // 売掛金
1018
- .creditAccountCode("41110") // 売上加工品
1019
- .validFrom(LocalDate.of(2024, 1, 1))
1020
- .validTo(LocalDate.of(9999, 12, 31))
1021
- .priority(100)
1022
- .build();
1023
-
1024
- // When
1025
- patternRepository.save(pattern);
1026
- var saved = patternRepository.findById("P001");
1027
-
1028
- // Then
1029
- assertThat(saved).isPresent();
1030
- assertThat(saved.get().getPatternName()).isEqualTo("加工品売上");
1031
- assertThat(saved.get().getProductGroup()).isEqualTo("加工品");
1032
- }
1033
-
1034
- @Test
1035
- @DisplayName("有効なパターンを優先順位順で取得できる")
1036
- void shouldFindValidPatternsSortedByPriority() {
1037
- // Given: 複数のパターンを登録
1038
- var pattern1 = AutoJournalPattern.builder()
1039
- .patternCode("P001")
1040
- .patternName("加工品売上")
1041
- .productGroup("加工品")
1042
- .customerGroup("ALL")
1043
- .debitAccountCode("11300")
1044
- .creditAccountCode("41110")
1045
- .validFrom(LocalDate.of(2024, 1, 1))
1046
- .validTo(LocalDate.of(9999, 12, 31))
1047
- .priority(200)
1048
- .build();
1049
-
1050
- var pattern2 = AutoJournalPattern.builder()
1051
- .patternCode("P002")
1052
- .patternName("生鮮品売上")
1053
- .productGroup("生鮮品")
1054
- .customerGroup("ALL")
1055
- .debitAccountCode("11300")
1056
- .creditAccountCode("41120")
1057
- .validFrom(LocalDate.of(2024, 1, 1))
1058
- .validTo(LocalDate.of(9999, 12, 31))
1059
- .priority(100)
1060
- .build();
1061
-
1062
- patternRepository.save(pattern1);
1063
- patternRepository.save(pattern2);
1064
-
1065
- // When
1066
- var patterns = patternRepository.findValidPatterns(LocalDate.of(2024, 4, 1));
1067
-
1068
- // Then
1069
- assertThat(patterns).isSortedAccordingTo(
1070
- (p1, p2) -> p1.getPriority().compareTo(p2.getPriority())
1071
- );
1072
- assertThat(patterns.get(0).getPatternCode()).isEqualTo("P002"); // priority 100
1073
- }
1074
- }
1075
- }
1076
- ```
1077
-
1078
- </details>
1079
-
1080
- ---
1081
-
1082
- ## 17.6 JpaRepository インターフェース
1083
-
1084
- ### 自動仕訳パターンマスタ Repository
1085
-
1086
- <details>
1087
- <summary>AutoJournalPatternJpaRepository.java</summary>
1088
-
1089
- ```java
1090
- package com.example.accounting.infrastructure.persistence.repository;
1091
-
1092
- import com.example.accounting.domain.model.autojournal.AutoJournalPattern;
1093
- import org.springframework.data.jpa.repository.EntityGraph;
1094
- import org.springframework.data.jpa.repository.JpaRepository;
1095
- import org.springframework.data.jpa.repository.Query;
1096
- import org.springframework.data.repository.query.Param;
1097
- import org.springframework.stereotype.Repository;
1098
-
1099
- import java.time.LocalDate;
1100
- import java.util.List;
1101
- import java.util.Optional;
1102
-
1103
- /**
1104
- * 自動仕訳パターンマスタ JpaRepository
1105
- *
1106
- * JPA 版では JpaRepository を継承(MyBatis 版では Mapper インターフェース)
1107
- * メソッド名規約で自動的にクエリを生成
1108
- */
1109
- @Repository
1110
- public interface AutoJournalPatternJpaRepository extends JpaRepository<AutoJournalPattern, String> {
1111
-
1112
- /**
1113
- * 有効なパターンを優先順位順で取得
1114
- * @Query で JPQL を明示的に記述
1115
- */
1116
- @Query("SELECT p FROM AutoJournalPattern p " +
1117
- "WHERE p.validFrom <= :date AND p.validTo >= :date " +
1118
- "ORDER BY p.priority, p.patternCode")
1119
- List<AutoJournalPattern> findValidPatterns(@Param("date") LocalDate date);
1120
-
1121
- /**
1122
- * パターンコードで検索(勘定科目も一緒に取得)
1123
- * @EntityGraph で N+1 問題を防止
1124
- */
1125
- @EntityGraph(attributePaths = {"debitAccount", "creditAccount"})
1126
- Optional<AutoJournalPattern> findWithAccountsByPatternCode(String patternCode);
1127
-
1128
- /**
1129
- * 商品グループで検索
1130
- */
1131
- List<AutoJournalPattern> findByProductGroupOrderByPriority(String productGroup);
1132
-
1133
- /**
1134
- * 顧客グループで検索
1135
- */
1136
- List<AutoJournalPattern> findByCustomerGroupOrderByPriority(String customerGroup);
1137
- }
1138
- ```
1139
-
1140
- </details>
1141
-
1142
- ### 自動仕訳データ Repository
1143
-
1144
- <details>
1145
- <summary>AutoJournalEntryJpaRepository.java</summary>
1146
-
1147
- ```java
1148
- package com.example.accounting.infrastructure.persistence.repository;
1149
-
1150
- import com.example.accounting.domain.model.autojournal.AutoJournalEntry;
1151
- import com.example.accounting.domain.model.autojournal.AutoJournalStatus;
1152
- import org.springframework.data.jpa.repository.EntityGraph;
1153
- import org.springframework.data.jpa.repository.JpaRepository;
1154
- import org.springframework.data.jpa.repository.JpaSpecificationExecutor;
1155
- import org.springframework.data.jpa.repository.Query;
1156
- import org.springframework.data.repository.query.Param;
1157
- import org.springframework.stereotype.Repository;
1158
-
1159
- import java.time.LocalDate;
1160
- import java.util.List;
1161
- import java.util.Optional;
1162
-
1163
- /**
1164
- * 自動仕訳データ JpaRepository
1165
- *
1166
- * JpaSpecificationExecutor を継承して動的クエリに対応
1167
- * MyBatis 版では動的 SQL を XML で記述していたが、
1168
- * JPA 版では Specification パターンを使用
1169
- */
1170
- @Repository
1171
- public interface AutoJournalEntryJpaRepository
1172
- extends JpaRepository<AutoJournalEntry, String>,
1173
- JpaSpecificationExecutor<AutoJournalEntry> {
1174
-
1175
- /**
1176
- * 売上番号で検索
1177
- */
1178
- List<AutoJournalEntry> findBySalesNumberOrderBySalesLineNumberAscDebitCreditTypeAsc(
1179
- String salesNumber);
1180
-
1181
- /**
1182
- * 未転記の自動仕訳を取得
1183
- */
1184
- @Query("SELECT e FROM AutoJournalEntry e " +
1185
- "WHERE e.postedFlag = false " +
1186
- "AND e.status = com.example.accounting.domain.model.autojournal.AutoJournalStatus.COMPLETED " +
1187
- "ORDER BY e.postingDate, e.autoJournalNumber")
1188
- List<AutoJournalEntry> findUnposted();
1189
-
1190
- /**
1191
- * 未転記の自動仕訳を日付で取得
1192
- */
1193
- @Query("SELECT e FROM AutoJournalEntry e " +
1194
- "WHERE e.postedFlag = false " +
1195
- "AND e.status = com.example.accounting.domain.model.autojournal.AutoJournalStatus.COMPLETED " +
1196
- "AND e.postingDate = :date " +
1197
- "ORDER BY e.autoJournalNumber")
1198
- List<AutoJournalEntry> findUnpostedByDate(@Param("date") LocalDate date);
1199
-
1200
- /**
1201
- * ステータスで検索
1202
- */
1203
- List<AutoJournalEntry> findByStatusOrderByPostingDateAscAutoJournalNumberAsc(
1204
- AutoJournalStatus status);
1205
-
1206
- /**
1207
- * パターンと関連エンティティを一緒に取得
1208
- * @EntityGraph で N+1 問題を防止
1209
- */
1210
- @EntityGraph(attributePaths = {"pattern", "account", "department"})
1211
- Optional<AutoJournalEntry> findWithRelationsById(String autoJournalNumber);
1212
- }
1213
- ```
1214
-
1215
- </details>
1216
-
1217
- ---
1218
-
1219
- ## 17.7 Spring Data JPA Specification による動的クエリ
1220
-
1221
- JPA 版では、複雑な検索条件を組み合わせる際に Specification パターンを使用します。MyBatis 版では XML の `<if>` タグで動的 SQL を構築していましたが、JPA 版では型安全な Specification で実現します。
1222
-
1223
- <details>
1224
- <summary>AutoJournalEntrySpecifications.java</summary>
1225
-
1226
- ```java
1227
- package com.example.accounting.infrastructure.persistence.specification;
1228
-
1229
- import com.example.accounting.domain.model.autojournal.AutoJournalEntry;
1230
- import com.example.accounting.domain.model.autojournal.AutoJournalStatus;
1231
- import org.springframework.data.jpa.domain.Specification;
1232
-
1233
- import java.time.LocalDate;
1234
-
1235
- /**
1236
- * 自動仕訳エントリ検索用 Specification
1237
- *
1238
- * JPA 版では Specification パターンで動的クエリを構築
1239
- * MyBatis 版の XML 動的 SQL を Java コードで表現
1240
- *
1241
- * 使用例:
1242
- * var spec = AutoJournalEntrySpecifications.unposted()
1243
- * .and(AutoJournalEntrySpecifications.postingDateBetween(from, to))
1244
- * .and(AutoJournalEntrySpecifications.hasStatus(AutoJournalStatus.COMPLETED));
1245
- * var entries = repository.findAll(spec);
1246
- */
1247
- public class AutoJournalEntrySpecifications {
1248
-
1249
- /**
1250
- * 未転記条件
1251
- */
1252
- public static Specification<AutoJournalEntry> unposted() {
1253
- return (root, query, cb) -> cb.equal(root.get("postedFlag"), false);
1254
- }
1255
-
1256
- /**
1257
- * ステータス条件
1258
- */
1259
- public static Specification<AutoJournalEntry> hasStatus(AutoJournalStatus status) {
1260
- return (root, query, cb) -> cb.equal(root.get("status"), status);
1261
- }
1262
-
1263
- /**
1264
- * 起票日範囲条件
1265
- */
1266
- public static Specification<AutoJournalEntry> postingDateBetween(
1267
- LocalDate from, LocalDate to) {
1268
- return (root, query, cb) -> cb.between(root.get("postingDate"), from, to);
1269
- }
1270
-
1271
- /**
1272
- * 起票日条件
1273
- */
1274
- public static Specification<AutoJournalEntry> postingDateEquals(LocalDate date) {
1275
- return (root, query, cb) -> cb.equal(root.get("postingDate"), date);
1276
- }
1277
-
1278
- /**
1279
- * 売上番号条件
1280
- */
1281
- public static Specification<AutoJournalEntry> salesNumberEquals(String salesNumber) {
1282
- return (root, query, cb) -> cb.equal(root.get("salesNumber"), salesNumber);
1283
- }
1284
-
1285
- /**
1286
- * パターンコード条件
1287
- */
1288
- public static Specification<AutoJournalEntry> patternCodeEquals(String patternCode) {
1289
- return (root, query, cb) -> cb.equal(root.get("patternCode"), patternCode);
1290
- }
1291
-
1292
- /**
1293
- * 勘定科目コード条件
1294
- */
1295
- public static Specification<AutoJournalEntry> accountCodeEquals(String accountCode) {
1296
- return (root, query, cb) -> cb.equal(root.get("accountCode"), accountCode);
1297
- }
1298
-
1299
- /**
1300
- * 部門コード条件
1301
- */
1302
- public static Specification<AutoJournalEntry> departmentCodeEquals(String departmentCode) {
1303
- return (root, query, cb) -> cb.equal(root.get("departmentCode"), departmentCode);
1304
- }
1305
-
1306
- /**
1307
- * エラー有無条件
1308
- */
1309
- public static Specification<AutoJournalEntry> hasError() {
1310
- return (root, query, cb) -> cb.isNotNull(root.get("errorCode"));
1311
- }
1312
-
1313
- /**
1314
- * エラー無し条件
1315
- */
1316
- public static Specification<AutoJournalEntry> noError() {
1317
- return (root, query, cb) -> cb.isNull(root.get("errorCode"));
1318
- }
1319
- }
1320
- ```
1321
-
1322
- </details>
1323
-
1324
- ### Specification の使用例
1325
-
1326
- ```java
1327
- // 動的クエリの組み立て
1328
- var spec = AutoJournalEntrySpecifications.unposted()
1329
- .and(AutoJournalEntrySpecifications.postingDateBetween(
1330
- LocalDate.of(2024, 4, 1),
1331
- LocalDate.of(2024, 4, 30)))
1332
- .and(AutoJournalEntrySpecifications.hasStatus(AutoJournalStatus.COMPLETED))
1333
- .and(AutoJournalEntrySpecifications.noError());
1334
-
1335
- // クエリ実行
1336
- var entries = entryJpaRepository.findAll(spec);
1337
- ```
1338
-
1339
- ---
1340
-
1341
- ## 17.8 自動仕訳サービスの実装
1342
-
1343
- <details>
1344
- <summary>AutoJournalService.java(一部抜粋)</summary>
1345
-
1346
- ```java
1347
- package com.example.accounting.application.service;
1348
-
1349
- import com.example.accounting.application.dto.AutoJournalResult;
1350
- import com.example.accounting.application.dto.SalesData;
1351
- import com.example.accounting.application.port.out.AutoJournalRepository;
1352
- import com.example.accounting.domain.model.autojournal.*;
1353
- import com.example.accounting.domain.model.journal.*;
1354
- import lombok.RequiredArgsConstructor;
1355
- import lombok.extern.slf4j.Slf4j;
1356
- import org.springframework.stereotype.Service;
1357
- import org.springframework.transaction.annotation.Transactional;
1358
-
1359
- import java.math.BigDecimal;
1360
- import java.time.LocalDate;
1361
- import java.time.LocalDateTime;
1362
- import java.util.*;
1363
-
1364
- /**
1365
- * 自動仕訳サービス
1366
- *
1367
- * JPA 版でもビジネスロジックは MyBatis 版と同じ
1368
- * リポジトリ経由でデータアクセスを行う
1369
- */
1370
- @Service
1371
- @RequiredArgsConstructor
1372
- @Slf4j
1373
- public class AutoJournalService {
1374
-
1375
- private final AutoJournalRepository autoJournalRepository;
1376
- private final JournalRepository journalRepository;
1377
-
1378
- /**
1379
- * 売上データから自動仕訳を生成する
1380
- */
1381
- @Transactional
1382
- public AutoJournalResult generateAutoJournals(List<SalesData> salesDataList) {
1383
- var processNumber = generateProcessNumber();
1384
- var startTime = LocalDateTime.now();
1385
-
1386
- int successCount = 0;
1387
- int errorCount = 0;
1388
- BigDecimal totalAmount = BigDecimal.ZERO;
1389
-
1390
- // 有効なパターンを取得
1391
- var patterns = autoJournalRepository.findValidPatterns(LocalDate.now());
1392
-
1393
- List<AutoJournalEntry> allEntries = new ArrayList<>();
1394
-
1395
- for (SalesData sale : salesDataList) {
1396
- try {
1397
- // マッチするパターンを検索
1398
- var matchedPattern = findMatchingPattern(patterns, sale);
1399
-
1400
- if (matchedPattern.isPresent()) {
1401
- // 自動仕訳データを生成
1402
- var entries = createAutoJournalEntries(sale, matchedPattern.get());
1403
- allEntries.addAll(entries);
1404
-
1405
- successCount++;
1406
- totalAmount = totalAmount.add(sale.getAmount());
1407
- } else {
1408
- // パターン不一致エラー
1409
- allEntries.add(createErrorEntry(sale, "E001", "マッチするパターンが見つかりません"));
1410
- errorCount++;
1411
- }
1412
- } catch (Exception e) {
1413
- log.error("自動仕訳処理エラー: {}", sale.getSalesNumber(), e);
1414
- allEntries.add(createErrorEntry(sale, "E999", e.getMessage()));
1415
- errorCount++;
1416
- }
1417
- }
1418
-
1419
- // JPA 版では saveAll でバッチ保存
1420
- autoJournalRepository.saveAllEntries(allEntries);
1421
-
1422
- // 処理履歴を保存
1423
- var history = AutoJournalHistory.builder()
1424
- .processNumber(processNumber)
1425
- .processDateTime(startTime)
1426
- .targetFromDate(salesDataList.stream()
1427
- .map(SalesData::getSalesDate)
1428
- .min(LocalDate::compareTo).orElse(LocalDate.now()))
1429
- .targetToDate(salesDataList.stream()
1430
- .map(SalesData::getSalesDate)
1431
- .max(LocalDate::compareTo).orElse(LocalDate.now()))
1432
- .totalCount(salesDataList.size())
1433
- .successCount(successCount)
1434
- .errorCount(errorCount)
1435
- .totalAmount(totalAmount)
1436
- .build();
1437
-
1438
- autoJournalRepository.saveHistory(history);
1439
-
1440
- return new AutoJournalResult(processNumber, successCount, errorCount, totalAmount);
1441
- }
1442
-
1443
- /**
1444
- * マッチするパターンを検索
1445
- */
1446
- private Optional<AutoJournalPattern> findMatchingPattern(
1447
- List<AutoJournalPattern> patterns, SalesData sale) {
1448
- return patterns.stream()
1449
- .filter(p -> p.matches(sale.getProductGroup(), sale.getCustomerGroup()))
1450
- .filter(p -> p.isValidAt(sale.getSalesDate()))
1451
- .findFirst();
1452
- }
1453
- }
1454
- ```
1455
-
1456
- </details>
1457
-
1458
- ---
1459
-
1460
- ## 17.9 ヘキサゴナルアーキテクチャでのポート設計
1461
-
1462
- ### Output Port(リポジトリインターフェース)
1463
-
1464
- <details>
1465
- <summary>AutoJournalRepository.java</summary>
1466
-
1467
- ```java
1468
- package com.example.accounting.application.port.out;
1469
-
1470
- import com.example.accounting.domain.model.autojournal.*;
1471
-
1472
- import java.time.LocalDate;
1473
- import java.util.List;
1474
- import java.util.Optional;
1475
-
1476
- /**
1477
- * 自動仕訳リポジトリ(Output Port)
1478
- *
1479
- * ヘキサゴナルアーキテクチャにおける出力ポート
1480
- * アプリケーション層はこのインターフェースに依存し、
1481
- * インフラストラクチャ層で実装する
1482
- */
1483
- public interface AutoJournalRepository {
1484
-
1485
- // パターンマスタ操作
1486
- void savePattern(AutoJournalPattern pattern);
1487
-
1488
- Optional<AutoJournalPattern> findPatternByCode(String patternCode);
1489
-
1490
- Optional<AutoJournalPattern> findPatternWithAccounts(String patternCode);
1491
-
1492
- List<AutoJournalPattern> findAllPatterns();
1493
-
1494
- List<AutoJournalPattern> findValidPatterns(LocalDate date);
1495
-
1496
- void updatePattern(AutoJournalPattern pattern);
1497
-
1498
- void deletePattern(String patternCode);
1499
-
1500
- // 自動仕訳エントリ操作
1501
- void saveEntry(AutoJournalEntry entry);
1502
-
1503
- void saveAllEntries(List<AutoJournalEntry> entries);
1504
-
1505
- Optional<AutoJournalEntry> findEntryByNumber(String autoJournalNumber);
1506
-
1507
- Optional<AutoJournalEntry> findEntryWithRelations(String autoJournalNumber);
1508
-
1509
- List<AutoJournalEntry> findEntriesBySalesNumber(String salesNumber);
1510
-
1511
- List<AutoJournalEntry> findUnpostedEntries();
1512
-
1513
- List<AutoJournalEntry> findUnpostedEntriesByDate(LocalDate date);
1514
-
1515
- List<AutoJournalEntry> findEntriesByStatus(AutoJournalStatus status);
1516
-
1517
- void updateEntry(AutoJournalEntry entry);
1518
-
1519
- // 処理履歴操作
1520
- void saveHistory(AutoJournalHistory history);
1521
-
1522
- Optional<AutoJournalHistory> findHistoryByNumber(String processNumber);
1523
-
1524
- List<AutoJournalHistory> findHistoriesByDateRange(LocalDate fromDate, LocalDate toDate);
1525
-
1526
- void deleteAll();
1527
- }
1528
- ```
1529
-
1530
- </details>
1531
-
1532
- ---
1533
-
1534
- ## 17.10 まとめ:MyBatis 版と JPA 版の比較
1535
-
1536
- ### データアクセス層のアーキテクチャ比較
1537
-
1538
- ```plantuml
1539
- @startuml
1540
-
1541
- title MyBatis 版と JPA 版のアーキテクチャ比較
1542
-
1543
- package "MyBatis 版" {
1544
- [AutoJournalService] as MService
1545
- [AutoJournalMapper] as MMapper
1546
- [AutoJournalMapper.xml] as MXml
1547
- database "PostgreSQL" as MDB
1548
-
1549
- MService --> MMapper
1550
- MMapper --> MXml
1551
- MXml --> MDB
1552
- }
1553
-
1554
- package "JPA 版" {
1555
- [AutoJournalService] as JService
1556
- [AutoJournalRepository\n(Output Port)] as JPort
1557
- [AutoJournalRepositoryImpl] as JImpl
1558
- [AutoJournalPatternJpaRepository] as JJpa
1559
- database "PostgreSQL" as JDB
1560
-
1561
- JService --> JPort
1562
- JPort <|.. JImpl
1563
- JImpl --> JJpa
1564
- JJpa --> JDB
1565
- }
1566
-
1567
- @enduml
1568
- ```
1569
-
1570
- ### JPA 版の特徴
1571
-
1572
- | 観点 | JPA 版の特徴 |
1573
- |-----|-------------|
1574
- | 型安全性 | Specification パターンによる型安全な動的クエリ |
1575
- | N+1 問題対策 | @EntityGraph による宣言的な解決 |
1576
- | バッチ処理 | saveAll() による効率的な一括保存 |
1577
- | ポート分離 | Output Port インターフェースによる抽象化 |
1578
- | テスト容易性 | @DataJpaTest + TestContainers |
1579
-
1580
- ---
1581
-
1582
- ## 次章予告
1583
-
1584
- [第18章](chapter18-orm.md)では、勘定科目残高テーブルの設計について、JPA 版での実装を解説します。
1
+ # 第17章:自動仕訳の設計(ORM版)
2
+
3
+ 販売管理システムなどの業務システムから会計システムへの自動仕訳処理を TDD で設計していきます。売上データから仕訳データへの自動変換ルールと、効率的なバッチ処理の設計を行います。
4
+
5
+ JPA 版では、@Entity によるリレーション定義、@EntityGraph による N+1 問題対策、Spring Data JPA Specification による動的クエリを活用します。
6
+
7
+ ---
8
+
9
+ ## 17.1 自動仕訳の概要
10
+
11
+ ### 自動仕訳とは
12
+
13
+ 自動仕訳は、業務システムのトランザクションデータを会計システムの仕訳データへ自動的に変換する機能です。
14
+
15
+ #### 従来のアナログ連携(手入力による連携)
16
+
17
+ 従来の方式では、販売管理システムで発行した売上伝票を紙で経理部門に回送し、経理担当者が仕訳入力を行っていました。
18
+
19
+ ```plantuml
20
+ @startuml
21
+ title 販売管理システムと会計システムの非連携(アナログ連携)
22
+
23
+ skinparam rectangle {
24
+ BackgroundColor White
25
+ BorderColor Black
26
+ }
27
+ skinparam database {
28
+ BackgroundColor #E8F5E9
29
+ BorderColor Black
30
+ }
31
+
32
+ package "営業部門" {
33
+ rectangle "売上入力" as SalesInput <<Screen>>
34
+ database "売上データ" as SalesDB
35
+ rectangle "売上伝票" as SalesVoucher <<Document>>
36
+ }
37
+
38
+ package "経理部門" {
39
+ rectangle "仕訳入力" as JournalInput <<Screen>>
40
+ database "仕訳データ" as JournalDB
41
+ rectangle "損益計算書\n貸借対照表" as FinancialReports <<Document>>
42
+ }
43
+
44
+ SalesInput -down-> SalesDB
45
+ SalesDB -down-> SalesVoucher
46
+
47
+ SalesVoucher -[#red,bold]right-> JournalInput : **紙伝票の回送**\n**(手入力による再登録)**
48
+
49
+ JournalInput -down-> JournalDB
50
+ JournalDB -down-> FinancialReports
51
+
52
+ @enduml
53
+ ```
54
+
55
+ **アナログ連携の問題点:**
56
+
57
+ | 問題 | 説明 |
58
+ |-----|------|
59
+ | 二重入力 | 営業部門と経理部門で同じ情報を入力する作業負荷 |
60
+ | 入力ミス | 手入力によるデータ不整合 |
61
+ | タイムラグ | 紙伝票の回送による情報遅延 |
62
+ | 消費税計算ミス | 手計算による誤り |
63
+ | 勘定科目選択ミス | 担当者の判断ばらつき |
64
+
65
+ #### 自動仕訳によるデジタル連携
66
+
67
+ 自動仕訳処理を導入することで、売上データから仕訳データへの変換を自動化します。
68
+
69
+ ```plantuml
70
+ @startuml
71
+ title 自動仕訳処理のデータの流れ
72
+
73
+ skinparam rectangle {
74
+ BackgroundColor White
75
+ BorderColor Black
76
+ }
77
+ skinparam database {
78
+ BackgroundColor #E8F5E9
79
+ BorderColor Black
80
+ }
81
+
82
+ package "営業部門" {
83
+ rectangle "売上入力" as SalesInput <<Screen>>
84
+ database "売上データ" as SalesDB
85
+ rectangle "売上\nチェックリスト" as SalesList <<Document>>
86
+ }
87
+
88
+ package "自動仕訳機能" #F1F8E9 {
89
+ database "自動仕訳\nパターンマスタ" as PatternMaster
90
+ rectangle "自動仕訳処理" as AutoJournalProcess <<Process>> #4CAF50
91
+ database "自動仕訳データ" as AutoJournalDB
92
+ database "エラーデータ" as ErrorDB
93
+ rectangle "エラーリスト" as ErrorList <<Document>>
94
+ rectangle "自動仕訳\nチェックリスト" as AutoJournalList <<Document>>
95
+
96
+ rectangle "転記処理" as PostingProcess <<Process>> #4CAF50
97
+ circle " " as Switch
98
+ }
99
+
100
+ package "経理部門" {
101
+ rectangle "仕訳入力" as JournalInput <<Screen>>
102
+ database "仕訳データ" as JournalDB
103
+ }
104
+
105
+ SalesInput -down-> SalesDB
106
+ SalesDB -down-> SalesList
107
+
108
+ SalesDB -right-> AutoJournalProcess
109
+ PatternMaster -down-> AutoJournalProcess
110
+ AutoJournalProcess -right-> AutoJournalDB
111
+ AutoJournalProcess -down-> ErrorDB
112
+ ErrorDB -down-> ErrorList
113
+ AutoJournalDB -down-> AutoJournalList
114
+
115
+ AutoJournalDB -right-> Switch
116
+ note top of Switch : (転記指示)
117
+ Switch -right-> PostingProcess
118
+ PostingProcess -right-> JournalDB
119
+
120
+ JournalInput -down-> JournalDB
121
+
122
+ @enduml
123
+ ```
124
+
125
+ ### 自動仕訳処理の流れ
126
+
127
+ ```plantuml
128
+ @startuml
129
+
130
+ title 自動仕訳処理フロー
131
+
132
+ |営業部門|
133
+ start
134
+ :売上登録;
135
+
136
+ |自動仕訳処理|
137
+ :売上データ抽出;
138
+ note right
139
+ 未処理の売上データを
140
+ 抽出条件に基づいて取得
141
+ end note
142
+
143
+ :パターンマッチング;
144
+ note right
145
+ 商品区分・顧客区分から
146
+ 適用する仕訳パターンを決定
147
+ end note
148
+
149
+ if (パターン適合?) then (はい)
150
+ :仕訳データ生成;
151
+ :自動仕訳データに保存;
152
+ else (いいえ)
153
+ :エラーデータに保存;
154
+ :エラーリスト出力;
155
+ stop
156
+ endif
157
+
158
+ :売上データに処理済フラグ設定;
159
+
160
+ |経理部門|
161
+ :自動仕訳チェックリスト確認;
162
+
163
+ if (確認OK?) then (はい)
164
+ :転記処理実行;
165
+ :仕訳データに登録;
166
+ else (いいえ)
167
+ :修正・再処理;
168
+ stop
169
+ endif
170
+
171
+ :仕訳完了;
172
+ stop
173
+
174
+ @enduml
175
+ ```
176
+
177
+ ### 売上伝票から仕訳伝票への変換例
178
+
179
+ #### 売上伝票(ビジネス上の事実)
180
+
181
+ | 項目 | 値 |
182
+ |-----|-----|
183
+ | 伝票日付 | 2024/04/01 |
184
+ | 顧客 | DBMフード新宿 |
185
+ | 明細1 | いちご蒸缶 1,000個 × 1,000円 = 1,000,000円 |
186
+ | 明細2 | アスパラ 200個 × 10,000円 = 2,000,000円 |
187
+ | 明細3 | さざえのエスカルゴ 1,500個 × 100円 = 150,000円 |
188
+ | 合計 | 3,150,000円(税抜)+ 消費税 315,000円 = 3,465,000円(税込) |
189
+
190
+ #### 仕訳伝票(会計上の記録)
191
+
192
+ | 借方科目 | 借方金額 | 貸方科目 | 貸方金額 | 摘要 |
193
+ |---------|---------|---------|---------|-----|
194
+ | 売掛金/DBMフード本社 | 3,465,000 | | | 売上計上 |
195
+ | | | 売上加工品/DBMフード新宿 | 1,150,000 | いちご蒸缶・さざえ |
196
+ | | | 売上生鮮品/DBMフード新宿 | 2,000,000 | アスパラ |
197
+ | | | 仮受消費税 | 315,000 | 消費税10% |
198
+
199
+ ---
200
+
201
+ ## 17.2 自動仕訳テーブルの設計
202
+
203
+ ### フラグ管理方式と日付管理方式
204
+
205
+ 売上データの処理状態を管理する方式には、2つのアプローチがあります。
206
+
207
+ #### フラグ管理方式
208
+
209
+ - **メリット**:シンプルな実装、処理状態が明確
210
+ - **デメリット**:再処理時にフラグリセットが必要、大量データでの更新負荷
211
+
212
+ #### 日付管理方式
213
+
214
+ - **メリット**:差分処理が容易、再処理時の柔軟性
215
+ - **デメリット**:管理テーブルが必要、更新日時の整合性管理が必要
216
+
217
+ ### セット中心のアプリケーション設計
218
+
219
+ 大量データを効率的に処理するには、ループ処理よりもセット中心処理が有効です。
220
+
221
+ | 方式 | 処理方法 | DB操作回数 |
222
+ |-----|---------|-----------|
223
+ | ループ処理 | 売上データを1件ずつ読み込み、各売上に対してパターンマスタを検索、1件ずつ仕訳データを挿入 | N×M回 |
224
+ | セット中心処理 | パターンマスタを1件読み込み、該当パターンの売上データを一括INSERT | M回(パターン数) |
225
+
226
+ ### 自動仕訳関連テーブルの ER 図
227
+
228
+ ```plantuml
229
+ @startuml
230
+
231
+ entity "自動仕訳パターンマスタ" as AutoJournalPattern {
232
+ * **パターンコード**: VARCHAR(10) <<PK>>
233
+ --
234
+ * **パターン名**: VARCHAR(50)
235
+ * **商品グループ**: VARCHAR(10)
236
+ * **顧客グループ**: VARCHAR(10)
237
+ * **売上区分**: VARCHAR(2)
238
+ * **借方勘定科目コード**: VARCHAR(5) <<FK>>
239
+ * **借方補助科目設定**: VARCHAR(20)
240
+ * **貸方勘定科目コード**: VARCHAR(5) <<FK>>
241
+ * **貸方補助科目設定**: VARCHAR(20)
242
+ * **返品時借方科目コード**: VARCHAR(5)
243
+ * **返品時貸方科目コード**: VARCHAR(5)
244
+ * **消費税処理区分**: VARCHAR(2)
245
+ * **有効開始日**: DATE
246
+ * **有効終了日**: DATE
247
+ * **優先順位**: INTEGER
248
+ }
249
+
250
+ entity "自動仕訳データ" as AutoJournal {
251
+ * **自動仕訳番号**: VARCHAR(15) <<PK>>
252
+ --
253
+ * **売上番号**: VARCHAR(10) <<FK>>
254
+ * **売上行番号**: SMALLINT
255
+ * **パターンコード**: VARCHAR(10) <<FK>>
256
+ * **起票日**: DATE
257
+ * **仕訳行貸借区分**: VARCHAR(1)
258
+ * **勘定科目コード**: VARCHAR(5) <<FK>>
259
+ * **補助科目コード**: VARCHAR(10)
260
+ * **部門コード**: VARCHAR(5)
261
+ * **仕訳金額**: DECIMAL(14,2)
262
+ * **消費税額**: DECIMAL(14,2)
263
+ * **処理ステータス**: 自動仕訳ステータス
264
+ * **転記済フラグ**: SMALLINT
265
+ }
266
+
267
+ entity "自動仕訳処理履歴" as AutoJournalHistory {
268
+ * **処理番号**: VARCHAR(15) <<PK>>
269
+ --
270
+ * **処理日時**: TIMESTAMP
271
+ * **処理対象開始日**: DATE
272
+ * **処理対象終了日**: DATE
273
+ * **処理件数**: INTEGER
274
+ * **成功件数**: INTEGER
275
+ * **エラー件数**: INTEGER
276
+ * **処理金額合計**: DECIMAL(15,2)
277
+ }
278
+
279
+ entity "勘定科目マスタ" as Account {
280
+ * **勘定科目コード** <<PK>>
281
+ --
282
+ 勘定科目名
283
+ ...
284
+ }
285
+
286
+ AutoJournalPattern }o--|| Account : 借方科目
287
+ AutoJournalPattern }o--|| Account : 貸方科目
288
+ AutoJournal }o--|| AutoJournalPattern
289
+ AutoJournal }o--|| Account
290
+
291
+ @enduml
292
+ ```
293
+
294
+ ---
295
+
296
+ ## 17.3 MyBatis 版との実装比較
297
+
298
+ ### データアクセス層の比較
299
+
300
+ | 機能 | MyBatis 版 | JPA 版 |
301
+ |-----|-----------|--------|
302
+ | ENUM 型変換 | TypeHandler | AttributeConverter |
303
+ | 関連エンティティ取得 | resultMap の association | @ManyToOne |
304
+ | N+1 問題対策 | JOIN を含む SQL | @EntityGraph |
305
+ | 動的クエリ | XML の `<if>` タグ | Specification パターン |
306
+ | バッチ INSERT | foreach 句 | saveAll() |
307
+ | 有効期間検索 | WHERE 句で条件指定 | JPQL で条件指定 |
308
+
309
+ ### Flyway マイグレーション
310
+
311
+ <details>
312
+ <summary>V007__create_auto_journal_tables.sql</summary>
313
+
314
+ ```sql
315
+ -- 自動仕訳処理ステータス
316
+ CREATE TYPE 自動仕訳ステータス AS ENUM ('処理待ち', '処理中', '処理完了', '転記済', 'エラー');
317
+
318
+ -- 自動仕訳パターンマスタ
319
+ CREATE TABLE "自動仕訳パターンマスタ" (
320
+ "パターンコード" VARCHAR(10) PRIMARY KEY,
321
+ "パターン名" VARCHAR(50) NOT NULL,
322
+ "商品グループ" VARCHAR(10) DEFAULT 'ALL',
323
+ "顧客グループ" VARCHAR(10) DEFAULT 'ALL',
324
+ "売上区分" VARCHAR(2) DEFAULT '01',
325
+ "借方勘定科目コード" VARCHAR(5) NOT NULL,
326
+ "借方補助科目設定" VARCHAR(20),
327
+ "貸方勘定科目コード" VARCHAR(5) NOT NULL,
328
+ "貸方補助科目設定" VARCHAR(20),
329
+ "返品時借方科目コード" VARCHAR(5),
330
+ "返品時貸方科目コード" VARCHAR(5),
331
+ "消費税処理区分" VARCHAR(2) DEFAULT '01',
332
+ "有効開始日" DATE DEFAULT CURRENT_DATE,
333
+ "有効終了日" DATE DEFAULT '9999-12-31',
334
+ "優先順位" INTEGER DEFAULT 100,
335
+ "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
336
+ "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
337
+ CONSTRAINT "fk_自動仕訳パターン_借方科目"
338
+ FOREIGN KEY ("借方勘定科目コード") REFERENCES "勘定科目マスタ"("勘定科目コード"),
339
+ CONSTRAINT "fk_自動仕訳パターン_貸方科目"
340
+ FOREIGN KEY ("貸方勘定科目コード") REFERENCES "勘定科目マスタ"("勘定科目コード")
341
+ );
342
+
343
+ -- 自動仕訳データ
344
+ CREATE TABLE "自動仕訳データ" (
345
+ "自動仕訳番号" VARCHAR(15) PRIMARY KEY,
346
+ "売上番号" VARCHAR(10) NOT NULL,
347
+ "売上行番号" SMALLINT NOT NULL,
348
+ "パターンコード" VARCHAR(10) NOT NULL,
349
+ "起票日" DATE NOT NULL,
350
+ "仕訳行貸借区分" 仕訳行貸借区分 NOT NULL,
351
+ "勘定科目コード" VARCHAR(5) NOT NULL,
352
+ "補助科目コード" VARCHAR(10),
353
+ "部門コード" VARCHAR(5),
354
+ "仕訳金額" DECIMAL(14,2) NOT NULL,
355
+ "消費税額" DECIMAL(14,2) DEFAULT 0,
356
+ "処理ステータス" 自動仕訳ステータス DEFAULT '処理待ち' NOT NULL,
357
+ "転記済フラグ" SMALLINT DEFAULT 0,
358
+ "転記日" DATE,
359
+ "仕訳伝票番号" VARCHAR(10),
360
+ "エラーコード" VARCHAR(10),
361
+ "エラーメッセージ" VARCHAR(200),
362
+ "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
363
+ "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL
364
+ );
365
+
366
+ -- 自動仕訳処理履歴
367
+ CREATE TABLE "自動仕訳処理履歴" (
368
+ "処理番号" VARCHAR(15) PRIMARY KEY,
369
+ "処理日時" TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
370
+ "処理対象開始日" DATE NOT NULL,
371
+ "処理対象終了日" DATE NOT NULL,
372
+ "処理件数" INTEGER DEFAULT 0,
373
+ "成功件数" INTEGER DEFAULT 0,
374
+ "エラー件数" INTEGER DEFAULT 0,
375
+ "処理金額合計" DECIMAL(15,2) DEFAULT 0,
376
+ "処理者" VARCHAR(50),
377
+ "備考" TEXT,
378
+ "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL
379
+ );
380
+
381
+ -- インデックス
382
+ CREATE INDEX "idx_自動仕訳パターン_優先順位" ON "自動仕訳パターンマスタ"("優先順位");
383
+ CREATE INDEX "idx_自動仕訳_売上番号" ON "自動仕訳データ"("売上番号");
384
+ CREATE INDEX "idx_自動仕訳_処理ステータス" ON "自動仕訳データ"("処理ステータス");
385
+ CREATE INDEX "idx_自動仕訳_転記済フラグ" ON "自動仕訳データ"("転記済フラグ");
386
+ ```
387
+
388
+ </details>
389
+
390
+ ---
391
+
392
+ ## 17.4 JPA エンティティの実装
393
+
394
+ ### 自動仕訳ステータス Enum と AttributeConverter
395
+
396
+ <details>
397
+ <summary>AutoJournalStatus.java</summary>
398
+
399
+ ```java
400
+ package com.example.accounting.domain.model.autojournal;
401
+
402
+ import lombok.Getter;
403
+ import lombok.RequiredArgsConstructor;
404
+
405
+ @Getter
406
+ @RequiredArgsConstructor
407
+ public enum AutoJournalStatus {
408
+ PENDING("処理待ち"),
409
+ PROCESSING("処理中"),
410
+ COMPLETED("処理完了"),
411
+ POSTED("転記済"),
412
+ ERROR("エラー");
413
+
414
+ private final String displayName;
415
+
416
+ public static AutoJournalStatus fromDisplayName(String displayName) {
417
+ for (AutoJournalStatus status : values()) {
418
+ if (status.displayName.equals(displayName)) {
419
+ return status;
420
+ }
421
+ }
422
+ throw new IllegalArgumentException("Unknown auto journal status: " + displayName);
423
+ }
424
+ }
425
+ ```
426
+
427
+ </details>
428
+
429
+ <details>
430
+ <summary>AutoJournalStatusConverter.java</summary>
431
+
432
+ ```java
433
+ package com.example.accounting.infrastructure.persistence.converter;
434
+
435
+ import com.example.accounting.domain.model.autojournal.AutoJournalStatus;
436
+ import jakarta.persistence.AttributeConverter;
437
+ import jakarta.persistence.Converter;
438
+
439
+ /**
440
+ * 自動仕訳ステータス ↔ PostgreSQL ENUM 変換
441
+ * JPA では AttributeConverter を使用して ENUM 型を変換する
442
+ */
443
+ @Converter(autoApply = true)
444
+ public class AutoJournalStatusConverter implements AttributeConverter<AutoJournalStatus, String> {
445
+
446
+ @Override
447
+ public String convertToDatabaseColumn(AutoJournalStatus attribute) {
448
+ return attribute != null ? attribute.getDisplayName() : null;
449
+ }
450
+
451
+ @Override
452
+ public AutoJournalStatus convertToEntityAttribute(String dbData) {
453
+ return dbData != null ? AutoJournalStatus.fromDisplayName(dbData) : null;
454
+ }
455
+ }
456
+ ```
457
+
458
+ </details>
459
+
460
+ ### 自動仕訳パターンマスタ Entity
461
+
462
+ <details>
463
+ <summary>AutoJournalPattern.java</summary>
464
+
465
+ ```java
466
+ package com.example.accounting.domain.model.autojournal;
467
+
468
+ import com.example.accounting.domain.model.account.Account;
469
+ import jakarta.persistence.*;
470
+ import lombok.*;
471
+
472
+ import java.time.LocalDate;
473
+ import java.time.LocalDateTime;
474
+
475
+ /**
476
+ * 自動仕訳パターンマスタ Entity
477
+ *
478
+ * JPA 版では @ManyToOne で勘定科目との関連を定義
479
+ * MyBatis 版では resultMap で手動マッピングしていたが、
480
+ * JPA では @JoinColumn で外部キー関連を自動的に解決
481
+ */
482
+ @Entity
483
+ @Table(name = "自動仕訳パターンマスタ")
484
+ @Data
485
+ @Builder
486
+ @NoArgsConstructor
487
+ @AllArgsConstructor
488
+ public class AutoJournalPattern {
489
+
490
+ @Id
491
+ @Column(name = "パターンコード", length = 10)
492
+ private String patternCode;
493
+
494
+ @Column(name = "パターン名", length = 50, nullable = false)
495
+ private String patternName;
496
+
497
+ @Column(name = "商品グループ", length = 10)
498
+ private String productGroup;
499
+
500
+ @Column(name = "顧客グループ", length = 10)
501
+ private String customerGroup;
502
+
503
+ @Column(name = "売上区分", length = 2)
504
+ private String salesType;
505
+
506
+ /**
507
+ * 借方勘定科目との関連
508
+ * @ManyToOne で遅延ロード(LAZY)を指定
509
+ * N+1 問題が発生する場合は @EntityGraph で対策
510
+ */
511
+ @ManyToOne(fetch = FetchType.LAZY)
512
+ @JoinColumn(name = "借方勘定科目コード", referencedColumnName = "勘定科目コード",
513
+ insertable = false, updatable = false)
514
+ private Account debitAccount;
515
+
516
+ @Column(name = "借方勘定科目コード", length = 5, nullable = false)
517
+ private String debitAccountCode;
518
+
519
+ @Column(name = "借方補助科目設定", length = 20)
520
+ private String debitSubAccountSetting;
521
+
522
+ /**
523
+ * 貸方勘定科目との関連
524
+ */
525
+ @ManyToOne(fetch = FetchType.LAZY)
526
+ @JoinColumn(name = "貸方勘定科目コード", referencedColumnName = "勘定科目コード",
527
+ insertable = false, updatable = false)
528
+ private Account creditAccount;
529
+
530
+ @Column(name = "貸方勘定科目コード", length = 5, nullable = false)
531
+ private String creditAccountCode;
532
+
533
+ @Column(name = "貸方補助科目設定", length = 20)
534
+ private String creditSubAccountSetting;
535
+
536
+ @Column(name = "返品時借方科目コード", length = 5)
537
+ private String returnDebitAccountCode;
538
+
539
+ @Column(name = "返品時貸方科目コード", length = 5)
540
+ private String returnCreditAccountCode;
541
+
542
+ @Column(name = "消費税処理区分", length = 2)
543
+ private String taxProcessingType;
544
+
545
+ @Column(name = "有効開始日")
546
+ private LocalDate validFrom;
547
+
548
+ @Column(name = "有効終了日")
549
+ private LocalDate validTo;
550
+
551
+ @Column(name = "優先順位")
552
+ private Integer priority;
553
+
554
+ @Column(name = "作成日時", nullable = false, updatable = false)
555
+ private LocalDateTime createdAt;
556
+
557
+ @Column(name = "更新日時", nullable = false)
558
+ private LocalDateTime updatedAt;
559
+
560
+ /**
561
+ * 指定日付に有効かどうかを判定
562
+ */
563
+ public boolean isValidAt(LocalDate date) {
564
+ return !date.isBefore(validFrom) && !date.isAfter(validTo);
565
+ }
566
+
567
+ /**
568
+ * 商品グループと顧客グループにマッチするか判定
569
+ */
570
+ public boolean matches(String productGroup, String customerGroup) {
571
+ boolean productMatch = "ALL".equals(this.productGroup) ||
572
+ this.productGroup.equals(productGroup);
573
+ boolean customerMatch = "ALL".equals(this.customerGroup) ||
574
+ this.customerGroup.equals(customerGroup);
575
+ return productMatch && customerMatch;
576
+ }
577
+
578
+ @PrePersist
579
+ protected void onCreate() {
580
+ LocalDateTime now = LocalDateTime.now();
581
+ this.createdAt = now;
582
+ this.updatedAt = now;
583
+ if (this.validFrom == null) {
584
+ this.validFrom = LocalDate.now();
585
+ }
586
+ if (this.validTo == null) {
587
+ this.validTo = LocalDate.of(9999, 12, 31);
588
+ }
589
+ if (this.priority == null) {
590
+ this.priority = 100;
591
+ }
592
+ }
593
+
594
+ @PreUpdate
595
+ protected void onUpdate() {
596
+ this.updatedAt = LocalDateTime.now();
597
+ }
598
+ }
599
+ ```
600
+
601
+ </details>
602
+
603
+ ### 自動仕訳データ Entity
604
+
605
+ <details>
606
+ <summary>AutoJournalEntry.java</summary>
607
+
608
+ ```java
609
+ package com.example.accounting.domain.model.autojournal;
610
+
611
+ import com.example.accounting.domain.model.account.Account;
612
+ import com.example.accounting.domain.model.department.Department;
613
+ import com.example.accounting.domain.model.journal.DebitCreditType;
614
+ import com.example.accounting.infrastructure.persistence.converter.AutoJournalStatusConverter;
615
+ import com.example.accounting.infrastructure.persistence.converter.DebitCreditTypeConverter;
616
+ import jakarta.persistence.*;
617
+ import lombok.*;
618
+
619
+ import java.math.BigDecimal;
620
+ import java.time.LocalDate;
621
+ import java.time.LocalDateTime;
622
+
623
+ /**
624
+ * 自動仕訳データ Entity
625
+ *
626
+ * JPA 版では @ManyToOne で関連エンティティを定義
627
+ * 状態管理には Enum + AttributeConverter を使用
628
+ */
629
+ @Entity
630
+ @Table(name = "自動仕訳データ")
631
+ @Data
632
+ @Builder
633
+ @NoArgsConstructor
634
+ @AllArgsConstructor
635
+ public class AutoJournalEntry {
636
+
637
+ @Id
638
+ @Column(name = "自動仕訳番号", length = 15)
639
+ private String autoJournalNumber;
640
+
641
+ @Column(name = "売上番号", length = 10, nullable = false)
642
+ private String salesNumber;
643
+
644
+ @Column(name = "売上行番号", nullable = false)
645
+ private Integer salesLineNumber;
646
+
647
+ /**
648
+ * パターンマスタとの関連
649
+ */
650
+ @ManyToOne(fetch = FetchType.LAZY)
651
+ @JoinColumn(name = "パターンコード", referencedColumnName = "パターンコード",
652
+ insertable = false, updatable = false)
653
+ private AutoJournalPattern pattern;
654
+
655
+ @Column(name = "パターンコード", length = 10, nullable = false)
656
+ private String patternCode;
657
+
658
+ @Column(name = "起票日", nullable = false)
659
+ private LocalDate postingDate;
660
+
661
+ /**
662
+ * 仕訳行貸借区分
663
+ * AttributeConverter で PostgreSQL ENUM と Java Enum を変換
664
+ */
665
+ @Convert(converter = DebitCreditTypeConverter.class)
666
+ @Column(name = "仕訳行貸借区分", nullable = false, columnDefinition = "仕訳行貸借区分")
667
+ private DebitCreditType debitCreditType;
668
+
669
+ /**
670
+ * 勘定科目との関連
671
+ */
672
+ @ManyToOne(fetch = FetchType.LAZY)
673
+ @JoinColumn(name = "勘定科目コード", referencedColumnName = "勘定科目コード",
674
+ insertable = false, updatable = false)
675
+ private Account account;
676
+
677
+ @Column(name = "勘定科目コード", length = 5, nullable = false)
678
+ private String accountCode;
679
+
680
+ @Column(name = "補助科目コード", length = 10)
681
+ private String subAccountCode;
682
+
683
+ /**
684
+ * 部門との関連
685
+ */
686
+ @ManyToOne(fetch = FetchType.LAZY)
687
+ @JoinColumn(name = "部門コード", referencedColumnName = "部門コード",
688
+ insertable = false, updatable = false)
689
+ private Department department;
690
+
691
+ @Column(name = "部門コード", length = 5)
692
+ private String departmentCode;
693
+
694
+ @Column(name = "仕訳金額", nullable = false, precision = 14, scale = 2)
695
+ private BigDecimal amount;
696
+
697
+ @Column(name = "消費税額", precision = 14, scale = 2)
698
+ private BigDecimal taxAmount;
699
+
700
+ /**
701
+ * 処理ステータス
702
+ * AttributeConverter で PostgreSQL ENUM と Java Enum を変換
703
+ */
704
+ @Convert(converter = AutoJournalStatusConverter.class)
705
+ @Column(name = "処理ステータス", nullable = false, columnDefinition = "自動仕訳ステータス")
706
+ private AutoJournalStatus status;
707
+
708
+ @Column(name = "転記済フラグ")
709
+ private Boolean postedFlag;
710
+
711
+ @Column(name = "転記日")
712
+ private LocalDate postedDate;
713
+
714
+ @Column(name = "仕訳伝票番号", length = 10)
715
+ private String journalVoucherNumber;
716
+
717
+ @Column(name = "エラーコード", length = 10)
718
+ private String errorCode;
719
+
720
+ @Column(name = "エラーメッセージ", length = 200)
721
+ private String errorMessage;
722
+
723
+ @Column(name = "作成日時", nullable = false, updatable = false)
724
+ private LocalDateTime createdAt;
725
+
726
+ @Column(name = "更新日時", nullable = false)
727
+ private LocalDateTime updatedAt;
728
+
729
+ @PrePersist
730
+ protected void onCreate() {
731
+ LocalDateTime now = LocalDateTime.now();
732
+ this.createdAt = now;
733
+ this.updatedAt = now;
734
+ if (this.status == null) {
735
+ this.status = AutoJournalStatus.PENDING;
736
+ }
737
+ if (this.postedFlag == null) {
738
+ this.postedFlag = false;
739
+ }
740
+ if (this.taxAmount == null) {
741
+ this.taxAmount = BigDecimal.ZERO;
742
+ }
743
+ }
744
+
745
+ @PreUpdate
746
+ protected void onUpdate() {
747
+ this.updatedAt = LocalDateTime.now();
748
+ }
749
+ }
750
+ ```
751
+
752
+ </details>
753
+
754
+ ### 自動仕訳処理履歴 Entity
755
+
756
+ <details>
757
+ <summary>AutoJournalHistory.java</summary>
758
+
759
+ ```java
760
+ package com.example.accounting.domain.model.autojournal;
761
+
762
+ import jakarta.persistence.*;
763
+ import lombok.*;
764
+
765
+ import java.math.BigDecimal;
766
+ import java.time.LocalDate;
767
+ import java.time.LocalDateTime;
768
+
769
+ /**
770
+ * 自動仕訳処理履歴 Entity
771
+ */
772
+ @Entity
773
+ @Table(name = "自動仕訳処理履歴")
774
+ @Data
775
+ @Builder
776
+ @NoArgsConstructor
777
+ @AllArgsConstructor
778
+ public class AutoJournalHistory {
779
+
780
+ @Id
781
+ @Column(name = "処理番号", length = 15)
782
+ private String processNumber;
783
+
784
+ @Column(name = "処理日時", nullable = false)
785
+ private LocalDateTime processDateTime;
786
+
787
+ @Column(name = "処理対象開始日", nullable = false)
788
+ private LocalDate targetFromDate;
789
+
790
+ @Column(name = "処理対象終了日", nullable = false)
791
+ private LocalDate targetToDate;
792
+
793
+ @Column(name = "処理件数")
794
+ private Integer totalCount;
795
+
796
+ @Column(name = "成功件数")
797
+ private Integer successCount;
798
+
799
+ @Column(name = "エラー件数")
800
+ private Integer errorCount;
801
+
802
+ @Column(name = "処理金額合計", precision = 15, scale = 2)
803
+ private BigDecimal totalAmount;
804
+
805
+ @Column(name = "処理者", length = 50)
806
+ private String processedBy;
807
+
808
+ @Column(name = "備考", columnDefinition = "TEXT")
809
+ private String remarks;
810
+
811
+ @Column(name = "作成日時", nullable = false, updatable = false)
812
+ private LocalDateTime createdAt;
813
+
814
+ @PrePersist
815
+ protected void onCreate() {
816
+ this.createdAt = LocalDateTime.now();
817
+ if (this.totalCount == null) {
818
+ this.totalCount = 0;
819
+ }
820
+ if (this.successCount == null) {
821
+ this.successCount = 0;
822
+ }
823
+ if (this.errorCount == null) {
824
+ this.errorCount = 0;
825
+ }
826
+ if (this.totalAmount == null) {
827
+ this.totalAmount = BigDecimal.ZERO;
828
+ }
829
+ }
830
+ }
831
+ ```
832
+
833
+ </details>
834
+
835
+ ---
836
+
837
+ ## 17.5 TDD によるパターンマッチングの実装
838
+
839
+ ### パターンマッチングのテスト
840
+
841
+ <details>
842
+ <summary>AutoJournalPatternTest.java</summary>
843
+
844
+ ```java
845
+ package com.example.accounting.domain.model.autojournal;
846
+
847
+ import org.junit.jupiter.api.DisplayName;
848
+ import org.junit.jupiter.api.Nested;
849
+ import org.junit.jupiter.api.Test;
850
+
851
+ import java.time.LocalDate;
852
+
853
+ import static org.assertj.core.api.Assertions.*;
854
+
855
+ @DisplayName("自動仕訳パターンのテスト")
856
+ class AutoJournalPatternTest {
857
+
858
+ @Nested
859
+ @DisplayName("パターンマッチング")
860
+ class PatternMatchingTest {
861
+
862
+ @Test
863
+ @DisplayName("商品グループALLは全ての商品グループにマッチする")
864
+ void shouldMatchAllProductGroups() {
865
+ // Given: 商品グループALLのパターン
866
+ var pattern = AutoJournalPattern.builder()
867
+ .patternCode("P001")
868
+ .productGroup("ALL")
869
+ .customerGroup("ALL")
870
+ .build();
871
+
872
+ // When & Then
873
+ assertThat(pattern.matches("加工品", "一般")).isTrue();
874
+ assertThat(pattern.matches("生鮮品", "一般")).isTrue();
875
+ assertThat(pattern.matches("雑貨", "特約店")).isTrue();
876
+ }
877
+
878
+ @Test
879
+ @DisplayName("特定の商品グループのみにマッチする")
880
+ void shouldMatchSpecificProductGroup() {
881
+ // Given: 加工品専用パターン
882
+ var pattern = AutoJournalPattern.builder()
883
+ .patternCode("P002")
884
+ .productGroup("加工品")
885
+ .customerGroup("ALL")
886
+ .build();
887
+
888
+ // When & Then
889
+ assertThat(pattern.matches("加工品", "一般")).isTrue();
890
+ assertThat(pattern.matches("生鮮品", "一般")).isFalse();
891
+ }
892
+
893
+ @Test
894
+ @DisplayName("商品グループと顧客グループの両方でマッチングする")
895
+ void shouldMatchBothProductAndCustomerGroup() {
896
+ // Given: 特定の組み合わせパターン
897
+ var pattern = AutoJournalPattern.builder()
898
+ .patternCode("P003")
899
+ .productGroup("加工品")
900
+ .customerGroup("特約店")
901
+ .build();
902
+
903
+ // When & Then
904
+ assertThat(pattern.matches("加工品", "特約店")).isTrue();
905
+ assertThat(pattern.matches("加工品", "一般")).isFalse();
906
+ assertThat(pattern.matches("生鮮品", "特約店")).isFalse();
907
+ }
908
+ }
909
+
910
+ @Nested
911
+ @DisplayName("有効期間チェック")
912
+ class ValidityCheckTest {
913
+
914
+ @Test
915
+ @DisplayName("有効期間内の日付でtrueを返す")
916
+ void shouldReturnTrueForValidDate() {
917
+ // Given
918
+ var pattern = AutoJournalPattern.builder()
919
+ .patternCode("P001")
920
+ .validFrom(LocalDate.of(2024, 1, 1))
921
+ .validTo(LocalDate.of(2024, 12, 31))
922
+ .build();
923
+
924
+ // When & Then
925
+ assertThat(pattern.isValidAt(LocalDate.of(2024, 6, 15))).isTrue();
926
+ assertThat(pattern.isValidAt(LocalDate.of(2024, 1, 1))).isTrue();
927
+ assertThat(pattern.isValidAt(LocalDate.of(2024, 12, 31))).isTrue();
928
+ }
929
+
930
+ @Test
931
+ @DisplayName("有効期間外の日付でfalseを返す")
932
+ void shouldReturnFalseForInvalidDate() {
933
+ // Given
934
+ var pattern = AutoJournalPattern.builder()
935
+ .patternCode("P001")
936
+ .validFrom(LocalDate.of(2024, 1, 1))
937
+ .validTo(LocalDate.of(2024, 12, 31))
938
+ .build();
939
+
940
+ // When & Then
941
+ assertThat(pattern.isValidAt(LocalDate.of(2023, 12, 31))).isFalse();
942
+ assertThat(pattern.isValidAt(LocalDate.of(2025, 1, 1))).isFalse();
943
+ }
944
+ }
945
+ }
946
+ ```
947
+
948
+ </details>
949
+
950
+ ### リポジトリのテスト
951
+
952
+ <details>
953
+ <summary>AutoJournalPatternRepositoryTest.java</summary>
954
+
955
+ ```java
956
+ package com.example.accounting.infrastructure.persistence;
957
+
958
+ import com.example.accounting.domain.model.autojournal.*;
959
+ import com.example.accounting.infrastructure.persistence.repository.AutoJournalPatternJpaRepository;
960
+ import org.junit.jupiter.api.DisplayName;
961
+ import org.junit.jupiter.api.Nested;
962
+ import org.junit.jupiter.api.Test;
963
+ import org.springframework.beans.factory.annotation.Autowired;
964
+ import org.springframework.boot.test.autoconfigure.jdbc.AutoConfigureTestDatabase;
965
+ import org.springframework.boot.test.autoconfigure.orm.jpa.DataJpaTest;
966
+ import org.springframework.test.context.DynamicPropertyRegistry;
967
+ import org.springframework.test.context.DynamicPropertySource;
968
+ import org.testcontainers.containers.PostgreSQLContainer;
969
+ import org.testcontainers.junit.jupiter.Container;
970
+ import org.testcontainers.junit.jupiter.Testcontainers;
971
+
972
+ import java.time.LocalDate;
973
+
974
+ import static org.assertj.core.api.Assertions.*;
975
+
976
+ /**
977
+ * 自動仕訳リポジトリのテスト
978
+ *
979
+ * JPA 版では @DataJpaTest を使用(MyBatis 版では @MybatisTest)
980
+ * TestContainers で PostgreSQL コンテナを起動し、実際のDBでテスト
981
+ */
982
+ @DataJpaTest
983
+ @AutoConfigureTestDatabase(replace = AutoConfigureTestDatabase.Replace.NONE)
984
+ @Testcontainers
985
+ @DisplayName("自動仕訳パターンリポジトリのテスト")
986
+ class AutoJournalPatternRepositoryTest {
987
+
988
+ @Container
989
+ static PostgreSQLContainer<?> postgres = new PostgreSQLContainer<>("postgres:16")
990
+ .withDatabaseName("testdb")
991
+ .withUsername("test")
992
+ .withPassword("test");
993
+
994
+ @DynamicPropertySource
995
+ static void configureProperties(DynamicPropertyRegistry registry) {
996
+ registry.add("spring.datasource.url", postgres::getJdbcUrl);
997
+ registry.add("spring.datasource.username", postgres::getUsername);
998
+ registry.add("spring.datasource.password", postgres::getPassword);
999
+ }
1000
+
1001
+ @Autowired
1002
+ private AutoJournalPatternJpaRepository patternRepository;
1003
+
1004
+ @Nested
1005
+ @DisplayName("パターンマスタの操作")
1006
+ class PatternMasterTest {
1007
+
1008
+ @Test
1009
+ @DisplayName("パターンマスタを登録・取得できる")
1010
+ void shouldSaveAndFindPattern() {
1011
+ // Given
1012
+ var pattern = AutoJournalPattern.builder()
1013
+ .patternCode("P001")
1014
+ .patternName("加工品売上")
1015
+ .productGroup("加工品")
1016
+ .customerGroup("ALL")
1017
+ .debitAccountCode("11300") // 売掛金
1018
+ .creditAccountCode("41110") // 売上加工品
1019
+ .validFrom(LocalDate.of(2024, 1, 1))
1020
+ .validTo(LocalDate.of(9999, 12, 31))
1021
+ .priority(100)
1022
+ .build();
1023
+
1024
+ // When
1025
+ patternRepository.save(pattern);
1026
+ var saved = patternRepository.findById("P001");
1027
+
1028
+ // Then
1029
+ assertThat(saved).isPresent();
1030
+ assertThat(saved.get().getPatternName()).isEqualTo("加工品売上");
1031
+ assertThat(saved.get().getProductGroup()).isEqualTo("加工品");
1032
+ }
1033
+
1034
+ @Test
1035
+ @DisplayName("有効なパターンを優先順位順で取得できる")
1036
+ void shouldFindValidPatternsSortedByPriority() {
1037
+ // Given: 複数のパターンを登録
1038
+ var pattern1 = AutoJournalPattern.builder()
1039
+ .patternCode("P001")
1040
+ .patternName("加工品売上")
1041
+ .productGroup("加工品")
1042
+ .customerGroup("ALL")
1043
+ .debitAccountCode("11300")
1044
+ .creditAccountCode("41110")
1045
+ .validFrom(LocalDate.of(2024, 1, 1))
1046
+ .validTo(LocalDate.of(9999, 12, 31))
1047
+ .priority(200)
1048
+ .build();
1049
+
1050
+ var pattern2 = AutoJournalPattern.builder()
1051
+ .patternCode("P002")
1052
+ .patternName("生鮮品売上")
1053
+ .productGroup("生鮮品")
1054
+ .customerGroup("ALL")
1055
+ .debitAccountCode("11300")
1056
+ .creditAccountCode("41120")
1057
+ .validFrom(LocalDate.of(2024, 1, 1))
1058
+ .validTo(LocalDate.of(9999, 12, 31))
1059
+ .priority(100)
1060
+ .build();
1061
+
1062
+ patternRepository.save(pattern1);
1063
+ patternRepository.save(pattern2);
1064
+
1065
+ // When
1066
+ var patterns = patternRepository.findValidPatterns(LocalDate.of(2024, 4, 1));
1067
+
1068
+ // Then
1069
+ assertThat(patterns).isSortedAccordingTo(
1070
+ (p1, p2) -> p1.getPriority().compareTo(p2.getPriority())
1071
+ );
1072
+ assertThat(patterns.get(0).getPatternCode()).isEqualTo("P002"); // priority 100
1073
+ }
1074
+ }
1075
+ }
1076
+ ```
1077
+
1078
+ </details>
1079
+
1080
+ ---
1081
+
1082
+ ## 17.6 JpaRepository インターフェース
1083
+
1084
+ ### 自動仕訳パターンマスタ Repository
1085
+
1086
+ <details>
1087
+ <summary>AutoJournalPatternJpaRepository.java</summary>
1088
+
1089
+ ```java
1090
+ package com.example.accounting.infrastructure.persistence.repository;
1091
+
1092
+ import com.example.accounting.domain.model.autojournal.AutoJournalPattern;
1093
+ import org.springframework.data.jpa.repository.EntityGraph;
1094
+ import org.springframework.data.jpa.repository.JpaRepository;
1095
+ import org.springframework.data.jpa.repository.Query;
1096
+ import org.springframework.data.repository.query.Param;
1097
+ import org.springframework.stereotype.Repository;
1098
+
1099
+ import java.time.LocalDate;
1100
+ import java.util.List;
1101
+ import java.util.Optional;
1102
+
1103
+ /**
1104
+ * 自動仕訳パターンマスタ JpaRepository
1105
+ *
1106
+ * JPA 版では JpaRepository を継承(MyBatis 版では Mapper インターフェース)
1107
+ * メソッド名規約で自動的にクエリを生成
1108
+ */
1109
+ @Repository
1110
+ public interface AutoJournalPatternJpaRepository extends JpaRepository<AutoJournalPattern, String> {
1111
+
1112
+ /**
1113
+ * 有効なパターンを優先順位順で取得
1114
+ * @Query で JPQL を明示的に記述
1115
+ */
1116
+ @Query("SELECT p FROM AutoJournalPattern p " +
1117
+ "WHERE p.validFrom <= :date AND p.validTo >= :date " +
1118
+ "ORDER BY p.priority, p.patternCode")
1119
+ List<AutoJournalPattern> findValidPatterns(@Param("date") LocalDate date);
1120
+
1121
+ /**
1122
+ * パターンコードで検索(勘定科目も一緒に取得)
1123
+ * @EntityGraph で N+1 問題を防止
1124
+ */
1125
+ @EntityGraph(attributePaths = {"debitAccount", "creditAccount"})
1126
+ Optional<AutoJournalPattern> findWithAccountsByPatternCode(String patternCode);
1127
+
1128
+ /**
1129
+ * 商品グループで検索
1130
+ */
1131
+ List<AutoJournalPattern> findByProductGroupOrderByPriority(String productGroup);
1132
+
1133
+ /**
1134
+ * 顧客グループで検索
1135
+ */
1136
+ List<AutoJournalPattern> findByCustomerGroupOrderByPriority(String customerGroup);
1137
+ }
1138
+ ```
1139
+
1140
+ </details>
1141
+
1142
+ ### 自動仕訳データ Repository
1143
+
1144
+ <details>
1145
+ <summary>AutoJournalEntryJpaRepository.java</summary>
1146
+
1147
+ ```java
1148
+ package com.example.accounting.infrastructure.persistence.repository;
1149
+
1150
+ import com.example.accounting.domain.model.autojournal.AutoJournalEntry;
1151
+ import com.example.accounting.domain.model.autojournal.AutoJournalStatus;
1152
+ import org.springframework.data.jpa.repository.EntityGraph;
1153
+ import org.springframework.data.jpa.repository.JpaRepository;
1154
+ import org.springframework.data.jpa.repository.JpaSpecificationExecutor;
1155
+ import org.springframework.data.jpa.repository.Query;
1156
+ import org.springframework.data.repository.query.Param;
1157
+ import org.springframework.stereotype.Repository;
1158
+
1159
+ import java.time.LocalDate;
1160
+ import java.util.List;
1161
+ import java.util.Optional;
1162
+
1163
+ /**
1164
+ * 自動仕訳データ JpaRepository
1165
+ *
1166
+ * JpaSpecificationExecutor を継承して動的クエリに対応
1167
+ * MyBatis 版では動的 SQL を XML で記述していたが、
1168
+ * JPA 版では Specification パターンを使用
1169
+ */
1170
+ @Repository
1171
+ public interface AutoJournalEntryJpaRepository
1172
+ extends JpaRepository<AutoJournalEntry, String>,
1173
+ JpaSpecificationExecutor<AutoJournalEntry> {
1174
+
1175
+ /**
1176
+ * 売上番号で検索
1177
+ */
1178
+ List<AutoJournalEntry> findBySalesNumberOrderBySalesLineNumberAscDebitCreditTypeAsc(
1179
+ String salesNumber);
1180
+
1181
+ /**
1182
+ * 未転記の自動仕訳を取得
1183
+ */
1184
+ @Query("SELECT e FROM AutoJournalEntry e " +
1185
+ "WHERE e.postedFlag = false " +
1186
+ "AND e.status = com.example.accounting.domain.model.autojournal.AutoJournalStatus.COMPLETED " +
1187
+ "ORDER BY e.postingDate, e.autoJournalNumber")
1188
+ List<AutoJournalEntry> findUnposted();
1189
+
1190
+ /**
1191
+ * 未転記の自動仕訳を日付で取得
1192
+ */
1193
+ @Query("SELECT e FROM AutoJournalEntry e " +
1194
+ "WHERE e.postedFlag = false " +
1195
+ "AND e.status = com.example.accounting.domain.model.autojournal.AutoJournalStatus.COMPLETED " +
1196
+ "AND e.postingDate = :date " +
1197
+ "ORDER BY e.autoJournalNumber")
1198
+ List<AutoJournalEntry> findUnpostedByDate(@Param("date") LocalDate date);
1199
+
1200
+ /**
1201
+ * ステータスで検索
1202
+ */
1203
+ List<AutoJournalEntry> findByStatusOrderByPostingDateAscAutoJournalNumberAsc(
1204
+ AutoJournalStatus status);
1205
+
1206
+ /**
1207
+ * パターンと関連エンティティを一緒に取得
1208
+ * @EntityGraph で N+1 問題を防止
1209
+ */
1210
+ @EntityGraph(attributePaths = {"pattern", "account", "department"})
1211
+ Optional<AutoJournalEntry> findWithRelationsById(String autoJournalNumber);
1212
+ }
1213
+ ```
1214
+
1215
+ </details>
1216
+
1217
+ ---
1218
+
1219
+ ## 17.7 Spring Data JPA Specification による動的クエリ
1220
+
1221
+ JPA 版では、複雑な検索条件を組み合わせる際に Specification パターンを使用します。MyBatis 版では XML の `<if>` タグで動的 SQL を構築していましたが、JPA 版では型安全な Specification で実現します。
1222
+
1223
+ <details>
1224
+ <summary>AutoJournalEntrySpecifications.java</summary>
1225
+
1226
+ ```java
1227
+ package com.example.accounting.infrastructure.persistence.specification;
1228
+
1229
+ import com.example.accounting.domain.model.autojournal.AutoJournalEntry;
1230
+ import com.example.accounting.domain.model.autojournal.AutoJournalStatus;
1231
+ import org.springframework.data.jpa.domain.Specification;
1232
+
1233
+ import java.time.LocalDate;
1234
+
1235
+ /**
1236
+ * 自動仕訳エントリ検索用 Specification
1237
+ *
1238
+ * JPA 版では Specification パターンで動的クエリを構築
1239
+ * MyBatis 版の XML 動的 SQL を Java コードで表現
1240
+ *
1241
+ * 使用例:
1242
+ * var spec = AutoJournalEntrySpecifications.unposted()
1243
+ * .and(AutoJournalEntrySpecifications.postingDateBetween(from, to))
1244
+ * .and(AutoJournalEntrySpecifications.hasStatus(AutoJournalStatus.COMPLETED));
1245
+ * var entries = repository.findAll(spec);
1246
+ */
1247
+ public class AutoJournalEntrySpecifications {
1248
+
1249
+ /**
1250
+ * 未転記条件
1251
+ */
1252
+ public static Specification<AutoJournalEntry> unposted() {
1253
+ return (root, query, cb) -> cb.equal(root.get("postedFlag"), false);
1254
+ }
1255
+
1256
+ /**
1257
+ * ステータス条件
1258
+ */
1259
+ public static Specification<AutoJournalEntry> hasStatus(AutoJournalStatus status) {
1260
+ return (root, query, cb) -> cb.equal(root.get("status"), status);
1261
+ }
1262
+
1263
+ /**
1264
+ * 起票日範囲条件
1265
+ */
1266
+ public static Specification<AutoJournalEntry> postingDateBetween(
1267
+ LocalDate from, LocalDate to) {
1268
+ return (root, query, cb) -> cb.between(root.get("postingDate"), from, to);
1269
+ }
1270
+
1271
+ /**
1272
+ * 起票日条件
1273
+ */
1274
+ public static Specification<AutoJournalEntry> postingDateEquals(LocalDate date) {
1275
+ return (root, query, cb) -> cb.equal(root.get("postingDate"), date);
1276
+ }
1277
+
1278
+ /**
1279
+ * 売上番号条件
1280
+ */
1281
+ public static Specification<AutoJournalEntry> salesNumberEquals(String salesNumber) {
1282
+ return (root, query, cb) -> cb.equal(root.get("salesNumber"), salesNumber);
1283
+ }
1284
+
1285
+ /**
1286
+ * パターンコード条件
1287
+ */
1288
+ public static Specification<AutoJournalEntry> patternCodeEquals(String patternCode) {
1289
+ return (root, query, cb) -> cb.equal(root.get("patternCode"), patternCode);
1290
+ }
1291
+
1292
+ /**
1293
+ * 勘定科目コード条件
1294
+ */
1295
+ public static Specification<AutoJournalEntry> accountCodeEquals(String accountCode) {
1296
+ return (root, query, cb) -> cb.equal(root.get("accountCode"), accountCode);
1297
+ }
1298
+
1299
+ /**
1300
+ * 部門コード条件
1301
+ */
1302
+ public static Specification<AutoJournalEntry> departmentCodeEquals(String departmentCode) {
1303
+ return (root, query, cb) -> cb.equal(root.get("departmentCode"), departmentCode);
1304
+ }
1305
+
1306
+ /**
1307
+ * エラー有無条件
1308
+ */
1309
+ public static Specification<AutoJournalEntry> hasError() {
1310
+ return (root, query, cb) -> cb.isNotNull(root.get("errorCode"));
1311
+ }
1312
+
1313
+ /**
1314
+ * エラー無し条件
1315
+ */
1316
+ public static Specification<AutoJournalEntry> noError() {
1317
+ return (root, query, cb) -> cb.isNull(root.get("errorCode"));
1318
+ }
1319
+ }
1320
+ ```
1321
+
1322
+ </details>
1323
+
1324
+ ### Specification の使用例
1325
+
1326
+ ```java
1327
+ // 動的クエリの組み立て
1328
+ var spec = AutoJournalEntrySpecifications.unposted()
1329
+ .and(AutoJournalEntrySpecifications.postingDateBetween(
1330
+ LocalDate.of(2024, 4, 1),
1331
+ LocalDate.of(2024, 4, 30)))
1332
+ .and(AutoJournalEntrySpecifications.hasStatus(AutoJournalStatus.COMPLETED))
1333
+ .and(AutoJournalEntrySpecifications.noError());
1334
+
1335
+ // クエリ実行
1336
+ var entries = entryJpaRepository.findAll(spec);
1337
+ ```
1338
+
1339
+ ---
1340
+
1341
+ ## 17.8 自動仕訳サービスの実装
1342
+
1343
+ <details>
1344
+ <summary>AutoJournalService.java(一部抜粋)</summary>
1345
+
1346
+ ```java
1347
+ package com.example.accounting.application.service;
1348
+
1349
+ import com.example.accounting.application.dto.AutoJournalResult;
1350
+ import com.example.accounting.application.dto.SalesData;
1351
+ import com.example.accounting.application.port.out.AutoJournalRepository;
1352
+ import com.example.accounting.domain.model.autojournal.*;
1353
+ import com.example.accounting.domain.model.journal.*;
1354
+ import lombok.RequiredArgsConstructor;
1355
+ import lombok.extern.slf4j.Slf4j;
1356
+ import org.springframework.stereotype.Service;
1357
+ import org.springframework.transaction.annotation.Transactional;
1358
+
1359
+ import java.math.BigDecimal;
1360
+ import java.time.LocalDate;
1361
+ import java.time.LocalDateTime;
1362
+ import java.util.*;
1363
+
1364
+ /**
1365
+ * 自動仕訳サービス
1366
+ *
1367
+ * JPA 版でもビジネスロジックは MyBatis 版と同じ
1368
+ * リポジトリ経由でデータアクセスを行う
1369
+ */
1370
+ @Service
1371
+ @RequiredArgsConstructor
1372
+ @Slf4j
1373
+ public class AutoJournalService {
1374
+
1375
+ private final AutoJournalRepository autoJournalRepository;
1376
+ private final JournalRepository journalRepository;
1377
+
1378
+ /**
1379
+ * 売上データから自動仕訳を生成する
1380
+ */
1381
+ @Transactional
1382
+ public AutoJournalResult generateAutoJournals(List<SalesData> salesDataList) {
1383
+ var processNumber = generateProcessNumber();
1384
+ var startTime = LocalDateTime.now();
1385
+
1386
+ int successCount = 0;
1387
+ int errorCount = 0;
1388
+ BigDecimal totalAmount = BigDecimal.ZERO;
1389
+
1390
+ // 有効なパターンを取得
1391
+ var patterns = autoJournalRepository.findValidPatterns(LocalDate.now());
1392
+
1393
+ List<AutoJournalEntry> allEntries = new ArrayList<>();
1394
+
1395
+ for (SalesData sale : salesDataList) {
1396
+ try {
1397
+ // マッチするパターンを検索
1398
+ var matchedPattern = findMatchingPattern(patterns, sale);
1399
+
1400
+ if (matchedPattern.isPresent()) {
1401
+ // 自動仕訳データを生成
1402
+ var entries = createAutoJournalEntries(sale, matchedPattern.get());
1403
+ allEntries.addAll(entries);
1404
+
1405
+ successCount++;
1406
+ totalAmount = totalAmount.add(sale.getAmount());
1407
+ } else {
1408
+ // パターン不一致エラー
1409
+ allEntries.add(createErrorEntry(sale, "E001", "マッチするパターンが見つかりません"));
1410
+ errorCount++;
1411
+ }
1412
+ } catch (Exception e) {
1413
+ log.error("自動仕訳処理エラー: {}", sale.getSalesNumber(), e);
1414
+ allEntries.add(createErrorEntry(sale, "E999", e.getMessage()));
1415
+ errorCount++;
1416
+ }
1417
+ }
1418
+
1419
+ // JPA 版では saveAll でバッチ保存
1420
+ autoJournalRepository.saveAllEntries(allEntries);
1421
+
1422
+ // 処理履歴を保存
1423
+ var history = AutoJournalHistory.builder()
1424
+ .processNumber(processNumber)
1425
+ .processDateTime(startTime)
1426
+ .targetFromDate(salesDataList.stream()
1427
+ .map(SalesData::getSalesDate)
1428
+ .min(LocalDate::compareTo).orElse(LocalDate.now()))
1429
+ .targetToDate(salesDataList.stream()
1430
+ .map(SalesData::getSalesDate)
1431
+ .max(LocalDate::compareTo).orElse(LocalDate.now()))
1432
+ .totalCount(salesDataList.size())
1433
+ .successCount(successCount)
1434
+ .errorCount(errorCount)
1435
+ .totalAmount(totalAmount)
1436
+ .build();
1437
+
1438
+ autoJournalRepository.saveHistory(history);
1439
+
1440
+ return new AutoJournalResult(processNumber, successCount, errorCount, totalAmount);
1441
+ }
1442
+
1443
+ /**
1444
+ * マッチするパターンを検索
1445
+ */
1446
+ private Optional<AutoJournalPattern> findMatchingPattern(
1447
+ List<AutoJournalPattern> patterns, SalesData sale) {
1448
+ return patterns.stream()
1449
+ .filter(p -> p.matches(sale.getProductGroup(), sale.getCustomerGroup()))
1450
+ .filter(p -> p.isValidAt(sale.getSalesDate()))
1451
+ .findFirst();
1452
+ }
1453
+ }
1454
+ ```
1455
+
1456
+ </details>
1457
+
1458
+ ---
1459
+
1460
+ ## 17.9 ヘキサゴナルアーキテクチャでのポート設計
1461
+
1462
+ ### Output Port(リポジトリインターフェース)
1463
+
1464
+ <details>
1465
+ <summary>AutoJournalRepository.java</summary>
1466
+
1467
+ ```java
1468
+ package com.example.accounting.application.port.out;
1469
+
1470
+ import com.example.accounting.domain.model.autojournal.*;
1471
+
1472
+ import java.time.LocalDate;
1473
+ import java.util.List;
1474
+ import java.util.Optional;
1475
+
1476
+ /**
1477
+ * 自動仕訳リポジトリ(Output Port)
1478
+ *
1479
+ * ヘキサゴナルアーキテクチャにおける出力ポート
1480
+ * アプリケーション層はこのインターフェースに依存し、
1481
+ * インフラストラクチャ層で実装する
1482
+ */
1483
+ public interface AutoJournalRepository {
1484
+
1485
+ // パターンマスタ操作
1486
+ void savePattern(AutoJournalPattern pattern);
1487
+
1488
+ Optional<AutoJournalPattern> findPatternByCode(String patternCode);
1489
+
1490
+ Optional<AutoJournalPattern> findPatternWithAccounts(String patternCode);
1491
+
1492
+ List<AutoJournalPattern> findAllPatterns();
1493
+
1494
+ List<AutoJournalPattern> findValidPatterns(LocalDate date);
1495
+
1496
+ void updatePattern(AutoJournalPattern pattern);
1497
+
1498
+ void deletePattern(String patternCode);
1499
+
1500
+ // 自動仕訳エントリ操作
1501
+ void saveEntry(AutoJournalEntry entry);
1502
+
1503
+ void saveAllEntries(List<AutoJournalEntry> entries);
1504
+
1505
+ Optional<AutoJournalEntry> findEntryByNumber(String autoJournalNumber);
1506
+
1507
+ Optional<AutoJournalEntry> findEntryWithRelations(String autoJournalNumber);
1508
+
1509
+ List<AutoJournalEntry> findEntriesBySalesNumber(String salesNumber);
1510
+
1511
+ List<AutoJournalEntry> findUnpostedEntries();
1512
+
1513
+ List<AutoJournalEntry> findUnpostedEntriesByDate(LocalDate date);
1514
+
1515
+ List<AutoJournalEntry> findEntriesByStatus(AutoJournalStatus status);
1516
+
1517
+ void updateEntry(AutoJournalEntry entry);
1518
+
1519
+ // 処理履歴操作
1520
+ void saveHistory(AutoJournalHistory history);
1521
+
1522
+ Optional<AutoJournalHistory> findHistoryByNumber(String processNumber);
1523
+
1524
+ List<AutoJournalHistory> findHistoriesByDateRange(LocalDate fromDate, LocalDate toDate);
1525
+
1526
+ void deleteAll();
1527
+ }
1528
+ ```
1529
+
1530
+ </details>
1531
+
1532
+ ---
1533
+
1534
+ ## 17.10 まとめ:MyBatis 版と JPA 版の比較
1535
+
1536
+ ### データアクセス層のアーキテクチャ比較
1537
+
1538
+ ```plantuml
1539
+ @startuml
1540
+
1541
+ title MyBatis 版と JPA 版のアーキテクチャ比較
1542
+
1543
+ package "MyBatis 版" {
1544
+ [AutoJournalService] as MService
1545
+ [AutoJournalMapper] as MMapper
1546
+ [AutoJournalMapper.xml] as MXml
1547
+ database "PostgreSQL" as MDB
1548
+
1549
+ MService --> MMapper
1550
+ MMapper --> MXml
1551
+ MXml --> MDB
1552
+ }
1553
+
1554
+ package "JPA 版" {
1555
+ [AutoJournalService] as JService
1556
+ [AutoJournalRepository\n(Output Port)] as JPort
1557
+ [AutoJournalRepositoryImpl] as JImpl
1558
+ [AutoJournalPatternJpaRepository] as JJpa
1559
+ database "PostgreSQL" as JDB
1560
+
1561
+ JService --> JPort
1562
+ JPort <|.. JImpl
1563
+ JImpl --> JJpa
1564
+ JJpa --> JDB
1565
+ }
1566
+
1567
+ @enduml
1568
+ ```
1569
+
1570
+ ### JPA 版の特徴
1571
+
1572
+ | 観点 | JPA 版の特徴 |
1573
+ |-----|-------------|
1574
+ | 型安全性 | Specification パターンによる型安全な動的クエリ |
1575
+ | N+1 問題対策 | @EntityGraph による宣言的な解決 |
1576
+ | バッチ処理 | saveAll() による効率的な一括保存 |
1577
+ | ポート分離 | Output Port インターフェースによる抽象化 |
1578
+ | テスト容易性 | @DataJpaTest + TestContainers |
1579
+
1580
+ ---
1581
+
1582
+ ## 次章予告
1583
+
1584
+ [第18章](chapter18-orm.md)では、勘定科目残高テーブルの設計について、JPA 版での実装を解説します。