@k2works/claude-code-booster 3.6.1 → 3.7.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -21
- package/README.md +42 -42
- package/bin/claude-code-booster +90 -90
- package/lib/assets/.claude/README.md +258 -239
- package/lib/assets/.claude/agent-memory/xp-programmer/MEMORY.md +6 -0
- package/lib/assets/.claude/agent-memory/xp-programmer/project_cargo_tracker.md +11 -0
- package/lib/assets/.claude/agent-memory/xp-programmer/project_ddd_patterns.md +27 -0
- package/lib/assets/.claude/agent-memory/xp-programmer/project_us07_route_assignment.md +19 -0
- package/lib/assets/.claude/scripts/generate-inception-deck.mjs +911 -911
- package/lib/assets/.claude/settings.json +11 -11
- package/lib/assets/.claude/skills/ai-agent-guidelines/SKILL.md +111 -111
- package/lib/assets/.claude/skills/analyzing-architecture/SKILL.md +83 -83
- package/lib/assets/.claude/skills/analyzing-business/SKILL.md +95 -95
- package/lib/assets/.claude/skills/analyzing-data-model/SKILL.md +77 -77
- package/lib/assets/.claude/skills/analyzing-domain-model/SKILL.md +117 -117
- package/lib/assets/.claude/skills/analyzing-inception-deck/SKILL.md +84 -84
- package/lib/assets/.claude/skills/analyzing-non-functional/SKILL.md +95 -95
- package/lib/assets/.claude/skills/analyzing-operation/SKILL.md +95 -95
- package/lib/assets/.claude/skills/analyzing-requirements/SKILL.md +91 -91
- package/lib/assets/.claude/skills/analyzing-tech-stack/SKILL.md +101 -101
- package/lib/assets/.claude/skills/analyzing-test-strategy/SKILL.md +89 -89
- package/lib/assets/.claude/skills/analyzing-ui-design/SKILL.md +80 -80
- package/lib/assets/.claude/skills/analyzing-usecases/SKILL.md +72 -72
- package/lib/assets/.claude/skills/creating-adr/SKILL.md +113 -113
- package/lib/assets/.claude/skills/developing-backend/SKILL.md +100 -100
- package/lib/assets/.claude/skills/developing-frontend/SKILL.md +93 -93
- package/lib/assets/.claude/skills/developing-release/SKILL.md +120 -120
- package/lib/assets/.claude/skills/generating-bmc/SKILL.md +97 -0
- package/lib/assets/.claude/skills/generating-slides/SKILL.md +94 -94
- package/lib/assets/.claude/skills/git-commit/SKILL.md +81 -81
- package/lib/assets/.claude/skills/killing-processes/SKILL.md +44 -44
- package/lib/assets/.claude/skills/operating-backup/SKILL.md +59 -59
- package/lib/assets/.claude/skills/operating-cicd/SKILL.md +54 -54
- package/lib/assets/.claude/skills/operating-deploy/SKILL.md +67 -67
- package/lib/assets/.claude/skills/operating-docs/SKILL.md +219 -219
- package/lib/assets/.claude/skills/operating-provision/SKILL.md +77 -77
- package/lib/assets/.claude/skills/operating-setup/SKILL.md +63 -63
- package/lib/assets/.claude/skills/orchestrating-analysis/SKILL.md +104 -104
- package/lib/assets/.claude/skills/orchestrating-development/SKILL.md +162 -162
- package/lib/assets/.claude/skills/orchestrating-operation/SKILL.md +158 -158
- package/lib/assets/.claude/skills/orchestrating-project/SKILL.md +144 -144
- package/lib/assets/.claude/skills/planning-releases/SKILL.md +119 -119
- package/lib/assets/.claude/skills/syncing-github-project/SKILL.md +151 -151
- package/lib/assets/.claude/skills/tracking-progress/SKILL.md +91 -91
- package/lib/assets/.claude/skills/validating-iteration-plan/SKILL.md +215 -215
- package/lib/assets/.devcontainer/devcontainer.json +34 -34
- package/lib/assets/.env.example +17 -17
- package/lib/assets/.gitattributes +4 -4
- package/lib/assets/.github/workflows/docker-publish.yml +77 -77
- package/lib/assets/.github/workflows/mkdocs.yml +39 -39
- package/lib/assets/AGENTS.md +94 -94
- package/lib/assets/CLAUDE.md +1 -0
- package/lib/assets/README.md +254 -254
- package/lib/assets/docker-compose.yml +33 -33
- package/lib/assets/docs/adr/index.md +10 -10
- package/lib/assets/docs/article/functional-desgin-ppp/all/01-immutability-and-data-transformation.md +475 -475
- package/lib/assets/docs/article/functional-desgin-ppp/all/02-function-composition.md +519 -519
- package/lib/assets/docs/article/functional-desgin-ppp/all/03-polymorphism.md +537 -537
- package/lib/assets/docs/article/functional-desgin-ppp/all/04-data-validation.md +300 -300
- package/lib/assets/docs/article/functional-desgin-ppp/all/05-property-based-testing.md +320 -320
- package/lib/assets/docs/article/functional-desgin-ppp/all/06-tdd-and-functional.md +498 -498
- package/lib/assets/docs/article/functional-desgin-ppp/all/07-composite-pattern.md +298 -298
- package/lib/assets/docs/article/functional-desgin-ppp/all/08-decorator-pattern.md +291 -291
- package/lib/assets/docs/article/functional-desgin-ppp/all/09-adapter-pattern.md +336 -336
- package/lib/assets/docs/article/functional-desgin-ppp/all/10-strategy-pattern.md +303 -303
- package/lib/assets/docs/article/functional-desgin-ppp/all/11-command-pattern.md +286 -286
- package/lib/assets/docs/article/functional-desgin-ppp/all/12-visitor-pattern.md +322 -322
- package/lib/assets/docs/article/functional-desgin-ppp/all/13-abstract-factory-pattern.md +319 -319
- package/lib/assets/docs/article/functional-desgin-ppp/all/14-abstract-server-pattern.md +365 -365
- package/lib/assets/docs/article/functional-desgin-ppp/all/15-gossiping-bus-drivers.md +156 -156
- package/lib/assets/docs/article/functional-desgin-ppp/all/16-payroll-system.md +178 -178
- package/lib/assets/docs/article/functional-desgin-ppp/all/17-video-rental-system.md +312 -312
- package/lib/assets/docs/article/functional-desgin-ppp/all/18-concurrency-system.md +287 -287
- package/lib/assets/docs/article/functional-desgin-ppp/all/19-wa-tor-simulation.md +286 -286
- package/lib/assets/docs/article/functional-desgin-ppp/all/20-pattern-interactions.md +274 -274
- package/lib/assets/docs/article/functional-desgin-ppp/all/21-best-practices.md +294 -294
- package/lib/assets/docs/article/functional-desgin-ppp/all/22-oo-to-fp-migration.md +337 -337
- package/lib/assets/docs/article/functional-desgin-ppp/all/index.md +388 -388
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/01-immutability-and-data-transformation.md +273 -273
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/02-function-composition.md +380 -380
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/03-polymorphism.md +384 -384
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/04-clojure-spec.md +350 -350
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/05-property-based-testing.md +352 -352
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/06-tdd-in-functional.md +383 -383
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/07-composite-pattern.md +529 -529
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/08-decorator-pattern.md +395 -395
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/09-adapter-pattern.md +399 -399
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/10-strategy-pattern.md +485 -485
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/11-command-pattern.md +566 -566
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/12-visitor-pattern.md +567 -567
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/13-abstract-factory-pattern.md +475 -475
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/14-abstract-server-pattern.md +462 -462
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/15-gossiping-bus-drivers.md +325 -325
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/16-payroll-system.md +401 -401
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/17-video-rental-system.md +450 -450
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/18-concurrency-system.md +475 -475
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/19-wator-simulation.md +739 -739
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/20-pattern-interactions.md +567 -567
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/21-best-practices.md +518 -518
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/22-oo-to-fp-migration.md +532 -532
- package/lib/assets/docs/article/functional-desgin-ppp/clojure/index.md +241 -241
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/01-immutability-and-data-transformation.md +383 -383
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/02-function-composition.md +374 -374
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/03-polymorphism.md +375 -375
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/04-data-validation.md +195 -195
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/05-property-based-testing.md +268 -268
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/06-tdd-and-fp.md +294 -294
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/07-effects-and-pure-functions.md +164 -164
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/08-error-handling-strategies.md +168 -168
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/09-io-and-external-systems.md +254 -254
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/10-concurrency-patterns.md +269 -269
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/11-command-pattern.md +148 -148
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/12-visitor-pattern.md +176 -176
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/13-abstract-factory-pattern.md +604 -604
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/14-abstract-server-pattern.md +729 -729
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/15-gossiping-bus-drivers.md +291 -291
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/16-payroll-system.md +420 -420
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/17-video-rental-system.md +319 -319
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/18-concurrency-system.md +466 -466
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/19-wator-simulation.md +523 -523
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/20-pattern-interactions.md +287 -287
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/21-best-practices.md +340 -340
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/22-oo-to-fp-migration.md +395 -395
- package/lib/assets/docs/article/functional-desgin-ppp/elixir/index.md +248 -248
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/01-immutability-and-data-transformation.md +384 -384
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/02-function-composition.md +452 -452
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/03-polymorphism.md +495 -495
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/04-data-validation.md +416 -416
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/05-property-based-testing.md +382 -382
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/06-tdd-functional.md +687 -687
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/07-composite-pattern.md +442 -442
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/08-decorator-pattern.md +479 -479
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/09-adapter-pattern.md +479 -479
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/10-strategy-pattern.md +427 -427
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/11-command-pattern.md +428 -428
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/12-visitor-pattern.md +339 -339
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/13-abstract-factory-pattern.md +309 -309
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/14-abstract-server-pattern.md +596 -596
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/15-gossiping-bus-drivers.md +355 -355
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/16-payroll-system.md +350 -350
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/17-video-rental-system.md +414 -414
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/18-concurrency-system.md +367 -367
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/19-wator-simulation.md +403 -403
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/20-pattern-interactions.md +291 -291
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/21-best-practices.md +324 -324
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/22-oo-to-fp-migration.md +332 -332
- package/lib/assets/docs/article/functional-desgin-ppp/fsharp/index.md +274 -274
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/01-immutability-and-data-transformation.md +298 -298
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/02-function-composition.md +304 -304
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/03-polymorphism.md +362 -362
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/04-data-validation.md +257 -257
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/05-property-based-testing.md +254 -254
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/06-tdd-functional.md +283 -283
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/07-composite-pattern.md +395 -395
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/08-decorator-pattern.md +319 -319
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/09-adapter-pattern.md +382 -382
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/10-strategy-pattern.md +287 -287
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/11-command-pattern.md +303 -303
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/12-visitor-pattern.md +326 -326
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/13-abstract-factory-pattern.md +332 -332
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/14-abstract-server-pattern.md +379 -379
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/15-gossiping-bus-drivers.md +177 -177
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/16-payroll-system.md +219 -219
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/17-video-rental-system.md +244 -244
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/18-concurrency-system.md +363 -363
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/19-wator-simulation.md +438 -438
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/20-pattern-interactions.md +325 -325
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/21-best-practices.md +403 -403
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/22-oo-to-fp-migration.md +469 -469
- package/lib/assets/docs/article/functional-desgin-ppp/haskell/index.md +174 -174
- package/lib/assets/docs/article/functional-desgin-ppp/index.md +90 -90
- package/lib/assets/docs/article/functional-desgin-ppp/rust/01-immutability-and-data-transformation.md +450 -450
- package/lib/assets/docs/article/functional-desgin-ppp/rust/02-function-composition.md +463 -463
- package/lib/assets/docs/article/functional-desgin-ppp/rust/03-polymorphism.md +425 -425
- package/lib/assets/docs/article/functional-desgin-ppp/rust/04-data-validation.md +273 -273
- package/lib/assets/docs/article/functional-desgin-ppp/rust/05-property-based-testing.md +247 -247
- package/lib/assets/docs/article/functional-desgin-ppp/rust/06-tdd-and-functional.md +841 -841
- package/lib/assets/docs/article/functional-desgin-ppp/rust/07-composite-pattern.md +384 -384
- package/lib/assets/docs/article/functional-desgin-ppp/rust/08-decorator-pattern.md +383 -383
- package/lib/assets/docs/article/functional-desgin-ppp/rust/09-adapter-pattern.md +339 -339
- package/lib/assets/docs/article/functional-desgin-ppp/rust/10-strategy-pattern.md +331 -331
- package/lib/assets/docs/article/functional-desgin-ppp/rust/11-command-pattern.md +356 -356
- package/lib/assets/docs/article/functional-desgin-ppp/rust/12-visitor-pattern.md +379 -379
- package/lib/assets/docs/article/functional-desgin-ppp/rust/13-abstract-factory-pattern.md +361 -361
- package/lib/assets/docs/article/functional-desgin-ppp/rust/14-abstract-server-pattern.md +392 -392
- package/lib/assets/docs/article/functional-desgin-ppp/rust/15-gossiping-bus-drivers.md +300 -300
- package/lib/assets/docs/article/functional-desgin-ppp/rust/16-payroll-system.md +297 -297
- package/lib/assets/docs/article/functional-desgin-ppp/rust/17-video-rental-system.md +304 -304
- package/lib/assets/docs/article/functional-desgin-ppp/rust/18-concurrency-system.md +315 -315
- package/lib/assets/docs/article/functional-desgin-ppp/rust/19-wator-simulation.md +311 -311
- package/lib/assets/docs/article/functional-desgin-ppp/rust/20-pattern-interactions.md +304 -304
- package/lib/assets/docs/article/functional-desgin-ppp/rust/21-best-practices.md +336 -336
- package/lib/assets/docs/article/functional-desgin-ppp/rust/22-oo-to-fp-migration.md +349 -349
- package/lib/assets/docs/article/functional-desgin-ppp/rust/index.md +243 -243
- package/lib/assets/docs/article/functional-desgin-ppp/scala/01-immutability-and-data-transformation.md +328 -328
- package/lib/assets/docs/article/functional-desgin-ppp/scala/02-function-composition.md +348 -348
- package/lib/assets/docs/article/functional-desgin-ppp/scala/03-polymorphism.md +357 -357
- package/lib/assets/docs/article/functional-desgin-ppp/scala/04-data-validation.md +364 -364
- package/lib/assets/docs/article/functional-desgin-ppp/scala/05-property-based-testing.md +515 -515
- package/lib/assets/docs/article/functional-desgin-ppp/scala/06-tdd-functional.md +557 -557
- package/lib/assets/docs/article/functional-desgin-ppp/scala/07-composite-pattern.md +363 -363
- package/lib/assets/docs/article/functional-desgin-ppp/scala/08-decorator-pattern.md +327 -327
- package/lib/assets/docs/article/functional-desgin-ppp/scala/09-adapter-pattern.md +517 -517
- package/lib/assets/docs/article/functional-desgin-ppp/scala/10-strategy-pattern.md +441 -441
- package/lib/assets/docs/article/functional-desgin-ppp/scala/11-command-pattern.md +407 -407
- package/lib/assets/docs/article/functional-desgin-ppp/scala/12-visitor-pattern.md +379 -379
- package/lib/assets/docs/article/functional-desgin-ppp/scala/13-abstract-factory-pattern.md +398 -398
- package/lib/assets/docs/article/functional-desgin-ppp/scala/14-abstract-server-pattern.md +476 -476
- package/lib/assets/docs/article/functional-desgin-ppp/scala/15-gossiping-bus-drivers.md +391 -391
- package/lib/assets/docs/article/functional-desgin-ppp/scala/16-payroll-system.md +342 -342
- package/lib/assets/docs/article/functional-desgin-ppp/scala/17-video-rental-system.md +324 -324
- package/lib/assets/docs/article/functional-desgin-ppp/scala/18-concurrency-system.md +730 -730
- package/lib/assets/docs/article/functional-desgin-ppp/scala/19-wator-simulation.md +624 -624
- package/lib/assets/docs/article/functional-desgin-ppp/scala/20-pattern-interactions.md +512 -512
- package/lib/assets/docs/article/functional-desgin-ppp/scala/21-best-practices.md +433 -433
- package/lib/assets/docs/article/functional-desgin-ppp/scala/22-oo-to-fp-migration.md +688 -688
- package/lib/assets/docs/article/functional-desgin-ppp/scala/index.md +243 -243
- package/lib/assets/docs/article/getting-start-tdd/clojure/01-todo-list-and-first-test.md +166 -166
- package/lib/assets/docs/article/getting-start-tdd/clojure/02-fake-it-and-triangulation.md +162 -162
- package/lib/assets/docs/article/getting-start-tdd/clojure/03-obvious-implementation-and-refactoring.md +135 -135
- package/lib/assets/docs/article/getting-start-tdd/clojure/04-version-control-and-conventional-commits.md +88 -88
- package/lib/assets/docs/article/getting-start-tdd/clojure/05-package-management-and-static-analysis.md +299 -299
- package/lib/assets/docs/article/getting-start-tdd/clojure/06-task-runner-and-ci-cd.md +241 -241
- package/lib/assets/docs/article/getting-start-tdd/clojure/07-protocols-and-records.md +131 -131
- package/lib/assets/docs/article/getting-start-tdd/clojure/08-multimethods-and-design-patterns.md +130 -130
- package/lib/assets/docs/article/getting-start-tdd/clojure/09-namespaces-and-module-design.md +127 -127
- package/lib/assets/docs/article/getting-start-tdd/clojure/10-higher-order-functions-and-composition.md +114 -114
- package/lib/assets/docs/article/getting-start-tdd/clojure/11-persistent-data-and-pipeline.md +138 -138
- package/lib/assets/docs/article/getting-start-tdd/clojure/12-error-handling-and-spec.md +161 -161
- package/lib/assets/docs/article/getting-start-tdd/clojure/index.md +65 -65
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter01.md +232 -232
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter02.md +244 -244
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter03.md +202 -202
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter04.md +92 -92
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter05.md +256 -256
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter06.md +195 -195
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter07.md +214 -214
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter08.md +249 -249
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter09.md +174 -174
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter10.md +166 -166
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter11.md +192 -192
- package/lib/assets/docs/article/getting-start-tdd/csharp/chapter12.md +211 -211
- package/lib/assets/docs/article/getting-start-tdd/csharp/index.md +83 -83
- package/lib/assets/docs/article/getting-start-tdd/elixir/01-todo-list-and-first-test.md +87 -87
- package/lib/assets/docs/article/getting-start-tdd/elixir/02-fake-it-and-triangulation.md +95 -95
- package/lib/assets/docs/article/getting-start-tdd/elixir/03-obvious-implementation-and-refactoring.md +109 -109
- package/lib/assets/docs/article/getting-start-tdd/elixir/04-version-control-and-conventional-commits.md +96 -96
- package/lib/assets/docs/article/getting-start-tdd/elixir/05-package-management-and-static-analysis.md +88 -88
- package/lib/assets/docs/article/getting-start-tdd/elixir/06-task-runner-and-ci-cd.md +71 -71
- package/lib/assets/docs/article/getting-start-tdd/elixir/07-structs-and-protocols.md +110 -110
- package/lib/assets/docs/article/getting-start-tdd/elixir/08-pattern-matching-and-guards.md +108 -108
- package/lib/assets/docs/article/getting-start-tdd/elixir/09-module-design-and-behaviours.md +104 -104
- package/lib/assets/docs/article/getting-start-tdd/elixir/10-higher-order-functions-and-pipeline.md +178 -178
- package/lib/assets/docs/article/getting-start-tdd/elixir/11-stream-and-lazy-evaluation.md +142 -142
- package/lib/assets/docs/article/getting-start-tdd/elixir/12-error-handling-and-with.md +145 -145
- package/lib/assets/docs/article/getting-start-tdd/elixir/index.md +35 -35
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter01.md +202 -202
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter02.md +246 -246
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter03.md +218 -218
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter04.md +179 -179
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter05.md +267 -267
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter06.md +190 -190
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter07.md +161 -161
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter08.md +175 -175
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter09.md +222 -222
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter10.md +189 -189
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter11.md +212 -212
- package/lib/assets/docs/article/getting-start-tdd/fsharp/chapter12.md +215 -215
- package/lib/assets/docs/article/getting-start-tdd/fsharp/index.md +71 -71
- package/lib/assets/docs/article/getting-start-tdd/go/01-todo-list-and-first-test.md +213 -213
- package/lib/assets/docs/article/getting-start-tdd/go/02-fake-it-and-triangulation.md +302 -302
- package/lib/assets/docs/article/getting-start-tdd/go/03-obvious-implementation-and-refactoring.md +339 -339
- package/lib/assets/docs/article/getting-start-tdd/go/04-version-control-and-conventional-commits.md +112 -112
- package/lib/assets/docs/article/getting-start-tdd/go/05-package-management-and-static-analysis.md +272 -272
- package/lib/assets/docs/article/getting-start-tdd/go/06-task-runner-and-ci-cd.md +233 -233
- package/lib/assets/docs/article/getting-start-tdd/go/07-encapsulation-and-polymorphism.md +394 -394
- package/lib/assets/docs/article/getting-start-tdd/go/08-design-patterns.md +422 -422
- package/lib/assets/docs/article/getting-start-tdd/go/09-solid-principles-and-module-design.md +400 -400
- package/lib/assets/docs/article/getting-start-tdd/go/10-higher-order-functions-and-composition.md +226 -226
- package/lib/assets/docs/article/getting-start-tdd/go/11-immutable-data-and-pipeline.md +296 -296
- package/lib/assets/docs/article/getting-start-tdd/go/12-error-handling-and-type-safety.md +411 -411
- package/lib/assets/docs/article/getting-start-tdd/go/index.md +83 -83
- package/lib/assets/docs/article/getting-start-tdd/haskell/01-todo-list-and-first-test.md +279 -279
- package/lib/assets/docs/article/getting-start-tdd/haskell/02-fake-it-and-triangulation.md +337 -337
- package/lib/assets/docs/article/getting-start-tdd/haskell/03-obvious-implementation-and-refactoring.md +257 -257
- package/lib/assets/docs/article/getting-start-tdd/haskell/04-version-control-and-conventional-commits.md +182 -182
- package/lib/assets/docs/article/getting-start-tdd/haskell/05-package-management-and-static-analysis.md +313 -313
- package/lib/assets/docs/article/getting-start-tdd/haskell/06-task-runner-and-ci-cd.md +309 -309
- package/lib/assets/docs/article/getting-start-tdd/haskell/07-algebraic-data-types-and-type-classes.md +412 -412
- package/lib/assets/docs/article/getting-start-tdd/haskell/08-pattern-matching-and-guards.md +390 -390
- package/lib/assets/docs/article/getting-start-tdd/haskell/09-module-design-and-smart-constructors.md +461 -461
- package/lib/assets/docs/article/getting-start-tdd/haskell/10-higher-order-functions-and-currying.md +434 -434
- package/lib/assets/docs/article/getting-start-tdd/haskell/11-function-composition-and-point-free.md +392 -392
- package/lib/assets/docs/article/getting-start-tdd/haskell/12-monad-and-error-handling.md +631 -631
- package/lib/assets/docs/article/getting-start-tdd/haskell/index.md +49 -49
- package/lib/assets/docs/article/getting-start-tdd/index.md +93 -93
- package/lib/assets/docs/article/getting-start-tdd/integration/01-language-overview.md +375 -375
- package/lib/assets/docs/article/getting-start-tdd/integration/02-test-framework-comparison.md +349 -349
- package/lib/assets/docs/article/getting-start-tdd/integration/03-tdd-pattern-comparison.md +445 -445
- package/lib/assets/docs/article/getting-start-tdd/integration/04-type-system-comparison.md +409 -409
- package/lib/assets/docs/article/getting-start-tdd/integration/05-dev-environment-comparison.md +330 -330
- package/lib/assets/docs/article/getting-start-tdd/integration/06-learning-roadmap.md +290 -290
- package/lib/assets/docs/article/getting-start-tdd/integration/index.md +69 -69
- package/lib/assets/docs/article/getting-start-tdd/java/01-todo-list-and-first-test.md +234 -234
- package/lib/assets/docs/article/getting-start-tdd/java/02-fake-it-and-triangulation.md +261 -261
- package/lib/assets/docs/article/getting-start-tdd/java/03-obvious-implementation-and-refactoring.md +185 -185
- package/lib/assets/docs/article/getting-start-tdd/java/04-version-control-and-conventional-commits.md +115 -115
- package/lib/assets/docs/article/getting-start-tdd/java/05-package-management-and-static-analysis.md +382 -382
- package/lib/assets/docs/article/getting-start-tdd/java/06-task-runner-and-ci-cd.md +272 -272
- package/lib/assets/docs/article/getting-start-tdd/java/07-encapsulation-and-polymorphism.md +626 -626
- package/lib/assets/docs/article/getting-start-tdd/java/08-design-patterns.md +393 -393
- package/lib/assets/docs/article/getting-start-tdd/java/09-solid-principles-and-module-design.md +310 -310
- package/lib/assets/docs/article/getting-start-tdd/java/10-higher-order-functions-and-composition.md +188 -188
- package/lib/assets/docs/article/getting-start-tdd/java/11-immutable-data-and-pipeline.md +167 -167
- package/lib/assets/docs/article/getting-start-tdd/java/12-error-handling-and-type-safety.md +205 -205
- package/lib/assets/docs/article/getting-start-tdd/java/index.md +61 -61
- package/lib/assets/docs/article/getting-start-tdd/node/01-todo-list-and-first-test.md +244 -244
- package/lib/assets/docs/article/getting-start-tdd/node/02-fake-it-and-triangulation.md +262 -262
- package/lib/assets/docs/article/getting-start-tdd/node/03-obvious-implementation-and-refactoring.md +169 -169
- package/lib/assets/docs/article/getting-start-tdd/node/04-version-control-and-conventional-commits.md +112 -112
- package/lib/assets/docs/article/getting-start-tdd/node/05-package-management-and-static-analysis.md +314 -314
- package/lib/assets/docs/article/getting-start-tdd/node/06-task-runner-and-ci-cd.md +235 -235
- package/lib/assets/docs/article/getting-start-tdd/node/07-encapsulation-and-polymorphism.md +327 -327
- package/lib/assets/docs/article/getting-start-tdd/node/08-design-patterns.md +322 -322
- package/lib/assets/docs/article/getting-start-tdd/node/09-solid-principles-and-module-design.md +285 -285
- package/lib/assets/docs/article/getting-start-tdd/node/10-higher-order-functions-and-composition.md +199 -199
- package/lib/assets/docs/article/getting-start-tdd/node/11-immutable-data-and-pipeline.md +207 -207
- package/lib/assets/docs/article/getting-start-tdd/node/12-error-handling-and-type-safety.md +295 -295
- package/lib/assets/docs/article/getting-start-tdd/node/index.md +56 -56
- package/lib/assets/docs/article/getting-start-tdd/php/01-todo-list-and-first-test.md +259 -259
- package/lib/assets/docs/article/getting-start-tdd/php/02-fake-it-and-triangulation.md +200 -200
- package/lib/assets/docs/article/getting-start-tdd/php/03-obvious-implementation-and-refactoring.md +248 -248
- package/lib/assets/docs/article/getting-start-tdd/php/04-version-control-and-conventional-commits.md +141 -141
- package/lib/assets/docs/article/getting-start-tdd/php/05-package-management-and-static-analysis.md +410 -410
- package/lib/assets/docs/article/getting-start-tdd/php/06-task-runner-and-ci-cd.md +321 -321
- package/lib/assets/docs/article/getting-start-tdd/php/07-encapsulation-and-polymorphism.md +372 -372
- package/lib/assets/docs/article/getting-start-tdd/php/08-design-patterns.md +453 -453
- package/lib/assets/docs/article/getting-start-tdd/php/09-solid-principles-and-module-design.md +460 -460
- package/lib/assets/docs/article/getting-start-tdd/php/10-higher-order-functions-and-composition.md +182 -182
- package/lib/assets/docs/article/getting-start-tdd/php/11-immutable-data-and-pipeline.md +266 -266
- package/lib/assets/docs/article/getting-start-tdd/php/12-error-handling-and-type-safety.md +308 -308
- package/lib/assets/docs/article/getting-start-tdd/php/index.md +84 -84
- package/lib/assets/docs/article/getting-start-tdd/python/01-todo-list-and-first-test.md +201 -201
- package/lib/assets/docs/article/getting-start-tdd/python/02-fake-it-and-triangulation.md +247 -247
- package/lib/assets/docs/article/getting-start-tdd/python/03-obvious-implementation-and-refactoring.md +199 -199
- package/lib/assets/docs/article/getting-start-tdd/python/04-version-control-and-conventional-commits.md +87 -87
- package/lib/assets/docs/article/getting-start-tdd/python/05-package-management-and-static-analysis.md +274 -274
- package/lib/assets/docs/article/getting-start-tdd/python/06-task-runner-and-ci-cd.md +190 -190
- package/lib/assets/docs/article/getting-start-tdd/python/07-encapsulation-and-polymorphism.md +208 -208
- package/lib/assets/docs/article/getting-start-tdd/python/08-design-patterns.md +172 -172
- package/lib/assets/docs/article/getting-start-tdd/python/09-solid-principles-and-module-design.md +130 -130
- package/lib/assets/docs/article/getting-start-tdd/python/10-higher-order-functions-and-composition.md +122 -122
- package/lib/assets/docs/article/getting-start-tdd/python/11-immutable-data-and-pipeline.md +116 -116
- package/lib/assets/docs/article/getting-start-tdd/python/12-error-handling-and-type-safety.md +126 -126
- package/lib/assets/docs/article/getting-start-tdd/python/index.md +55 -55
- package/lib/assets/docs/article/getting-start-tdd/ruby/01-todo-list-and-first-test.md +231 -231
- package/lib/assets/docs/article/getting-start-tdd/ruby/02-fake-it-and-triangulation.md +238 -238
- package/lib/assets/docs/article/getting-start-tdd/ruby/03-obvious-implementation-and-refactoring.md +228 -228
- package/lib/assets/docs/article/getting-start-tdd/ruby/04-version-control-and-conventional-commits.md +112 -112
- package/lib/assets/docs/article/getting-start-tdd/ruby/05-package-management-and-static-analysis.md +287 -287
- package/lib/assets/docs/article/getting-start-tdd/ruby/06-task-runner-and-ci-cd.md +248 -248
- package/lib/assets/docs/article/getting-start-tdd/ruby/07-encapsulation-and-polymorphism.md +279 -279
- package/lib/assets/docs/article/getting-start-tdd/ruby/08-design-patterns.md +329 -329
- package/lib/assets/docs/article/getting-start-tdd/ruby/09-solid-principles-and-module-design.md +196 -196
- package/lib/assets/docs/article/getting-start-tdd/ruby/10-higher-order-functions-and-composition.md +175 -175
- package/lib/assets/docs/article/getting-start-tdd/ruby/11-immutable-data-and-pipeline.md +237 -237
- package/lib/assets/docs/article/getting-start-tdd/ruby/12-error-handling-and-type-safety.md +398 -398
- package/lib/assets/docs/article/getting-start-tdd/ruby/index.md +83 -83
- package/lib/assets/docs/article/getting-start-tdd/rust/01-todo-list-and-first-test.md +211 -211
- package/lib/assets/docs/article/getting-start-tdd/rust/02-fake-it-and-triangulation.md +264 -264
- package/lib/assets/docs/article/getting-start-tdd/rust/03-obvious-implementation-and-refactoring.md +233 -233
- package/lib/assets/docs/article/getting-start-tdd/rust/04-version-control-and-conventional-commits.md +92 -92
- package/lib/assets/docs/article/getting-start-tdd/rust/05-package-management-and-static-analysis.md +212 -212
- package/lib/assets/docs/article/getting-start-tdd/rust/06-task-runner-and-ci-cd.md +164 -164
- package/lib/assets/docs/article/getting-start-tdd/rust/07-encapsulation-and-polymorphism.md +142 -142
- package/lib/assets/docs/article/getting-start-tdd/rust/08-design-patterns.md +145 -145
- package/lib/assets/docs/article/getting-start-tdd/rust/09-solid-principles-and-module-design.md +110 -110
- package/lib/assets/docs/article/getting-start-tdd/rust/10-higher-order-functions-and-composition.md +94 -94
- package/lib/assets/docs/article/getting-start-tdd/rust/11-immutable-data-and-pipeline.md +105 -105
- package/lib/assets/docs/article/getting-start-tdd/rust/12-error-handling-and-type-safety.md +112 -112
- package/lib/assets/docs/article/getting-start-tdd/rust/index.md +83 -83
- package/lib/assets/docs/article/getting-start-tdd/scala/01-todo-list-and-first-test.md +111 -111
- package/lib/assets/docs/article/getting-start-tdd/scala/02-fake-it-and-triangulation.md +107 -107
- package/lib/assets/docs/article/getting-start-tdd/scala/03-obvious-implementation-and-refactoring.md +99 -99
- package/lib/assets/docs/article/getting-start-tdd/scala/04-version-control-and-conventional-commits.md +123 -123
- package/lib/assets/docs/article/getting-start-tdd/scala/05-package-management-and-static-analysis.md +196 -196
- package/lib/assets/docs/article/getting-start-tdd/scala/06-task-runner-and-ci-cd.md +186 -186
- package/lib/assets/docs/article/getting-start-tdd/scala/07-case-classes-and-traits.md +139 -139
- package/lib/assets/docs/article/getting-start-tdd/scala/08-pattern-matching-and-sealed-traits.md +106 -106
- package/lib/assets/docs/article/getting-start-tdd/scala/09-packages-and-module-design.md +75 -75
- package/lib/assets/docs/article/getting-start-tdd/scala/10-higher-order-functions-and-composition.md +104 -104
- package/lib/assets/docs/article/getting-start-tdd/scala/11-collections-and-lazy-evaluation.md +94 -94
- package/lib/assets/docs/article/getting-start-tdd/scala/12-error-handling-and-type-safety.md +92 -92
- package/lib/assets/docs/article/getting-start-tdd/scala/index.md +65 -65
- package/lib/assets/docs/article/grokking-concurrency/all/index.md +404 -404
- package/lib/assets/docs/article/grokking-concurrency/all/part-1-ch02-sequential.md +554 -554
- package/lib/assets/docs/article/grokking-concurrency/all/part-2-ch04-05-threads.md +469 -469
- package/lib/assets/docs/article/grokking-concurrency/all/part-3-ch06-multitasking.md +520 -520
- package/lib/assets/docs/article/grokking-concurrency/all/part-4-ch07-parallel-patterns.md +420 -420
- package/lib/assets/docs/article/grokking-concurrency/all/part-5-ch08-09-synchronization.md +510 -510
- package/lib/assets/docs/article/grokking-concurrency/all/part-6-ch10-11-nonblocking-io.md +435 -435
- package/lib/assets/docs/article/grokking-concurrency/all/part-7-ch12-async.md +465 -465
- package/lib/assets/docs/article/grokking-concurrency/all/part-8-ch13-mapreduce.md +377 -377
- package/lib/assets/docs/article/grokking-concurrency/clojure/index.md +116 -116
- package/lib/assets/docs/article/grokking-concurrency/clojure/part-1.md +108 -108
- package/lib/assets/docs/article/grokking-concurrency/clojure/part-2.md +101 -101
- package/lib/assets/docs/article/grokking-concurrency/clojure/part-3.md +122 -122
- package/lib/assets/docs/article/grokking-concurrency/clojure/part-4.md +123 -123
- package/lib/assets/docs/article/grokking-concurrency/clojure/part-5.md +118 -118
- package/lib/assets/docs/article/grokking-concurrency/clojure/part-6.md +89 -89
- package/lib/assets/docs/article/grokking-concurrency/clojure/part-7.md +100 -100
- package/lib/assets/docs/article/grokking-concurrency/clojure/part-8.md +120 -120
- package/lib/assets/docs/article/grokking-concurrency/csharp/index.md +101 -101
- package/lib/assets/docs/article/grokking-concurrency/csharp/part-1.md +97 -97
- package/lib/assets/docs/article/grokking-concurrency/csharp/part-2.md +123 -123
- package/lib/assets/docs/article/grokking-concurrency/csharp/part-3.md +101 -101
- package/lib/assets/docs/article/grokking-concurrency/csharp/part-4.md +112 -112
- package/lib/assets/docs/article/grokking-concurrency/csharp/part-5.md +99 -99
- package/lib/assets/docs/article/grokking-concurrency/csharp/part-6.md +61 -61
- package/lib/assets/docs/article/grokking-concurrency/csharp/part-7.md +84 -84
- package/lib/assets/docs/article/grokking-concurrency/csharp/part-8.md +92 -92
- package/lib/assets/docs/article/grokking-concurrency/fsharp/index.md +65 -65
- package/lib/assets/docs/article/grokking-concurrency/fsharp/part-1.md +80 -80
- package/lib/assets/docs/article/grokking-concurrency/fsharp/part-2.md +103 -103
- package/lib/assets/docs/article/grokking-concurrency/fsharp/part-3.md +94 -94
- package/lib/assets/docs/article/grokking-concurrency/fsharp/part-4.md +110 -110
- package/lib/assets/docs/article/grokking-concurrency/fsharp/part-5.md +104 -104
- package/lib/assets/docs/article/grokking-concurrency/fsharp/part-6.md +93 -93
- package/lib/assets/docs/article/grokking-concurrency/fsharp/part-7.md +121 -121
- package/lib/assets/docs/article/grokking-concurrency/fsharp/part-8.md +107 -107
- package/lib/assets/docs/article/grokking-concurrency/haskell/index.md +248 -248
- package/lib/assets/docs/article/grokking-concurrency/haskell/part-1.md +96 -96
- package/lib/assets/docs/article/grokking-concurrency/haskell/part-2.md +96 -96
- package/lib/assets/docs/article/grokking-concurrency/haskell/part-3.md +91 -91
- package/lib/assets/docs/article/grokking-concurrency/haskell/part-4.md +106 -106
- package/lib/assets/docs/article/grokking-concurrency/haskell/part-5.md +99 -99
- package/lib/assets/docs/article/grokking-concurrency/haskell/part-6.md +95 -95
- package/lib/assets/docs/article/grokking-concurrency/haskell/part-7.md +111 -111
- package/lib/assets/docs/article/grokking-concurrency/haskell/part-8.md +118 -118
- package/lib/assets/docs/article/grokking-concurrency/index.md +66 -66
- package/lib/assets/docs/article/grokking-concurrency/java/index.md +102 -102
- package/lib/assets/docs/article/grokking-concurrency/java/part-1.md +308 -308
- package/lib/assets/docs/article/grokking-concurrency/java/part-2.md +334 -334
- package/lib/assets/docs/article/grokking-concurrency/java/part-3.md +221 -221
- package/lib/assets/docs/article/grokking-concurrency/java/part-4.md +213 -213
- package/lib/assets/docs/article/grokking-concurrency/java/part-5.md +112 -112
- package/lib/assets/docs/article/grokking-concurrency/java/part-6.md +69 -69
- package/lib/assets/docs/article/grokking-concurrency/java/part-7.md +101 -101
- package/lib/assets/docs/article/grokking-concurrency/java/part-8.md +101 -101
- package/lib/assets/docs/article/grokking-concurrency/python/index.md +313 -313
- package/lib/assets/docs/article/grokking-concurrency/python/part-1.md +239 -239
- package/lib/assets/docs/article/grokking-concurrency/python/part-2.md +418 -418
- package/lib/assets/docs/article/grokking-concurrency/python/part-3.md +227 -227
- package/lib/assets/docs/article/grokking-concurrency/python/part-4.md +299 -299
- package/lib/assets/docs/article/grokking-concurrency/python/part-5.md +315 -315
- package/lib/assets/docs/article/grokking-concurrency/python/part-6.md +297 -297
- package/lib/assets/docs/article/grokking-concurrency/python/part-7.md +314 -314
- package/lib/assets/docs/article/grokking-concurrency/python/part-8.md +360 -360
- package/lib/assets/docs/article/grokking-concurrency/rust/index.md +270 -270
- package/lib/assets/docs/article/grokking-concurrency/rust/part-1.md +108 -108
- package/lib/assets/docs/article/grokking-concurrency/rust/part-2.md +120 -120
- package/lib/assets/docs/article/grokking-concurrency/rust/part-3.md +126 -126
- package/lib/assets/docs/article/grokking-concurrency/rust/part-4.md +175 -175
- package/lib/assets/docs/article/grokking-concurrency/rust/part-5.md +158 -158
- package/lib/assets/docs/article/grokking-concurrency/rust/part-6.md +94 -94
- package/lib/assets/docs/article/grokking-concurrency/rust/part-7.md +133 -133
- package/lib/assets/docs/article/grokking-concurrency/rust/part-8.md +155 -155
- package/lib/assets/docs/article/grokking-concurrency/scala/index.md +69 -69
- package/lib/assets/docs/article/grokking-concurrency/scala/part-1.md +78 -78
- package/lib/assets/docs/article/grokking-concurrency/scala/part-2.md +112 -112
- package/lib/assets/docs/article/grokking-concurrency/scala/part-3.md +93 -93
- package/lib/assets/docs/article/grokking-concurrency/scala/part-4.md +110 -110
- package/lib/assets/docs/article/grokking-concurrency/scala/part-5.md +119 -119
- package/lib/assets/docs/article/grokking-concurrency/scala/part-6.md +83 -83
- package/lib/assets/docs/article/grokking-concurrency/scala/part-7.md +131 -131
- package/lib/assets/docs/article/grokking-concurrency/scala/part-8.md +129 -129
- package/lib/assets/docs/article/grokkingfp/all/index.md +368 -368
- package/lib/assets/docs/article/grokkingfp/all/part-1-ch01-fp-introduction.md +530 -530
- package/lib/assets/docs/article/grokkingfp/all/part-1-ch02-pure-functions.md +923 -923
- package/lib/assets/docs/article/grokkingfp/all/part-2-ch03-immutable-data.md +1128 -1128
- package/lib/assets/docs/article/grokkingfp/all/part-2-ch04-higher-order-functions.md +1104 -1104
- package/lib/assets/docs/article/grokkingfp/all/part-2-ch05-flatmap.md +1026 -1026
- package/lib/assets/docs/article/grokkingfp/all/part-3-ch06-option.md +785 -785
- package/lib/assets/docs/article/grokkingfp/all/part-3-ch07-either-adt.md +871 -871
- package/lib/assets/docs/article/grokkingfp/all/part-4-ch08-io-monad.md +972 -972
- package/lib/assets/docs/article/grokkingfp/all/part-4-ch09-streams.md +926 -926
- package/lib/assets/docs/article/grokkingfp/all/part-5-ch10-concurrency.md +870 -870
- package/lib/assets/docs/article/grokkingfp/all/part-6-ch11-application.md +715 -715
- package/lib/assets/docs/article/grokkingfp/all/part-6-ch12-testing.md +626 -626
- package/lib/assets/docs/article/grokkingfp/all/writing-plan.md +712 -712
- package/lib/assets/docs/article/grokkingfp/clojure/index.md +276 -276
- package/lib/assets/docs/article/grokkingfp/clojure/part-1.md +667 -667
- package/lib/assets/docs/article/grokkingfp/clojure/part-2.md +643 -643
- package/lib/assets/docs/article/grokkingfp/clojure/part-3.md +620 -620
- package/lib/assets/docs/article/grokkingfp/clojure/part-4.md +697 -697
- package/lib/assets/docs/article/grokkingfp/clojure/part-5.md +751 -751
- package/lib/assets/docs/article/grokkingfp/clojure/part-6.md +721 -721
- package/lib/assets/docs/article/grokkingfp/csharp/index.md +246 -246
- package/lib/assets/docs/article/grokkingfp/csharp/part-1.md +811 -811
- package/lib/assets/docs/article/grokkingfp/csharp/part-2.md +971 -971
- package/lib/assets/docs/article/grokkingfp/csharp/part-3.md +981 -981
- package/lib/assets/docs/article/grokkingfp/csharp/part-4.md +949 -949
- package/lib/assets/docs/article/grokkingfp/csharp/part-5.md +947 -947
- package/lib/assets/docs/article/grokkingfp/csharp/part-6.md +739 -739
- package/lib/assets/docs/article/grokkingfp/elixir/index.md +203 -203
- package/lib/assets/docs/article/grokkingfp/elixir/part-1.md +712 -712
- package/lib/assets/docs/article/grokkingfp/elixir/part-2.md +838 -838
- package/lib/assets/docs/article/grokkingfp/elixir/part-3.md +985 -985
- package/lib/assets/docs/article/grokkingfp/elixir/part-4.md +974 -974
- package/lib/assets/docs/article/grokkingfp/elixir/part-5.md +1286 -1286
- package/lib/assets/docs/article/grokkingfp/elixir/part-6.md +1049 -1049
- package/lib/assets/docs/article/grokkingfp/fsharp/index.md +210 -210
- package/lib/assets/docs/article/grokkingfp/fsharp/part-1.md +714 -714
- package/lib/assets/docs/article/grokkingfp/fsharp/part-2.md +961 -961
- package/lib/assets/docs/article/grokkingfp/fsharp/part-3.md +972 -972
- package/lib/assets/docs/article/grokkingfp/fsharp/part-4.md +832 -832
- package/lib/assets/docs/article/grokkingfp/fsharp/part-5.md +911 -911
- package/lib/assets/docs/article/grokkingfp/fsharp/part-6.md +922 -922
- package/lib/assets/docs/article/grokkingfp/haskell/index.md +234 -234
- package/lib/assets/docs/article/grokkingfp/haskell/part-1.md +591 -591
- package/lib/assets/docs/article/grokkingfp/haskell/part-2.md +866 -866
- package/lib/assets/docs/article/grokkingfp/haskell/part-3.md +915 -915
- package/lib/assets/docs/article/grokkingfp/haskell/part-4.md +878 -878
- package/lib/assets/docs/article/grokkingfp/haskell/part-5.md +845 -845
- package/lib/assets/docs/article/grokkingfp/haskell/part-6.md +844 -844
- package/lib/assets/docs/article/grokkingfp/index.md +143 -143
- package/lib/assets/docs/article/grokkingfp/java/index.md +211 -211
- package/lib/assets/docs/article/grokkingfp/java/part-1.md +648 -648
- package/lib/assets/docs/article/grokkingfp/java/part-2.md +675 -675
- package/lib/assets/docs/article/grokkingfp/java/part-3.md +672 -672
- package/lib/assets/docs/article/grokkingfp/java/part-4.md +771 -771
- package/lib/assets/docs/article/grokkingfp/java/part-5.md +959 -959
- package/lib/assets/docs/article/grokkingfp/java/part-6.md +1328 -1328
- package/lib/assets/docs/article/grokkingfp/python/index.md +258 -258
- package/lib/assets/docs/article/grokkingfp/python/part-1.md +443 -443
- package/lib/assets/docs/article/grokkingfp/python/part-2.md +958 -958
- package/lib/assets/docs/article/grokkingfp/python/part-3.md +1004 -1004
- package/lib/assets/docs/article/grokkingfp/python/part-4.md +765 -765
- package/lib/assets/docs/article/grokkingfp/python/part-5.md +747 -747
- package/lib/assets/docs/article/grokkingfp/python/part-6.md +861 -861
- package/lib/assets/docs/article/grokkingfp/ruby/index.md +330 -330
- package/lib/assets/docs/article/grokkingfp/ruby/part-1.md +755 -755
- package/lib/assets/docs/article/grokkingfp/ruby/part-2.md +938 -938
- package/lib/assets/docs/article/grokkingfp/ruby/part-3.md +946 -946
- package/lib/assets/docs/article/grokkingfp/ruby/part-4.md +921 -921
- package/lib/assets/docs/article/grokkingfp/ruby/part-5.md +908 -908
- package/lib/assets/docs/article/grokkingfp/ruby/part-6.md +1412 -1412
- package/lib/assets/docs/article/grokkingfp/rust/index.md +242 -242
- package/lib/assets/docs/article/grokkingfp/rust/part-1.md +634 -634
- package/lib/assets/docs/article/grokkingfp/rust/part-2.md +1060 -1060
- package/lib/assets/docs/article/grokkingfp/rust/part-3.md +994 -994
- package/lib/assets/docs/article/grokkingfp/rust/part-4.md +573 -573
- package/lib/assets/docs/article/grokkingfp/rust/part-5.md +705 -705
- package/lib/assets/docs/article/grokkingfp/rust/part-6.md +508 -508
- package/lib/assets/docs/article/grokkingfp/scala/index.md +171 -171
- package/lib/assets/docs/article/grokkingfp/scala/part-1.md +543 -543
- package/lib/assets/docs/article/grokkingfp/scala/part-2.md +946 -946
- package/lib/assets/docs/article/grokkingfp/scala/part-3.md +919 -919
- package/lib/assets/docs/article/grokkingfp/scala/part-4.md +742 -742
- package/lib/assets/docs/article/grokkingfp/scala/part-5.md +722 -722
- package/lib/assets/docs/article/grokkingfp/scala/part-6.md +867 -867
- package/lib/assets/docs/article/grokkingfp/typescript/index.md +273 -273
- package/lib/assets/docs/article/grokkingfp/typescript/part-1.md +561 -561
- package/lib/assets/docs/article/grokkingfp/typescript/part-2.md +1129 -1129
- package/lib/assets/docs/article/grokkingfp/typescript/part-3.md +842 -842
- package/lib/assets/docs/article/grokkingfp/typescript/part-4.md +1087 -1087
- package/lib/assets/docs/article/grokkingfp/typescript/part-5.md +717 -717
- package/lib/assets/docs/article/grokkingfp/typescript/part-6.md +982 -982
- package/lib/assets/docs/article/practical-database-design/index.md +121 -121
- package/lib/assets/docs/article/practical-database-design/part1/chapter01.md +288 -288
- package/lib/assets/docs/article/practical-database-design/part1/chapter02.md +518 -518
- package/lib/assets/docs/article/practical-database-design/part1/chapter03.md +557 -557
- package/lib/assets/docs/article/practical-database-design/part2/chapter04.md +924 -924
- package/lib/assets/docs/article/practical-database-design/part2/chapter05.md +1627 -1627
- package/lib/assets/docs/article/practical-database-design/part2/chapter06.md +2716 -2716
- package/lib/assets/docs/article/practical-database-design/part2/chapter07.md +2082 -2082
- package/lib/assets/docs/article/practical-database-design/part2/chapter08.md +2105 -2105
- package/lib/assets/docs/article/practical-database-design/part2/chapter09.md +2031 -2031
- package/lib/assets/docs/article/practical-database-design/part2/chapter10.md +1387 -1387
- package/lib/assets/docs/article/practical-database-design/part2/chapter11.md +1677 -1677
- package/lib/assets/docs/article/practical-database-design/part2/chapter12.md +1417 -1417
- package/lib/assets/docs/article/practical-database-design/part2/chapter13.md +1434 -1434
- package/lib/assets/docs/article/practical-database-design/part3/chapter14.md +667 -667
- package/lib/assets/docs/article/practical-database-design/part3/chapter15.md +1625 -1625
- package/lib/assets/docs/article/practical-database-design/part3/chapter16.md +1915 -1915
- package/lib/assets/docs/article/practical-database-design/part3/chapter17.md +1708 -1708
- package/lib/assets/docs/article/practical-database-design/part3/chapter18.md +2095 -2095
- package/lib/assets/docs/article/practical-database-design/part3/chapter19.md +1123 -1123
- package/lib/assets/docs/article/practical-database-design/part3/chapter20.md +1031 -1031
- package/lib/assets/docs/article/practical-database-design/part3/chapter21.md +1382 -1382
- package/lib/assets/docs/article/practical-database-design/part3-orm/chapter14-orm.md +991 -991
- package/lib/assets/docs/article/practical-database-design/part3-orm/chapter15-orm.md +1300 -1300
- package/lib/assets/docs/article/practical-database-design/part3-orm/chapter16-orm.md +1166 -1166
- package/lib/assets/docs/article/practical-database-design/part3-orm/chapter17-orm.md +1584 -1584
- package/lib/assets/docs/article/practical-database-design/part3-orm/chapter18-orm.md +1183 -1183
- package/lib/assets/docs/article/practical-database-design/part3-orm/chapter19-orm.md +1016 -1016
- package/lib/assets/docs/article/practical-database-design/part3-orm/chapter20-orm.md +1753 -1753
- package/lib/assets/docs/article/practical-database-design/part3-orm/chapter21-orm.md +1447 -1447
- package/lib/assets/docs/article/practical-database-design/part3-orm/chapter22-orm.md +1878 -1878
- package/lib/assets/docs/article/practical-database-design/part4/chapter22.md +965 -965
- package/lib/assets/docs/article/practical-database-design/part4/chapter23.md +2069 -2069
- package/lib/assets/docs/article/practical-database-design/part4/chapter24.md +2439 -2439
- package/lib/assets/docs/article/practical-database-design/part4/chapter25.md +3661 -3661
- package/lib/assets/docs/article/practical-database-design/part4/chapter26.md +2916 -2916
- package/lib/assets/docs/article/practical-database-design/part4/chapter27.md +3105 -3105
- package/lib/assets/docs/article/practical-database-design/part4/chapter28.md +2697 -2697
- package/lib/assets/docs/article/practical-database-design/part4/chapter29.md +2544 -2544
- package/lib/assets/docs/article/practical-database-design/part4/chapter30.md +2180 -2180
- package/lib/assets/docs/article/practical-database-design/part4/chapter31.md +1192 -1192
- package/lib/assets/docs/article/practical-database-design/part4/chapter32.md +2101 -2101
- package/lib/assets/docs/article/practical-database-design/part5/chapter33.md +1032 -1032
- package/lib/assets/docs/article/practical-database-design/part5/chapter34.md +1609 -1609
- package/lib/assets/docs/article/practical-database-design/part5/chapter35.md +1453 -1453
- package/lib/assets/docs/article/practical-database-design/part5/chapter36.md +1292 -1292
- package/lib/assets/docs/article/practical-database-design/part5/chapter37.md +1470 -1470
- package/lib/assets/docs/article/practical-database-design/part5/chapter38.md +1698 -1698
- package/lib/assets/docs/article/practical-database-design/part5/chapter39.md +2334 -2334
- package/lib/assets/docs/article/practical-database-design/study/study2-1.md +1693 -1693
- package/lib/assets/docs/article/practical-database-design/study/study2-2.md +1347 -1347
- package/lib/assets/docs/article/practical-database-design/study/study2-3.md +2044 -2044
- package/lib/assets/docs/article/practical-database-design/study/study2-4.md +2229 -2229
- package/lib/assets/docs/article/practical-database-design/study/study2-5.md +2418 -2418
- package/lib/assets/docs/article/practical-database-design/study/study3-1.md +2205 -2205
- package/lib/assets/docs/article/practical-database-design/study/study3-2.md +2221 -2221
- package/lib/assets/docs/article/practical-database-design/study/study3-3.md +2253 -2253
- package/lib/assets/docs/article/practical-database-design/study/study3-4.md +2106 -2106
- package/lib/assets/docs/article/practical-database-design/study/study3-5.md +2507 -2507
- package/lib/assets/docs/article/practical-database-design/study/study4-1.md +2587 -2587
- package/lib/assets/docs/article/practical-database-design/study/study4-2.md +2075 -2075
- package/lib/assets/docs/article/practical-database-design/study/study4-3.md +1805 -1805
- package/lib/assets/docs/article/practical-database-design/study/study4-4.md +1895 -1895
- package/lib/assets/docs/article/practical-database-design/study/study4-5.md +2878 -2878
- package/lib/assets/docs/assets/css/extra.css +29 -29
- package/lib/assets/docs/assets/js/extra.js +44 -44
- package/lib/assets/docs/development/index.md +39 -39
- package/lib/assets/docs/operation/index.md +11 -11
- package/lib/assets/docs/reference/CodexCLIMCP/343/202/242/343/203/227/343/203/252/343/202/261/343/203/274/343/202/267/343/203/247/343/203/263/351/226/213/347/231/272/343/203/225/343/203/255/343/203/274.md +532 -532
- package/lib/assets/docs/reference/CodexCLIMCP/343/202/265/343/203/274/343/203/220/343/203/274/350/250/255/345/256/232/346/211/213/351/240/206.md +341 -341
- package/lib/assets/docs/reference/Java/343/202/242/343/203/227/343/203/252/343/202/261/343/203/274/343/202/267/343/203/247/343/203/263/347/222/260/345/242/203/346/247/213/347/257/211/343/202/254/343/202/244/343/203/211.md +581 -581
- package/lib/assets/docs/reference/SonarQube/343/203/255/343/203/274/343/202/253/343/203/253/347/222/260/345/242/203/343/202/273/343/203/203/343/203/210/343/202/242/343/203/203/343/203/227/346/211/213/351/240/206/346/233/270.md +642 -642
- package/lib/assets/docs/reference/TypeScript/343/202/242/343/203/227/343/203/252/343/202/261/343/203/274/343/202/267/343/203/247/343/203/263/347/222/260/345/242/203/346/247/213/347/257/211/343/202/254/343/202/244/343/203/211.md +465 -465
- package/lib/assets/docs/reference/UI/350/250/255/350/250/210/343/202/254/343/202/244/343/203/211.md +450 -450
- package/lib/assets/docs/reference/images/Ansoff.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/BrandBasicStrategy.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/BrandCategorization.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/BrandRecurutementStrategy.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/BrandValue.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/BusinessActivitiy.svg +3 -3
- package/lib/assets/docs/reference/images/HRM.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/MarketingStructure.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/OrganizationElemnts.svg +3 -3
- package/lib/assets/docs/reference/images/PPM.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/PositioningMap.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/ProductLayer.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/ProductMix.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/SWOT.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/TargetMarket.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/ThreeGenericStrategies.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/VRIO.drawio.svg +3 -3
- package/lib/assets/docs/reference/images/ValueChain.drawio.svg +3 -3
- package/lib/assets/docs/reference/index.md +52 -52
- package/lib/assets/docs/reference//343/202/210/343/201/204/343/202/275/343/203/225/343/203/210/343/202/246/343/202/247/343/202/242/343/201/250/343/201/257.md +250 -250
- package/lib/assets/docs/reference//343/202/242/343/203/274/343/202/255/343/203/206/343/202/257/343/203/201/343/203/243/350/250/255/350/250/210/343/202/254/343/202/244/343/203/211.md +2216 -2216
- package/lib/assets/docs/reference//343/202/244/343/203/263/343/203/225/343/203/251/350/250/255/350/250/210/343/202/254/343/202/244/343/203/211.md +1878 -1878
- package/lib/assets/docs/reference//343/202/250/343/202/257/343/202/271/343/203/210/343/203/252/343/203/274/343/203/240/343/203/227/343/203/255/343/202/260/343/203/251/343/203/237/343/203/263/343/202/260.md +550 -550
- package/lib/assets/docs/reference//343/202/263/343/203/274/343/203/207/343/202/243/343/203/263/343/202/260/343/201/250/343/203/206/343/202/271/343/203/210/343/202/254/343/202/244/343/203/211.md +705 -705
- package/lib/assets/docs/reference//343/203/206/343/202/271/343/203/210/346/210/246/347/225/245/343/202/254/343/202/244/343/203/211.md +1313 -1313
- package/lib/assets/docs/reference//343/203/207/343/203/274/343/202/277/343/203/242/343/203/207/343/203/253/350/250/255/350/250/210/343/202/254/343/202/244/343/203/211.md +311 -311
- package/lib/assets/docs/reference//343/203/211/343/203/241/343/202/244/343/203/263/343/203/242/343/203/207/343/203/253/350/250/255/350/250/210/343/202/254/343/202/244/343/203/211.md +599 -599
- package/lib/assets/docs/reference//343/203/223/343/202/270/343/203/215/343/202/271/343/202/242/343/203/274/343/202/255/343/203/206/343/202/257/343/203/201/343/203/243/345/210/206/346/236/220/343/202/254/343/202/244/343/203/211.md +528 -528
- package/lib/assets/docs/reference//343/203/246/343/203/274/343/202/271/343/202/261/343/203/274/343/202/271/344/275/234/346/210/220/343/202/254/343/202/244/343/203/211.md +689 -689
- package/lib/assets/docs/reference//343/203/252/343/203/252/343/203/274/343/202/271/343/202/254/343/202/244/343/203/211.md +461 -461
- package/lib/assets/docs/reference//343/203/252/343/203/252/343/203/274/343/202/271/343/203/273/343/202/244/343/203/206/343/203/254/343/203/274/343/202/267/343/203/247/343/203/263/350/250/210/347/224/273/343/202/254/343/202/244/343/203/211.md +580 -580
- package/lib/assets/docs/reference//343/203/255/343/202/270/343/202/253/343/203/253/343/202/267/343/203/263/343/202/255/343/203/263/343/202/260.md +1367 -1367
- package/lib/assets/docs/reference//344/274/201/346/245/255/347/265/214/345/226/266/350/253/226.md +2637 -2637
- package/lib/assets/docs/reference//347/222/260/345/242/203/345/244/211/346/225/260/347/256/241/347/220/206/343/202/254/343/202/244/343/203/211.md +665 -665
- package/lib/assets/docs/reference//350/246/201/344/273/266/345/256/232/347/276/251/343/202/254/343/202/244/343/203/211.md +1248 -1248
- package/lib/assets/docs/reference//350/250/200/350/252/236/345/210/245/351/226/213/347/231/272/343/202/254/343/202/244/343/203/211.md +518 -518
- package/lib/assets/docs/reference//351/201/213/345/226/266/347/256/241/347/220/206.md +1482 -1482
- package/lib/assets/docs/reference//351/201/213/347/224/250/343/202/271/343/202/257/343/203/252/343/203/227/343/203/210/344/275/234/346/210/220/343/202/254/343/202/244/343/203/211.md +421 -421
- package/lib/assets/docs/reference//351/201/213/347/224/250/350/246/201/344/273/266/345/256/232/347/276/251/343/202/254/343/202/244/343/203/211.md +392 -392
- package/lib/assets/docs/reference//351/226/213/347/231/272/343/202/254/343/202/244/343/203/211.md +299 -299
- package/lib/assets/docs/reference//351/235/236/346/251/237/350/203/275/350/246/201/344/273/266/345/256/232/347/276/251/343/202/254/343/202/244/343/203/211.md +1236 -1236
- package/lib/assets/docs/review/index.md +5 -5
- package/lib/assets/docs/strategy/index.md +1 -1
- package/lib/assets/docs/template/ADR.md +30 -30
- package/lib/assets/docs/template/AWS/343/202/271/343/203/206/343/203/274/343/202/270/343/203/263/343/202/260/347/222/260/345/242/203/343/202/273/343/203/203/343/203/210/343/202/242/343/203/203/343/203/227/346/211/213/351/240/206/346/233/270.md +1366 -1366
- package/lib/assets/docs/template/AWS/343/203/227/343/203/255/343/203/200/343/202/257/343/202/267/343/203/247/343/203/263/347/222/260/345/242/203/343/202/273/343/203/203/343/203/210/343/202/242/343/203/203/343/203/227/346/211/213/351/240/206/346/233/270.md +634 -634
- package/lib/assets/docs/template/README.md +50 -50
- package/lib/assets/docs/template/index.md +23 -23
- package/lib/assets/docs/template//343/201/276/343/201/232/343/201/223/343/202/214/343/202/222/350/252/255/343/202/202/343/201/206/343/203/252/343/202/271/343/203/210.md +12 -12
- package/lib/assets/docs/template//343/202/242/343/203/227/343/203/252/343/202/261/343/203/274/343/202/267/343/203/247/343/203/263/351/226/213/347/231/272/347/222/260/345/242/203/343/202/273/343/203/203/343/203/210/343/202/242/343/203/203/343/203/227/346/211/213/351/240/206/346/233/270.md +547 -547
- package/lib/assets/docs/template//343/202/244/343/203/206/343/203/254/343/203/274/343/202/267/343/203/247/343/203/263/345/256/214/344/272/206/345/240/261/345/221/212/346/233/270.md +58 -58
- package/lib/assets/docs/template//343/202/244/343/203/263/343/202/273/343/203/227/343/202/267/343/203/247/343/203/263/343/203/207/343/203/203/343/202/255.md +13 -13
- package/lib/assets/docs/template//343/203/223/343/202/270/343/203/215/343/202/271/343/202/242/343/203/274/343/202/255/343/203/206/343/202/257/343/203/201/343/203/243.md +379 -379
- package/lib/assets/docs/template//344/274/201/346/245/255/345/210/206/346/236/220.md +573 -573
- package/lib/assets/docs/template//345/256/214/345/205/250/345/275/242/345/274/217/343/201/256/343/203/246/343/203/274/343/202/271/343/202/261/343/203/274/343/202/271.md +69 -69
- package/lib/assets/docs/template//350/246/201/344/273/266/345/256/232/347/276/251.md +669 -669
- package/lib/assets/docs/template//350/250/255/350/250/210.md +173 -173
- package/lib/assets/docs/template//351/226/213/347/231/272/347/222/260/345/242/203/343/202/273/343/203/203/343/203/210/343/202/242/343/203/203/343/203/227/346/211/213/351/240/206/346/233/270.md +688 -688
- package/lib/assets/gulpfile.js +25 -25
- package/lib/assets/mkdocs.yml +136 -136
- package/lib/assets/ops/docker/mkdoc/Dockerfile +19 -19
- package/lib/assets/ops/scripts/journal.js +180 -180
- package/lib/assets/ops/scripts/mkdocs.js +82 -82
- package/lib/assets/ops/scripts/release.js +431 -431
- package/lib/assets/ops/scripts/sonar_local.js +726 -726
- package/lib/assets/ops/scripts/ssh.js +190 -190
- package/lib/assets/ops/scripts/vault.js +299 -299
- package/lib/assets/package-lock.json +1653 -1653
- package/lib/assets/package.json +40 -40
- package/lib/gulpfile.js +37 -37
- package/package.json +41 -41
|
@@ -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 版での実装を解説します。
|