@k2works/claude-code-booster 3.5.0 → 3.6.1
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 +239 -239
- 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 -88
- 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-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 -161
- 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 +183 -183
- 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 -580
- 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 -242
- 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 -544
- 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 -682
- 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 -560
- 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 -2636
- package/lib/assets/docs/reference//347/222/260/345/242/203/345/244/211/346/225/260/347/256/241/347/220/206/343/202/254/343/202/244/343/203/211.md +665 -663
- 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 +28 -0
- 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 -68
- 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 -135
- 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
- package/lib/assets/.claude/agent-memory/xp-programmer/MEMORY.md +0 -6
- package/lib/assets/.claude/agent-memory/xp-programmer/project_cargo_tracker.md +0 -11
- package/lib/assets/.claude/agent-memory/xp-programmer/project_ddd_patterns.md +0 -27
- package/lib/assets/.claude/agent-memory/xp-programmer/project_us07_route_assignment.md +0 -19
|
@@ -1,2101 +1,2101 @@
|
|
|
1
|
-
# 第32章:API サービスの実装
|
|
2
|
-
|
|
3
|
-
## 32.1 ヘキサゴナルアーキテクチャの復習
|
|
4
|
-
|
|
5
|
-
### ヘキサゴナルアーキテクチャとは
|
|
6
|
-
|
|
7
|
-
ヘキサゴナルアーキテクチャ(Ports and Adapters パターン)は、アプリケーションのドメインロジックを外部の技術的詳細から分離するアーキテクチャスタイルです。
|
|
8
|
-
|
|
9
|
-
```plantuml
|
|
10
|
-
@startuml
|
|
11
|
-
skinparam componentStyle uml2
|
|
12
|
-
skinparam component {
|
|
13
|
-
BackgroundColor<<adapter>> LightBlue
|
|
14
|
-
BackgroundColor<<port>> LightGreen
|
|
15
|
-
BackgroundColor<<domain>> LightYellow
|
|
16
|
-
}
|
|
17
|
-
|
|
18
|
-
package "外部" {
|
|
19
|
-
[REST Client] as rest
|
|
20
|
-
[Database] as db
|
|
21
|
-
}
|
|
22
|
-
|
|
23
|
-
package "Application" {
|
|
24
|
-
package "Infrastructure" {
|
|
25
|
-
[REST Controller] <<adapter>> as controller
|
|
26
|
-
[MyBatis Repository] <<adapter>> as mybatis
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
package "Application Service" {
|
|
30
|
-
interface "Input Port\n(UseCase)" <<port>> as input
|
|
31
|
-
interface "Output Port\n(Repository)" <<port>> as output
|
|
32
|
-
[Service] as service
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
package "Domain" <<domain>> {
|
|
36
|
-
[Entity]
|
|
37
|
-
[Value Object]
|
|
38
|
-
[Domain Service]
|
|
39
|
-
}
|
|
40
|
-
}
|
|
41
|
-
|
|
42
|
-
rest --> controller
|
|
43
|
-
controller --> input
|
|
44
|
-
input <|.. service
|
|
45
|
-
service --> output
|
|
46
|
-
output <|.. mybatis
|
|
47
|
-
mybatis --> db
|
|
48
|
-
|
|
49
|
-
service --> Entity
|
|
50
|
-
service --> [Domain Service]
|
|
51
|
-
@enduml
|
|
52
|
-
```
|
|
53
|
-
|
|
54
|
-
### 主要な概念
|
|
55
|
-
|
|
56
|
-
| 概念 | 説明 | 例 |
|
|
57
|
-
|------|------|-----|
|
|
58
|
-
| Input Port | ユースケースを定義するインターフェース | ItemUseCase |
|
|
59
|
-
| Output Port | 外部リソースへのアクセスを定義するインターフェース | ItemRepository |
|
|
60
|
-
| Input Adapter | 外部からの入力を受け付けるコンポーネント | REST Controller |
|
|
61
|
-
| Output Adapter | 外部リソースへの出力を行うコンポーネント | MyBatis Repository |
|
|
62
|
-
| Domain | ビジネスロジックを含む中心部分 | Entity, Value Object |
|
|
63
|
-
|
|
64
|
-
### 依存関係の方向
|
|
65
|
-
|
|
66
|
-
ヘキサゴナルアーキテクチャでは、依存関係は常に外側から内側に向かいます。
|
|
67
|
-
|
|
68
|
-
```plantuml
|
|
69
|
-
@startuml
|
|
70
|
-
skinparam rectangle {
|
|
71
|
-
BackgroundColor<<outer>> LightBlue
|
|
72
|
-
BackgroundColor<<middle>> LightGreen
|
|
73
|
-
BackgroundColor<<inner>> LightYellow
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
rectangle "Infrastructure\n(Adapters)" <<outer>> as infra {
|
|
77
|
-
}
|
|
78
|
-
|
|
79
|
-
rectangle "Application\n(Ports & Services)" <<middle>> as app {
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
rectangle "Domain\n(Entities & Rules)" <<inner>> as domain {
|
|
83
|
-
}
|
|
84
|
-
|
|
85
|
-
infra --> app : 依存
|
|
86
|
-
app --> domain : 依存
|
|
87
|
-
@enduml
|
|
88
|
-
```
|
|
89
|
-
|
|
90
|
-
---
|
|
91
|
-
|
|
92
|
-
## 32.2 アーキテクチャ構造
|
|
93
|
-
|
|
94
|
-
### プロジェクト構造
|
|
95
|
-
|
|
96
|
-
```
|
|
97
|
-
apps/pms/backend/
|
|
98
|
-
├── src/
|
|
99
|
-
│ ├── main/
|
|
100
|
-
│ │ ├── java/
|
|
101
|
-
│ │ │ └── com/example/pms/
|
|
102
|
-
│ │ │ ├── Application.java
|
|
103
|
-
│ │ │ ├── domain/
|
|
104
|
-
│ │ │ │ ├── model/ # エンティティ・値オブジェクト
|
|
105
|
-
│ │ │ │ │ ├── item/
|
|
106
|
-
│ │ │ │ │ ├── bom/
|
|
107
|
-
│ │ │ │ │ ├── inventory/
|
|
108
|
-
│ │ │ │ │ ├── purchase/
|
|
109
|
-
│ │ │ │ │ └── process/
|
|
110
|
-
│ │ │ │ ├── service/ # ドメインサービス
|
|
111
|
-
│ │ │ │ └── exception/ # ドメイン例外
|
|
112
|
-
│ │ │ ├── application/
|
|
113
|
-
│ │ │ │ ├── port/
|
|
114
|
-
│ │ │ │ │ ├── in/ # Input Port(ユースケース)
|
|
115
|
-
│ │ │ │ │ │ └── command/ # コマンドオブジェクト
|
|
116
|
-
│ │ │ │ │ └── out/ # Output Port(リポジトリ)
|
|
117
|
-
│ │ │ │ └── service/ # アプリケーションサービス
|
|
118
|
-
│ │ │ └── infrastructure/
|
|
119
|
-
│ │ │ ├── in/
|
|
120
|
-
│ │ │ │ ├── rest/ # REST Controller
|
|
121
|
-
│ │ │ │ │ └── dto/ # Request/Response DTO
|
|
122
|
-
│ │ │ │ └── web/
|
|
123
|
-
│ │ │ │ └── exception/ # 例外ハンドラ
|
|
124
|
-
│ │ │ └── out/
|
|
125
|
-
│ │ │ └── persistence/
|
|
126
|
-
│ │ │ └── repository/ # Repository 実装
|
|
127
|
-
│ │ └── resources/
|
|
128
|
-
│ │ ├── mapper/ # MyBatis マッパー XML
|
|
129
|
-
│ │ └── application.yml
|
|
130
|
-
│ └── test/
|
|
131
|
-
│ └── java/
|
|
132
|
-
│ └── com/example/pms/
|
|
133
|
-
│ └── integration/ # 統合テスト
|
|
134
|
-
└── build.gradle.kts
|
|
135
|
-
```
|
|
136
|
-
|
|
137
|
-
### 技術スタックの導入
|
|
138
|
-
|
|
139
|
-
<details>
|
|
140
|
-
<summary>build.gradle.kts</summary>
|
|
141
|
-
|
|
142
|
-
```kotlin
|
|
143
|
-
plugins {
|
|
144
|
-
java
|
|
145
|
-
id("org.springframework.boot") version "3.2.0"
|
|
146
|
-
id("io.spring.dependency-management") version "1.1.4"
|
|
147
|
-
}
|
|
148
|
-
|
|
149
|
-
group = "com.example"
|
|
150
|
-
version = "0.0.1-SNAPSHOT"
|
|
151
|
-
|
|
152
|
-
java {
|
|
153
|
-
sourceCompatibility = JavaVersion.VERSION_21
|
|
154
|
-
}
|
|
155
|
-
|
|
156
|
-
repositories {
|
|
157
|
-
mavenCentral()
|
|
158
|
-
}
|
|
159
|
-
|
|
160
|
-
dependencies {
|
|
161
|
-
// Spring Boot
|
|
162
|
-
implementation("org.springframework.boot:spring-boot-starter-web")
|
|
163
|
-
implementation("org.springframework.boot:spring-boot-starter-validation")
|
|
164
|
-
|
|
165
|
-
// MyBatis
|
|
166
|
-
implementation("org.mybatis.spring.boot:mybatis-spring-boot-starter:3.0.3")
|
|
167
|
-
|
|
168
|
-
// PostgreSQL
|
|
169
|
-
runtimeOnly("org.postgresql:postgresql")
|
|
170
|
-
|
|
171
|
-
// OpenAPI / Swagger
|
|
172
|
-
implementation("org.springdoc:springdoc-openapi-starter-webmvc-ui:2.3.0")
|
|
173
|
-
|
|
174
|
-
// Lombok
|
|
175
|
-
compileOnly("org.projectlombok:lombok")
|
|
176
|
-
annotationProcessor("org.projectlombok:lombok")
|
|
177
|
-
|
|
178
|
-
// Test
|
|
179
|
-
testImplementation("org.springframework.boot:spring-boot-starter-test")
|
|
180
|
-
testImplementation("org.testcontainers:postgresql:1.19.3")
|
|
181
|
-
testImplementation("org.testcontainers:junit-jupiter:1.19.3")
|
|
182
|
-
}
|
|
183
|
-
|
|
184
|
-
tasks.withType<Test> {
|
|
185
|
-
useJUnitPlatform()
|
|
186
|
-
}
|
|
187
|
-
```
|
|
188
|
-
|
|
189
|
-
</details>
|
|
190
|
-
|
|
191
|
-
### アプリケーション設定
|
|
192
|
-
|
|
193
|
-
<details>
|
|
194
|
-
<summary>Application.java</summary>
|
|
195
|
-
|
|
196
|
-
```java
|
|
197
|
-
package com.example.pms;
|
|
198
|
-
|
|
199
|
-
import org.springframework.boot.SpringApplication;
|
|
200
|
-
import org.springframework.boot.autoconfigure.SpringBootApplication;
|
|
201
|
-
|
|
202
|
-
/**
|
|
203
|
-
* 生産管理システム(Production Management System)のメインクラス.
|
|
204
|
-
*/
|
|
205
|
-
@SpringBootApplication
|
|
206
|
-
public class Application {
|
|
207
|
-
|
|
208
|
-
public static void main(String[] args) {
|
|
209
|
-
SpringApplication.run(Application.class, args);
|
|
210
|
-
}
|
|
211
|
-
}
|
|
212
|
-
```
|
|
213
|
-
|
|
214
|
-
</details>
|
|
215
|
-
|
|
216
|
-
<details>
|
|
217
|
-
<summary>OpenApiConfig.java</summary>
|
|
218
|
-
|
|
219
|
-
```java
|
|
220
|
-
package com.example.pms.infrastructure.config;
|
|
221
|
-
|
|
222
|
-
import io.swagger.v3.oas.models.OpenAPI;
|
|
223
|
-
import io.swagger.v3.oas.models.info.Contact;
|
|
224
|
-
import io.swagger.v3.oas.models.info.Info;
|
|
225
|
-
import org.springframework.context.annotation.Bean;
|
|
226
|
-
import org.springframework.context.annotation.Configuration;
|
|
227
|
-
|
|
228
|
-
/**
|
|
229
|
-
* OpenAPI 設定.
|
|
230
|
-
*/
|
|
231
|
-
@Configuration
|
|
232
|
-
public class OpenApiConfig {
|
|
233
|
-
|
|
234
|
-
@Bean
|
|
235
|
-
public OpenAPI customOpenAPI() {
|
|
236
|
-
return new OpenAPI()
|
|
237
|
-
.info(new Info()
|
|
238
|
-
.title("生産管理 API")
|
|
239
|
-
.version("1.0")
|
|
240
|
-
.description("生産管理システムの RESTful API")
|
|
241
|
-
.contact(new Contact()
|
|
242
|
-
.name("開発チーム")
|
|
243
|
-
.email("dev@example.com")));
|
|
244
|
-
}
|
|
245
|
-
}
|
|
246
|
-
```
|
|
247
|
-
|
|
248
|
-
</details>
|
|
249
|
-
|
|
250
|
-
<details>
|
|
251
|
-
<summary>RootController.java</summary>
|
|
252
|
-
|
|
253
|
-
```java
|
|
254
|
-
package com.example.pms.infrastructure.in.rest;
|
|
255
|
-
|
|
256
|
-
import io.swagger.v3.oas.annotations.Operation;
|
|
257
|
-
import io.swagger.v3.oas.annotations.tags.Tag;
|
|
258
|
-
import org.springframework.http.ResponseEntity;
|
|
259
|
-
import org.springframework.web.bind.annotation.GetMapping;
|
|
260
|
-
import org.springframework.web.bind.annotation.RequestMapping;
|
|
261
|
-
import org.springframework.web.bind.annotation.RestController;
|
|
262
|
-
|
|
263
|
-
import java.util.Map;
|
|
264
|
-
|
|
265
|
-
/**
|
|
266
|
-
* API ルート Controller.
|
|
267
|
-
*/
|
|
268
|
-
@RestController
|
|
269
|
-
@RequestMapping("/api")
|
|
270
|
-
@Tag(name = "root", description = "API ルート")
|
|
271
|
-
public class RootController {
|
|
272
|
-
|
|
273
|
-
@GetMapping
|
|
274
|
-
@Operation(summary = "API 情報の取得")
|
|
275
|
-
public ResponseEntity<Map<String, String>> root() {
|
|
276
|
-
return ResponseEntity.ok(Map.of(
|
|
277
|
-
"service", "Production Management API",
|
|
278
|
-
"version", "1.0",
|
|
279
|
-
"docs", "/swagger-ui.html"
|
|
280
|
-
));
|
|
281
|
-
}
|
|
282
|
-
|
|
283
|
-
@GetMapping("/health")
|
|
284
|
-
@Operation(summary = "ヘルスチェック")
|
|
285
|
-
public ResponseEntity<Map<String, String>> health() {
|
|
286
|
-
return ResponseEntity.ok(Map.of("status", "UP"));
|
|
287
|
-
}
|
|
288
|
-
}
|
|
289
|
-
```
|
|
290
|
-
|
|
291
|
-
</details>
|
|
292
|
-
|
|
293
|
-
---
|
|
294
|
-
|
|
295
|
-
## 32.3 マスタ API の実装
|
|
296
|
-
|
|
297
|
-
### 32.3.1 品目マスタ API
|
|
298
|
-
|
|
299
|
-
#### Output Port(リポジトリインターフェース)
|
|
300
|
-
|
|
301
|
-
<details>
|
|
302
|
-
<summary>ItemRepository.java</summary>
|
|
303
|
-
|
|
304
|
-
```java
|
|
305
|
-
package com.example.pms.application.port.out;
|
|
306
|
-
|
|
307
|
-
import com.example.pms.domain.model.item.Item;
|
|
308
|
-
|
|
309
|
-
import java.time.LocalDate;
|
|
310
|
-
import java.util.List;
|
|
311
|
-
import java.util.Optional;
|
|
312
|
-
|
|
313
|
-
/**
|
|
314
|
-
* 品目リポジトリ(Output Port)
|
|
315
|
-
* ドメイン層がデータアクセスに依存しないためのインターフェース
|
|
316
|
-
*/
|
|
317
|
-
public interface ItemRepository {
|
|
318
|
-
|
|
319
|
-
void save(Item item);
|
|
320
|
-
|
|
321
|
-
Optional<Item> findByItemCode(String itemCode);
|
|
322
|
-
|
|
323
|
-
Optional<Item> findByItemCodeAndDate(String itemCode, LocalDate baseDate);
|
|
324
|
-
|
|
325
|
-
List<Item> findAll();
|
|
326
|
-
|
|
327
|
-
void update(Item item);
|
|
328
|
-
|
|
329
|
-
void deleteAll();
|
|
330
|
-
}
|
|
331
|
-
```
|
|
332
|
-
|
|
333
|
-
</details>
|
|
334
|
-
|
|
335
|
-
#### Input Port(ユースケースインターフェース)
|
|
336
|
-
|
|
337
|
-
<details>
|
|
338
|
-
<summary>ItemUseCase.java</summary>
|
|
339
|
-
|
|
340
|
-
```java
|
|
341
|
-
package com.example.pms.application.port.in;
|
|
342
|
-
|
|
343
|
-
import com.example.pms.application.port.in.command.CreateItemCommand;
|
|
344
|
-
import com.example.pms.application.port.in.command.UpdateItemCommand;
|
|
345
|
-
import com.example.pms.domain.model.item.Item;
|
|
346
|
-
|
|
347
|
-
import java.util.List;
|
|
348
|
-
|
|
349
|
-
/**
|
|
350
|
-
* 品目ユースケース(Input Port).
|
|
351
|
-
*/
|
|
352
|
-
public interface ItemUseCase {
|
|
353
|
-
|
|
354
|
-
List<Item> getAllItems();
|
|
355
|
-
|
|
356
|
-
Item getItem(String itemCode);
|
|
357
|
-
|
|
358
|
-
Item createItem(CreateItemCommand command);
|
|
359
|
-
|
|
360
|
-
Item updateItem(String itemCode, UpdateItemCommand command);
|
|
361
|
-
|
|
362
|
-
void deleteItem(String itemCode);
|
|
363
|
-
}
|
|
364
|
-
```
|
|
365
|
-
|
|
366
|
-
</details>
|
|
367
|
-
|
|
368
|
-
<details>
|
|
369
|
-
<summary>CreateItemCommand.java</summary>
|
|
370
|
-
|
|
371
|
-
```java
|
|
372
|
-
package com.example.pms.application.port.in.command;
|
|
373
|
-
|
|
374
|
-
import com.example.pms.domain.model.item.ItemCategory;
|
|
375
|
-
import lombok.Builder;
|
|
376
|
-
import lombok.Value;
|
|
377
|
-
|
|
378
|
-
import java.math.BigDecimal;
|
|
379
|
-
import java.time.LocalDate;
|
|
380
|
-
|
|
381
|
-
/**
|
|
382
|
-
* 品目登録コマンド.
|
|
383
|
-
*/
|
|
384
|
-
@Value
|
|
385
|
-
@Builder
|
|
386
|
-
public class CreateItemCommand {
|
|
387
|
-
String itemCode;
|
|
388
|
-
String itemName;
|
|
389
|
-
ItemCategory itemCategory;
|
|
390
|
-
String unitCode;
|
|
391
|
-
LocalDate effectiveFrom;
|
|
392
|
-
LocalDate effectiveTo;
|
|
393
|
-
Integer leadTime;
|
|
394
|
-
Integer safetyLeadTime;
|
|
395
|
-
BigDecimal safetyStock;
|
|
396
|
-
BigDecimal yieldRate;
|
|
397
|
-
BigDecimal minLotSize;
|
|
398
|
-
BigDecimal lotIncrement;
|
|
399
|
-
BigDecimal maxLotSize;
|
|
400
|
-
Integer shelfLife;
|
|
401
|
-
}
|
|
402
|
-
```
|
|
403
|
-
|
|
404
|
-
</details>
|
|
405
|
-
|
|
406
|
-
#### Application Service
|
|
407
|
-
|
|
408
|
-
<details>
|
|
409
|
-
<summary>ItemService.java</summary>
|
|
410
|
-
|
|
411
|
-
```java
|
|
412
|
-
package com.example.pms.application.service;
|
|
413
|
-
|
|
414
|
-
import com.example.pms.application.port.in.ItemUseCase;
|
|
415
|
-
import com.example.pms.application.port.in.command.CreateItemCommand;
|
|
416
|
-
import com.example.pms.application.port.in.command.UpdateItemCommand;
|
|
417
|
-
import com.example.pms.application.port.out.ItemRepository;
|
|
418
|
-
import com.example.pms.domain.exception.DuplicateItemException;
|
|
419
|
-
import com.example.pms.domain.exception.ItemNotFoundException;
|
|
420
|
-
import com.example.pms.domain.model.item.Item;
|
|
421
|
-
import org.springframework.stereotype.Service;
|
|
422
|
-
import org.springframework.transaction.annotation.Transactional;
|
|
423
|
-
|
|
424
|
-
import java.time.LocalDate;
|
|
425
|
-
import java.util.List;
|
|
426
|
-
|
|
427
|
-
/**
|
|
428
|
-
* 品目サービス(Application Service).
|
|
429
|
-
*/
|
|
430
|
-
@Service
|
|
431
|
-
@Transactional
|
|
432
|
-
public class ItemService implements ItemUseCase {
|
|
433
|
-
|
|
434
|
-
private final ItemRepository itemRepository;
|
|
435
|
-
|
|
436
|
-
public ItemService(ItemRepository itemRepository) {
|
|
437
|
-
this.itemRepository = itemRepository;
|
|
438
|
-
}
|
|
439
|
-
|
|
440
|
-
@Override
|
|
441
|
-
@Transactional(readOnly = true)
|
|
442
|
-
public List<Item> getAllItems() {
|
|
443
|
-
return itemRepository.findAll();
|
|
444
|
-
}
|
|
445
|
-
|
|
446
|
-
@Override
|
|
447
|
-
@Transactional(readOnly = true)
|
|
448
|
-
public Item getItem(String itemCode) {
|
|
449
|
-
return itemRepository.findByItemCode(itemCode)
|
|
450
|
-
.orElseThrow(() -> new ItemNotFoundException(itemCode));
|
|
451
|
-
}
|
|
452
|
-
|
|
453
|
-
@Override
|
|
454
|
-
public Item createItem(CreateItemCommand command) {
|
|
455
|
-
if (itemRepository.findByItemCode(command.getItemCode()).isPresent()) {
|
|
456
|
-
throw new DuplicateItemException(command.getItemCode());
|
|
457
|
-
}
|
|
458
|
-
|
|
459
|
-
Item item = Item.builder()
|
|
460
|
-
.itemCode(command.getItemCode())
|
|
461
|
-
.itemName(command.getItemName())
|
|
462
|
-
.itemCategory(command.getItemCategory())
|
|
463
|
-
.unitCode(command.getUnitCode())
|
|
464
|
-
.effectiveFrom(command.getEffectiveFrom() != null
|
|
465
|
-
? command.getEffectiveFrom() : LocalDate.now())
|
|
466
|
-
.effectiveTo(command.getEffectiveTo())
|
|
467
|
-
.leadTime(command.getLeadTime())
|
|
468
|
-
.safetyLeadTime(command.getSafetyLeadTime())
|
|
469
|
-
.safetyStock(command.getSafetyStock())
|
|
470
|
-
.yieldRate(command.getYieldRate())
|
|
471
|
-
.minLotSize(command.getMinLotSize())
|
|
472
|
-
.lotIncrement(command.getLotIncrement())
|
|
473
|
-
.maxLotSize(command.getMaxLotSize())
|
|
474
|
-
.shelfLife(command.getShelfLife())
|
|
475
|
-
.build();
|
|
476
|
-
|
|
477
|
-
itemRepository.save(item);
|
|
478
|
-
return item;
|
|
479
|
-
}
|
|
480
|
-
|
|
481
|
-
@Override
|
|
482
|
-
public Item updateItem(String itemCode, UpdateItemCommand command) {
|
|
483
|
-
Item existing = itemRepository.findByItemCode(itemCode)
|
|
484
|
-
.orElseThrow(() -> new ItemNotFoundException(itemCode));
|
|
485
|
-
|
|
486
|
-
Item updated = Item.builder()
|
|
487
|
-
.id(existing.getId())
|
|
488
|
-
.itemCode(existing.getItemCode())
|
|
489
|
-
.itemName(command.getItemName() != null
|
|
490
|
-
? command.getItemName() : existing.getItemName())
|
|
491
|
-
.itemCategory(command.getItemCategory() != null
|
|
492
|
-
? command.getItemCategory() : existing.getItemCategory())
|
|
493
|
-
.unitCode(command.getUnitCode() != null
|
|
494
|
-
? command.getUnitCode() : existing.getUnitCode())
|
|
495
|
-
.effectiveFrom(existing.getEffectiveFrom())
|
|
496
|
-
.createdAt(existing.getCreatedAt())
|
|
497
|
-
.build();
|
|
498
|
-
|
|
499
|
-
itemRepository.update(updated);
|
|
500
|
-
return updated;
|
|
501
|
-
}
|
|
502
|
-
|
|
503
|
-
@Override
|
|
504
|
-
public void deleteItem(String itemCode) {
|
|
505
|
-
itemRepository.findByItemCode(itemCode)
|
|
506
|
-
.orElseThrow(() -> new ItemNotFoundException(itemCode));
|
|
507
|
-
throw new UnsupportedOperationException("品目の削除は現在サポートされていません");
|
|
508
|
-
}
|
|
509
|
-
}
|
|
510
|
-
```
|
|
511
|
-
|
|
512
|
-
</details>
|
|
513
|
-
|
|
514
|
-
#### Input Adapter(REST Controller)
|
|
515
|
-
|
|
516
|
-
<details>
|
|
517
|
-
<summary>ItemControllerTest.java(テスト駆動開発)</summary>
|
|
518
|
-
|
|
519
|
-
```java
|
|
520
|
-
package com.example.pms.infrastructure.in.rest;
|
|
521
|
-
|
|
522
|
-
import com.example.pms.application.port.in.ItemUseCase;
|
|
523
|
-
import com.example.pms.domain.model.item.Item;
|
|
524
|
-
import com.example.pms.domain.model.item.ItemCategory;
|
|
525
|
-
import com.fasterxml.jackson.databind.ObjectMapper;
|
|
526
|
-
import org.junit.jupiter.api.BeforeEach;
|
|
527
|
-
import org.junit.jupiter.api.Test;
|
|
528
|
-
import org.springframework.beans.factory.annotation.Autowired;
|
|
529
|
-
import org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest;
|
|
530
|
-
import org.springframework.boot.test.mock.mockito.MockBean;
|
|
531
|
-
import org.springframework.http.MediaType;
|
|
532
|
-
import org.springframework.test.web.servlet.MockMvc;
|
|
533
|
-
|
|
534
|
-
import java.time.LocalDate;
|
|
535
|
-
import java.util.List;
|
|
536
|
-
|
|
537
|
-
import static org.mockito.ArgumentMatchers.any;
|
|
538
|
-
import static org.mockito.Mockito.when;
|
|
539
|
-
import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.*;
|
|
540
|
-
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.*;
|
|
541
|
-
|
|
542
|
-
@WebMvcTest(ItemController.class)
|
|
543
|
-
class ItemControllerTest {
|
|
544
|
-
|
|
545
|
-
@Autowired
|
|
546
|
-
private MockMvc mockMvc;
|
|
547
|
-
|
|
548
|
-
@Autowired
|
|
549
|
-
private ObjectMapper objectMapper;
|
|
550
|
-
|
|
551
|
-
@MockBean
|
|
552
|
-
private ItemUseCase itemUseCase;
|
|
553
|
-
|
|
554
|
-
private Item sampleItem;
|
|
555
|
-
|
|
556
|
-
@BeforeEach
|
|
557
|
-
void setUp() {
|
|
558
|
-
sampleItem = Item.builder()
|
|
559
|
-
.itemCode("ITEM001")
|
|
560
|
-
.itemName("テスト品目")
|
|
561
|
-
.itemCategory(ItemCategory.PRODUCT)
|
|
562
|
-
.unitCode("個")
|
|
563
|
-
.effectiveFrom(LocalDate.now())
|
|
564
|
-
.leadTime(5)
|
|
565
|
-
.build();
|
|
566
|
-
}
|
|
567
|
-
|
|
568
|
-
@Test
|
|
569
|
-
void 品目一覧を取得できる() throws Exception {
|
|
570
|
-
when(itemUseCase.getAllItems()).thenReturn(List.of(sampleItem));
|
|
571
|
-
|
|
572
|
-
mockMvc.perform(get("/api/items"))
|
|
573
|
-
.andExpect(status().isOk())
|
|
574
|
-
.andExpect(jsonPath("$[0].itemCode").value("ITEM001"))
|
|
575
|
-
.andExpect(jsonPath("$[0].itemName").value("テスト品目"));
|
|
576
|
-
}
|
|
577
|
-
|
|
578
|
-
@Test
|
|
579
|
-
void 品目を登録できる() throws Exception {
|
|
580
|
-
when(itemUseCase.createItem(any())).thenReturn(sampleItem);
|
|
581
|
-
|
|
582
|
-
var request = """
|
|
583
|
-
{
|
|
584
|
-
"itemCode": "ITEM001",
|
|
585
|
-
"itemName": "テスト品目",
|
|
586
|
-
"itemCategory": "PRODUCT",
|
|
587
|
-
"unitCode": "個",
|
|
588
|
-
"leadTime": 5
|
|
589
|
-
}
|
|
590
|
-
""";
|
|
591
|
-
|
|
592
|
-
mockMvc.perform(post("/api/items")
|
|
593
|
-
.contentType(MediaType.APPLICATION_JSON)
|
|
594
|
-
.content(request))
|
|
595
|
-
.andExpect(status().isCreated())
|
|
596
|
-
.andExpect(jsonPath("$.itemCode").value("ITEM001"));
|
|
597
|
-
}
|
|
598
|
-
|
|
599
|
-
@Test
|
|
600
|
-
void 品目を削除できる() throws Exception {
|
|
601
|
-
mockMvc.perform(delete("/api/items/ITEM001"))
|
|
602
|
-
.andExpect(status().isNoContent());
|
|
603
|
-
}
|
|
604
|
-
}
|
|
605
|
-
```
|
|
606
|
-
|
|
607
|
-
</details>
|
|
608
|
-
|
|
609
|
-
<details>
|
|
610
|
-
<summary>ItemController.java</summary>
|
|
611
|
-
|
|
612
|
-
```java
|
|
613
|
-
package com.example.pms.infrastructure.in.rest;
|
|
614
|
-
|
|
615
|
-
import com.example.pms.application.port.in.ItemUseCase;
|
|
616
|
-
import com.example.pms.domain.model.item.Item;
|
|
617
|
-
import com.example.pms.infrastructure.in.rest.dto.CreateItemRequest;
|
|
618
|
-
import com.example.pms.infrastructure.in.rest.dto.ItemResponse;
|
|
619
|
-
import com.example.pms.infrastructure.in.rest.dto.UpdateItemRequest;
|
|
620
|
-
import io.swagger.v3.oas.annotations.Operation;
|
|
621
|
-
import io.swagger.v3.oas.annotations.tags.Tag;
|
|
622
|
-
import jakarta.validation.Valid;
|
|
623
|
-
import org.springframework.http.HttpStatus;
|
|
624
|
-
import org.springframework.http.ResponseEntity;
|
|
625
|
-
import org.springframework.web.bind.annotation.*;
|
|
626
|
-
|
|
627
|
-
import java.util.List;
|
|
628
|
-
|
|
629
|
-
/**
|
|
630
|
-
* 品目マスタ API Controller.
|
|
631
|
-
*/
|
|
632
|
-
@RestController
|
|
633
|
-
@RequestMapping("/api/items")
|
|
634
|
-
@Tag(name = "items", description = "品目マスタ API")
|
|
635
|
-
public class ItemController {
|
|
636
|
-
|
|
637
|
-
private final ItemUseCase itemUseCase;
|
|
638
|
-
|
|
639
|
-
public ItemController(ItemUseCase itemUseCase) {
|
|
640
|
-
this.itemUseCase = itemUseCase;
|
|
641
|
-
}
|
|
642
|
-
|
|
643
|
-
@GetMapping
|
|
644
|
-
@Operation(summary = "品目一覧の取得")
|
|
645
|
-
public ResponseEntity<List<ItemResponse>> getAllItems() {
|
|
646
|
-
List<Item> items = itemUseCase.getAllItems();
|
|
647
|
-
return ResponseEntity.ok(items.stream()
|
|
648
|
-
.map(ItemResponse::from)
|
|
649
|
-
.toList());
|
|
650
|
-
}
|
|
651
|
-
|
|
652
|
-
@GetMapping("/{itemCode}")
|
|
653
|
-
@Operation(summary = "品目の取得")
|
|
654
|
-
public ResponseEntity<ItemResponse> getItem(@PathVariable String itemCode) {
|
|
655
|
-
Item item = itemUseCase.getItem(itemCode);
|
|
656
|
-
return ResponseEntity.ok(ItemResponse.from(item));
|
|
657
|
-
}
|
|
658
|
-
|
|
659
|
-
@PostMapping
|
|
660
|
-
@Operation(summary = "品目の登録")
|
|
661
|
-
public ResponseEntity<ItemResponse> createItem(
|
|
662
|
-
@Valid @RequestBody CreateItemRequest request) {
|
|
663
|
-
Item item = itemUseCase.createItem(request.toCommand());
|
|
664
|
-
return ResponseEntity.status(HttpStatus.CREATED)
|
|
665
|
-
.body(ItemResponse.from(item));
|
|
666
|
-
}
|
|
667
|
-
|
|
668
|
-
@PutMapping("/{itemCode}")
|
|
669
|
-
@Operation(summary = "品目の更新")
|
|
670
|
-
public ResponseEntity<ItemResponse> updateItem(
|
|
671
|
-
@PathVariable String itemCode,
|
|
672
|
-
@Valid @RequestBody UpdateItemRequest request) {
|
|
673
|
-
Item item = itemUseCase.updateItem(itemCode, request.toCommand());
|
|
674
|
-
return ResponseEntity.ok(ItemResponse.from(item));
|
|
675
|
-
}
|
|
676
|
-
|
|
677
|
-
@DeleteMapping("/{itemCode}")
|
|
678
|
-
@Operation(summary = "品目の削除")
|
|
679
|
-
@ResponseStatus(HttpStatus.NO_CONTENT)
|
|
680
|
-
public void deleteItem(@PathVariable String itemCode) {
|
|
681
|
-
itemUseCase.deleteItem(itemCode);
|
|
682
|
-
}
|
|
683
|
-
}
|
|
684
|
-
```
|
|
685
|
-
|
|
686
|
-
</details>
|
|
687
|
-
|
|
688
|
-
#### DTO(Request/Response)
|
|
689
|
-
|
|
690
|
-
<details>
|
|
691
|
-
<summary>CreateItemRequest.java</summary>
|
|
692
|
-
|
|
693
|
-
```java
|
|
694
|
-
package com.example.pms.infrastructure.in.rest.dto;
|
|
695
|
-
|
|
696
|
-
import com.example.pms.application.port.in.command.CreateItemCommand;
|
|
697
|
-
import com.example.pms.domain.model.item.ItemCategory;
|
|
698
|
-
import jakarta.validation.constraints.NotBlank;
|
|
699
|
-
import jakarta.validation.constraints.NotNull;
|
|
700
|
-
import lombok.Data;
|
|
701
|
-
|
|
702
|
-
import java.math.BigDecimal;
|
|
703
|
-
import java.time.LocalDate;
|
|
704
|
-
|
|
705
|
-
/**
|
|
706
|
-
* 品目登録リクエスト DTO.
|
|
707
|
-
*/
|
|
708
|
-
@Data
|
|
709
|
-
public class CreateItemRequest {
|
|
710
|
-
|
|
711
|
-
@NotBlank(message = "品目コードは必須です")
|
|
712
|
-
private String itemCode;
|
|
713
|
-
|
|
714
|
-
@NotBlank(message = "品目名は必須です")
|
|
715
|
-
private String itemName;
|
|
716
|
-
|
|
717
|
-
@NotNull(message = "品目区分は必須です")
|
|
718
|
-
private ItemCategory itemCategory;
|
|
719
|
-
|
|
720
|
-
@NotBlank(message = "単位コードは必須です")
|
|
721
|
-
private String unitCode;
|
|
722
|
-
|
|
723
|
-
private LocalDate effectiveFrom;
|
|
724
|
-
private LocalDate effectiveTo;
|
|
725
|
-
private Integer leadTime;
|
|
726
|
-
private Integer safetyLeadTime;
|
|
727
|
-
private BigDecimal safetyStock;
|
|
728
|
-
private BigDecimal yieldRate;
|
|
729
|
-
private BigDecimal minLotSize;
|
|
730
|
-
private BigDecimal lotIncrement;
|
|
731
|
-
private BigDecimal maxLotSize;
|
|
732
|
-
private Integer shelfLife;
|
|
733
|
-
|
|
734
|
-
public CreateItemCommand toCommand() {
|
|
735
|
-
return CreateItemCommand.builder()
|
|
736
|
-
.itemCode(itemCode)
|
|
737
|
-
.itemName(itemName)
|
|
738
|
-
.itemCategory(itemCategory)
|
|
739
|
-
.unitCode(unitCode)
|
|
740
|
-
.effectiveFrom(effectiveFrom)
|
|
741
|
-
.effectiveTo(effectiveTo)
|
|
742
|
-
.leadTime(leadTime)
|
|
743
|
-
.safetyLeadTime(safetyLeadTime)
|
|
744
|
-
.safetyStock(safetyStock)
|
|
745
|
-
.yieldRate(yieldRate)
|
|
746
|
-
.minLotSize(minLotSize)
|
|
747
|
-
.lotIncrement(lotIncrement)
|
|
748
|
-
.maxLotSize(maxLotSize)
|
|
749
|
-
.shelfLife(shelfLife)
|
|
750
|
-
.build();
|
|
751
|
-
}
|
|
752
|
-
}
|
|
753
|
-
```
|
|
754
|
-
|
|
755
|
-
</details>
|
|
756
|
-
|
|
757
|
-
<details>
|
|
758
|
-
<summary>UpdateItemRequest.java</summary>
|
|
759
|
-
|
|
760
|
-
```java
|
|
761
|
-
package com.example.pms.infrastructure.in.rest.dto;
|
|
762
|
-
|
|
763
|
-
import com.example.pms.application.port.in.command.UpdateItemCommand;
|
|
764
|
-
import com.example.pms.domain.model.item.ItemCategory;
|
|
765
|
-
import lombok.Data;
|
|
766
|
-
|
|
767
|
-
import java.math.BigDecimal;
|
|
768
|
-
import java.time.LocalDate;
|
|
769
|
-
|
|
770
|
-
/**
|
|
771
|
-
* 品目更新リクエスト DTO.
|
|
772
|
-
*/
|
|
773
|
-
@Data
|
|
774
|
-
public class UpdateItemRequest {
|
|
775
|
-
|
|
776
|
-
private String itemName;
|
|
777
|
-
private ItemCategory itemCategory;
|
|
778
|
-
private String unitCode;
|
|
779
|
-
private LocalDate effectiveFrom;
|
|
780
|
-
private LocalDate effectiveTo;
|
|
781
|
-
private Integer leadTime;
|
|
782
|
-
private Integer safetyLeadTime;
|
|
783
|
-
private BigDecimal safetyStock;
|
|
784
|
-
private BigDecimal yieldRate;
|
|
785
|
-
private BigDecimal minLotSize;
|
|
786
|
-
private BigDecimal lotIncrement;
|
|
787
|
-
private BigDecimal maxLotSize;
|
|
788
|
-
private Integer shelfLife;
|
|
789
|
-
|
|
790
|
-
public UpdateItemCommand toCommand() {
|
|
791
|
-
return UpdateItemCommand.builder()
|
|
792
|
-
.itemName(itemName)
|
|
793
|
-
.itemCategory(itemCategory)
|
|
794
|
-
.unitCode(unitCode)
|
|
795
|
-
.effectiveFrom(effectiveFrom)
|
|
796
|
-
.effectiveTo(effectiveTo)
|
|
797
|
-
.leadTime(leadTime)
|
|
798
|
-
.safetyLeadTime(safetyLeadTime)
|
|
799
|
-
.safetyStock(safetyStock)
|
|
800
|
-
.yieldRate(yieldRate)
|
|
801
|
-
.minLotSize(minLotSize)
|
|
802
|
-
.lotIncrement(lotIncrement)
|
|
803
|
-
.maxLotSize(maxLotSize)
|
|
804
|
-
.shelfLife(shelfLife)
|
|
805
|
-
.build();
|
|
806
|
-
}
|
|
807
|
-
}
|
|
808
|
-
```
|
|
809
|
-
|
|
810
|
-
</details>
|
|
811
|
-
|
|
812
|
-
<details>
|
|
813
|
-
<summary>ItemResponse.java</summary>
|
|
814
|
-
|
|
815
|
-
```java
|
|
816
|
-
package com.example.pms.infrastructure.in.rest.dto;
|
|
817
|
-
|
|
818
|
-
import com.example.pms.domain.model.item.Item;
|
|
819
|
-
import com.example.pms.domain.model.item.ItemCategory;
|
|
820
|
-
import lombok.AllArgsConstructor;
|
|
821
|
-
import lombok.Builder;
|
|
822
|
-
import lombok.Data;
|
|
823
|
-
import lombok.NoArgsConstructor;
|
|
824
|
-
|
|
825
|
-
import java.math.BigDecimal;
|
|
826
|
-
import java.time.LocalDate;
|
|
827
|
-
import java.time.LocalDateTime;
|
|
828
|
-
|
|
829
|
-
/**
|
|
830
|
-
* 品目レスポンス DTO.
|
|
831
|
-
* Note: Jackson シリアライゼーションのため @Data + @NoArgsConstructor + @AllArgsConstructor を使用
|
|
832
|
-
*/
|
|
833
|
-
@Data
|
|
834
|
-
@Builder
|
|
835
|
-
@NoArgsConstructor
|
|
836
|
-
@AllArgsConstructor
|
|
837
|
-
public class ItemResponse {
|
|
838
|
-
private Integer id;
|
|
839
|
-
private String itemCode;
|
|
840
|
-
private String itemName;
|
|
841
|
-
private ItemCategory itemCategory;
|
|
842
|
-
private String unitCode;
|
|
843
|
-
private LocalDate effectiveFrom;
|
|
844
|
-
private LocalDate effectiveTo;
|
|
845
|
-
private Integer leadTime;
|
|
846
|
-
private Integer safetyLeadTime;
|
|
847
|
-
private BigDecimal safetyStock;
|
|
848
|
-
private BigDecimal yieldRate;
|
|
849
|
-
private BigDecimal minLotSize;
|
|
850
|
-
private BigDecimal lotIncrement;
|
|
851
|
-
private BigDecimal maxLotSize;
|
|
852
|
-
private Integer shelfLife;
|
|
853
|
-
private LocalDateTime createdAt;
|
|
854
|
-
private LocalDateTime updatedAt;
|
|
855
|
-
|
|
856
|
-
public static ItemResponse from(Item item) {
|
|
857
|
-
return ItemResponse.builder()
|
|
858
|
-
.id(item.getId())
|
|
859
|
-
.itemCode(item.getItemCode())
|
|
860
|
-
.itemName(item.getItemName())
|
|
861
|
-
.itemCategory(item.getItemCategory())
|
|
862
|
-
.unitCode(item.getUnitCode())
|
|
863
|
-
.effectiveFrom(item.getEffectiveFrom())
|
|
864
|
-
.effectiveTo(item.getEffectiveTo())
|
|
865
|
-
.leadTime(item.getLeadTime())
|
|
866
|
-
.safetyLeadTime(item.getSafetyLeadTime())
|
|
867
|
-
.safetyStock(item.getSafetyStock())
|
|
868
|
-
.yieldRate(item.getYieldRate())
|
|
869
|
-
.minLotSize(item.getMinLotSize())
|
|
870
|
-
.lotIncrement(item.getLotIncrement())
|
|
871
|
-
.maxLotSize(item.getMaxLotSize())
|
|
872
|
-
.shelfLife(item.getShelfLife())
|
|
873
|
-
.createdAt(item.getCreatedAt())
|
|
874
|
-
.updatedAt(item.getUpdatedAt())
|
|
875
|
-
.build();
|
|
876
|
-
}
|
|
877
|
-
}
|
|
878
|
-
```
|
|
879
|
-
|
|
880
|
-
</details>
|
|
881
|
-
|
|
882
|
-
#### Output Adapter(MyBatis Repository)
|
|
883
|
-
|
|
884
|
-
<details>
|
|
885
|
-
<summary>MyBatisItemRepository.java</summary>
|
|
886
|
-
|
|
887
|
-
```java
|
|
888
|
-
package com.example.pms.infrastructure.out.persistence.repository;
|
|
889
|
-
|
|
890
|
-
import com.example.pms.application.port.out.ItemRepository;
|
|
891
|
-
import com.example.pms.domain.model.item.Item;
|
|
892
|
-
import org.apache.ibatis.annotations.*;
|
|
893
|
-
import org.springframework.stereotype.Repository;
|
|
894
|
-
|
|
895
|
-
import java.time.LocalDate;
|
|
896
|
-
import java.util.List;
|
|
897
|
-
import java.util.Optional;
|
|
898
|
-
|
|
899
|
-
/**
|
|
900
|
-
* 品目リポジトリの MyBatis 実装(Output Adapter).
|
|
901
|
-
*/
|
|
902
|
-
@Repository
|
|
903
|
-
@Mapper
|
|
904
|
-
public interface MyBatisItemRepository extends ItemRepository {
|
|
905
|
-
|
|
906
|
-
@Override
|
|
907
|
-
@Insert("""
|
|
908
|
-
INSERT INTO "品目マスタ" ("品目コード", "適用開始日", "品名", "品目区分", "単位コード",
|
|
909
|
-
"リードタイム", "安全リードタイム", "安全在庫数", "歩留まり率",
|
|
910
|
-
"最小ロットサイズ", "ロット増分", "最大ロットサイズ", "賞味期限日数")
|
|
911
|
-
VALUES (#{itemCode}, #{effectiveFrom}, #{itemName}, #{itemCategory}::品目区分, #{unitCode},
|
|
912
|
-
#{leadTime}, #{safetyLeadTime}, #{safetyStock}, #{yieldRate},
|
|
913
|
-
#{minLotSize}, #{lotIncrement}, #{maxLotSize}, #{shelfLife})
|
|
914
|
-
""")
|
|
915
|
-
void save(Item item);
|
|
916
|
-
|
|
917
|
-
@Override
|
|
918
|
-
@Select("""
|
|
919
|
-
SELECT * FROM "品目マスタ"
|
|
920
|
-
WHERE "品目コード" = #{itemCode}
|
|
921
|
-
ORDER BY "適用開始日" DESC LIMIT 1
|
|
922
|
-
""")
|
|
923
|
-
@ResultMap("itemMap")
|
|
924
|
-
Optional<Item> findByItemCode(String itemCode);
|
|
925
|
-
|
|
926
|
-
@Override
|
|
927
|
-
@Select("""
|
|
928
|
-
SELECT * FROM "品目マスタ"
|
|
929
|
-
WHERE "品目コード" = #{itemCode}
|
|
930
|
-
AND "適用開始日" <= #{baseDate}
|
|
931
|
-
ORDER BY "適用開始日" DESC LIMIT 1
|
|
932
|
-
""")
|
|
933
|
-
@ResultMap("itemMap")
|
|
934
|
-
Optional<Item> findByItemCodeAndDate(String itemCode, LocalDate baseDate);
|
|
935
|
-
|
|
936
|
-
@Override
|
|
937
|
-
@Select("SELECT * FROM \"品目マスタ\"")
|
|
938
|
-
@Results(id = "itemMap", value = {
|
|
939
|
-
@Result(property = "id", column = "品目ID"),
|
|
940
|
-
@Result(property = "itemCode", column = "品目コード"),
|
|
941
|
-
@Result(property = "itemName", column = "品名"),
|
|
942
|
-
@Result(property = "itemCategory", column = "品目区分"),
|
|
943
|
-
@Result(property = "unitCode", column = "単位コード"),
|
|
944
|
-
@Result(property = "effectiveFrom", column = "適用開始日"),
|
|
945
|
-
@Result(property = "effectiveTo", column = "適用終了日"),
|
|
946
|
-
@Result(property = "leadTime", column = "リードタイム"),
|
|
947
|
-
@Result(property = "safetyLeadTime", column = "安全リードタイム"),
|
|
948
|
-
@Result(property = "safetyStock", column = "安全在庫数"),
|
|
949
|
-
@Result(property = "yieldRate", column = "歩留まり率"),
|
|
950
|
-
@Result(property = "minLotSize", column = "最小ロットサイズ"),
|
|
951
|
-
@Result(property = "lotIncrement", column = "ロット増分"),
|
|
952
|
-
@Result(property = "maxLotSize", column = "最大ロットサイズ"),
|
|
953
|
-
@Result(property = "shelfLife", column = "賞味期限日数"),
|
|
954
|
-
@Result(property = "createdAt", column = "作成日時"),
|
|
955
|
-
@Result(property = "updatedAt", column = "更新日時")
|
|
956
|
-
})
|
|
957
|
-
List<Item> findAll();
|
|
958
|
-
|
|
959
|
-
@Override
|
|
960
|
-
@Update("""
|
|
961
|
-
UPDATE "品目マスタ"
|
|
962
|
-
SET "品名" = #{itemName}, "品目区分" = #{itemCategory}::品目区分,
|
|
963
|
-
"単位コード" = #{unitCode}
|
|
964
|
-
WHERE "品目ID" = #{id}
|
|
965
|
-
""")
|
|
966
|
-
void update(Item item);
|
|
967
|
-
|
|
968
|
-
@Override
|
|
969
|
-
@Delete("DELETE FROM \"品目マスタ\"")
|
|
970
|
-
void deleteAll();
|
|
971
|
-
}
|
|
972
|
-
```
|
|
973
|
-
|
|
974
|
-
</details>
|
|
975
|
-
|
|
976
|
-
### 32.3.2 BOM マスタ API
|
|
977
|
-
|
|
978
|
-
#### BOM 展開サービス
|
|
979
|
-
|
|
980
|
-
<details>
|
|
981
|
-
<summary>BomService.java</summary>
|
|
982
|
-
|
|
983
|
-
```java
|
|
984
|
-
package com.example.pms.application.service;
|
|
985
|
-
|
|
986
|
-
import com.example.pms.application.port.out.BomRepository;
|
|
987
|
-
import com.example.pms.domain.model.bom.Bom;
|
|
988
|
-
import lombok.Builder;
|
|
989
|
-
import lombok.Value;
|
|
990
|
-
import org.springframework.stereotype.Service;
|
|
991
|
-
import org.springframework.transaction.annotation.Transactional;
|
|
992
|
-
|
|
993
|
-
import java.math.BigDecimal;
|
|
994
|
-
import java.util.ArrayList;
|
|
995
|
-
import java.util.List;
|
|
996
|
-
|
|
997
|
-
/**
|
|
998
|
-
* BOM サービス(Application Service).
|
|
999
|
-
*/
|
|
1000
|
-
@Service
|
|
1001
|
-
@Transactional(readOnly = true)
|
|
1002
|
-
public class BomService {
|
|
1003
|
-
|
|
1004
|
-
private final BomRepository bomRepository;
|
|
1005
|
-
|
|
1006
|
-
public BomService(BomRepository bomRepository) {
|
|
1007
|
-
this.bomRepository = bomRepository;
|
|
1008
|
-
}
|
|
1009
|
-
|
|
1010
|
-
/**
|
|
1011
|
-
* BOM 展開(部品展開)
|
|
1012
|
-
*/
|
|
1013
|
-
public List<BomNode> explodeBom(String itemCode, int maxLevel) {
|
|
1014
|
-
List<BomNode> result = new ArrayList<>();
|
|
1015
|
-
explodeRecursive(itemCode, 1, maxLevel, BigDecimal.ONE, result);
|
|
1016
|
-
return result;
|
|
1017
|
-
}
|
|
1018
|
-
|
|
1019
|
-
private void explodeRecursive(
|
|
1020
|
-
String parentCode,
|
|
1021
|
-
int currentLevel,
|
|
1022
|
-
int maxLevel,
|
|
1023
|
-
BigDecimal parentQuantity,
|
|
1024
|
-
List<BomNode> result) {
|
|
1025
|
-
|
|
1026
|
-
if (currentLevel > maxLevel) return;
|
|
1027
|
-
|
|
1028
|
-
List<Bom> children = bomRepository.findByParentCode(parentCode);
|
|
1029
|
-
|
|
1030
|
-
for (Bom bom : children) {
|
|
1031
|
-
BigDecimal requiredQty = bom.getQuantity().multiply(parentQuantity);
|
|
1032
|
-
|
|
1033
|
-
result.add(BomNode.builder()
|
|
1034
|
-
.level(currentLevel)
|
|
1035
|
-
.itemCode(bom.getChildItemCode())
|
|
1036
|
-
.itemName(bom.getChildItemName())
|
|
1037
|
-
.quantity(requiredQty)
|
|
1038
|
-
.unit(bom.getUnit())
|
|
1039
|
-
.build());
|
|
1040
|
-
|
|
1041
|
-
explodeRecursive(
|
|
1042
|
-
bom.getChildItemCode(),
|
|
1043
|
-
currentLevel + 1,
|
|
1044
|
-
maxLevel,
|
|
1045
|
-
requiredQty,
|
|
1046
|
-
result);
|
|
1047
|
-
}
|
|
1048
|
-
}
|
|
1049
|
-
|
|
1050
|
-
/**
|
|
1051
|
-
* 逆展開(使用先照会)
|
|
1052
|
-
*/
|
|
1053
|
-
public List<WhereUsedResult> whereUsed(String itemCode) {
|
|
1054
|
-
List<WhereUsedResult> result = new ArrayList<>();
|
|
1055
|
-
whereUsedRecursive(itemCode, 1, result);
|
|
1056
|
-
return result;
|
|
1057
|
-
}
|
|
1058
|
-
|
|
1059
|
-
private void whereUsedRecursive(
|
|
1060
|
-
String childCode,
|
|
1061
|
-
int level,
|
|
1062
|
-
List<WhereUsedResult> result) {
|
|
1063
|
-
|
|
1064
|
-
List<Bom> parents = bomRepository.findByChildCode(childCode);
|
|
1065
|
-
|
|
1066
|
-
for (Bom bom : parents) {
|
|
1067
|
-
result.add(WhereUsedResult.builder()
|
|
1068
|
-
.level(level)
|
|
1069
|
-
.parentItemCode(bom.getParentItemCode())
|
|
1070
|
-
.parentItemName(bom.getParentItemName())
|
|
1071
|
-
.quantity(bom.getQuantity())
|
|
1072
|
-
.build());
|
|
1073
|
-
|
|
1074
|
-
whereUsedRecursive(bom.getParentItemCode(), level + 1, result);
|
|
1075
|
-
}
|
|
1076
|
-
}
|
|
1077
|
-
}
|
|
1078
|
-
|
|
1079
|
-
@Value
|
|
1080
|
-
@Builder
|
|
1081
|
-
class BomNode {
|
|
1082
|
-
int level;
|
|
1083
|
-
String itemCode;
|
|
1084
|
-
String itemName;
|
|
1085
|
-
BigDecimal quantity;
|
|
1086
|
-
String unit;
|
|
1087
|
-
}
|
|
1088
|
-
|
|
1089
|
-
@Value
|
|
1090
|
-
@Builder
|
|
1091
|
-
class WhereUsedResult {
|
|
1092
|
-
int level;
|
|
1093
|
-
String parentItemCode;
|
|
1094
|
-
String parentItemName;
|
|
1095
|
-
BigDecimal quantity;
|
|
1096
|
-
}
|
|
1097
|
-
```
|
|
1098
|
-
|
|
1099
|
-
</details>
|
|
1100
|
-
|
|
1101
|
-
#### BOM Controller
|
|
1102
|
-
|
|
1103
|
-
<details>
|
|
1104
|
-
<summary>BomController.java</summary>
|
|
1105
|
-
|
|
1106
|
-
```java
|
|
1107
|
-
package com.example.pms.infrastructure.in.rest;
|
|
1108
|
-
|
|
1109
|
-
import com.example.pms.application.service.BomNode;
|
|
1110
|
-
import com.example.pms.application.service.BomService;
|
|
1111
|
-
import com.example.pms.application.service.WhereUsedResult;
|
|
1112
|
-
import io.swagger.v3.oas.annotations.Operation;
|
|
1113
|
-
import io.swagger.v3.oas.annotations.Parameter;
|
|
1114
|
-
import io.swagger.v3.oas.annotations.tags.Tag;
|
|
1115
|
-
import org.springframework.http.ResponseEntity;
|
|
1116
|
-
import org.springframework.web.bind.annotation.*;
|
|
1117
|
-
|
|
1118
|
-
import java.util.List;
|
|
1119
|
-
|
|
1120
|
-
/**
|
|
1121
|
-
* BOM API Controller.
|
|
1122
|
-
*/
|
|
1123
|
-
@RestController
|
|
1124
|
-
@RequestMapping("/api/bom")
|
|
1125
|
-
@Tag(name = "bom", description = "BOM API")
|
|
1126
|
-
public class BomController {
|
|
1127
|
-
|
|
1128
|
-
private final BomService bomService;
|
|
1129
|
-
|
|
1130
|
-
public BomController(BomService bomService) {
|
|
1131
|
-
this.bomService = bomService;
|
|
1132
|
-
}
|
|
1133
|
-
|
|
1134
|
-
@GetMapping("/{itemCode}/explode")
|
|
1135
|
-
@Operation(summary = "BOM 展開(部品展開)")
|
|
1136
|
-
public ResponseEntity<List<BomNode>> explodeBom(
|
|
1137
|
-
@PathVariable String itemCode,
|
|
1138
|
-
@Parameter(description = "展開レベル(デフォルト: 10)")
|
|
1139
|
-
@RequestParam(defaultValue = "10") int maxLevel) {
|
|
1140
|
-
List<BomNode> nodes = bomService.explodeBom(itemCode, maxLevel);
|
|
1141
|
-
return ResponseEntity.ok(nodes);
|
|
1142
|
-
}
|
|
1143
|
-
|
|
1144
|
-
@GetMapping("/{itemCode}/where-used")
|
|
1145
|
-
@Operation(summary = "逆展開(使用先照会)")
|
|
1146
|
-
public ResponseEntity<List<WhereUsedResult>> whereUsed(
|
|
1147
|
-
@PathVariable String itemCode) {
|
|
1148
|
-
List<WhereUsedResult> results = bomService.whereUsed(itemCode);
|
|
1149
|
-
return ResponseEntity.ok(results);
|
|
1150
|
-
}
|
|
1151
|
-
}
|
|
1152
|
-
```
|
|
1153
|
-
|
|
1154
|
-
</details>
|
|
1155
|
-
|
|
1156
|
-
---
|
|
1157
|
-
|
|
1158
|
-
## 32.4 トランザクション API の実装
|
|
1159
|
-
|
|
1160
|
-
### 32.4.1 発注 API
|
|
1161
|
-
|
|
1162
|
-
#### Input Port
|
|
1163
|
-
|
|
1164
|
-
<details>
|
|
1165
|
-
<summary>PurchaseOrderUseCase.java</summary>
|
|
1166
|
-
|
|
1167
|
-
```java
|
|
1168
|
-
package com.example.pms.application.port.in;
|
|
1169
|
-
|
|
1170
|
-
import com.example.pms.application.port.in.command.CreatePurchaseOrderCommand;
|
|
1171
|
-
import com.example.pms.domain.model.purchase.PurchaseOrder;
|
|
1172
|
-
|
|
1173
|
-
import java.util.List;
|
|
1174
|
-
|
|
1175
|
-
/**
|
|
1176
|
-
* 発注ユースケース(Input Port).
|
|
1177
|
-
*/
|
|
1178
|
-
public interface PurchaseOrderUseCase {
|
|
1179
|
-
|
|
1180
|
-
List<PurchaseOrder> getAllOrders();
|
|
1181
|
-
|
|
1182
|
-
PurchaseOrder getOrder(String orderNumber);
|
|
1183
|
-
|
|
1184
|
-
PurchaseOrder createOrder(CreatePurchaseOrderCommand command);
|
|
1185
|
-
|
|
1186
|
-
PurchaseOrder confirmOrder(String orderNumber);
|
|
1187
|
-
|
|
1188
|
-
void cancelOrder(String orderNumber);
|
|
1189
|
-
}
|
|
1190
|
-
```
|
|
1191
|
-
|
|
1192
|
-
</details>
|
|
1193
|
-
|
|
1194
|
-
#### Controller
|
|
1195
|
-
|
|
1196
|
-
<details>
|
|
1197
|
-
<summary>PurchaseOrderController.java</summary>
|
|
1198
|
-
|
|
1199
|
-
```java
|
|
1200
|
-
package com.example.pms.infrastructure.in.rest;
|
|
1201
|
-
|
|
1202
|
-
import com.example.pms.application.port.in.PurchaseOrderUseCase;
|
|
1203
|
-
import com.example.pms.domain.model.purchase.PurchaseOrder;
|
|
1204
|
-
import com.example.pms.infrastructure.in.rest.dto.CreatePurchaseOrderRequest;
|
|
1205
|
-
import com.example.pms.infrastructure.in.rest.dto.PurchaseOrderResponse;
|
|
1206
|
-
import io.swagger.v3.oas.annotations.Operation;
|
|
1207
|
-
import io.swagger.v3.oas.annotations.tags.Tag;
|
|
1208
|
-
import jakarta.validation.Valid;
|
|
1209
|
-
import org.springframework.http.HttpStatus;
|
|
1210
|
-
import org.springframework.http.ResponseEntity;
|
|
1211
|
-
import org.springframework.web.bind.annotation.*;
|
|
1212
|
-
|
|
1213
|
-
import java.util.List;
|
|
1214
|
-
|
|
1215
|
-
/**
|
|
1216
|
-
* 発注 API Controller.
|
|
1217
|
-
*/
|
|
1218
|
-
@RestController
|
|
1219
|
-
@RequestMapping("/api/purchase-orders")
|
|
1220
|
-
@Tag(name = "purchase-orders", description = "発注 API")
|
|
1221
|
-
public class PurchaseOrderController {
|
|
1222
|
-
|
|
1223
|
-
private final PurchaseOrderUseCase useCase;
|
|
1224
|
-
|
|
1225
|
-
public PurchaseOrderController(PurchaseOrderUseCase useCase) {
|
|
1226
|
-
this.useCase = useCase;
|
|
1227
|
-
}
|
|
1228
|
-
|
|
1229
|
-
@GetMapping
|
|
1230
|
-
@Operation(summary = "発注一覧の取得")
|
|
1231
|
-
public ResponseEntity<List<PurchaseOrderResponse>> getAllOrders() {
|
|
1232
|
-
List<PurchaseOrder> orders = useCase.getAllOrders();
|
|
1233
|
-
return ResponseEntity.ok(orders.stream()
|
|
1234
|
-
.map(PurchaseOrderResponse::from)
|
|
1235
|
-
.toList());
|
|
1236
|
-
}
|
|
1237
|
-
|
|
1238
|
-
@GetMapping("/{orderNumber}")
|
|
1239
|
-
@Operation(summary = "発注詳細の取得")
|
|
1240
|
-
public ResponseEntity<PurchaseOrderResponse> getOrder(
|
|
1241
|
-
@PathVariable String orderNumber) {
|
|
1242
|
-
PurchaseOrder order = useCase.getOrder(orderNumber);
|
|
1243
|
-
return ResponseEntity.ok(PurchaseOrderResponse.from(order));
|
|
1244
|
-
}
|
|
1245
|
-
|
|
1246
|
-
@PostMapping
|
|
1247
|
-
@Operation(summary = "発注の登録")
|
|
1248
|
-
public ResponseEntity<PurchaseOrderResponse> createOrder(
|
|
1249
|
-
@Valid @RequestBody CreatePurchaseOrderRequest request) {
|
|
1250
|
-
PurchaseOrder order = useCase.createOrder(request.toCommand());
|
|
1251
|
-
return ResponseEntity.status(HttpStatus.CREATED)
|
|
1252
|
-
.body(PurchaseOrderResponse.from(order));
|
|
1253
|
-
}
|
|
1254
|
-
|
|
1255
|
-
@PostMapping("/{orderNumber}/confirm")
|
|
1256
|
-
@Operation(summary = "発注の確定")
|
|
1257
|
-
public ResponseEntity<PurchaseOrderResponse> confirmOrder(
|
|
1258
|
-
@PathVariable String orderNumber) {
|
|
1259
|
-
PurchaseOrder order = useCase.confirmOrder(orderNumber);
|
|
1260
|
-
return ResponseEntity.ok(PurchaseOrderResponse.from(order));
|
|
1261
|
-
}
|
|
1262
|
-
|
|
1263
|
-
@DeleteMapping("/{orderNumber}")
|
|
1264
|
-
@Operation(summary = "発注の取消")
|
|
1265
|
-
@ResponseStatus(HttpStatus.NO_CONTENT)
|
|
1266
|
-
public void cancelOrder(@PathVariable String orderNumber) {
|
|
1267
|
-
useCase.cancelOrder(orderNumber);
|
|
1268
|
-
}
|
|
1269
|
-
}
|
|
1270
|
-
```
|
|
1271
|
-
|
|
1272
|
-
</details>
|
|
1273
|
-
|
|
1274
|
-
### 32.4.2 在庫照会 API
|
|
1275
|
-
|
|
1276
|
-
#### Input Port
|
|
1277
|
-
|
|
1278
|
-
<details>
|
|
1279
|
-
<summary>InventoryUseCase.java</summary>
|
|
1280
|
-
|
|
1281
|
-
```java
|
|
1282
|
-
package com.example.pms.application.port.in;
|
|
1283
|
-
|
|
1284
|
-
import com.example.pms.domain.model.inventory.Stock;
|
|
1285
|
-
import lombok.Builder;
|
|
1286
|
-
import lombok.Value;
|
|
1287
|
-
|
|
1288
|
-
import java.math.BigDecimal;
|
|
1289
|
-
import java.util.List;
|
|
1290
|
-
|
|
1291
|
-
/**
|
|
1292
|
-
* 在庫ユースケース(Input Port).
|
|
1293
|
-
*/
|
|
1294
|
-
public interface InventoryUseCase {
|
|
1295
|
-
|
|
1296
|
-
/**
|
|
1297
|
-
* 在庫一覧を取得
|
|
1298
|
-
*/
|
|
1299
|
-
List<Stock> getInventory(InventoryQuery query);
|
|
1300
|
-
|
|
1301
|
-
/**
|
|
1302
|
-
* 在庫サマリーを取得
|
|
1303
|
-
*/
|
|
1304
|
-
List<InventorySummary> getInventorySummary();
|
|
1305
|
-
|
|
1306
|
-
/**
|
|
1307
|
-
* 在庫不足品目を取得
|
|
1308
|
-
*/
|
|
1309
|
-
List<InventorySummary> getShortageItems();
|
|
1310
|
-
}
|
|
1311
|
-
|
|
1312
|
-
@Value
|
|
1313
|
-
@Builder
|
|
1314
|
-
class InventoryQuery {
|
|
1315
|
-
String itemCode;
|
|
1316
|
-
String locationCode;
|
|
1317
|
-
String stockStatus;
|
|
1318
|
-
}
|
|
1319
|
-
|
|
1320
|
-
@Value
|
|
1321
|
-
@Builder
|
|
1322
|
-
class InventorySummary {
|
|
1323
|
-
String itemCode;
|
|
1324
|
-
String itemName;
|
|
1325
|
-
BigDecimal totalQuantity;
|
|
1326
|
-
Integer safetyStock;
|
|
1327
|
-
StockState stockState;
|
|
1328
|
-
|
|
1329
|
-
public enum StockState {
|
|
1330
|
-
NORMAL, SHORTAGE, EXCESS
|
|
1331
|
-
}
|
|
1332
|
-
}
|
|
1333
|
-
```
|
|
1334
|
-
|
|
1335
|
-
</details>
|
|
1336
|
-
|
|
1337
|
-
#### Controller
|
|
1338
|
-
|
|
1339
|
-
<details>
|
|
1340
|
-
<summary>InventoryController.java</summary>
|
|
1341
|
-
|
|
1342
|
-
```java
|
|
1343
|
-
package com.example.pms.infrastructure.in.rest;
|
|
1344
|
-
|
|
1345
|
-
import com.example.pms.application.port.in.InventoryQuery;
|
|
1346
|
-
import com.example.pms.application.port.in.InventorySummary;
|
|
1347
|
-
import com.example.pms.application.port.in.InventoryUseCase;
|
|
1348
|
-
import com.example.pms.domain.model.inventory.Stock;
|
|
1349
|
-
import com.example.pms.infrastructure.in.rest.dto.StockResponse;
|
|
1350
|
-
import io.swagger.v3.oas.annotations.Operation;
|
|
1351
|
-
import io.swagger.v3.oas.annotations.Parameter;
|
|
1352
|
-
import io.swagger.v3.oas.annotations.tags.Tag;
|
|
1353
|
-
import org.springframework.http.ResponseEntity;
|
|
1354
|
-
import org.springframework.web.bind.annotation.*;
|
|
1355
|
-
|
|
1356
|
-
import java.util.List;
|
|
1357
|
-
|
|
1358
|
-
/**
|
|
1359
|
-
* 在庫 API Controller.
|
|
1360
|
-
*/
|
|
1361
|
-
@RestController
|
|
1362
|
-
@RequestMapping("/api/inventory")
|
|
1363
|
-
@Tag(name = "inventory", description = "在庫 API")
|
|
1364
|
-
public class InventoryController {
|
|
1365
|
-
|
|
1366
|
-
private final InventoryUseCase useCase;
|
|
1367
|
-
|
|
1368
|
-
public InventoryController(InventoryUseCase useCase) {
|
|
1369
|
-
this.useCase = useCase;
|
|
1370
|
-
}
|
|
1371
|
-
|
|
1372
|
-
@GetMapping
|
|
1373
|
-
@Operation(summary = "在庫一覧の取得")
|
|
1374
|
-
public ResponseEntity<List<StockResponse>> getInventory(
|
|
1375
|
-
@Parameter(description = "品目コード")
|
|
1376
|
-
@RequestParam(required = false) String itemCode,
|
|
1377
|
-
@Parameter(description = "場所コード")
|
|
1378
|
-
@RequestParam(required = false) String locationCode,
|
|
1379
|
-
@Parameter(description = "在庫状態")
|
|
1380
|
-
@RequestParam(required = false) String status) {
|
|
1381
|
-
|
|
1382
|
-
InventoryQuery query = InventoryQuery.builder()
|
|
1383
|
-
.itemCode(itemCode)
|
|
1384
|
-
.locationCode(locationCode)
|
|
1385
|
-
.stockStatus(status)
|
|
1386
|
-
.build();
|
|
1387
|
-
|
|
1388
|
-
List<Stock> stocks = useCase.getInventory(query);
|
|
1389
|
-
return ResponseEntity.ok(stocks.stream()
|
|
1390
|
-
.map(StockResponse::from)
|
|
1391
|
-
.toList());
|
|
1392
|
-
}
|
|
1393
|
-
|
|
1394
|
-
@GetMapping("/summary")
|
|
1395
|
-
@Operation(summary = "在庫サマリーの取得")
|
|
1396
|
-
public ResponseEntity<List<InventorySummary>> getInventorySummary() {
|
|
1397
|
-
List<InventorySummary> summary = useCase.getInventorySummary();
|
|
1398
|
-
return ResponseEntity.ok(summary);
|
|
1399
|
-
}
|
|
1400
|
-
|
|
1401
|
-
@GetMapping("/shortage")
|
|
1402
|
-
@Operation(summary = "在庫不足品目の取得")
|
|
1403
|
-
public ResponseEntity<List<InventorySummary>> getShortageItems() {
|
|
1404
|
-
List<InventorySummary> items = useCase.getShortageItems();
|
|
1405
|
-
return ResponseEntity.ok(items);
|
|
1406
|
-
}
|
|
1407
|
-
}
|
|
1408
|
-
```
|
|
1409
|
-
|
|
1410
|
-
</details>
|
|
1411
|
-
|
|
1412
|
-
### 32.4.3 作業指示 API
|
|
1413
|
-
|
|
1414
|
-
<details>
|
|
1415
|
-
<summary>WorkOrderController.java</summary>
|
|
1416
|
-
|
|
1417
|
-
```java
|
|
1418
|
-
package com.example.pms.infrastructure.in.rest;
|
|
1419
|
-
|
|
1420
|
-
import com.example.pms.application.port.in.WorkOrderUseCase;
|
|
1421
|
-
import com.example.pms.domain.model.process.WorkOrder;
|
|
1422
|
-
import com.example.pms.infrastructure.in.rest.dto.CreateWorkOrderRequest;
|
|
1423
|
-
import com.example.pms.infrastructure.in.rest.dto.WorkOrderResponse;
|
|
1424
|
-
import io.swagger.v3.oas.annotations.Operation;
|
|
1425
|
-
import io.swagger.v3.oas.annotations.tags.Tag;
|
|
1426
|
-
import jakarta.validation.Valid;
|
|
1427
|
-
import org.springframework.http.HttpStatus;
|
|
1428
|
-
import org.springframework.http.ResponseEntity;
|
|
1429
|
-
import org.springframework.web.bind.annotation.*;
|
|
1430
|
-
|
|
1431
|
-
import java.util.List;
|
|
1432
|
-
|
|
1433
|
-
/**
|
|
1434
|
-
* 作業指示 API Controller.
|
|
1435
|
-
*/
|
|
1436
|
-
@RestController
|
|
1437
|
-
@RequestMapping("/api/work-orders")
|
|
1438
|
-
@Tag(name = "work-orders", description = "作業指示 API")
|
|
1439
|
-
public class WorkOrderController {
|
|
1440
|
-
|
|
1441
|
-
private final WorkOrderUseCase useCase;
|
|
1442
|
-
|
|
1443
|
-
public WorkOrderController(WorkOrderUseCase useCase) {
|
|
1444
|
-
this.useCase = useCase;
|
|
1445
|
-
}
|
|
1446
|
-
|
|
1447
|
-
@GetMapping
|
|
1448
|
-
@Operation(summary = "作業指示一覧の取得")
|
|
1449
|
-
public ResponseEntity<List<WorkOrderResponse>> getAllWorkOrders() {
|
|
1450
|
-
List<WorkOrder> workOrders = useCase.getAllWorkOrders();
|
|
1451
|
-
return ResponseEntity.ok(workOrders.stream()
|
|
1452
|
-
.map(WorkOrderResponse::from)
|
|
1453
|
-
.toList());
|
|
1454
|
-
}
|
|
1455
|
-
|
|
1456
|
-
@GetMapping("/{workOrderNumber}")
|
|
1457
|
-
@Operation(summary = "作業指示詳細の取得")
|
|
1458
|
-
public ResponseEntity<WorkOrderResponse> getWorkOrder(
|
|
1459
|
-
@PathVariable String workOrderNumber) {
|
|
1460
|
-
WorkOrder workOrder = useCase.getWorkOrder(workOrderNumber);
|
|
1461
|
-
return ResponseEntity.ok(WorkOrderResponse.from(workOrder));
|
|
1462
|
-
}
|
|
1463
|
-
|
|
1464
|
-
@PostMapping
|
|
1465
|
-
@Operation(summary = "作業指示の登録")
|
|
1466
|
-
public ResponseEntity<WorkOrderResponse> createWorkOrder(
|
|
1467
|
-
@Valid @RequestBody CreateWorkOrderRequest request) {
|
|
1468
|
-
WorkOrder workOrder = useCase.createWorkOrder(request.toCommand());
|
|
1469
|
-
return ResponseEntity.status(HttpStatus.CREATED)
|
|
1470
|
-
.body(WorkOrderResponse.from(workOrder));
|
|
1471
|
-
}
|
|
1472
|
-
}
|
|
1473
|
-
```
|
|
1474
|
-
|
|
1475
|
-
</details>
|
|
1476
|
-
|
|
1477
|
-
### 32.4.4 MRP 実行 API
|
|
1478
|
-
|
|
1479
|
-
#### MRP サービス
|
|
1480
|
-
|
|
1481
|
-
<details>
|
|
1482
|
-
<summary>MrpUseCase.java</summary>
|
|
1483
|
-
|
|
1484
|
-
```java
|
|
1485
|
-
package com.example.pms.application.port.in;
|
|
1486
|
-
|
|
1487
|
-
import lombok.Builder;
|
|
1488
|
-
import lombok.Value;
|
|
1489
|
-
|
|
1490
|
-
import java.math.BigDecimal;
|
|
1491
|
-
import java.time.LocalDate;
|
|
1492
|
-
import java.time.LocalDateTime;
|
|
1493
|
-
import java.util.List;
|
|
1494
|
-
|
|
1495
|
-
/**
|
|
1496
|
-
* MRP ユースケース(Input Port).
|
|
1497
|
-
*/
|
|
1498
|
-
public interface MrpUseCase {
|
|
1499
|
-
|
|
1500
|
-
/**
|
|
1501
|
-
* MRP(所要量展開)を実行
|
|
1502
|
-
*/
|
|
1503
|
-
MrpResult execute(LocalDate startDate, LocalDate endDate);
|
|
1504
|
-
|
|
1505
|
-
@Value
|
|
1506
|
-
@Builder
|
|
1507
|
-
class MrpResult {
|
|
1508
|
-
LocalDateTime executionTime;
|
|
1509
|
-
LocalDate periodStart;
|
|
1510
|
-
LocalDate periodEnd;
|
|
1511
|
-
List<PlannedOrder> plannedOrders;
|
|
1512
|
-
List<ShortageItem> shortageItems;
|
|
1513
|
-
}
|
|
1514
|
-
|
|
1515
|
-
@Value
|
|
1516
|
-
@Builder
|
|
1517
|
-
class PlannedOrder {
|
|
1518
|
-
String itemCode;
|
|
1519
|
-
String itemName;
|
|
1520
|
-
String orderType; // MANUFACTURING or PURCHASE
|
|
1521
|
-
BigDecimal quantity;
|
|
1522
|
-
LocalDate dueDate;
|
|
1523
|
-
}
|
|
1524
|
-
|
|
1525
|
-
@Value
|
|
1526
|
-
@Builder
|
|
1527
|
-
class ShortageItem {
|
|
1528
|
-
String itemCode;
|
|
1529
|
-
String itemName;
|
|
1530
|
-
BigDecimal shortageQuantity;
|
|
1531
|
-
LocalDate recommendedOrderDate;
|
|
1532
|
-
}
|
|
1533
|
-
}
|
|
1534
|
-
```
|
|
1535
|
-
|
|
1536
|
-
</details>
|
|
1537
|
-
|
|
1538
|
-
#### MRP Controller
|
|
1539
|
-
|
|
1540
|
-
<details>
|
|
1541
|
-
<summary>MrpController.java</summary>
|
|
1542
|
-
|
|
1543
|
-
```java
|
|
1544
|
-
package com.example.pms.infrastructure.in.rest;
|
|
1545
|
-
|
|
1546
|
-
import com.example.pms.application.port.in.MrpUseCase;
|
|
1547
|
-
import com.example.pms.infrastructure.in.rest.dto.ExecuteMrpRequest;
|
|
1548
|
-
import com.example.pms.infrastructure.in.rest.dto.MrpResultResponse;
|
|
1549
|
-
import io.swagger.v3.oas.annotations.Operation;
|
|
1550
|
-
import io.swagger.v3.oas.annotations.tags.Tag;
|
|
1551
|
-
import jakarta.validation.Valid;
|
|
1552
|
-
import org.springframework.http.ResponseEntity;
|
|
1553
|
-
import org.springframework.web.bind.annotation.*;
|
|
1554
|
-
|
|
1555
|
-
/**
|
|
1556
|
-
* MRP API Controller.
|
|
1557
|
-
*/
|
|
1558
|
-
@RestController
|
|
1559
|
-
@RequestMapping("/api/mrp")
|
|
1560
|
-
@Tag(name = "mrp", description = "MRP API")
|
|
1561
|
-
public class MrpController {
|
|
1562
|
-
|
|
1563
|
-
private final MrpUseCase mrpUseCase;
|
|
1564
|
-
|
|
1565
|
-
public MrpController(MrpUseCase mrpUseCase) {
|
|
1566
|
-
this.mrpUseCase = mrpUseCase;
|
|
1567
|
-
}
|
|
1568
|
-
|
|
1569
|
-
@PostMapping("/execute")
|
|
1570
|
-
@Operation(
|
|
1571
|
-
summary = "MRP の実行",
|
|
1572
|
-
description = "指定期間の所要量展開を実行し、計画オーダを生成します"
|
|
1573
|
-
)
|
|
1574
|
-
public ResponseEntity<MrpResultResponse> execute(
|
|
1575
|
-
@Valid @RequestBody ExecuteMrpRequest request) {
|
|
1576
|
-
var result = mrpUseCase.execute(
|
|
1577
|
-
request.getStartDate(),
|
|
1578
|
-
request.getEndDate()
|
|
1579
|
-
);
|
|
1580
|
-
return ResponseEntity.ok(MrpResultResponse.from(result));
|
|
1581
|
-
}
|
|
1582
|
-
}
|
|
1583
|
-
```
|
|
1584
|
-
|
|
1585
|
-
</details>
|
|
1586
|
-
|
|
1587
|
-
---
|
|
1588
|
-
|
|
1589
|
-
## 32.5 エラーハンドリング
|
|
1590
|
-
|
|
1591
|
-
### 32.5.1 ドメイン例外の定義
|
|
1592
|
-
|
|
1593
|
-
<details>
|
|
1594
|
-
<summary>DomainException.java</summary>
|
|
1595
|
-
|
|
1596
|
-
```java
|
|
1597
|
-
package com.example.pms.domain.exception;
|
|
1598
|
-
|
|
1599
|
-
/**
|
|
1600
|
-
* ドメイン例外の基底クラス.
|
|
1601
|
-
*/
|
|
1602
|
-
public abstract class DomainException extends RuntimeException {
|
|
1603
|
-
|
|
1604
|
-
protected DomainException(String message) {
|
|
1605
|
-
super(message);
|
|
1606
|
-
}
|
|
1607
|
-
|
|
1608
|
-
protected DomainException(String message, Throwable cause) {
|
|
1609
|
-
super(message, cause);
|
|
1610
|
-
}
|
|
1611
|
-
}
|
|
1612
|
-
```
|
|
1613
|
-
|
|
1614
|
-
</details>
|
|
1615
|
-
|
|
1616
|
-
<details>
|
|
1617
|
-
<summary>ItemNotFoundException.java</summary>
|
|
1618
|
-
|
|
1619
|
-
```java
|
|
1620
|
-
package com.example.pms.domain.exception;
|
|
1621
|
-
|
|
1622
|
-
/**
|
|
1623
|
-
* 品目が見つからない例外.
|
|
1624
|
-
*/
|
|
1625
|
-
public class ItemNotFoundException extends DomainException {
|
|
1626
|
-
|
|
1627
|
-
public ItemNotFoundException(String itemCode) {
|
|
1628
|
-
super("品目が見つかりません: " + itemCode);
|
|
1629
|
-
}
|
|
1630
|
-
}
|
|
1631
|
-
```
|
|
1632
|
-
|
|
1633
|
-
</details>
|
|
1634
|
-
|
|
1635
|
-
<details>
|
|
1636
|
-
<summary>DuplicateItemException.java</summary>
|
|
1637
|
-
|
|
1638
|
-
```java
|
|
1639
|
-
package com.example.pms.domain.exception;
|
|
1640
|
-
|
|
1641
|
-
/**
|
|
1642
|
-
* 品目コード重複例外.
|
|
1643
|
-
*/
|
|
1644
|
-
public class DuplicateItemException extends DomainException {
|
|
1645
|
-
|
|
1646
|
-
public DuplicateItemException(String itemCode) {
|
|
1647
|
-
super("品目コードが既に存在します: " + itemCode);
|
|
1648
|
-
}
|
|
1649
|
-
}
|
|
1650
|
-
```
|
|
1651
|
-
|
|
1652
|
-
</details>
|
|
1653
|
-
|
|
1654
|
-
<details>
|
|
1655
|
-
<summary>InsufficientInventoryException.java</summary>
|
|
1656
|
-
|
|
1657
|
-
```java
|
|
1658
|
-
package com.example.pms.domain.exception;
|
|
1659
|
-
|
|
1660
|
-
import java.math.BigDecimal;
|
|
1661
|
-
|
|
1662
|
-
/**
|
|
1663
|
-
* 在庫不足例外.
|
|
1664
|
-
*/
|
|
1665
|
-
public class InsufficientInventoryException extends DomainException {
|
|
1666
|
-
|
|
1667
|
-
public InsufficientInventoryException(String itemCode, BigDecimal required, BigDecimal available) {
|
|
1668
|
-
super(String.format("在庫が不足しています: %s (必要: %s, 有効: %s)",
|
|
1669
|
-
itemCode, required, available));
|
|
1670
|
-
}
|
|
1671
|
-
}
|
|
1672
|
-
```
|
|
1673
|
-
|
|
1674
|
-
</details>
|
|
1675
|
-
|
|
1676
|
-
### 32.5.2 グローバル例外ハンドラ
|
|
1677
|
-
|
|
1678
|
-
<details>
|
|
1679
|
-
<summary>GlobalExceptionHandler.java</summary>
|
|
1680
|
-
|
|
1681
|
-
```java
|
|
1682
|
-
package com.example.pms.infrastructure.in.web.exception;
|
|
1683
|
-
|
|
1684
|
-
import com.example.pms.domain.exception.DomainException;
|
|
1685
|
-
import com.example.pms.domain.exception.DuplicateItemException;
|
|
1686
|
-
import com.example.pms.domain.exception.ItemNotFoundException;
|
|
1687
|
-
import org.springframework.http.HttpStatus;
|
|
1688
|
-
import org.springframework.http.ProblemDetail;
|
|
1689
|
-
import org.springframework.web.bind.MethodArgumentNotValidException;
|
|
1690
|
-
import org.springframework.web.bind.annotation.ExceptionHandler;
|
|
1691
|
-
import org.springframework.web.bind.annotation.RestControllerAdvice;
|
|
1692
|
-
|
|
1693
|
-
import java.net.URI;
|
|
1694
|
-
import java.time.Instant;
|
|
1695
|
-
import java.util.stream.Collectors;
|
|
1696
|
-
|
|
1697
|
-
/**
|
|
1698
|
-
* グローバル例外ハンドラ.
|
|
1699
|
-
*/
|
|
1700
|
-
@RestControllerAdvice
|
|
1701
|
-
public class GlobalExceptionHandler {
|
|
1702
|
-
|
|
1703
|
-
@ExceptionHandler(ItemNotFoundException.class)
|
|
1704
|
-
public ProblemDetail handleItemNotFoundException(ItemNotFoundException ex) {
|
|
1705
|
-
ProblemDetail problem = ProblemDetail.forStatusAndDetail(
|
|
1706
|
-
HttpStatus.NOT_FOUND, ex.getMessage());
|
|
1707
|
-
problem.setTitle("品目が見つかりません");
|
|
1708
|
-
problem.setType(URI.create("https://api.example.com/errors/item-not-found"));
|
|
1709
|
-
problem.setProperty("timestamp", Instant.now());
|
|
1710
|
-
return problem;
|
|
1711
|
-
}
|
|
1712
|
-
|
|
1713
|
-
@ExceptionHandler(DuplicateItemException.class)
|
|
1714
|
-
public ProblemDetail handleDuplicateItemException(DuplicateItemException ex) {
|
|
1715
|
-
ProblemDetail problem = ProblemDetail.forStatusAndDetail(
|
|
1716
|
-
HttpStatus.CONFLICT, ex.getMessage());
|
|
1717
|
-
problem.setTitle("品目コード重複");
|
|
1718
|
-
problem.setType(URI.create("https://api.example.com/errors/duplicate-item"));
|
|
1719
|
-
problem.setProperty("timestamp", Instant.now());
|
|
1720
|
-
return problem;
|
|
1721
|
-
}
|
|
1722
|
-
|
|
1723
|
-
@ExceptionHandler(MethodArgumentNotValidException.class)
|
|
1724
|
-
public ProblemDetail handleValidationException(MethodArgumentNotValidException ex) {
|
|
1725
|
-
String errors = ex.getBindingResult().getFieldErrors().stream()
|
|
1726
|
-
.map(e -> e.getField() + ": " + e.getDefaultMessage())
|
|
1727
|
-
.collect(Collectors.joining(", "));
|
|
1728
|
-
|
|
1729
|
-
ProblemDetail problem = ProblemDetail.forStatusAndDetail(
|
|
1730
|
-
HttpStatus.BAD_REQUEST, errors);
|
|
1731
|
-
problem.setTitle("入力値が不正です");
|
|
1732
|
-
problem.setType(URI.create("https://api.example.com/errors/validation-error"));
|
|
1733
|
-
problem.setProperty("timestamp", Instant.now());
|
|
1734
|
-
return problem;
|
|
1735
|
-
}
|
|
1736
|
-
|
|
1737
|
-
@ExceptionHandler(DomainException.class)
|
|
1738
|
-
public ProblemDetail handleDomainException(DomainException ex) {
|
|
1739
|
-
ProblemDetail problem = ProblemDetail.forStatusAndDetail(
|
|
1740
|
-
HttpStatus.BAD_REQUEST, ex.getMessage());
|
|
1741
|
-
problem.setTitle("ドメインエラー");
|
|
1742
|
-
problem.setType(URI.create("https://api.example.com/errors/domain-error"));
|
|
1743
|
-
problem.setProperty("timestamp", Instant.now());
|
|
1744
|
-
return problem;
|
|
1745
|
-
}
|
|
1746
|
-
|
|
1747
|
-
@ExceptionHandler(Exception.class)
|
|
1748
|
-
public ProblemDetail handleGenericException(Exception ex) {
|
|
1749
|
-
ProblemDetail problem = ProblemDetail.forStatusAndDetail(
|
|
1750
|
-
HttpStatus.INTERNAL_SERVER_ERROR, "予期しないエラーが発生しました");
|
|
1751
|
-
problem.setTitle("内部エラー");
|
|
1752
|
-
problem.setType(URI.create("https://api.example.com/errors/internal-error"));
|
|
1753
|
-
problem.setProperty("timestamp", Instant.now());
|
|
1754
|
-
return problem;
|
|
1755
|
-
}
|
|
1756
|
-
}
|
|
1757
|
-
```
|
|
1758
|
-
|
|
1759
|
-
</details>
|
|
1760
|
-
|
|
1761
|
-
### 32.5.3 エラーレスポンス形式
|
|
1762
|
-
|
|
1763
|
-
RFC 7807(Problem Details for HTTP APIs)に準拠したエラーレスポンスを返します。
|
|
1764
|
-
|
|
1765
|
-
```json
|
|
1766
|
-
{
|
|
1767
|
-
"type": "https://api.example.com/errors/item-not-found",
|
|
1768
|
-
"title": "品目が見つかりません",
|
|
1769
|
-
"status": 404,
|
|
1770
|
-
"detail": "品目が見つかりません: ITEM001",
|
|
1771
|
-
"instance": "/api/items/ITEM001",
|
|
1772
|
-
"timestamp": "2024-01-15T10:30:00Z"
|
|
1773
|
-
}
|
|
1774
|
-
```
|
|
1775
|
-
|
|
1776
|
-
| フィールド | 説明 |
|
|
1777
|
-
|-----------|------|
|
|
1778
|
-
| type | エラー種別を識別する URI |
|
|
1779
|
-
| title | エラーの概要 |
|
|
1780
|
-
| status | HTTP ステータスコード |
|
|
1781
|
-
| detail | エラーの詳細説明 |
|
|
1782
|
-
| instance | エラーが発生したリソースの URI |
|
|
1783
|
-
| timestamp | エラー発生時刻 |
|
|
1784
|
-
|
|
1785
|
-
---
|
|
1786
|
-
|
|
1787
|
-
## 32.6 API ドキュメント
|
|
1788
|
-
|
|
1789
|
-
### OpenAPI / Swagger UI
|
|
1790
|
-
|
|
1791
|
-
サーバー起動後、以下の URL で Swagger UI を確認できます:
|
|
1792
|
-
|
|
1793
|
-
```
|
|
1794
|
-
http://localhost:8080/swagger-ui.html
|
|
1795
|
-
```
|
|
1796
|
-
|
|
1797
|
-
### API 一覧
|
|
1798
|
-
|
|
1799
|
-
| カテゴリ | エンドポイント | メソッド | 説明 |
|
|
1800
|
-
|---------|---------------|----------|------|
|
|
1801
|
-
| **品目** | /api/items | GET | 品目一覧の取得 |
|
|
1802
|
-
| | /api/items/{itemCode} | GET | 品目の取得 |
|
|
1803
|
-
| | /api/items | POST | 品目の登録 |
|
|
1804
|
-
| | /api/items/{itemCode} | PUT | 品目の更新 |
|
|
1805
|
-
| | /api/items/{itemCode} | DELETE | 品目の削除 |
|
|
1806
|
-
| **BOM** | /api/bom/{itemCode}/explode | GET | 部品展開 |
|
|
1807
|
-
| | /api/bom/{itemCode}/where-used | GET | 使用先照会 |
|
|
1808
|
-
| **発注** | /api/purchase-orders | GET | 発注一覧の取得 |
|
|
1809
|
-
| | /api/purchase-orders/{orderNumber} | GET | 発注詳細の取得 |
|
|
1810
|
-
| | /api/purchase-orders | POST | 発注の登録 |
|
|
1811
|
-
| | /api/purchase-orders/{orderNumber}/confirm | POST | 発注の確定 |
|
|
1812
|
-
| | /api/purchase-orders/{orderNumber} | DELETE | 発注の取消 |
|
|
1813
|
-
| **在庫** | /api/inventory | GET | 在庫一覧の取得 |
|
|
1814
|
-
| | /api/inventory/summary | GET | 在庫サマリーの取得 |
|
|
1815
|
-
| | /api/inventory/shortage | GET | 在庫不足品目の取得 |
|
|
1816
|
-
| **作業指示** | /api/work-orders | GET | 作業指示一覧の取得 |
|
|
1817
|
-
| | /api/work-orders/{workOrderNumber} | GET | 作業指示詳細の取得 |
|
|
1818
|
-
| | /api/work-orders | POST | 作業指示の登録 |
|
|
1819
|
-
| | /api/work-orders/{workOrderNumber}/completion | POST | 完成実績の登録 |
|
|
1820
|
-
| | /api/work-orders/{workOrderNumber}/progress | PATCH | 作業進捗の更新 |
|
|
1821
|
-
| **MRP** | /api/mrp/execute | POST | MRP の実行 |
|
|
1822
|
-
| | /api/mrp/results | GET | MRP 実行結果の照会 |
|
|
1823
|
-
| | /api/mrp/planned-orders | GET | 計画オーダ一覧の取得 |
|
|
1824
|
-
|
|
1825
|
-
---
|
|
1826
|
-
|
|
1827
|
-
## 32.7 API インテグレーションテスト
|
|
1828
|
-
|
|
1829
|
-
### RestClient を使用した API インテグレーションテスト
|
|
1830
|
-
|
|
1831
|
-
本プロジェクトでは、MockMvc の代わりに RestClient を使用した実際の HTTP リクエストベースのテストを採用しています。
|
|
1832
|
-
|
|
1833
|
-
<details>
|
|
1834
|
-
<summary>IntegrationTestBase.java</summary>
|
|
1835
|
-
|
|
1836
|
-
```java
|
|
1837
|
-
package com.example.pms.integration;
|
|
1838
|
-
|
|
1839
|
-
import com.example.pms.TestcontainersConfiguration;
|
|
1840
|
-
import org.springframework.beans.factory.annotation.Autowired;
|
|
1841
|
-
import org.springframework.boot.test.context.SpringBootTest;
|
|
1842
|
-
import org.springframework.boot.test.web.server.LocalServerPort;
|
|
1843
|
-
import org.springframework.context.annotation.Import;
|
|
1844
|
-
import org.springframework.http.MediaType;
|
|
1845
|
-
import org.springframework.jdbc.core.JdbcTemplate;
|
|
1846
|
-
import org.springframework.test.context.ActiveProfiles;
|
|
1847
|
-
import org.springframework.web.client.RestClient;
|
|
1848
|
-
|
|
1849
|
-
/**
|
|
1850
|
-
* API インテグレーションテストの基底クラス.
|
|
1851
|
-
* TestContainers を使用して PostgreSQL コンテナを起動し、
|
|
1852
|
-
* 実際の HTTP リクエストでテストを実行する。
|
|
1853
|
-
*/
|
|
1854
|
-
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)
|
|
1855
|
-
@Import(TestcontainersConfiguration.class)
|
|
1856
|
-
@ActiveProfiles("test")
|
|
1857
|
-
public abstract class IntegrationTestBase {
|
|
1858
|
-
|
|
1859
|
-
@LocalServerPort
|
|
1860
|
-
protected int port;
|
|
1861
|
-
|
|
1862
|
-
@Autowired
|
|
1863
|
-
protected JdbcTemplate jdbcTemplate;
|
|
1864
|
-
|
|
1865
|
-
private RestClient restClient;
|
|
1866
|
-
|
|
1867
|
-
/**
|
|
1868
|
-
* REST クライアントを取得.
|
|
1869
|
-
*/
|
|
1870
|
-
protected RestClient getRestClient() {
|
|
1871
|
-
if (restClient == null) {
|
|
1872
|
-
restClient = RestClient.builder()
|
|
1873
|
-
.baseUrl("http://localhost:" + port)
|
|
1874
|
-
.defaultHeader("Content-Type", MediaType.APPLICATION_JSON_VALUE)
|
|
1875
|
-
.defaultHeader("Accept", MediaType.APPLICATION_JSON_VALUE)
|
|
1876
|
-
.build();
|
|
1877
|
-
}
|
|
1878
|
-
return restClient;
|
|
1879
|
-
}
|
|
1880
|
-
|
|
1881
|
-
protected void cleanupItemData() {
|
|
1882
|
-
jdbcTemplate.execute("DELETE FROM \"品目マスタ\" WHERE \"品目コード\" LIKE 'TEST%'");
|
|
1883
|
-
}
|
|
1884
|
-
}
|
|
1885
|
-
```
|
|
1886
|
-
|
|
1887
|
-
</details>
|
|
1888
|
-
|
|
1889
|
-
<details>
|
|
1890
|
-
<summary>ItemApiIntegrationTest.java</summary>
|
|
1891
|
-
|
|
1892
|
-
```java
|
|
1893
|
-
package com.example.pms.integration;
|
|
1894
|
-
|
|
1895
|
-
import static org.assertj.core.api.Assertions.assertThat;
|
|
1896
|
-
import static org.assertj.core.api.Assertions.assertThatThrownBy;
|
|
1897
|
-
|
|
1898
|
-
import com.example.pms.domain.model.item.ItemCategory;
|
|
1899
|
-
import com.example.pms.infrastructure.in.rest.dto.CreateItemRequest;
|
|
1900
|
-
import com.example.pms.infrastructure.in.rest.dto.ItemResponse;
|
|
1901
|
-
import com.example.pms.infrastructure.in.rest.dto.UpdateItemRequest;
|
|
1902
|
-
import org.junit.jupiter.api.AfterEach;
|
|
1903
|
-
import org.junit.jupiter.api.BeforeEach;
|
|
1904
|
-
import org.junit.jupiter.api.DisplayName;
|
|
1905
|
-
import org.junit.jupiter.api.Nested;
|
|
1906
|
-
import org.junit.jupiter.api.Test;
|
|
1907
|
-
import org.springframework.http.HttpStatus;
|
|
1908
|
-
import org.springframework.http.MediaType;
|
|
1909
|
-
import org.springframework.web.client.HttpClientErrorException;
|
|
1910
|
-
|
|
1911
|
-
@DisplayName("品目 API 統合テスト")
|
|
1912
|
-
class ItemApiIntegrationTest extends IntegrationTestBase {
|
|
1913
|
-
|
|
1914
|
-
private static final String API_PATH = "/api/items";
|
|
1915
|
-
|
|
1916
|
-
@BeforeEach
|
|
1917
|
-
void setUp() {
|
|
1918
|
-
createUnit("個", "個", "個");
|
|
1919
|
-
cleanupItemData();
|
|
1920
|
-
}
|
|
1921
|
-
|
|
1922
|
-
@AfterEach
|
|
1923
|
-
void tearDown() {
|
|
1924
|
-
cleanupItemData();
|
|
1925
|
-
}
|
|
1926
|
-
|
|
1927
|
-
@Nested
|
|
1928
|
-
@DisplayName("品目登録・取得フロー")
|
|
1929
|
-
class ItemCrudFlow {
|
|
1930
|
-
|
|
1931
|
-
@Test
|
|
1932
|
-
@DisplayName("品目を登録して取得できる")
|
|
1933
|
-
void shouldCreateAndRetrieveItem() {
|
|
1934
|
-
CreateItemRequest createRequest = new CreateItemRequest();
|
|
1935
|
-
createRequest.setItemCode("TEST001");
|
|
1936
|
-
createRequest.setItemName("テスト品目");
|
|
1937
|
-
createRequest.setItemCategory(ItemCategory.PRODUCT);
|
|
1938
|
-
createRequest.setUnitCode("個");
|
|
1939
|
-
|
|
1940
|
-
// 品目を登録
|
|
1941
|
-
ItemResponse createResponse = getRestClient()
|
|
1942
|
-
.post()
|
|
1943
|
-
.uri(API_PATH)
|
|
1944
|
-
.contentType(MediaType.APPLICATION_JSON)
|
|
1945
|
-
.body(createRequest)
|
|
1946
|
-
.retrieve()
|
|
1947
|
-
.body(ItemResponse.class);
|
|
1948
|
-
|
|
1949
|
-
assertThat(createResponse).isNotNull();
|
|
1950
|
-
assertThat(createResponse.getItemCode()).isEqualTo("TEST001");
|
|
1951
|
-
|
|
1952
|
-
// 登録した品目を取得
|
|
1953
|
-
ItemResponse getResponse = getRestClient()
|
|
1954
|
-
.get()
|
|
1955
|
-
.uri(API_PATH + "/TEST001")
|
|
1956
|
-
.retrieve()
|
|
1957
|
-
.body(ItemResponse.class);
|
|
1958
|
-
|
|
1959
|
-
assertThat(getResponse).isNotNull();
|
|
1960
|
-
assertThat(getResponse.getItemCode()).isEqualTo("TEST001");
|
|
1961
|
-
assertThat(getResponse.getItemName()).isEqualTo("テスト品目");
|
|
1962
|
-
}
|
|
1963
|
-
|
|
1964
|
-
@Test
|
|
1965
|
-
@DisplayName("存在しない品目を取得すると404エラー")
|
|
1966
|
-
void shouldReturn404WhenItemNotFound() {
|
|
1967
|
-
assertThatThrownBy(() ->
|
|
1968
|
-
getRestClient()
|
|
1969
|
-
.get()
|
|
1970
|
-
.uri(API_PATH + "/NOT-EXIST")
|
|
1971
|
-
.retrieve()
|
|
1972
|
-
.body(ItemResponse.class)
|
|
1973
|
-
).isInstanceOf(HttpClientErrorException.class)
|
|
1974
|
-
.satisfies(ex -> {
|
|
1975
|
-
HttpClientErrorException httpEx = (HttpClientErrorException) ex;
|
|
1976
|
-
assertThat(httpEx.getStatusCode()).isEqualTo(HttpStatus.NOT_FOUND);
|
|
1977
|
-
});
|
|
1978
|
-
}
|
|
1979
|
-
|
|
1980
|
-
@Test
|
|
1981
|
-
@DisplayName("重複した品目コードで登録すると409エラー")
|
|
1982
|
-
void shouldReturn409WhenDuplicateItemCode() {
|
|
1983
|
-
CreateItemRequest createRequest = new CreateItemRequest();
|
|
1984
|
-
createRequest.setItemCode("TEST002");
|
|
1985
|
-
createRequest.setItemName("テスト品目2");
|
|
1986
|
-
createRequest.setItemCategory(ItemCategory.PRODUCT);
|
|
1987
|
-
createRequest.setUnitCode("個");
|
|
1988
|
-
|
|
1989
|
-
// 1回目は成功
|
|
1990
|
-
getRestClient()
|
|
1991
|
-
.post()
|
|
1992
|
-
.uri(API_PATH)
|
|
1993
|
-
.contentType(MediaType.APPLICATION_JSON)
|
|
1994
|
-
.body(createRequest)
|
|
1995
|
-
.retrieve()
|
|
1996
|
-
.body(ItemResponse.class);
|
|
1997
|
-
|
|
1998
|
-
// 2回目は409
|
|
1999
|
-
assertThatThrownBy(() ->
|
|
2000
|
-
getRestClient()
|
|
2001
|
-
.post()
|
|
2002
|
-
.uri(API_PATH)
|
|
2003
|
-
.contentType(MediaType.APPLICATION_JSON)
|
|
2004
|
-
.body(createRequest)
|
|
2005
|
-
.retrieve()
|
|
2006
|
-
.body(ItemResponse.class)
|
|
2007
|
-
).isInstanceOf(HttpClientErrorException.class)
|
|
2008
|
-
.satisfies(ex -> {
|
|
2009
|
-
HttpClientErrorException httpEx = (HttpClientErrorException) ex;
|
|
2010
|
-
assertThat(httpEx.getStatusCode()).isEqualTo(HttpStatus.CONFLICT);
|
|
2011
|
-
});
|
|
2012
|
-
}
|
|
2013
|
-
}
|
|
2014
|
-
}
|
|
2015
|
-
```
|
|
2016
|
-
|
|
2017
|
-
</details>
|
|
2018
|
-
|
|
2019
|
-
### テスト用初期化スクリプト
|
|
2020
|
-
|
|
2021
|
-
<details>
|
|
2022
|
-
<summary>init.sql</summary>
|
|
2023
|
-
|
|
2024
|
-
```sql
|
|
2025
|
-
-- テスト用テーブル作成
|
|
2026
|
-
CREATE TABLE 品目マスタ (
|
|
2027
|
-
品目コード VARCHAR(20) PRIMARY KEY,
|
|
2028
|
-
品目名 VARCHAR(100) NOT NULL,
|
|
2029
|
-
品目種別 VARCHAR(20),
|
|
2030
|
-
単位 VARCHAR(10),
|
|
2031
|
-
標準原価 DECIMAL(15,2),
|
|
2032
|
-
安全在庫 INTEGER,
|
|
2033
|
-
リードタイム INTEGER
|
|
2034
|
-
);
|
|
2035
|
-
|
|
2036
|
-
CREATE TABLE 部品表 (
|
|
2037
|
-
親品目コード VARCHAR(20) NOT NULL,
|
|
2038
|
-
子品目コード VARCHAR(20) NOT NULL,
|
|
2039
|
-
員数 DECIMAL(10,4) NOT NULL,
|
|
2040
|
-
PRIMARY KEY (親品目コード, 子品目コード)
|
|
2041
|
-
);
|
|
2042
|
-
|
|
2043
|
-
CREATE TABLE 発注 (
|
|
2044
|
-
発注番号 VARCHAR(20) PRIMARY KEY,
|
|
2045
|
-
仕入先コード VARCHAR(20) NOT NULL,
|
|
2046
|
-
発注日 DATE NOT NULL,
|
|
2047
|
-
ステータス VARCHAR(10) DEFAULT '作成中'
|
|
2048
|
-
);
|
|
2049
|
-
|
|
2050
|
-
CREATE TABLE 発注明細 (
|
|
2051
|
-
発注番号 VARCHAR(20) NOT NULL,
|
|
2052
|
-
明細番号 INTEGER NOT NULL,
|
|
2053
|
-
品目コード VARCHAR(20) NOT NULL,
|
|
2054
|
-
発注数量 DECIMAL(10,2) NOT NULL,
|
|
2055
|
-
単価 DECIMAL(15,2),
|
|
2056
|
-
納期 DATE,
|
|
2057
|
-
PRIMARY KEY (発注番号, 明細番号)
|
|
2058
|
-
);
|
|
2059
|
-
|
|
2060
|
-
CREATE TABLE 在庫 (
|
|
2061
|
-
品目コード VARCHAR(20) NOT NULL,
|
|
2062
|
-
保管場所コード VARCHAR(20) NOT NULL,
|
|
2063
|
-
在庫数量 DECIMAL(10,2) NOT NULL DEFAULT 0,
|
|
2064
|
-
PRIMARY KEY (品目コード, 保管場所コード)
|
|
2065
|
-
);
|
|
2066
|
-
```
|
|
2067
|
-
|
|
2068
|
-
</details>
|
|
2069
|
-
|
|
2070
|
-
---
|
|
2071
|
-
|
|
2072
|
-
## まとめ
|
|
2073
|
-
|
|
2074
|
-
本章では、ヘキサゴナルアーキテクチャを採用した生産管理システムの REST API を実装しました。
|
|
2075
|
-
|
|
2076
|
-
### 実装したコンポーネント
|
|
2077
|
-
|
|
2078
|
-
| レイヤー | コンポーネント | 役割 |
|
|
2079
|
-
|---------|--------------|------|
|
|
2080
|
-
| Domain | Entity, Exception | ビジネスロジック、ドメインルール |
|
|
2081
|
-
| Application | UseCase, Service | ユースケースの実装、トランザクション管理 |
|
|
2082
|
-
| Infrastructure | Controller, Repository | 外部との入出力 |
|
|
2083
|
-
|
|
2084
|
-
### アーキテクチャの特徴
|
|
2085
|
-
|
|
2086
|
-
1. **ヘキサゴナルアーキテクチャ**: ドメインを中心に据えた設計
|
|
2087
|
-
2. **Ports and Adapters**: インターフェースと実装の分離
|
|
2088
|
-
3. **TDD**: テスト駆動で API を開発
|
|
2089
|
-
4. **OpenAPI**: Swagger UI で API 仕様を可視化
|
|
2090
|
-
|
|
2091
|
-
### 技術スタック
|
|
2092
|
-
|
|
2093
|
-
- **Spring Boot 3.2**: REST API フレームワーク
|
|
2094
|
-
- **MyBatis**: O/R マッピング
|
|
2095
|
-
- **Bean Validation**: リクエスト検証
|
|
2096
|
-
- **springdoc-openapi**: API ドキュメント生成
|
|
2097
|
-
- **Testcontainers**: 統合テスト
|
|
2098
|
-
|
|
2099
|
-
---
|
|
2100
|
-
|
|
2101
|
-
[前へ: 第31章 生産管理データ設計(E 社事例)](chapter31.md) | [目次へ戻る](../index.md)
|
|
1
|
+
# 第32章:API サービスの実装
|
|
2
|
+
|
|
3
|
+
## 32.1 ヘキサゴナルアーキテクチャの復習
|
|
4
|
+
|
|
5
|
+
### ヘキサゴナルアーキテクチャとは
|
|
6
|
+
|
|
7
|
+
ヘキサゴナルアーキテクチャ(Ports and Adapters パターン)は、アプリケーションのドメインロジックを外部の技術的詳細から分離するアーキテクチャスタイルです。
|
|
8
|
+
|
|
9
|
+
```plantuml
|
|
10
|
+
@startuml
|
|
11
|
+
skinparam componentStyle uml2
|
|
12
|
+
skinparam component {
|
|
13
|
+
BackgroundColor<<adapter>> LightBlue
|
|
14
|
+
BackgroundColor<<port>> LightGreen
|
|
15
|
+
BackgroundColor<<domain>> LightYellow
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
package "外部" {
|
|
19
|
+
[REST Client] as rest
|
|
20
|
+
[Database] as db
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
package "Application" {
|
|
24
|
+
package "Infrastructure" {
|
|
25
|
+
[REST Controller] <<adapter>> as controller
|
|
26
|
+
[MyBatis Repository] <<adapter>> as mybatis
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
package "Application Service" {
|
|
30
|
+
interface "Input Port\n(UseCase)" <<port>> as input
|
|
31
|
+
interface "Output Port\n(Repository)" <<port>> as output
|
|
32
|
+
[Service] as service
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
package "Domain" <<domain>> {
|
|
36
|
+
[Entity]
|
|
37
|
+
[Value Object]
|
|
38
|
+
[Domain Service]
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
rest --> controller
|
|
43
|
+
controller --> input
|
|
44
|
+
input <|.. service
|
|
45
|
+
service --> output
|
|
46
|
+
output <|.. mybatis
|
|
47
|
+
mybatis --> db
|
|
48
|
+
|
|
49
|
+
service --> Entity
|
|
50
|
+
service --> [Domain Service]
|
|
51
|
+
@enduml
|
|
52
|
+
```
|
|
53
|
+
|
|
54
|
+
### 主要な概念
|
|
55
|
+
|
|
56
|
+
| 概念 | 説明 | 例 |
|
|
57
|
+
|------|------|-----|
|
|
58
|
+
| Input Port | ユースケースを定義するインターフェース | ItemUseCase |
|
|
59
|
+
| Output Port | 外部リソースへのアクセスを定義するインターフェース | ItemRepository |
|
|
60
|
+
| Input Adapter | 外部からの入力を受け付けるコンポーネント | REST Controller |
|
|
61
|
+
| Output Adapter | 外部リソースへの出力を行うコンポーネント | MyBatis Repository |
|
|
62
|
+
| Domain | ビジネスロジックを含む中心部分 | Entity, Value Object |
|
|
63
|
+
|
|
64
|
+
### 依存関係の方向
|
|
65
|
+
|
|
66
|
+
ヘキサゴナルアーキテクチャでは、依存関係は常に外側から内側に向かいます。
|
|
67
|
+
|
|
68
|
+
```plantuml
|
|
69
|
+
@startuml
|
|
70
|
+
skinparam rectangle {
|
|
71
|
+
BackgroundColor<<outer>> LightBlue
|
|
72
|
+
BackgroundColor<<middle>> LightGreen
|
|
73
|
+
BackgroundColor<<inner>> LightYellow
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
rectangle "Infrastructure\n(Adapters)" <<outer>> as infra {
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
rectangle "Application\n(Ports & Services)" <<middle>> as app {
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
rectangle "Domain\n(Entities & Rules)" <<inner>> as domain {
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
infra --> app : 依存
|
|
86
|
+
app --> domain : 依存
|
|
87
|
+
@enduml
|
|
88
|
+
```
|
|
89
|
+
|
|
90
|
+
---
|
|
91
|
+
|
|
92
|
+
## 32.2 アーキテクチャ構造
|
|
93
|
+
|
|
94
|
+
### プロジェクト構造
|
|
95
|
+
|
|
96
|
+
```
|
|
97
|
+
apps/pms/backend/
|
|
98
|
+
├── src/
|
|
99
|
+
│ ├── main/
|
|
100
|
+
│ │ ├── java/
|
|
101
|
+
│ │ │ └── com/example/pms/
|
|
102
|
+
│ │ │ ├── Application.java
|
|
103
|
+
│ │ │ ├── domain/
|
|
104
|
+
│ │ │ │ ├── model/ # エンティティ・値オブジェクト
|
|
105
|
+
│ │ │ │ │ ├── item/
|
|
106
|
+
│ │ │ │ │ ├── bom/
|
|
107
|
+
│ │ │ │ │ ├── inventory/
|
|
108
|
+
│ │ │ │ │ ├── purchase/
|
|
109
|
+
│ │ │ │ │ └── process/
|
|
110
|
+
│ │ │ │ ├── service/ # ドメインサービス
|
|
111
|
+
│ │ │ │ └── exception/ # ドメイン例外
|
|
112
|
+
│ │ │ ├── application/
|
|
113
|
+
│ │ │ │ ├── port/
|
|
114
|
+
│ │ │ │ │ ├── in/ # Input Port(ユースケース)
|
|
115
|
+
│ │ │ │ │ │ └── command/ # コマンドオブジェクト
|
|
116
|
+
│ │ │ │ │ └── out/ # Output Port(リポジトリ)
|
|
117
|
+
│ │ │ │ └── service/ # アプリケーションサービス
|
|
118
|
+
│ │ │ └── infrastructure/
|
|
119
|
+
│ │ │ ├── in/
|
|
120
|
+
│ │ │ │ ├── rest/ # REST Controller
|
|
121
|
+
│ │ │ │ │ └── dto/ # Request/Response DTO
|
|
122
|
+
│ │ │ │ └── web/
|
|
123
|
+
│ │ │ │ └── exception/ # 例外ハンドラ
|
|
124
|
+
│ │ │ └── out/
|
|
125
|
+
│ │ │ └── persistence/
|
|
126
|
+
│ │ │ └── repository/ # Repository 実装
|
|
127
|
+
│ │ └── resources/
|
|
128
|
+
│ │ ├── mapper/ # MyBatis マッパー XML
|
|
129
|
+
│ │ └── application.yml
|
|
130
|
+
│ └── test/
|
|
131
|
+
│ └── java/
|
|
132
|
+
│ └── com/example/pms/
|
|
133
|
+
│ └── integration/ # 統合テスト
|
|
134
|
+
└── build.gradle.kts
|
|
135
|
+
```
|
|
136
|
+
|
|
137
|
+
### 技術スタックの導入
|
|
138
|
+
|
|
139
|
+
<details>
|
|
140
|
+
<summary>build.gradle.kts</summary>
|
|
141
|
+
|
|
142
|
+
```kotlin
|
|
143
|
+
plugins {
|
|
144
|
+
java
|
|
145
|
+
id("org.springframework.boot") version "3.2.0"
|
|
146
|
+
id("io.spring.dependency-management") version "1.1.4"
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
group = "com.example"
|
|
150
|
+
version = "0.0.1-SNAPSHOT"
|
|
151
|
+
|
|
152
|
+
java {
|
|
153
|
+
sourceCompatibility = JavaVersion.VERSION_21
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
repositories {
|
|
157
|
+
mavenCentral()
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
dependencies {
|
|
161
|
+
// Spring Boot
|
|
162
|
+
implementation("org.springframework.boot:spring-boot-starter-web")
|
|
163
|
+
implementation("org.springframework.boot:spring-boot-starter-validation")
|
|
164
|
+
|
|
165
|
+
// MyBatis
|
|
166
|
+
implementation("org.mybatis.spring.boot:mybatis-spring-boot-starter:3.0.3")
|
|
167
|
+
|
|
168
|
+
// PostgreSQL
|
|
169
|
+
runtimeOnly("org.postgresql:postgresql")
|
|
170
|
+
|
|
171
|
+
// OpenAPI / Swagger
|
|
172
|
+
implementation("org.springdoc:springdoc-openapi-starter-webmvc-ui:2.3.0")
|
|
173
|
+
|
|
174
|
+
// Lombok
|
|
175
|
+
compileOnly("org.projectlombok:lombok")
|
|
176
|
+
annotationProcessor("org.projectlombok:lombok")
|
|
177
|
+
|
|
178
|
+
// Test
|
|
179
|
+
testImplementation("org.springframework.boot:spring-boot-starter-test")
|
|
180
|
+
testImplementation("org.testcontainers:postgresql:1.19.3")
|
|
181
|
+
testImplementation("org.testcontainers:junit-jupiter:1.19.3")
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
tasks.withType<Test> {
|
|
185
|
+
useJUnitPlatform()
|
|
186
|
+
}
|
|
187
|
+
```
|
|
188
|
+
|
|
189
|
+
</details>
|
|
190
|
+
|
|
191
|
+
### アプリケーション設定
|
|
192
|
+
|
|
193
|
+
<details>
|
|
194
|
+
<summary>Application.java</summary>
|
|
195
|
+
|
|
196
|
+
```java
|
|
197
|
+
package com.example.pms;
|
|
198
|
+
|
|
199
|
+
import org.springframework.boot.SpringApplication;
|
|
200
|
+
import org.springframework.boot.autoconfigure.SpringBootApplication;
|
|
201
|
+
|
|
202
|
+
/**
|
|
203
|
+
* 生産管理システム(Production Management System)のメインクラス.
|
|
204
|
+
*/
|
|
205
|
+
@SpringBootApplication
|
|
206
|
+
public class Application {
|
|
207
|
+
|
|
208
|
+
public static void main(String[] args) {
|
|
209
|
+
SpringApplication.run(Application.class, args);
|
|
210
|
+
}
|
|
211
|
+
}
|
|
212
|
+
```
|
|
213
|
+
|
|
214
|
+
</details>
|
|
215
|
+
|
|
216
|
+
<details>
|
|
217
|
+
<summary>OpenApiConfig.java</summary>
|
|
218
|
+
|
|
219
|
+
```java
|
|
220
|
+
package com.example.pms.infrastructure.config;
|
|
221
|
+
|
|
222
|
+
import io.swagger.v3.oas.models.OpenAPI;
|
|
223
|
+
import io.swagger.v3.oas.models.info.Contact;
|
|
224
|
+
import io.swagger.v3.oas.models.info.Info;
|
|
225
|
+
import org.springframework.context.annotation.Bean;
|
|
226
|
+
import org.springframework.context.annotation.Configuration;
|
|
227
|
+
|
|
228
|
+
/**
|
|
229
|
+
* OpenAPI 設定.
|
|
230
|
+
*/
|
|
231
|
+
@Configuration
|
|
232
|
+
public class OpenApiConfig {
|
|
233
|
+
|
|
234
|
+
@Bean
|
|
235
|
+
public OpenAPI customOpenAPI() {
|
|
236
|
+
return new OpenAPI()
|
|
237
|
+
.info(new Info()
|
|
238
|
+
.title("生産管理 API")
|
|
239
|
+
.version("1.0")
|
|
240
|
+
.description("生産管理システムの RESTful API")
|
|
241
|
+
.contact(new Contact()
|
|
242
|
+
.name("開発チーム")
|
|
243
|
+
.email("dev@example.com")));
|
|
244
|
+
}
|
|
245
|
+
}
|
|
246
|
+
```
|
|
247
|
+
|
|
248
|
+
</details>
|
|
249
|
+
|
|
250
|
+
<details>
|
|
251
|
+
<summary>RootController.java</summary>
|
|
252
|
+
|
|
253
|
+
```java
|
|
254
|
+
package com.example.pms.infrastructure.in.rest;
|
|
255
|
+
|
|
256
|
+
import io.swagger.v3.oas.annotations.Operation;
|
|
257
|
+
import io.swagger.v3.oas.annotations.tags.Tag;
|
|
258
|
+
import org.springframework.http.ResponseEntity;
|
|
259
|
+
import org.springframework.web.bind.annotation.GetMapping;
|
|
260
|
+
import org.springframework.web.bind.annotation.RequestMapping;
|
|
261
|
+
import org.springframework.web.bind.annotation.RestController;
|
|
262
|
+
|
|
263
|
+
import java.util.Map;
|
|
264
|
+
|
|
265
|
+
/**
|
|
266
|
+
* API ルート Controller.
|
|
267
|
+
*/
|
|
268
|
+
@RestController
|
|
269
|
+
@RequestMapping("/api")
|
|
270
|
+
@Tag(name = "root", description = "API ルート")
|
|
271
|
+
public class RootController {
|
|
272
|
+
|
|
273
|
+
@GetMapping
|
|
274
|
+
@Operation(summary = "API 情報の取得")
|
|
275
|
+
public ResponseEntity<Map<String, String>> root() {
|
|
276
|
+
return ResponseEntity.ok(Map.of(
|
|
277
|
+
"service", "Production Management API",
|
|
278
|
+
"version", "1.0",
|
|
279
|
+
"docs", "/swagger-ui.html"
|
|
280
|
+
));
|
|
281
|
+
}
|
|
282
|
+
|
|
283
|
+
@GetMapping("/health")
|
|
284
|
+
@Operation(summary = "ヘルスチェック")
|
|
285
|
+
public ResponseEntity<Map<String, String>> health() {
|
|
286
|
+
return ResponseEntity.ok(Map.of("status", "UP"));
|
|
287
|
+
}
|
|
288
|
+
}
|
|
289
|
+
```
|
|
290
|
+
|
|
291
|
+
</details>
|
|
292
|
+
|
|
293
|
+
---
|
|
294
|
+
|
|
295
|
+
## 32.3 マスタ API の実装
|
|
296
|
+
|
|
297
|
+
### 32.3.1 品目マスタ API
|
|
298
|
+
|
|
299
|
+
#### Output Port(リポジトリインターフェース)
|
|
300
|
+
|
|
301
|
+
<details>
|
|
302
|
+
<summary>ItemRepository.java</summary>
|
|
303
|
+
|
|
304
|
+
```java
|
|
305
|
+
package com.example.pms.application.port.out;
|
|
306
|
+
|
|
307
|
+
import com.example.pms.domain.model.item.Item;
|
|
308
|
+
|
|
309
|
+
import java.time.LocalDate;
|
|
310
|
+
import java.util.List;
|
|
311
|
+
import java.util.Optional;
|
|
312
|
+
|
|
313
|
+
/**
|
|
314
|
+
* 品目リポジトリ(Output Port)
|
|
315
|
+
* ドメイン層がデータアクセスに依存しないためのインターフェース
|
|
316
|
+
*/
|
|
317
|
+
public interface ItemRepository {
|
|
318
|
+
|
|
319
|
+
void save(Item item);
|
|
320
|
+
|
|
321
|
+
Optional<Item> findByItemCode(String itemCode);
|
|
322
|
+
|
|
323
|
+
Optional<Item> findByItemCodeAndDate(String itemCode, LocalDate baseDate);
|
|
324
|
+
|
|
325
|
+
List<Item> findAll();
|
|
326
|
+
|
|
327
|
+
void update(Item item);
|
|
328
|
+
|
|
329
|
+
void deleteAll();
|
|
330
|
+
}
|
|
331
|
+
```
|
|
332
|
+
|
|
333
|
+
</details>
|
|
334
|
+
|
|
335
|
+
#### Input Port(ユースケースインターフェース)
|
|
336
|
+
|
|
337
|
+
<details>
|
|
338
|
+
<summary>ItemUseCase.java</summary>
|
|
339
|
+
|
|
340
|
+
```java
|
|
341
|
+
package com.example.pms.application.port.in;
|
|
342
|
+
|
|
343
|
+
import com.example.pms.application.port.in.command.CreateItemCommand;
|
|
344
|
+
import com.example.pms.application.port.in.command.UpdateItemCommand;
|
|
345
|
+
import com.example.pms.domain.model.item.Item;
|
|
346
|
+
|
|
347
|
+
import java.util.List;
|
|
348
|
+
|
|
349
|
+
/**
|
|
350
|
+
* 品目ユースケース(Input Port).
|
|
351
|
+
*/
|
|
352
|
+
public interface ItemUseCase {
|
|
353
|
+
|
|
354
|
+
List<Item> getAllItems();
|
|
355
|
+
|
|
356
|
+
Item getItem(String itemCode);
|
|
357
|
+
|
|
358
|
+
Item createItem(CreateItemCommand command);
|
|
359
|
+
|
|
360
|
+
Item updateItem(String itemCode, UpdateItemCommand command);
|
|
361
|
+
|
|
362
|
+
void deleteItem(String itemCode);
|
|
363
|
+
}
|
|
364
|
+
```
|
|
365
|
+
|
|
366
|
+
</details>
|
|
367
|
+
|
|
368
|
+
<details>
|
|
369
|
+
<summary>CreateItemCommand.java</summary>
|
|
370
|
+
|
|
371
|
+
```java
|
|
372
|
+
package com.example.pms.application.port.in.command;
|
|
373
|
+
|
|
374
|
+
import com.example.pms.domain.model.item.ItemCategory;
|
|
375
|
+
import lombok.Builder;
|
|
376
|
+
import lombok.Value;
|
|
377
|
+
|
|
378
|
+
import java.math.BigDecimal;
|
|
379
|
+
import java.time.LocalDate;
|
|
380
|
+
|
|
381
|
+
/**
|
|
382
|
+
* 品目登録コマンド.
|
|
383
|
+
*/
|
|
384
|
+
@Value
|
|
385
|
+
@Builder
|
|
386
|
+
public class CreateItemCommand {
|
|
387
|
+
String itemCode;
|
|
388
|
+
String itemName;
|
|
389
|
+
ItemCategory itemCategory;
|
|
390
|
+
String unitCode;
|
|
391
|
+
LocalDate effectiveFrom;
|
|
392
|
+
LocalDate effectiveTo;
|
|
393
|
+
Integer leadTime;
|
|
394
|
+
Integer safetyLeadTime;
|
|
395
|
+
BigDecimal safetyStock;
|
|
396
|
+
BigDecimal yieldRate;
|
|
397
|
+
BigDecimal minLotSize;
|
|
398
|
+
BigDecimal lotIncrement;
|
|
399
|
+
BigDecimal maxLotSize;
|
|
400
|
+
Integer shelfLife;
|
|
401
|
+
}
|
|
402
|
+
```
|
|
403
|
+
|
|
404
|
+
</details>
|
|
405
|
+
|
|
406
|
+
#### Application Service
|
|
407
|
+
|
|
408
|
+
<details>
|
|
409
|
+
<summary>ItemService.java</summary>
|
|
410
|
+
|
|
411
|
+
```java
|
|
412
|
+
package com.example.pms.application.service;
|
|
413
|
+
|
|
414
|
+
import com.example.pms.application.port.in.ItemUseCase;
|
|
415
|
+
import com.example.pms.application.port.in.command.CreateItemCommand;
|
|
416
|
+
import com.example.pms.application.port.in.command.UpdateItemCommand;
|
|
417
|
+
import com.example.pms.application.port.out.ItemRepository;
|
|
418
|
+
import com.example.pms.domain.exception.DuplicateItemException;
|
|
419
|
+
import com.example.pms.domain.exception.ItemNotFoundException;
|
|
420
|
+
import com.example.pms.domain.model.item.Item;
|
|
421
|
+
import org.springframework.stereotype.Service;
|
|
422
|
+
import org.springframework.transaction.annotation.Transactional;
|
|
423
|
+
|
|
424
|
+
import java.time.LocalDate;
|
|
425
|
+
import java.util.List;
|
|
426
|
+
|
|
427
|
+
/**
|
|
428
|
+
* 品目サービス(Application Service).
|
|
429
|
+
*/
|
|
430
|
+
@Service
|
|
431
|
+
@Transactional
|
|
432
|
+
public class ItemService implements ItemUseCase {
|
|
433
|
+
|
|
434
|
+
private final ItemRepository itemRepository;
|
|
435
|
+
|
|
436
|
+
public ItemService(ItemRepository itemRepository) {
|
|
437
|
+
this.itemRepository = itemRepository;
|
|
438
|
+
}
|
|
439
|
+
|
|
440
|
+
@Override
|
|
441
|
+
@Transactional(readOnly = true)
|
|
442
|
+
public List<Item> getAllItems() {
|
|
443
|
+
return itemRepository.findAll();
|
|
444
|
+
}
|
|
445
|
+
|
|
446
|
+
@Override
|
|
447
|
+
@Transactional(readOnly = true)
|
|
448
|
+
public Item getItem(String itemCode) {
|
|
449
|
+
return itemRepository.findByItemCode(itemCode)
|
|
450
|
+
.orElseThrow(() -> new ItemNotFoundException(itemCode));
|
|
451
|
+
}
|
|
452
|
+
|
|
453
|
+
@Override
|
|
454
|
+
public Item createItem(CreateItemCommand command) {
|
|
455
|
+
if (itemRepository.findByItemCode(command.getItemCode()).isPresent()) {
|
|
456
|
+
throw new DuplicateItemException(command.getItemCode());
|
|
457
|
+
}
|
|
458
|
+
|
|
459
|
+
Item item = Item.builder()
|
|
460
|
+
.itemCode(command.getItemCode())
|
|
461
|
+
.itemName(command.getItemName())
|
|
462
|
+
.itemCategory(command.getItemCategory())
|
|
463
|
+
.unitCode(command.getUnitCode())
|
|
464
|
+
.effectiveFrom(command.getEffectiveFrom() != null
|
|
465
|
+
? command.getEffectiveFrom() : LocalDate.now())
|
|
466
|
+
.effectiveTo(command.getEffectiveTo())
|
|
467
|
+
.leadTime(command.getLeadTime())
|
|
468
|
+
.safetyLeadTime(command.getSafetyLeadTime())
|
|
469
|
+
.safetyStock(command.getSafetyStock())
|
|
470
|
+
.yieldRate(command.getYieldRate())
|
|
471
|
+
.minLotSize(command.getMinLotSize())
|
|
472
|
+
.lotIncrement(command.getLotIncrement())
|
|
473
|
+
.maxLotSize(command.getMaxLotSize())
|
|
474
|
+
.shelfLife(command.getShelfLife())
|
|
475
|
+
.build();
|
|
476
|
+
|
|
477
|
+
itemRepository.save(item);
|
|
478
|
+
return item;
|
|
479
|
+
}
|
|
480
|
+
|
|
481
|
+
@Override
|
|
482
|
+
public Item updateItem(String itemCode, UpdateItemCommand command) {
|
|
483
|
+
Item existing = itemRepository.findByItemCode(itemCode)
|
|
484
|
+
.orElseThrow(() -> new ItemNotFoundException(itemCode));
|
|
485
|
+
|
|
486
|
+
Item updated = Item.builder()
|
|
487
|
+
.id(existing.getId())
|
|
488
|
+
.itemCode(existing.getItemCode())
|
|
489
|
+
.itemName(command.getItemName() != null
|
|
490
|
+
? command.getItemName() : existing.getItemName())
|
|
491
|
+
.itemCategory(command.getItemCategory() != null
|
|
492
|
+
? command.getItemCategory() : existing.getItemCategory())
|
|
493
|
+
.unitCode(command.getUnitCode() != null
|
|
494
|
+
? command.getUnitCode() : existing.getUnitCode())
|
|
495
|
+
.effectiveFrom(existing.getEffectiveFrom())
|
|
496
|
+
.createdAt(existing.getCreatedAt())
|
|
497
|
+
.build();
|
|
498
|
+
|
|
499
|
+
itemRepository.update(updated);
|
|
500
|
+
return updated;
|
|
501
|
+
}
|
|
502
|
+
|
|
503
|
+
@Override
|
|
504
|
+
public void deleteItem(String itemCode) {
|
|
505
|
+
itemRepository.findByItemCode(itemCode)
|
|
506
|
+
.orElseThrow(() -> new ItemNotFoundException(itemCode));
|
|
507
|
+
throw new UnsupportedOperationException("品目の削除は現在サポートされていません");
|
|
508
|
+
}
|
|
509
|
+
}
|
|
510
|
+
```
|
|
511
|
+
|
|
512
|
+
</details>
|
|
513
|
+
|
|
514
|
+
#### Input Adapter(REST Controller)
|
|
515
|
+
|
|
516
|
+
<details>
|
|
517
|
+
<summary>ItemControllerTest.java(テスト駆動開発)</summary>
|
|
518
|
+
|
|
519
|
+
```java
|
|
520
|
+
package com.example.pms.infrastructure.in.rest;
|
|
521
|
+
|
|
522
|
+
import com.example.pms.application.port.in.ItemUseCase;
|
|
523
|
+
import com.example.pms.domain.model.item.Item;
|
|
524
|
+
import com.example.pms.domain.model.item.ItemCategory;
|
|
525
|
+
import com.fasterxml.jackson.databind.ObjectMapper;
|
|
526
|
+
import org.junit.jupiter.api.BeforeEach;
|
|
527
|
+
import org.junit.jupiter.api.Test;
|
|
528
|
+
import org.springframework.beans.factory.annotation.Autowired;
|
|
529
|
+
import org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest;
|
|
530
|
+
import org.springframework.boot.test.mock.mockito.MockBean;
|
|
531
|
+
import org.springframework.http.MediaType;
|
|
532
|
+
import org.springframework.test.web.servlet.MockMvc;
|
|
533
|
+
|
|
534
|
+
import java.time.LocalDate;
|
|
535
|
+
import java.util.List;
|
|
536
|
+
|
|
537
|
+
import static org.mockito.ArgumentMatchers.any;
|
|
538
|
+
import static org.mockito.Mockito.when;
|
|
539
|
+
import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.*;
|
|
540
|
+
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.*;
|
|
541
|
+
|
|
542
|
+
@WebMvcTest(ItemController.class)
|
|
543
|
+
class ItemControllerTest {
|
|
544
|
+
|
|
545
|
+
@Autowired
|
|
546
|
+
private MockMvc mockMvc;
|
|
547
|
+
|
|
548
|
+
@Autowired
|
|
549
|
+
private ObjectMapper objectMapper;
|
|
550
|
+
|
|
551
|
+
@MockBean
|
|
552
|
+
private ItemUseCase itemUseCase;
|
|
553
|
+
|
|
554
|
+
private Item sampleItem;
|
|
555
|
+
|
|
556
|
+
@BeforeEach
|
|
557
|
+
void setUp() {
|
|
558
|
+
sampleItem = Item.builder()
|
|
559
|
+
.itemCode("ITEM001")
|
|
560
|
+
.itemName("テスト品目")
|
|
561
|
+
.itemCategory(ItemCategory.PRODUCT)
|
|
562
|
+
.unitCode("個")
|
|
563
|
+
.effectiveFrom(LocalDate.now())
|
|
564
|
+
.leadTime(5)
|
|
565
|
+
.build();
|
|
566
|
+
}
|
|
567
|
+
|
|
568
|
+
@Test
|
|
569
|
+
void 品目一覧を取得できる() throws Exception {
|
|
570
|
+
when(itemUseCase.getAllItems()).thenReturn(List.of(sampleItem));
|
|
571
|
+
|
|
572
|
+
mockMvc.perform(get("/api/items"))
|
|
573
|
+
.andExpect(status().isOk())
|
|
574
|
+
.andExpect(jsonPath("$[0].itemCode").value("ITEM001"))
|
|
575
|
+
.andExpect(jsonPath("$[0].itemName").value("テスト品目"));
|
|
576
|
+
}
|
|
577
|
+
|
|
578
|
+
@Test
|
|
579
|
+
void 品目を登録できる() throws Exception {
|
|
580
|
+
when(itemUseCase.createItem(any())).thenReturn(sampleItem);
|
|
581
|
+
|
|
582
|
+
var request = """
|
|
583
|
+
{
|
|
584
|
+
"itemCode": "ITEM001",
|
|
585
|
+
"itemName": "テスト品目",
|
|
586
|
+
"itemCategory": "PRODUCT",
|
|
587
|
+
"unitCode": "個",
|
|
588
|
+
"leadTime": 5
|
|
589
|
+
}
|
|
590
|
+
""";
|
|
591
|
+
|
|
592
|
+
mockMvc.perform(post("/api/items")
|
|
593
|
+
.contentType(MediaType.APPLICATION_JSON)
|
|
594
|
+
.content(request))
|
|
595
|
+
.andExpect(status().isCreated())
|
|
596
|
+
.andExpect(jsonPath("$.itemCode").value("ITEM001"));
|
|
597
|
+
}
|
|
598
|
+
|
|
599
|
+
@Test
|
|
600
|
+
void 品目を削除できる() throws Exception {
|
|
601
|
+
mockMvc.perform(delete("/api/items/ITEM001"))
|
|
602
|
+
.andExpect(status().isNoContent());
|
|
603
|
+
}
|
|
604
|
+
}
|
|
605
|
+
```
|
|
606
|
+
|
|
607
|
+
</details>
|
|
608
|
+
|
|
609
|
+
<details>
|
|
610
|
+
<summary>ItemController.java</summary>
|
|
611
|
+
|
|
612
|
+
```java
|
|
613
|
+
package com.example.pms.infrastructure.in.rest;
|
|
614
|
+
|
|
615
|
+
import com.example.pms.application.port.in.ItemUseCase;
|
|
616
|
+
import com.example.pms.domain.model.item.Item;
|
|
617
|
+
import com.example.pms.infrastructure.in.rest.dto.CreateItemRequest;
|
|
618
|
+
import com.example.pms.infrastructure.in.rest.dto.ItemResponse;
|
|
619
|
+
import com.example.pms.infrastructure.in.rest.dto.UpdateItemRequest;
|
|
620
|
+
import io.swagger.v3.oas.annotations.Operation;
|
|
621
|
+
import io.swagger.v3.oas.annotations.tags.Tag;
|
|
622
|
+
import jakarta.validation.Valid;
|
|
623
|
+
import org.springframework.http.HttpStatus;
|
|
624
|
+
import org.springframework.http.ResponseEntity;
|
|
625
|
+
import org.springframework.web.bind.annotation.*;
|
|
626
|
+
|
|
627
|
+
import java.util.List;
|
|
628
|
+
|
|
629
|
+
/**
|
|
630
|
+
* 品目マスタ API Controller.
|
|
631
|
+
*/
|
|
632
|
+
@RestController
|
|
633
|
+
@RequestMapping("/api/items")
|
|
634
|
+
@Tag(name = "items", description = "品目マスタ API")
|
|
635
|
+
public class ItemController {
|
|
636
|
+
|
|
637
|
+
private final ItemUseCase itemUseCase;
|
|
638
|
+
|
|
639
|
+
public ItemController(ItemUseCase itemUseCase) {
|
|
640
|
+
this.itemUseCase = itemUseCase;
|
|
641
|
+
}
|
|
642
|
+
|
|
643
|
+
@GetMapping
|
|
644
|
+
@Operation(summary = "品目一覧の取得")
|
|
645
|
+
public ResponseEntity<List<ItemResponse>> getAllItems() {
|
|
646
|
+
List<Item> items = itemUseCase.getAllItems();
|
|
647
|
+
return ResponseEntity.ok(items.stream()
|
|
648
|
+
.map(ItemResponse::from)
|
|
649
|
+
.toList());
|
|
650
|
+
}
|
|
651
|
+
|
|
652
|
+
@GetMapping("/{itemCode}")
|
|
653
|
+
@Operation(summary = "品目の取得")
|
|
654
|
+
public ResponseEntity<ItemResponse> getItem(@PathVariable String itemCode) {
|
|
655
|
+
Item item = itemUseCase.getItem(itemCode);
|
|
656
|
+
return ResponseEntity.ok(ItemResponse.from(item));
|
|
657
|
+
}
|
|
658
|
+
|
|
659
|
+
@PostMapping
|
|
660
|
+
@Operation(summary = "品目の登録")
|
|
661
|
+
public ResponseEntity<ItemResponse> createItem(
|
|
662
|
+
@Valid @RequestBody CreateItemRequest request) {
|
|
663
|
+
Item item = itemUseCase.createItem(request.toCommand());
|
|
664
|
+
return ResponseEntity.status(HttpStatus.CREATED)
|
|
665
|
+
.body(ItemResponse.from(item));
|
|
666
|
+
}
|
|
667
|
+
|
|
668
|
+
@PutMapping("/{itemCode}")
|
|
669
|
+
@Operation(summary = "品目の更新")
|
|
670
|
+
public ResponseEntity<ItemResponse> updateItem(
|
|
671
|
+
@PathVariable String itemCode,
|
|
672
|
+
@Valid @RequestBody UpdateItemRequest request) {
|
|
673
|
+
Item item = itemUseCase.updateItem(itemCode, request.toCommand());
|
|
674
|
+
return ResponseEntity.ok(ItemResponse.from(item));
|
|
675
|
+
}
|
|
676
|
+
|
|
677
|
+
@DeleteMapping("/{itemCode}")
|
|
678
|
+
@Operation(summary = "品目の削除")
|
|
679
|
+
@ResponseStatus(HttpStatus.NO_CONTENT)
|
|
680
|
+
public void deleteItem(@PathVariable String itemCode) {
|
|
681
|
+
itemUseCase.deleteItem(itemCode);
|
|
682
|
+
}
|
|
683
|
+
}
|
|
684
|
+
```
|
|
685
|
+
|
|
686
|
+
</details>
|
|
687
|
+
|
|
688
|
+
#### DTO(Request/Response)
|
|
689
|
+
|
|
690
|
+
<details>
|
|
691
|
+
<summary>CreateItemRequest.java</summary>
|
|
692
|
+
|
|
693
|
+
```java
|
|
694
|
+
package com.example.pms.infrastructure.in.rest.dto;
|
|
695
|
+
|
|
696
|
+
import com.example.pms.application.port.in.command.CreateItemCommand;
|
|
697
|
+
import com.example.pms.domain.model.item.ItemCategory;
|
|
698
|
+
import jakarta.validation.constraints.NotBlank;
|
|
699
|
+
import jakarta.validation.constraints.NotNull;
|
|
700
|
+
import lombok.Data;
|
|
701
|
+
|
|
702
|
+
import java.math.BigDecimal;
|
|
703
|
+
import java.time.LocalDate;
|
|
704
|
+
|
|
705
|
+
/**
|
|
706
|
+
* 品目登録リクエスト DTO.
|
|
707
|
+
*/
|
|
708
|
+
@Data
|
|
709
|
+
public class CreateItemRequest {
|
|
710
|
+
|
|
711
|
+
@NotBlank(message = "品目コードは必須です")
|
|
712
|
+
private String itemCode;
|
|
713
|
+
|
|
714
|
+
@NotBlank(message = "品目名は必須です")
|
|
715
|
+
private String itemName;
|
|
716
|
+
|
|
717
|
+
@NotNull(message = "品目区分は必須です")
|
|
718
|
+
private ItemCategory itemCategory;
|
|
719
|
+
|
|
720
|
+
@NotBlank(message = "単位コードは必須です")
|
|
721
|
+
private String unitCode;
|
|
722
|
+
|
|
723
|
+
private LocalDate effectiveFrom;
|
|
724
|
+
private LocalDate effectiveTo;
|
|
725
|
+
private Integer leadTime;
|
|
726
|
+
private Integer safetyLeadTime;
|
|
727
|
+
private BigDecimal safetyStock;
|
|
728
|
+
private BigDecimal yieldRate;
|
|
729
|
+
private BigDecimal minLotSize;
|
|
730
|
+
private BigDecimal lotIncrement;
|
|
731
|
+
private BigDecimal maxLotSize;
|
|
732
|
+
private Integer shelfLife;
|
|
733
|
+
|
|
734
|
+
public CreateItemCommand toCommand() {
|
|
735
|
+
return CreateItemCommand.builder()
|
|
736
|
+
.itemCode(itemCode)
|
|
737
|
+
.itemName(itemName)
|
|
738
|
+
.itemCategory(itemCategory)
|
|
739
|
+
.unitCode(unitCode)
|
|
740
|
+
.effectiveFrom(effectiveFrom)
|
|
741
|
+
.effectiveTo(effectiveTo)
|
|
742
|
+
.leadTime(leadTime)
|
|
743
|
+
.safetyLeadTime(safetyLeadTime)
|
|
744
|
+
.safetyStock(safetyStock)
|
|
745
|
+
.yieldRate(yieldRate)
|
|
746
|
+
.minLotSize(minLotSize)
|
|
747
|
+
.lotIncrement(lotIncrement)
|
|
748
|
+
.maxLotSize(maxLotSize)
|
|
749
|
+
.shelfLife(shelfLife)
|
|
750
|
+
.build();
|
|
751
|
+
}
|
|
752
|
+
}
|
|
753
|
+
```
|
|
754
|
+
|
|
755
|
+
</details>
|
|
756
|
+
|
|
757
|
+
<details>
|
|
758
|
+
<summary>UpdateItemRequest.java</summary>
|
|
759
|
+
|
|
760
|
+
```java
|
|
761
|
+
package com.example.pms.infrastructure.in.rest.dto;
|
|
762
|
+
|
|
763
|
+
import com.example.pms.application.port.in.command.UpdateItemCommand;
|
|
764
|
+
import com.example.pms.domain.model.item.ItemCategory;
|
|
765
|
+
import lombok.Data;
|
|
766
|
+
|
|
767
|
+
import java.math.BigDecimal;
|
|
768
|
+
import java.time.LocalDate;
|
|
769
|
+
|
|
770
|
+
/**
|
|
771
|
+
* 品目更新リクエスト DTO.
|
|
772
|
+
*/
|
|
773
|
+
@Data
|
|
774
|
+
public class UpdateItemRequest {
|
|
775
|
+
|
|
776
|
+
private String itemName;
|
|
777
|
+
private ItemCategory itemCategory;
|
|
778
|
+
private String unitCode;
|
|
779
|
+
private LocalDate effectiveFrom;
|
|
780
|
+
private LocalDate effectiveTo;
|
|
781
|
+
private Integer leadTime;
|
|
782
|
+
private Integer safetyLeadTime;
|
|
783
|
+
private BigDecimal safetyStock;
|
|
784
|
+
private BigDecimal yieldRate;
|
|
785
|
+
private BigDecimal minLotSize;
|
|
786
|
+
private BigDecimal lotIncrement;
|
|
787
|
+
private BigDecimal maxLotSize;
|
|
788
|
+
private Integer shelfLife;
|
|
789
|
+
|
|
790
|
+
public UpdateItemCommand toCommand() {
|
|
791
|
+
return UpdateItemCommand.builder()
|
|
792
|
+
.itemName(itemName)
|
|
793
|
+
.itemCategory(itemCategory)
|
|
794
|
+
.unitCode(unitCode)
|
|
795
|
+
.effectiveFrom(effectiveFrom)
|
|
796
|
+
.effectiveTo(effectiveTo)
|
|
797
|
+
.leadTime(leadTime)
|
|
798
|
+
.safetyLeadTime(safetyLeadTime)
|
|
799
|
+
.safetyStock(safetyStock)
|
|
800
|
+
.yieldRate(yieldRate)
|
|
801
|
+
.minLotSize(minLotSize)
|
|
802
|
+
.lotIncrement(lotIncrement)
|
|
803
|
+
.maxLotSize(maxLotSize)
|
|
804
|
+
.shelfLife(shelfLife)
|
|
805
|
+
.build();
|
|
806
|
+
}
|
|
807
|
+
}
|
|
808
|
+
```
|
|
809
|
+
|
|
810
|
+
</details>
|
|
811
|
+
|
|
812
|
+
<details>
|
|
813
|
+
<summary>ItemResponse.java</summary>
|
|
814
|
+
|
|
815
|
+
```java
|
|
816
|
+
package com.example.pms.infrastructure.in.rest.dto;
|
|
817
|
+
|
|
818
|
+
import com.example.pms.domain.model.item.Item;
|
|
819
|
+
import com.example.pms.domain.model.item.ItemCategory;
|
|
820
|
+
import lombok.AllArgsConstructor;
|
|
821
|
+
import lombok.Builder;
|
|
822
|
+
import lombok.Data;
|
|
823
|
+
import lombok.NoArgsConstructor;
|
|
824
|
+
|
|
825
|
+
import java.math.BigDecimal;
|
|
826
|
+
import java.time.LocalDate;
|
|
827
|
+
import java.time.LocalDateTime;
|
|
828
|
+
|
|
829
|
+
/**
|
|
830
|
+
* 品目レスポンス DTO.
|
|
831
|
+
* Note: Jackson シリアライゼーションのため @Data + @NoArgsConstructor + @AllArgsConstructor を使用
|
|
832
|
+
*/
|
|
833
|
+
@Data
|
|
834
|
+
@Builder
|
|
835
|
+
@NoArgsConstructor
|
|
836
|
+
@AllArgsConstructor
|
|
837
|
+
public class ItemResponse {
|
|
838
|
+
private Integer id;
|
|
839
|
+
private String itemCode;
|
|
840
|
+
private String itemName;
|
|
841
|
+
private ItemCategory itemCategory;
|
|
842
|
+
private String unitCode;
|
|
843
|
+
private LocalDate effectiveFrom;
|
|
844
|
+
private LocalDate effectiveTo;
|
|
845
|
+
private Integer leadTime;
|
|
846
|
+
private Integer safetyLeadTime;
|
|
847
|
+
private BigDecimal safetyStock;
|
|
848
|
+
private BigDecimal yieldRate;
|
|
849
|
+
private BigDecimal minLotSize;
|
|
850
|
+
private BigDecimal lotIncrement;
|
|
851
|
+
private BigDecimal maxLotSize;
|
|
852
|
+
private Integer shelfLife;
|
|
853
|
+
private LocalDateTime createdAt;
|
|
854
|
+
private LocalDateTime updatedAt;
|
|
855
|
+
|
|
856
|
+
public static ItemResponse from(Item item) {
|
|
857
|
+
return ItemResponse.builder()
|
|
858
|
+
.id(item.getId())
|
|
859
|
+
.itemCode(item.getItemCode())
|
|
860
|
+
.itemName(item.getItemName())
|
|
861
|
+
.itemCategory(item.getItemCategory())
|
|
862
|
+
.unitCode(item.getUnitCode())
|
|
863
|
+
.effectiveFrom(item.getEffectiveFrom())
|
|
864
|
+
.effectiveTo(item.getEffectiveTo())
|
|
865
|
+
.leadTime(item.getLeadTime())
|
|
866
|
+
.safetyLeadTime(item.getSafetyLeadTime())
|
|
867
|
+
.safetyStock(item.getSafetyStock())
|
|
868
|
+
.yieldRate(item.getYieldRate())
|
|
869
|
+
.minLotSize(item.getMinLotSize())
|
|
870
|
+
.lotIncrement(item.getLotIncrement())
|
|
871
|
+
.maxLotSize(item.getMaxLotSize())
|
|
872
|
+
.shelfLife(item.getShelfLife())
|
|
873
|
+
.createdAt(item.getCreatedAt())
|
|
874
|
+
.updatedAt(item.getUpdatedAt())
|
|
875
|
+
.build();
|
|
876
|
+
}
|
|
877
|
+
}
|
|
878
|
+
```
|
|
879
|
+
|
|
880
|
+
</details>
|
|
881
|
+
|
|
882
|
+
#### Output Adapter(MyBatis Repository)
|
|
883
|
+
|
|
884
|
+
<details>
|
|
885
|
+
<summary>MyBatisItemRepository.java</summary>
|
|
886
|
+
|
|
887
|
+
```java
|
|
888
|
+
package com.example.pms.infrastructure.out.persistence.repository;
|
|
889
|
+
|
|
890
|
+
import com.example.pms.application.port.out.ItemRepository;
|
|
891
|
+
import com.example.pms.domain.model.item.Item;
|
|
892
|
+
import org.apache.ibatis.annotations.*;
|
|
893
|
+
import org.springframework.stereotype.Repository;
|
|
894
|
+
|
|
895
|
+
import java.time.LocalDate;
|
|
896
|
+
import java.util.List;
|
|
897
|
+
import java.util.Optional;
|
|
898
|
+
|
|
899
|
+
/**
|
|
900
|
+
* 品目リポジトリの MyBatis 実装(Output Adapter).
|
|
901
|
+
*/
|
|
902
|
+
@Repository
|
|
903
|
+
@Mapper
|
|
904
|
+
public interface MyBatisItemRepository extends ItemRepository {
|
|
905
|
+
|
|
906
|
+
@Override
|
|
907
|
+
@Insert("""
|
|
908
|
+
INSERT INTO "品目マスタ" ("品目コード", "適用開始日", "品名", "品目区分", "単位コード",
|
|
909
|
+
"リードタイム", "安全リードタイム", "安全在庫数", "歩留まり率",
|
|
910
|
+
"最小ロットサイズ", "ロット増分", "最大ロットサイズ", "賞味期限日数")
|
|
911
|
+
VALUES (#{itemCode}, #{effectiveFrom}, #{itemName}, #{itemCategory}::品目区分, #{unitCode},
|
|
912
|
+
#{leadTime}, #{safetyLeadTime}, #{safetyStock}, #{yieldRate},
|
|
913
|
+
#{minLotSize}, #{lotIncrement}, #{maxLotSize}, #{shelfLife})
|
|
914
|
+
""")
|
|
915
|
+
void save(Item item);
|
|
916
|
+
|
|
917
|
+
@Override
|
|
918
|
+
@Select("""
|
|
919
|
+
SELECT * FROM "品目マスタ"
|
|
920
|
+
WHERE "品目コード" = #{itemCode}
|
|
921
|
+
ORDER BY "適用開始日" DESC LIMIT 1
|
|
922
|
+
""")
|
|
923
|
+
@ResultMap("itemMap")
|
|
924
|
+
Optional<Item> findByItemCode(String itemCode);
|
|
925
|
+
|
|
926
|
+
@Override
|
|
927
|
+
@Select("""
|
|
928
|
+
SELECT * FROM "品目マスタ"
|
|
929
|
+
WHERE "品目コード" = #{itemCode}
|
|
930
|
+
AND "適用開始日" <= #{baseDate}
|
|
931
|
+
ORDER BY "適用開始日" DESC LIMIT 1
|
|
932
|
+
""")
|
|
933
|
+
@ResultMap("itemMap")
|
|
934
|
+
Optional<Item> findByItemCodeAndDate(String itemCode, LocalDate baseDate);
|
|
935
|
+
|
|
936
|
+
@Override
|
|
937
|
+
@Select("SELECT * FROM \"品目マスタ\"")
|
|
938
|
+
@Results(id = "itemMap", value = {
|
|
939
|
+
@Result(property = "id", column = "品目ID"),
|
|
940
|
+
@Result(property = "itemCode", column = "品目コード"),
|
|
941
|
+
@Result(property = "itemName", column = "品名"),
|
|
942
|
+
@Result(property = "itemCategory", column = "品目区分"),
|
|
943
|
+
@Result(property = "unitCode", column = "単位コード"),
|
|
944
|
+
@Result(property = "effectiveFrom", column = "適用開始日"),
|
|
945
|
+
@Result(property = "effectiveTo", column = "適用終了日"),
|
|
946
|
+
@Result(property = "leadTime", column = "リードタイム"),
|
|
947
|
+
@Result(property = "safetyLeadTime", column = "安全リードタイム"),
|
|
948
|
+
@Result(property = "safetyStock", column = "安全在庫数"),
|
|
949
|
+
@Result(property = "yieldRate", column = "歩留まり率"),
|
|
950
|
+
@Result(property = "minLotSize", column = "最小ロットサイズ"),
|
|
951
|
+
@Result(property = "lotIncrement", column = "ロット増分"),
|
|
952
|
+
@Result(property = "maxLotSize", column = "最大ロットサイズ"),
|
|
953
|
+
@Result(property = "shelfLife", column = "賞味期限日数"),
|
|
954
|
+
@Result(property = "createdAt", column = "作成日時"),
|
|
955
|
+
@Result(property = "updatedAt", column = "更新日時")
|
|
956
|
+
})
|
|
957
|
+
List<Item> findAll();
|
|
958
|
+
|
|
959
|
+
@Override
|
|
960
|
+
@Update("""
|
|
961
|
+
UPDATE "品目マスタ"
|
|
962
|
+
SET "品名" = #{itemName}, "品目区分" = #{itemCategory}::品目区分,
|
|
963
|
+
"単位コード" = #{unitCode}
|
|
964
|
+
WHERE "品目ID" = #{id}
|
|
965
|
+
""")
|
|
966
|
+
void update(Item item);
|
|
967
|
+
|
|
968
|
+
@Override
|
|
969
|
+
@Delete("DELETE FROM \"品目マスタ\"")
|
|
970
|
+
void deleteAll();
|
|
971
|
+
}
|
|
972
|
+
```
|
|
973
|
+
|
|
974
|
+
</details>
|
|
975
|
+
|
|
976
|
+
### 32.3.2 BOM マスタ API
|
|
977
|
+
|
|
978
|
+
#### BOM 展開サービス
|
|
979
|
+
|
|
980
|
+
<details>
|
|
981
|
+
<summary>BomService.java</summary>
|
|
982
|
+
|
|
983
|
+
```java
|
|
984
|
+
package com.example.pms.application.service;
|
|
985
|
+
|
|
986
|
+
import com.example.pms.application.port.out.BomRepository;
|
|
987
|
+
import com.example.pms.domain.model.bom.Bom;
|
|
988
|
+
import lombok.Builder;
|
|
989
|
+
import lombok.Value;
|
|
990
|
+
import org.springframework.stereotype.Service;
|
|
991
|
+
import org.springframework.transaction.annotation.Transactional;
|
|
992
|
+
|
|
993
|
+
import java.math.BigDecimal;
|
|
994
|
+
import java.util.ArrayList;
|
|
995
|
+
import java.util.List;
|
|
996
|
+
|
|
997
|
+
/**
|
|
998
|
+
* BOM サービス(Application Service).
|
|
999
|
+
*/
|
|
1000
|
+
@Service
|
|
1001
|
+
@Transactional(readOnly = true)
|
|
1002
|
+
public class BomService {
|
|
1003
|
+
|
|
1004
|
+
private final BomRepository bomRepository;
|
|
1005
|
+
|
|
1006
|
+
public BomService(BomRepository bomRepository) {
|
|
1007
|
+
this.bomRepository = bomRepository;
|
|
1008
|
+
}
|
|
1009
|
+
|
|
1010
|
+
/**
|
|
1011
|
+
* BOM 展開(部品展開)
|
|
1012
|
+
*/
|
|
1013
|
+
public List<BomNode> explodeBom(String itemCode, int maxLevel) {
|
|
1014
|
+
List<BomNode> result = new ArrayList<>();
|
|
1015
|
+
explodeRecursive(itemCode, 1, maxLevel, BigDecimal.ONE, result);
|
|
1016
|
+
return result;
|
|
1017
|
+
}
|
|
1018
|
+
|
|
1019
|
+
private void explodeRecursive(
|
|
1020
|
+
String parentCode,
|
|
1021
|
+
int currentLevel,
|
|
1022
|
+
int maxLevel,
|
|
1023
|
+
BigDecimal parentQuantity,
|
|
1024
|
+
List<BomNode> result) {
|
|
1025
|
+
|
|
1026
|
+
if (currentLevel > maxLevel) return;
|
|
1027
|
+
|
|
1028
|
+
List<Bom> children = bomRepository.findByParentCode(parentCode);
|
|
1029
|
+
|
|
1030
|
+
for (Bom bom : children) {
|
|
1031
|
+
BigDecimal requiredQty = bom.getQuantity().multiply(parentQuantity);
|
|
1032
|
+
|
|
1033
|
+
result.add(BomNode.builder()
|
|
1034
|
+
.level(currentLevel)
|
|
1035
|
+
.itemCode(bom.getChildItemCode())
|
|
1036
|
+
.itemName(bom.getChildItemName())
|
|
1037
|
+
.quantity(requiredQty)
|
|
1038
|
+
.unit(bom.getUnit())
|
|
1039
|
+
.build());
|
|
1040
|
+
|
|
1041
|
+
explodeRecursive(
|
|
1042
|
+
bom.getChildItemCode(),
|
|
1043
|
+
currentLevel + 1,
|
|
1044
|
+
maxLevel,
|
|
1045
|
+
requiredQty,
|
|
1046
|
+
result);
|
|
1047
|
+
}
|
|
1048
|
+
}
|
|
1049
|
+
|
|
1050
|
+
/**
|
|
1051
|
+
* 逆展開(使用先照会)
|
|
1052
|
+
*/
|
|
1053
|
+
public List<WhereUsedResult> whereUsed(String itemCode) {
|
|
1054
|
+
List<WhereUsedResult> result = new ArrayList<>();
|
|
1055
|
+
whereUsedRecursive(itemCode, 1, result);
|
|
1056
|
+
return result;
|
|
1057
|
+
}
|
|
1058
|
+
|
|
1059
|
+
private void whereUsedRecursive(
|
|
1060
|
+
String childCode,
|
|
1061
|
+
int level,
|
|
1062
|
+
List<WhereUsedResult> result) {
|
|
1063
|
+
|
|
1064
|
+
List<Bom> parents = bomRepository.findByChildCode(childCode);
|
|
1065
|
+
|
|
1066
|
+
for (Bom bom : parents) {
|
|
1067
|
+
result.add(WhereUsedResult.builder()
|
|
1068
|
+
.level(level)
|
|
1069
|
+
.parentItemCode(bom.getParentItemCode())
|
|
1070
|
+
.parentItemName(bom.getParentItemName())
|
|
1071
|
+
.quantity(bom.getQuantity())
|
|
1072
|
+
.build());
|
|
1073
|
+
|
|
1074
|
+
whereUsedRecursive(bom.getParentItemCode(), level + 1, result);
|
|
1075
|
+
}
|
|
1076
|
+
}
|
|
1077
|
+
}
|
|
1078
|
+
|
|
1079
|
+
@Value
|
|
1080
|
+
@Builder
|
|
1081
|
+
class BomNode {
|
|
1082
|
+
int level;
|
|
1083
|
+
String itemCode;
|
|
1084
|
+
String itemName;
|
|
1085
|
+
BigDecimal quantity;
|
|
1086
|
+
String unit;
|
|
1087
|
+
}
|
|
1088
|
+
|
|
1089
|
+
@Value
|
|
1090
|
+
@Builder
|
|
1091
|
+
class WhereUsedResult {
|
|
1092
|
+
int level;
|
|
1093
|
+
String parentItemCode;
|
|
1094
|
+
String parentItemName;
|
|
1095
|
+
BigDecimal quantity;
|
|
1096
|
+
}
|
|
1097
|
+
```
|
|
1098
|
+
|
|
1099
|
+
</details>
|
|
1100
|
+
|
|
1101
|
+
#### BOM Controller
|
|
1102
|
+
|
|
1103
|
+
<details>
|
|
1104
|
+
<summary>BomController.java</summary>
|
|
1105
|
+
|
|
1106
|
+
```java
|
|
1107
|
+
package com.example.pms.infrastructure.in.rest;
|
|
1108
|
+
|
|
1109
|
+
import com.example.pms.application.service.BomNode;
|
|
1110
|
+
import com.example.pms.application.service.BomService;
|
|
1111
|
+
import com.example.pms.application.service.WhereUsedResult;
|
|
1112
|
+
import io.swagger.v3.oas.annotations.Operation;
|
|
1113
|
+
import io.swagger.v3.oas.annotations.Parameter;
|
|
1114
|
+
import io.swagger.v3.oas.annotations.tags.Tag;
|
|
1115
|
+
import org.springframework.http.ResponseEntity;
|
|
1116
|
+
import org.springframework.web.bind.annotation.*;
|
|
1117
|
+
|
|
1118
|
+
import java.util.List;
|
|
1119
|
+
|
|
1120
|
+
/**
|
|
1121
|
+
* BOM API Controller.
|
|
1122
|
+
*/
|
|
1123
|
+
@RestController
|
|
1124
|
+
@RequestMapping("/api/bom")
|
|
1125
|
+
@Tag(name = "bom", description = "BOM API")
|
|
1126
|
+
public class BomController {
|
|
1127
|
+
|
|
1128
|
+
private final BomService bomService;
|
|
1129
|
+
|
|
1130
|
+
public BomController(BomService bomService) {
|
|
1131
|
+
this.bomService = bomService;
|
|
1132
|
+
}
|
|
1133
|
+
|
|
1134
|
+
@GetMapping("/{itemCode}/explode")
|
|
1135
|
+
@Operation(summary = "BOM 展開(部品展開)")
|
|
1136
|
+
public ResponseEntity<List<BomNode>> explodeBom(
|
|
1137
|
+
@PathVariable String itemCode,
|
|
1138
|
+
@Parameter(description = "展開レベル(デフォルト: 10)")
|
|
1139
|
+
@RequestParam(defaultValue = "10") int maxLevel) {
|
|
1140
|
+
List<BomNode> nodes = bomService.explodeBom(itemCode, maxLevel);
|
|
1141
|
+
return ResponseEntity.ok(nodes);
|
|
1142
|
+
}
|
|
1143
|
+
|
|
1144
|
+
@GetMapping("/{itemCode}/where-used")
|
|
1145
|
+
@Operation(summary = "逆展開(使用先照会)")
|
|
1146
|
+
public ResponseEntity<List<WhereUsedResult>> whereUsed(
|
|
1147
|
+
@PathVariable String itemCode) {
|
|
1148
|
+
List<WhereUsedResult> results = bomService.whereUsed(itemCode);
|
|
1149
|
+
return ResponseEntity.ok(results);
|
|
1150
|
+
}
|
|
1151
|
+
}
|
|
1152
|
+
```
|
|
1153
|
+
|
|
1154
|
+
</details>
|
|
1155
|
+
|
|
1156
|
+
---
|
|
1157
|
+
|
|
1158
|
+
## 32.4 トランザクション API の実装
|
|
1159
|
+
|
|
1160
|
+
### 32.4.1 発注 API
|
|
1161
|
+
|
|
1162
|
+
#### Input Port
|
|
1163
|
+
|
|
1164
|
+
<details>
|
|
1165
|
+
<summary>PurchaseOrderUseCase.java</summary>
|
|
1166
|
+
|
|
1167
|
+
```java
|
|
1168
|
+
package com.example.pms.application.port.in;
|
|
1169
|
+
|
|
1170
|
+
import com.example.pms.application.port.in.command.CreatePurchaseOrderCommand;
|
|
1171
|
+
import com.example.pms.domain.model.purchase.PurchaseOrder;
|
|
1172
|
+
|
|
1173
|
+
import java.util.List;
|
|
1174
|
+
|
|
1175
|
+
/**
|
|
1176
|
+
* 発注ユースケース(Input Port).
|
|
1177
|
+
*/
|
|
1178
|
+
public interface PurchaseOrderUseCase {
|
|
1179
|
+
|
|
1180
|
+
List<PurchaseOrder> getAllOrders();
|
|
1181
|
+
|
|
1182
|
+
PurchaseOrder getOrder(String orderNumber);
|
|
1183
|
+
|
|
1184
|
+
PurchaseOrder createOrder(CreatePurchaseOrderCommand command);
|
|
1185
|
+
|
|
1186
|
+
PurchaseOrder confirmOrder(String orderNumber);
|
|
1187
|
+
|
|
1188
|
+
void cancelOrder(String orderNumber);
|
|
1189
|
+
}
|
|
1190
|
+
```
|
|
1191
|
+
|
|
1192
|
+
</details>
|
|
1193
|
+
|
|
1194
|
+
#### Controller
|
|
1195
|
+
|
|
1196
|
+
<details>
|
|
1197
|
+
<summary>PurchaseOrderController.java</summary>
|
|
1198
|
+
|
|
1199
|
+
```java
|
|
1200
|
+
package com.example.pms.infrastructure.in.rest;
|
|
1201
|
+
|
|
1202
|
+
import com.example.pms.application.port.in.PurchaseOrderUseCase;
|
|
1203
|
+
import com.example.pms.domain.model.purchase.PurchaseOrder;
|
|
1204
|
+
import com.example.pms.infrastructure.in.rest.dto.CreatePurchaseOrderRequest;
|
|
1205
|
+
import com.example.pms.infrastructure.in.rest.dto.PurchaseOrderResponse;
|
|
1206
|
+
import io.swagger.v3.oas.annotations.Operation;
|
|
1207
|
+
import io.swagger.v3.oas.annotations.tags.Tag;
|
|
1208
|
+
import jakarta.validation.Valid;
|
|
1209
|
+
import org.springframework.http.HttpStatus;
|
|
1210
|
+
import org.springframework.http.ResponseEntity;
|
|
1211
|
+
import org.springframework.web.bind.annotation.*;
|
|
1212
|
+
|
|
1213
|
+
import java.util.List;
|
|
1214
|
+
|
|
1215
|
+
/**
|
|
1216
|
+
* 発注 API Controller.
|
|
1217
|
+
*/
|
|
1218
|
+
@RestController
|
|
1219
|
+
@RequestMapping("/api/purchase-orders")
|
|
1220
|
+
@Tag(name = "purchase-orders", description = "発注 API")
|
|
1221
|
+
public class PurchaseOrderController {
|
|
1222
|
+
|
|
1223
|
+
private final PurchaseOrderUseCase useCase;
|
|
1224
|
+
|
|
1225
|
+
public PurchaseOrderController(PurchaseOrderUseCase useCase) {
|
|
1226
|
+
this.useCase = useCase;
|
|
1227
|
+
}
|
|
1228
|
+
|
|
1229
|
+
@GetMapping
|
|
1230
|
+
@Operation(summary = "発注一覧の取得")
|
|
1231
|
+
public ResponseEntity<List<PurchaseOrderResponse>> getAllOrders() {
|
|
1232
|
+
List<PurchaseOrder> orders = useCase.getAllOrders();
|
|
1233
|
+
return ResponseEntity.ok(orders.stream()
|
|
1234
|
+
.map(PurchaseOrderResponse::from)
|
|
1235
|
+
.toList());
|
|
1236
|
+
}
|
|
1237
|
+
|
|
1238
|
+
@GetMapping("/{orderNumber}")
|
|
1239
|
+
@Operation(summary = "発注詳細の取得")
|
|
1240
|
+
public ResponseEntity<PurchaseOrderResponse> getOrder(
|
|
1241
|
+
@PathVariable String orderNumber) {
|
|
1242
|
+
PurchaseOrder order = useCase.getOrder(orderNumber);
|
|
1243
|
+
return ResponseEntity.ok(PurchaseOrderResponse.from(order));
|
|
1244
|
+
}
|
|
1245
|
+
|
|
1246
|
+
@PostMapping
|
|
1247
|
+
@Operation(summary = "発注の登録")
|
|
1248
|
+
public ResponseEntity<PurchaseOrderResponse> createOrder(
|
|
1249
|
+
@Valid @RequestBody CreatePurchaseOrderRequest request) {
|
|
1250
|
+
PurchaseOrder order = useCase.createOrder(request.toCommand());
|
|
1251
|
+
return ResponseEntity.status(HttpStatus.CREATED)
|
|
1252
|
+
.body(PurchaseOrderResponse.from(order));
|
|
1253
|
+
}
|
|
1254
|
+
|
|
1255
|
+
@PostMapping("/{orderNumber}/confirm")
|
|
1256
|
+
@Operation(summary = "発注の確定")
|
|
1257
|
+
public ResponseEntity<PurchaseOrderResponse> confirmOrder(
|
|
1258
|
+
@PathVariable String orderNumber) {
|
|
1259
|
+
PurchaseOrder order = useCase.confirmOrder(orderNumber);
|
|
1260
|
+
return ResponseEntity.ok(PurchaseOrderResponse.from(order));
|
|
1261
|
+
}
|
|
1262
|
+
|
|
1263
|
+
@DeleteMapping("/{orderNumber}")
|
|
1264
|
+
@Operation(summary = "発注の取消")
|
|
1265
|
+
@ResponseStatus(HttpStatus.NO_CONTENT)
|
|
1266
|
+
public void cancelOrder(@PathVariable String orderNumber) {
|
|
1267
|
+
useCase.cancelOrder(orderNumber);
|
|
1268
|
+
}
|
|
1269
|
+
}
|
|
1270
|
+
```
|
|
1271
|
+
|
|
1272
|
+
</details>
|
|
1273
|
+
|
|
1274
|
+
### 32.4.2 在庫照会 API
|
|
1275
|
+
|
|
1276
|
+
#### Input Port
|
|
1277
|
+
|
|
1278
|
+
<details>
|
|
1279
|
+
<summary>InventoryUseCase.java</summary>
|
|
1280
|
+
|
|
1281
|
+
```java
|
|
1282
|
+
package com.example.pms.application.port.in;
|
|
1283
|
+
|
|
1284
|
+
import com.example.pms.domain.model.inventory.Stock;
|
|
1285
|
+
import lombok.Builder;
|
|
1286
|
+
import lombok.Value;
|
|
1287
|
+
|
|
1288
|
+
import java.math.BigDecimal;
|
|
1289
|
+
import java.util.List;
|
|
1290
|
+
|
|
1291
|
+
/**
|
|
1292
|
+
* 在庫ユースケース(Input Port).
|
|
1293
|
+
*/
|
|
1294
|
+
public interface InventoryUseCase {
|
|
1295
|
+
|
|
1296
|
+
/**
|
|
1297
|
+
* 在庫一覧を取得
|
|
1298
|
+
*/
|
|
1299
|
+
List<Stock> getInventory(InventoryQuery query);
|
|
1300
|
+
|
|
1301
|
+
/**
|
|
1302
|
+
* 在庫サマリーを取得
|
|
1303
|
+
*/
|
|
1304
|
+
List<InventorySummary> getInventorySummary();
|
|
1305
|
+
|
|
1306
|
+
/**
|
|
1307
|
+
* 在庫不足品目を取得
|
|
1308
|
+
*/
|
|
1309
|
+
List<InventorySummary> getShortageItems();
|
|
1310
|
+
}
|
|
1311
|
+
|
|
1312
|
+
@Value
|
|
1313
|
+
@Builder
|
|
1314
|
+
class InventoryQuery {
|
|
1315
|
+
String itemCode;
|
|
1316
|
+
String locationCode;
|
|
1317
|
+
String stockStatus;
|
|
1318
|
+
}
|
|
1319
|
+
|
|
1320
|
+
@Value
|
|
1321
|
+
@Builder
|
|
1322
|
+
class InventorySummary {
|
|
1323
|
+
String itemCode;
|
|
1324
|
+
String itemName;
|
|
1325
|
+
BigDecimal totalQuantity;
|
|
1326
|
+
Integer safetyStock;
|
|
1327
|
+
StockState stockState;
|
|
1328
|
+
|
|
1329
|
+
public enum StockState {
|
|
1330
|
+
NORMAL, SHORTAGE, EXCESS
|
|
1331
|
+
}
|
|
1332
|
+
}
|
|
1333
|
+
```
|
|
1334
|
+
|
|
1335
|
+
</details>
|
|
1336
|
+
|
|
1337
|
+
#### Controller
|
|
1338
|
+
|
|
1339
|
+
<details>
|
|
1340
|
+
<summary>InventoryController.java</summary>
|
|
1341
|
+
|
|
1342
|
+
```java
|
|
1343
|
+
package com.example.pms.infrastructure.in.rest;
|
|
1344
|
+
|
|
1345
|
+
import com.example.pms.application.port.in.InventoryQuery;
|
|
1346
|
+
import com.example.pms.application.port.in.InventorySummary;
|
|
1347
|
+
import com.example.pms.application.port.in.InventoryUseCase;
|
|
1348
|
+
import com.example.pms.domain.model.inventory.Stock;
|
|
1349
|
+
import com.example.pms.infrastructure.in.rest.dto.StockResponse;
|
|
1350
|
+
import io.swagger.v3.oas.annotations.Operation;
|
|
1351
|
+
import io.swagger.v3.oas.annotations.Parameter;
|
|
1352
|
+
import io.swagger.v3.oas.annotations.tags.Tag;
|
|
1353
|
+
import org.springframework.http.ResponseEntity;
|
|
1354
|
+
import org.springframework.web.bind.annotation.*;
|
|
1355
|
+
|
|
1356
|
+
import java.util.List;
|
|
1357
|
+
|
|
1358
|
+
/**
|
|
1359
|
+
* 在庫 API Controller.
|
|
1360
|
+
*/
|
|
1361
|
+
@RestController
|
|
1362
|
+
@RequestMapping("/api/inventory")
|
|
1363
|
+
@Tag(name = "inventory", description = "在庫 API")
|
|
1364
|
+
public class InventoryController {
|
|
1365
|
+
|
|
1366
|
+
private final InventoryUseCase useCase;
|
|
1367
|
+
|
|
1368
|
+
public InventoryController(InventoryUseCase useCase) {
|
|
1369
|
+
this.useCase = useCase;
|
|
1370
|
+
}
|
|
1371
|
+
|
|
1372
|
+
@GetMapping
|
|
1373
|
+
@Operation(summary = "在庫一覧の取得")
|
|
1374
|
+
public ResponseEntity<List<StockResponse>> getInventory(
|
|
1375
|
+
@Parameter(description = "品目コード")
|
|
1376
|
+
@RequestParam(required = false) String itemCode,
|
|
1377
|
+
@Parameter(description = "場所コード")
|
|
1378
|
+
@RequestParam(required = false) String locationCode,
|
|
1379
|
+
@Parameter(description = "在庫状態")
|
|
1380
|
+
@RequestParam(required = false) String status) {
|
|
1381
|
+
|
|
1382
|
+
InventoryQuery query = InventoryQuery.builder()
|
|
1383
|
+
.itemCode(itemCode)
|
|
1384
|
+
.locationCode(locationCode)
|
|
1385
|
+
.stockStatus(status)
|
|
1386
|
+
.build();
|
|
1387
|
+
|
|
1388
|
+
List<Stock> stocks = useCase.getInventory(query);
|
|
1389
|
+
return ResponseEntity.ok(stocks.stream()
|
|
1390
|
+
.map(StockResponse::from)
|
|
1391
|
+
.toList());
|
|
1392
|
+
}
|
|
1393
|
+
|
|
1394
|
+
@GetMapping("/summary")
|
|
1395
|
+
@Operation(summary = "在庫サマリーの取得")
|
|
1396
|
+
public ResponseEntity<List<InventorySummary>> getInventorySummary() {
|
|
1397
|
+
List<InventorySummary> summary = useCase.getInventorySummary();
|
|
1398
|
+
return ResponseEntity.ok(summary);
|
|
1399
|
+
}
|
|
1400
|
+
|
|
1401
|
+
@GetMapping("/shortage")
|
|
1402
|
+
@Operation(summary = "在庫不足品目の取得")
|
|
1403
|
+
public ResponseEntity<List<InventorySummary>> getShortageItems() {
|
|
1404
|
+
List<InventorySummary> items = useCase.getShortageItems();
|
|
1405
|
+
return ResponseEntity.ok(items);
|
|
1406
|
+
}
|
|
1407
|
+
}
|
|
1408
|
+
```
|
|
1409
|
+
|
|
1410
|
+
</details>
|
|
1411
|
+
|
|
1412
|
+
### 32.4.3 作業指示 API
|
|
1413
|
+
|
|
1414
|
+
<details>
|
|
1415
|
+
<summary>WorkOrderController.java</summary>
|
|
1416
|
+
|
|
1417
|
+
```java
|
|
1418
|
+
package com.example.pms.infrastructure.in.rest;
|
|
1419
|
+
|
|
1420
|
+
import com.example.pms.application.port.in.WorkOrderUseCase;
|
|
1421
|
+
import com.example.pms.domain.model.process.WorkOrder;
|
|
1422
|
+
import com.example.pms.infrastructure.in.rest.dto.CreateWorkOrderRequest;
|
|
1423
|
+
import com.example.pms.infrastructure.in.rest.dto.WorkOrderResponse;
|
|
1424
|
+
import io.swagger.v3.oas.annotations.Operation;
|
|
1425
|
+
import io.swagger.v3.oas.annotations.tags.Tag;
|
|
1426
|
+
import jakarta.validation.Valid;
|
|
1427
|
+
import org.springframework.http.HttpStatus;
|
|
1428
|
+
import org.springframework.http.ResponseEntity;
|
|
1429
|
+
import org.springframework.web.bind.annotation.*;
|
|
1430
|
+
|
|
1431
|
+
import java.util.List;
|
|
1432
|
+
|
|
1433
|
+
/**
|
|
1434
|
+
* 作業指示 API Controller.
|
|
1435
|
+
*/
|
|
1436
|
+
@RestController
|
|
1437
|
+
@RequestMapping("/api/work-orders")
|
|
1438
|
+
@Tag(name = "work-orders", description = "作業指示 API")
|
|
1439
|
+
public class WorkOrderController {
|
|
1440
|
+
|
|
1441
|
+
private final WorkOrderUseCase useCase;
|
|
1442
|
+
|
|
1443
|
+
public WorkOrderController(WorkOrderUseCase useCase) {
|
|
1444
|
+
this.useCase = useCase;
|
|
1445
|
+
}
|
|
1446
|
+
|
|
1447
|
+
@GetMapping
|
|
1448
|
+
@Operation(summary = "作業指示一覧の取得")
|
|
1449
|
+
public ResponseEntity<List<WorkOrderResponse>> getAllWorkOrders() {
|
|
1450
|
+
List<WorkOrder> workOrders = useCase.getAllWorkOrders();
|
|
1451
|
+
return ResponseEntity.ok(workOrders.stream()
|
|
1452
|
+
.map(WorkOrderResponse::from)
|
|
1453
|
+
.toList());
|
|
1454
|
+
}
|
|
1455
|
+
|
|
1456
|
+
@GetMapping("/{workOrderNumber}")
|
|
1457
|
+
@Operation(summary = "作業指示詳細の取得")
|
|
1458
|
+
public ResponseEntity<WorkOrderResponse> getWorkOrder(
|
|
1459
|
+
@PathVariable String workOrderNumber) {
|
|
1460
|
+
WorkOrder workOrder = useCase.getWorkOrder(workOrderNumber);
|
|
1461
|
+
return ResponseEntity.ok(WorkOrderResponse.from(workOrder));
|
|
1462
|
+
}
|
|
1463
|
+
|
|
1464
|
+
@PostMapping
|
|
1465
|
+
@Operation(summary = "作業指示の登録")
|
|
1466
|
+
public ResponseEntity<WorkOrderResponse> createWorkOrder(
|
|
1467
|
+
@Valid @RequestBody CreateWorkOrderRequest request) {
|
|
1468
|
+
WorkOrder workOrder = useCase.createWorkOrder(request.toCommand());
|
|
1469
|
+
return ResponseEntity.status(HttpStatus.CREATED)
|
|
1470
|
+
.body(WorkOrderResponse.from(workOrder));
|
|
1471
|
+
}
|
|
1472
|
+
}
|
|
1473
|
+
```
|
|
1474
|
+
|
|
1475
|
+
</details>
|
|
1476
|
+
|
|
1477
|
+
### 32.4.4 MRP 実行 API
|
|
1478
|
+
|
|
1479
|
+
#### MRP サービス
|
|
1480
|
+
|
|
1481
|
+
<details>
|
|
1482
|
+
<summary>MrpUseCase.java</summary>
|
|
1483
|
+
|
|
1484
|
+
```java
|
|
1485
|
+
package com.example.pms.application.port.in;
|
|
1486
|
+
|
|
1487
|
+
import lombok.Builder;
|
|
1488
|
+
import lombok.Value;
|
|
1489
|
+
|
|
1490
|
+
import java.math.BigDecimal;
|
|
1491
|
+
import java.time.LocalDate;
|
|
1492
|
+
import java.time.LocalDateTime;
|
|
1493
|
+
import java.util.List;
|
|
1494
|
+
|
|
1495
|
+
/**
|
|
1496
|
+
* MRP ユースケース(Input Port).
|
|
1497
|
+
*/
|
|
1498
|
+
public interface MrpUseCase {
|
|
1499
|
+
|
|
1500
|
+
/**
|
|
1501
|
+
* MRP(所要量展開)を実行
|
|
1502
|
+
*/
|
|
1503
|
+
MrpResult execute(LocalDate startDate, LocalDate endDate);
|
|
1504
|
+
|
|
1505
|
+
@Value
|
|
1506
|
+
@Builder
|
|
1507
|
+
class MrpResult {
|
|
1508
|
+
LocalDateTime executionTime;
|
|
1509
|
+
LocalDate periodStart;
|
|
1510
|
+
LocalDate periodEnd;
|
|
1511
|
+
List<PlannedOrder> plannedOrders;
|
|
1512
|
+
List<ShortageItem> shortageItems;
|
|
1513
|
+
}
|
|
1514
|
+
|
|
1515
|
+
@Value
|
|
1516
|
+
@Builder
|
|
1517
|
+
class PlannedOrder {
|
|
1518
|
+
String itemCode;
|
|
1519
|
+
String itemName;
|
|
1520
|
+
String orderType; // MANUFACTURING or PURCHASE
|
|
1521
|
+
BigDecimal quantity;
|
|
1522
|
+
LocalDate dueDate;
|
|
1523
|
+
}
|
|
1524
|
+
|
|
1525
|
+
@Value
|
|
1526
|
+
@Builder
|
|
1527
|
+
class ShortageItem {
|
|
1528
|
+
String itemCode;
|
|
1529
|
+
String itemName;
|
|
1530
|
+
BigDecimal shortageQuantity;
|
|
1531
|
+
LocalDate recommendedOrderDate;
|
|
1532
|
+
}
|
|
1533
|
+
}
|
|
1534
|
+
```
|
|
1535
|
+
|
|
1536
|
+
</details>
|
|
1537
|
+
|
|
1538
|
+
#### MRP Controller
|
|
1539
|
+
|
|
1540
|
+
<details>
|
|
1541
|
+
<summary>MrpController.java</summary>
|
|
1542
|
+
|
|
1543
|
+
```java
|
|
1544
|
+
package com.example.pms.infrastructure.in.rest;
|
|
1545
|
+
|
|
1546
|
+
import com.example.pms.application.port.in.MrpUseCase;
|
|
1547
|
+
import com.example.pms.infrastructure.in.rest.dto.ExecuteMrpRequest;
|
|
1548
|
+
import com.example.pms.infrastructure.in.rest.dto.MrpResultResponse;
|
|
1549
|
+
import io.swagger.v3.oas.annotations.Operation;
|
|
1550
|
+
import io.swagger.v3.oas.annotations.tags.Tag;
|
|
1551
|
+
import jakarta.validation.Valid;
|
|
1552
|
+
import org.springframework.http.ResponseEntity;
|
|
1553
|
+
import org.springframework.web.bind.annotation.*;
|
|
1554
|
+
|
|
1555
|
+
/**
|
|
1556
|
+
* MRP API Controller.
|
|
1557
|
+
*/
|
|
1558
|
+
@RestController
|
|
1559
|
+
@RequestMapping("/api/mrp")
|
|
1560
|
+
@Tag(name = "mrp", description = "MRP API")
|
|
1561
|
+
public class MrpController {
|
|
1562
|
+
|
|
1563
|
+
private final MrpUseCase mrpUseCase;
|
|
1564
|
+
|
|
1565
|
+
public MrpController(MrpUseCase mrpUseCase) {
|
|
1566
|
+
this.mrpUseCase = mrpUseCase;
|
|
1567
|
+
}
|
|
1568
|
+
|
|
1569
|
+
@PostMapping("/execute")
|
|
1570
|
+
@Operation(
|
|
1571
|
+
summary = "MRP の実行",
|
|
1572
|
+
description = "指定期間の所要量展開を実行し、計画オーダを生成します"
|
|
1573
|
+
)
|
|
1574
|
+
public ResponseEntity<MrpResultResponse> execute(
|
|
1575
|
+
@Valid @RequestBody ExecuteMrpRequest request) {
|
|
1576
|
+
var result = mrpUseCase.execute(
|
|
1577
|
+
request.getStartDate(),
|
|
1578
|
+
request.getEndDate()
|
|
1579
|
+
);
|
|
1580
|
+
return ResponseEntity.ok(MrpResultResponse.from(result));
|
|
1581
|
+
}
|
|
1582
|
+
}
|
|
1583
|
+
```
|
|
1584
|
+
|
|
1585
|
+
</details>
|
|
1586
|
+
|
|
1587
|
+
---
|
|
1588
|
+
|
|
1589
|
+
## 32.5 エラーハンドリング
|
|
1590
|
+
|
|
1591
|
+
### 32.5.1 ドメイン例外の定義
|
|
1592
|
+
|
|
1593
|
+
<details>
|
|
1594
|
+
<summary>DomainException.java</summary>
|
|
1595
|
+
|
|
1596
|
+
```java
|
|
1597
|
+
package com.example.pms.domain.exception;
|
|
1598
|
+
|
|
1599
|
+
/**
|
|
1600
|
+
* ドメイン例外の基底クラス.
|
|
1601
|
+
*/
|
|
1602
|
+
public abstract class DomainException extends RuntimeException {
|
|
1603
|
+
|
|
1604
|
+
protected DomainException(String message) {
|
|
1605
|
+
super(message);
|
|
1606
|
+
}
|
|
1607
|
+
|
|
1608
|
+
protected DomainException(String message, Throwable cause) {
|
|
1609
|
+
super(message, cause);
|
|
1610
|
+
}
|
|
1611
|
+
}
|
|
1612
|
+
```
|
|
1613
|
+
|
|
1614
|
+
</details>
|
|
1615
|
+
|
|
1616
|
+
<details>
|
|
1617
|
+
<summary>ItemNotFoundException.java</summary>
|
|
1618
|
+
|
|
1619
|
+
```java
|
|
1620
|
+
package com.example.pms.domain.exception;
|
|
1621
|
+
|
|
1622
|
+
/**
|
|
1623
|
+
* 品目が見つからない例外.
|
|
1624
|
+
*/
|
|
1625
|
+
public class ItemNotFoundException extends DomainException {
|
|
1626
|
+
|
|
1627
|
+
public ItemNotFoundException(String itemCode) {
|
|
1628
|
+
super("品目が見つかりません: " + itemCode);
|
|
1629
|
+
}
|
|
1630
|
+
}
|
|
1631
|
+
```
|
|
1632
|
+
|
|
1633
|
+
</details>
|
|
1634
|
+
|
|
1635
|
+
<details>
|
|
1636
|
+
<summary>DuplicateItemException.java</summary>
|
|
1637
|
+
|
|
1638
|
+
```java
|
|
1639
|
+
package com.example.pms.domain.exception;
|
|
1640
|
+
|
|
1641
|
+
/**
|
|
1642
|
+
* 品目コード重複例外.
|
|
1643
|
+
*/
|
|
1644
|
+
public class DuplicateItemException extends DomainException {
|
|
1645
|
+
|
|
1646
|
+
public DuplicateItemException(String itemCode) {
|
|
1647
|
+
super("品目コードが既に存在します: " + itemCode);
|
|
1648
|
+
}
|
|
1649
|
+
}
|
|
1650
|
+
```
|
|
1651
|
+
|
|
1652
|
+
</details>
|
|
1653
|
+
|
|
1654
|
+
<details>
|
|
1655
|
+
<summary>InsufficientInventoryException.java</summary>
|
|
1656
|
+
|
|
1657
|
+
```java
|
|
1658
|
+
package com.example.pms.domain.exception;
|
|
1659
|
+
|
|
1660
|
+
import java.math.BigDecimal;
|
|
1661
|
+
|
|
1662
|
+
/**
|
|
1663
|
+
* 在庫不足例外.
|
|
1664
|
+
*/
|
|
1665
|
+
public class InsufficientInventoryException extends DomainException {
|
|
1666
|
+
|
|
1667
|
+
public InsufficientInventoryException(String itemCode, BigDecimal required, BigDecimal available) {
|
|
1668
|
+
super(String.format("在庫が不足しています: %s (必要: %s, 有効: %s)",
|
|
1669
|
+
itemCode, required, available));
|
|
1670
|
+
}
|
|
1671
|
+
}
|
|
1672
|
+
```
|
|
1673
|
+
|
|
1674
|
+
</details>
|
|
1675
|
+
|
|
1676
|
+
### 32.5.2 グローバル例外ハンドラ
|
|
1677
|
+
|
|
1678
|
+
<details>
|
|
1679
|
+
<summary>GlobalExceptionHandler.java</summary>
|
|
1680
|
+
|
|
1681
|
+
```java
|
|
1682
|
+
package com.example.pms.infrastructure.in.web.exception;
|
|
1683
|
+
|
|
1684
|
+
import com.example.pms.domain.exception.DomainException;
|
|
1685
|
+
import com.example.pms.domain.exception.DuplicateItemException;
|
|
1686
|
+
import com.example.pms.domain.exception.ItemNotFoundException;
|
|
1687
|
+
import org.springframework.http.HttpStatus;
|
|
1688
|
+
import org.springframework.http.ProblemDetail;
|
|
1689
|
+
import org.springframework.web.bind.MethodArgumentNotValidException;
|
|
1690
|
+
import org.springframework.web.bind.annotation.ExceptionHandler;
|
|
1691
|
+
import org.springframework.web.bind.annotation.RestControllerAdvice;
|
|
1692
|
+
|
|
1693
|
+
import java.net.URI;
|
|
1694
|
+
import java.time.Instant;
|
|
1695
|
+
import java.util.stream.Collectors;
|
|
1696
|
+
|
|
1697
|
+
/**
|
|
1698
|
+
* グローバル例外ハンドラ.
|
|
1699
|
+
*/
|
|
1700
|
+
@RestControllerAdvice
|
|
1701
|
+
public class GlobalExceptionHandler {
|
|
1702
|
+
|
|
1703
|
+
@ExceptionHandler(ItemNotFoundException.class)
|
|
1704
|
+
public ProblemDetail handleItemNotFoundException(ItemNotFoundException ex) {
|
|
1705
|
+
ProblemDetail problem = ProblemDetail.forStatusAndDetail(
|
|
1706
|
+
HttpStatus.NOT_FOUND, ex.getMessage());
|
|
1707
|
+
problem.setTitle("品目が見つかりません");
|
|
1708
|
+
problem.setType(URI.create("https://api.example.com/errors/item-not-found"));
|
|
1709
|
+
problem.setProperty("timestamp", Instant.now());
|
|
1710
|
+
return problem;
|
|
1711
|
+
}
|
|
1712
|
+
|
|
1713
|
+
@ExceptionHandler(DuplicateItemException.class)
|
|
1714
|
+
public ProblemDetail handleDuplicateItemException(DuplicateItemException ex) {
|
|
1715
|
+
ProblemDetail problem = ProblemDetail.forStatusAndDetail(
|
|
1716
|
+
HttpStatus.CONFLICT, ex.getMessage());
|
|
1717
|
+
problem.setTitle("品目コード重複");
|
|
1718
|
+
problem.setType(URI.create("https://api.example.com/errors/duplicate-item"));
|
|
1719
|
+
problem.setProperty("timestamp", Instant.now());
|
|
1720
|
+
return problem;
|
|
1721
|
+
}
|
|
1722
|
+
|
|
1723
|
+
@ExceptionHandler(MethodArgumentNotValidException.class)
|
|
1724
|
+
public ProblemDetail handleValidationException(MethodArgumentNotValidException ex) {
|
|
1725
|
+
String errors = ex.getBindingResult().getFieldErrors().stream()
|
|
1726
|
+
.map(e -> e.getField() + ": " + e.getDefaultMessage())
|
|
1727
|
+
.collect(Collectors.joining(", "));
|
|
1728
|
+
|
|
1729
|
+
ProblemDetail problem = ProblemDetail.forStatusAndDetail(
|
|
1730
|
+
HttpStatus.BAD_REQUEST, errors);
|
|
1731
|
+
problem.setTitle("入力値が不正です");
|
|
1732
|
+
problem.setType(URI.create("https://api.example.com/errors/validation-error"));
|
|
1733
|
+
problem.setProperty("timestamp", Instant.now());
|
|
1734
|
+
return problem;
|
|
1735
|
+
}
|
|
1736
|
+
|
|
1737
|
+
@ExceptionHandler(DomainException.class)
|
|
1738
|
+
public ProblemDetail handleDomainException(DomainException ex) {
|
|
1739
|
+
ProblemDetail problem = ProblemDetail.forStatusAndDetail(
|
|
1740
|
+
HttpStatus.BAD_REQUEST, ex.getMessage());
|
|
1741
|
+
problem.setTitle("ドメインエラー");
|
|
1742
|
+
problem.setType(URI.create("https://api.example.com/errors/domain-error"));
|
|
1743
|
+
problem.setProperty("timestamp", Instant.now());
|
|
1744
|
+
return problem;
|
|
1745
|
+
}
|
|
1746
|
+
|
|
1747
|
+
@ExceptionHandler(Exception.class)
|
|
1748
|
+
public ProblemDetail handleGenericException(Exception ex) {
|
|
1749
|
+
ProblemDetail problem = ProblemDetail.forStatusAndDetail(
|
|
1750
|
+
HttpStatus.INTERNAL_SERVER_ERROR, "予期しないエラーが発生しました");
|
|
1751
|
+
problem.setTitle("内部エラー");
|
|
1752
|
+
problem.setType(URI.create("https://api.example.com/errors/internal-error"));
|
|
1753
|
+
problem.setProperty("timestamp", Instant.now());
|
|
1754
|
+
return problem;
|
|
1755
|
+
}
|
|
1756
|
+
}
|
|
1757
|
+
```
|
|
1758
|
+
|
|
1759
|
+
</details>
|
|
1760
|
+
|
|
1761
|
+
### 32.5.3 エラーレスポンス形式
|
|
1762
|
+
|
|
1763
|
+
RFC 7807(Problem Details for HTTP APIs)に準拠したエラーレスポンスを返します。
|
|
1764
|
+
|
|
1765
|
+
```json
|
|
1766
|
+
{
|
|
1767
|
+
"type": "https://api.example.com/errors/item-not-found",
|
|
1768
|
+
"title": "品目が見つかりません",
|
|
1769
|
+
"status": 404,
|
|
1770
|
+
"detail": "品目が見つかりません: ITEM001",
|
|
1771
|
+
"instance": "/api/items/ITEM001",
|
|
1772
|
+
"timestamp": "2024-01-15T10:30:00Z"
|
|
1773
|
+
}
|
|
1774
|
+
```
|
|
1775
|
+
|
|
1776
|
+
| フィールド | 説明 |
|
|
1777
|
+
|-----------|------|
|
|
1778
|
+
| type | エラー種別を識別する URI |
|
|
1779
|
+
| title | エラーの概要 |
|
|
1780
|
+
| status | HTTP ステータスコード |
|
|
1781
|
+
| detail | エラーの詳細説明 |
|
|
1782
|
+
| instance | エラーが発生したリソースの URI |
|
|
1783
|
+
| timestamp | エラー発生時刻 |
|
|
1784
|
+
|
|
1785
|
+
---
|
|
1786
|
+
|
|
1787
|
+
## 32.6 API ドキュメント
|
|
1788
|
+
|
|
1789
|
+
### OpenAPI / Swagger UI
|
|
1790
|
+
|
|
1791
|
+
サーバー起動後、以下の URL で Swagger UI を確認できます:
|
|
1792
|
+
|
|
1793
|
+
```
|
|
1794
|
+
http://localhost:8080/swagger-ui.html
|
|
1795
|
+
```
|
|
1796
|
+
|
|
1797
|
+
### API 一覧
|
|
1798
|
+
|
|
1799
|
+
| カテゴリ | エンドポイント | メソッド | 説明 |
|
|
1800
|
+
|---------|---------------|----------|------|
|
|
1801
|
+
| **品目** | /api/items | GET | 品目一覧の取得 |
|
|
1802
|
+
| | /api/items/{itemCode} | GET | 品目の取得 |
|
|
1803
|
+
| | /api/items | POST | 品目の登録 |
|
|
1804
|
+
| | /api/items/{itemCode} | PUT | 品目の更新 |
|
|
1805
|
+
| | /api/items/{itemCode} | DELETE | 品目の削除 |
|
|
1806
|
+
| **BOM** | /api/bom/{itemCode}/explode | GET | 部品展開 |
|
|
1807
|
+
| | /api/bom/{itemCode}/where-used | GET | 使用先照会 |
|
|
1808
|
+
| **発注** | /api/purchase-orders | GET | 発注一覧の取得 |
|
|
1809
|
+
| | /api/purchase-orders/{orderNumber} | GET | 発注詳細の取得 |
|
|
1810
|
+
| | /api/purchase-orders | POST | 発注の登録 |
|
|
1811
|
+
| | /api/purchase-orders/{orderNumber}/confirm | POST | 発注の確定 |
|
|
1812
|
+
| | /api/purchase-orders/{orderNumber} | DELETE | 発注の取消 |
|
|
1813
|
+
| **在庫** | /api/inventory | GET | 在庫一覧の取得 |
|
|
1814
|
+
| | /api/inventory/summary | GET | 在庫サマリーの取得 |
|
|
1815
|
+
| | /api/inventory/shortage | GET | 在庫不足品目の取得 |
|
|
1816
|
+
| **作業指示** | /api/work-orders | GET | 作業指示一覧の取得 |
|
|
1817
|
+
| | /api/work-orders/{workOrderNumber} | GET | 作業指示詳細の取得 |
|
|
1818
|
+
| | /api/work-orders | POST | 作業指示の登録 |
|
|
1819
|
+
| | /api/work-orders/{workOrderNumber}/completion | POST | 完成実績の登録 |
|
|
1820
|
+
| | /api/work-orders/{workOrderNumber}/progress | PATCH | 作業進捗の更新 |
|
|
1821
|
+
| **MRP** | /api/mrp/execute | POST | MRP の実行 |
|
|
1822
|
+
| | /api/mrp/results | GET | MRP 実行結果の照会 |
|
|
1823
|
+
| | /api/mrp/planned-orders | GET | 計画オーダ一覧の取得 |
|
|
1824
|
+
|
|
1825
|
+
---
|
|
1826
|
+
|
|
1827
|
+
## 32.7 API インテグレーションテスト
|
|
1828
|
+
|
|
1829
|
+
### RestClient を使用した API インテグレーションテスト
|
|
1830
|
+
|
|
1831
|
+
本プロジェクトでは、MockMvc の代わりに RestClient を使用した実際の HTTP リクエストベースのテストを採用しています。
|
|
1832
|
+
|
|
1833
|
+
<details>
|
|
1834
|
+
<summary>IntegrationTestBase.java</summary>
|
|
1835
|
+
|
|
1836
|
+
```java
|
|
1837
|
+
package com.example.pms.integration;
|
|
1838
|
+
|
|
1839
|
+
import com.example.pms.TestcontainersConfiguration;
|
|
1840
|
+
import org.springframework.beans.factory.annotation.Autowired;
|
|
1841
|
+
import org.springframework.boot.test.context.SpringBootTest;
|
|
1842
|
+
import org.springframework.boot.test.web.server.LocalServerPort;
|
|
1843
|
+
import org.springframework.context.annotation.Import;
|
|
1844
|
+
import org.springframework.http.MediaType;
|
|
1845
|
+
import org.springframework.jdbc.core.JdbcTemplate;
|
|
1846
|
+
import org.springframework.test.context.ActiveProfiles;
|
|
1847
|
+
import org.springframework.web.client.RestClient;
|
|
1848
|
+
|
|
1849
|
+
/**
|
|
1850
|
+
* API インテグレーションテストの基底クラス.
|
|
1851
|
+
* TestContainers を使用して PostgreSQL コンテナを起動し、
|
|
1852
|
+
* 実際の HTTP リクエストでテストを実行する。
|
|
1853
|
+
*/
|
|
1854
|
+
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)
|
|
1855
|
+
@Import(TestcontainersConfiguration.class)
|
|
1856
|
+
@ActiveProfiles("test")
|
|
1857
|
+
public abstract class IntegrationTestBase {
|
|
1858
|
+
|
|
1859
|
+
@LocalServerPort
|
|
1860
|
+
protected int port;
|
|
1861
|
+
|
|
1862
|
+
@Autowired
|
|
1863
|
+
protected JdbcTemplate jdbcTemplate;
|
|
1864
|
+
|
|
1865
|
+
private RestClient restClient;
|
|
1866
|
+
|
|
1867
|
+
/**
|
|
1868
|
+
* REST クライアントを取得.
|
|
1869
|
+
*/
|
|
1870
|
+
protected RestClient getRestClient() {
|
|
1871
|
+
if (restClient == null) {
|
|
1872
|
+
restClient = RestClient.builder()
|
|
1873
|
+
.baseUrl("http://localhost:" + port)
|
|
1874
|
+
.defaultHeader("Content-Type", MediaType.APPLICATION_JSON_VALUE)
|
|
1875
|
+
.defaultHeader("Accept", MediaType.APPLICATION_JSON_VALUE)
|
|
1876
|
+
.build();
|
|
1877
|
+
}
|
|
1878
|
+
return restClient;
|
|
1879
|
+
}
|
|
1880
|
+
|
|
1881
|
+
protected void cleanupItemData() {
|
|
1882
|
+
jdbcTemplate.execute("DELETE FROM \"品目マスタ\" WHERE \"品目コード\" LIKE 'TEST%'");
|
|
1883
|
+
}
|
|
1884
|
+
}
|
|
1885
|
+
```
|
|
1886
|
+
|
|
1887
|
+
</details>
|
|
1888
|
+
|
|
1889
|
+
<details>
|
|
1890
|
+
<summary>ItemApiIntegrationTest.java</summary>
|
|
1891
|
+
|
|
1892
|
+
```java
|
|
1893
|
+
package com.example.pms.integration;
|
|
1894
|
+
|
|
1895
|
+
import static org.assertj.core.api.Assertions.assertThat;
|
|
1896
|
+
import static org.assertj.core.api.Assertions.assertThatThrownBy;
|
|
1897
|
+
|
|
1898
|
+
import com.example.pms.domain.model.item.ItemCategory;
|
|
1899
|
+
import com.example.pms.infrastructure.in.rest.dto.CreateItemRequest;
|
|
1900
|
+
import com.example.pms.infrastructure.in.rest.dto.ItemResponse;
|
|
1901
|
+
import com.example.pms.infrastructure.in.rest.dto.UpdateItemRequest;
|
|
1902
|
+
import org.junit.jupiter.api.AfterEach;
|
|
1903
|
+
import org.junit.jupiter.api.BeforeEach;
|
|
1904
|
+
import org.junit.jupiter.api.DisplayName;
|
|
1905
|
+
import org.junit.jupiter.api.Nested;
|
|
1906
|
+
import org.junit.jupiter.api.Test;
|
|
1907
|
+
import org.springframework.http.HttpStatus;
|
|
1908
|
+
import org.springframework.http.MediaType;
|
|
1909
|
+
import org.springframework.web.client.HttpClientErrorException;
|
|
1910
|
+
|
|
1911
|
+
@DisplayName("品目 API 統合テスト")
|
|
1912
|
+
class ItemApiIntegrationTest extends IntegrationTestBase {
|
|
1913
|
+
|
|
1914
|
+
private static final String API_PATH = "/api/items";
|
|
1915
|
+
|
|
1916
|
+
@BeforeEach
|
|
1917
|
+
void setUp() {
|
|
1918
|
+
createUnit("個", "個", "個");
|
|
1919
|
+
cleanupItemData();
|
|
1920
|
+
}
|
|
1921
|
+
|
|
1922
|
+
@AfterEach
|
|
1923
|
+
void tearDown() {
|
|
1924
|
+
cleanupItemData();
|
|
1925
|
+
}
|
|
1926
|
+
|
|
1927
|
+
@Nested
|
|
1928
|
+
@DisplayName("品目登録・取得フロー")
|
|
1929
|
+
class ItemCrudFlow {
|
|
1930
|
+
|
|
1931
|
+
@Test
|
|
1932
|
+
@DisplayName("品目を登録して取得できる")
|
|
1933
|
+
void shouldCreateAndRetrieveItem() {
|
|
1934
|
+
CreateItemRequest createRequest = new CreateItemRequest();
|
|
1935
|
+
createRequest.setItemCode("TEST001");
|
|
1936
|
+
createRequest.setItemName("テスト品目");
|
|
1937
|
+
createRequest.setItemCategory(ItemCategory.PRODUCT);
|
|
1938
|
+
createRequest.setUnitCode("個");
|
|
1939
|
+
|
|
1940
|
+
// 品目を登録
|
|
1941
|
+
ItemResponse createResponse = getRestClient()
|
|
1942
|
+
.post()
|
|
1943
|
+
.uri(API_PATH)
|
|
1944
|
+
.contentType(MediaType.APPLICATION_JSON)
|
|
1945
|
+
.body(createRequest)
|
|
1946
|
+
.retrieve()
|
|
1947
|
+
.body(ItemResponse.class);
|
|
1948
|
+
|
|
1949
|
+
assertThat(createResponse).isNotNull();
|
|
1950
|
+
assertThat(createResponse.getItemCode()).isEqualTo("TEST001");
|
|
1951
|
+
|
|
1952
|
+
// 登録した品目を取得
|
|
1953
|
+
ItemResponse getResponse = getRestClient()
|
|
1954
|
+
.get()
|
|
1955
|
+
.uri(API_PATH + "/TEST001")
|
|
1956
|
+
.retrieve()
|
|
1957
|
+
.body(ItemResponse.class);
|
|
1958
|
+
|
|
1959
|
+
assertThat(getResponse).isNotNull();
|
|
1960
|
+
assertThat(getResponse.getItemCode()).isEqualTo("TEST001");
|
|
1961
|
+
assertThat(getResponse.getItemName()).isEqualTo("テスト品目");
|
|
1962
|
+
}
|
|
1963
|
+
|
|
1964
|
+
@Test
|
|
1965
|
+
@DisplayName("存在しない品目を取得すると404エラー")
|
|
1966
|
+
void shouldReturn404WhenItemNotFound() {
|
|
1967
|
+
assertThatThrownBy(() ->
|
|
1968
|
+
getRestClient()
|
|
1969
|
+
.get()
|
|
1970
|
+
.uri(API_PATH + "/NOT-EXIST")
|
|
1971
|
+
.retrieve()
|
|
1972
|
+
.body(ItemResponse.class)
|
|
1973
|
+
).isInstanceOf(HttpClientErrorException.class)
|
|
1974
|
+
.satisfies(ex -> {
|
|
1975
|
+
HttpClientErrorException httpEx = (HttpClientErrorException) ex;
|
|
1976
|
+
assertThat(httpEx.getStatusCode()).isEqualTo(HttpStatus.NOT_FOUND);
|
|
1977
|
+
});
|
|
1978
|
+
}
|
|
1979
|
+
|
|
1980
|
+
@Test
|
|
1981
|
+
@DisplayName("重複した品目コードで登録すると409エラー")
|
|
1982
|
+
void shouldReturn409WhenDuplicateItemCode() {
|
|
1983
|
+
CreateItemRequest createRequest = new CreateItemRequest();
|
|
1984
|
+
createRequest.setItemCode("TEST002");
|
|
1985
|
+
createRequest.setItemName("テスト品目2");
|
|
1986
|
+
createRequest.setItemCategory(ItemCategory.PRODUCT);
|
|
1987
|
+
createRequest.setUnitCode("個");
|
|
1988
|
+
|
|
1989
|
+
// 1回目は成功
|
|
1990
|
+
getRestClient()
|
|
1991
|
+
.post()
|
|
1992
|
+
.uri(API_PATH)
|
|
1993
|
+
.contentType(MediaType.APPLICATION_JSON)
|
|
1994
|
+
.body(createRequest)
|
|
1995
|
+
.retrieve()
|
|
1996
|
+
.body(ItemResponse.class);
|
|
1997
|
+
|
|
1998
|
+
// 2回目は409
|
|
1999
|
+
assertThatThrownBy(() ->
|
|
2000
|
+
getRestClient()
|
|
2001
|
+
.post()
|
|
2002
|
+
.uri(API_PATH)
|
|
2003
|
+
.contentType(MediaType.APPLICATION_JSON)
|
|
2004
|
+
.body(createRequest)
|
|
2005
|
+
.retrieve()
|
|
2006
|
+
.body(ItemResponse.class)
|
|
2007
|
+
).isInstanceOf(HttpClientErrorException.class)
|
|
2008
|
+
.satisfies(ex -> {
|
|
2009
|
+
HttpClientErrorException httpEx = (HttpClientErrorException) ex;
|
|
2010
|
+
assertThat(httpEx.getStatusCode()).isEqualTo(HttpStatus.CONFLICT);
|
|
2011
|
+
});
|
|
2012
|
+
}
|
|
2013
|
+
}
|
|
2014
|
+
}
|
|
2015
|
+
```
|
|
2016
|
+
|
|
2017
|
+
</details>
|
|
2018
|
+
|
|
2019
|
+
### テスト用初期化スクリプト
|
|
2020
|
+
|
|
2021
|
+
<details>
|
|
2022
|
+
<summary>init.sql</summary>
|
|
2023
|
+
|
|
2024
|
+
```sql
|
|
2025
|
+
-- テスト用テーブル作成
|
|
2026
|
+
CREATE TABLE 品目マスタ (
|
|
2027
|
+
品目コード VARCHAR(20) PRIMARY KEY,
|
|
2028
|
+
品目名 VARCHAR(100) NOT NULL,
|
|
2029
|
+
品目種別 VARCHAR(20),
|
|
2030
|
+
単位 VARCHAR(10),
|
|
2031
|
+
標準原価 DECIMAL(15,2),
|
|
2032
|
+
安全在庫 INTEGER,
|
|
2033
|
+
リードタイム INTEGER
|
|
2034
|
+
);
|
|
2035
|
+
|
|
2036
|
+
CREATE TABLE 部品表 (
|
|
2037
|
+
親品目コード VARCHAR(20) NOT NULL,
|
|
2038
|
+
子品目コード VARCHAR(20) NOT NULL,
|
|
2039
|
+
員数 DECIMAL(10,4) NOT NULL,
|
|
2040
|
+
PRIMARY KEY (親品目コード, 子品目コード)
|
|
2041
|
+
);
|
|
2042
|
+
|
|
2043
|
+
CREATE TABLE 発注 (
|
|
2044
|
+
発注番号 VARCHAR(20) PRIMARY KEY,
|
|
2045
|
+
仕入先コード VARCHAR(20) NOT NULL,
|
|
2046
|
+
発注日 DATE NOT NULL,
|
|
2047
|
+
ステータス VARCHAR(10) DEFAULT '作成中'
|
|
2048
|
+
);
|
|
2049
|
+
|
|
2050
|
+
CREATE TABLE 発注明細 (
|
|
2051
|
+
発注番号 VARCHAR(20) NOT NULL,
|
|
2052
|
+
明細番号 INTEGER NOT NULL,
|
|
2053
|
+
品目コード VARCHAR(20) NOT NULL,
|
|
2054
|
+
発注数量 DECIMAL(10,2) NOT NULL,
|
|
2055
|
+
単価 DECIMAL(15,2),
|
|
2056
|
+
納期 DATE,
|
|
2057
|
+
PRIMARY KEY (発注番号, 明細番号)
|
|
2058
|
+
);
|
|
2059
|
+
|
|
2060
|
+
CREATE TABLE 在庫 (
|
|
2061
|
+
品目コード VARCHAR(20) NOT NULL,
|
|
2062
|
+
保管場所コード VARCHAR(20) NOT NULL,
|
|
2063
|
+
在庫数量 DECIMAL(10,2) NOT NULL DEFAULT 0,
|
|
2064
|
+
PRIMARY KEY (品目コード, 保管場所コード)
|
|
2065
|
+
);
|
|
2066
|
+
```
|
|
2067
|
+
|
|
2068
|
+
</details>
|
|
2069
|
+
|
|
2070
|
+
---
|
|
2071
|
+
|
|
2072
|
+
## まとめ
|
|
2073
|
+
|
|
2074
|
+
本章では、ヘキサゴナルアーキテクチャを採用した生産管理システムの REST API を実装しました。
|
|
2075
|
+
|
|
2076
|
+
### 実装したコンポーネント
|
|
2077
|
+
|
|
2078
|
+
| レイヤー | コンポーネント | 役割 |
|
|
2079
|
+
|---------|--------------|------|
|
|
2080
|
+
| Domain | Entity, Exception | ビジネスロジック、ドメインルール |
|
|
2081
|
+
| Application | UseCase, Service | ユースケースの実装、トランザクション管理 |
|
|
2082
|
+
| Infrastructure | Controller, Repository | 外部との入出力 |
|
|
2083
|
+
|
|
2084
|
+
### アーキテクチャの特徴
|
|
2085
|
+
|
|
2086
|
+
1. **ヘキサゴナルアーキテクチャ**: ドメインを中心に据えた設計
|
|
2087
|
+
2. **Ports and Adapters**: インターフェースと実装の分離
|
|
2088
|
+
3. **TDD**: テスト駆動で API を開発
|
|
2089
|
+
4. **OpenAPI**: Swagger UI で API 仕様を可視化
|
|
2090
|
+
|
|
2091
|
+
### 技術スタック
|
|
2092
|
+
|
|
2093
|
+
- **Spring Boot 3.2**: REST API フレームワーク
|
|
2094
|
+
- **MyBatis**: O/R マッピング
|
|
2095
|
+
- **Bean Validation**: リクエスト検証
|
|
2096
|
+
- **springdoc-openapi**: API ドキュメント生成
|
|
2097
|
+
- **Testcontainers**: 統合テスト
|
|
2098
|
+
|
|
2099
|
+
---
|
|
2100
|
+
|
|
2101
|
+
[前へ: 第31章 生産管理データ設計(E 社事例)](chapter31.md) | [目次へ戻る](../index.md)
|