@k2works/claude-code-booster 3.2.1 → 3.3.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.
Files changed (70) hide show
  1. package/lib/assets/.claude/skills/analyzing-business/SKILL.md +2 -2
  2. package/lib/assets/.claude/skills/analyzing-inception-deck/SKILL.md +5 -5
  3. package/lib/assets/.claude/skills/analyzing-requirements/SKILL.md +2 -2
  4. package/lib/assets/.claude/skills/generating-slides/SKILL.md +7 -7
  5. package/lib/assets/docs/article/index.md +4 -1
  6. package/lib/assets/docs/article/practical-database-design/index.md +121 -0
  7. package/lib/assets/docs/article/practical-database-design/part1/chapter01.md +288 -0
  8. package/lib/assets/docs/article/practical-database-design/part1/chapter02.md +518 -0
  9. package/lib/assets/docs/article/practical-database-design/part1/chapter03.md +557 -0
  10. package/lib/assets/docs/article/practical-database-design/part2/chapter04.md +924 -0
  11. package/lib/assets/docs/article/practical-database-design/part2/chapter05.md +1627 -0
  12. package/lib/assets/docs/article/practical-database-design/part2/chapter06.md +2716 -0
  13. package/lib/assets/docs/article/practical-database-design/part2/chapter07.md +2082 -0
  14. package/lib/assets/docs/article/practical-database-design/part2/chapter08.md +2105 -0
  15. package/lib/assets/docs/article/practical-database-design/part2/chapter09.md +2031 -0
  16. package/lib/assets/docs/article/practical-database-design/part2/chapter10.md +1387 -0
  17. package/lib/assets/docs/article/practical-database-design/part2/chapter11.md +1677 -0
  18. package/lib/assets/docs/article/practical-database-design/part2/chapter12.md +1417 -0
  19. package/lib/assets/docs/article/practical-database-design/part2/chapter13.md +1434 -0
  20. package/lib/assets/docs/article/practical-database-design/part3/chapter14.md +667 -0
  21. package/lib/assets/docs/article/practical-database-design/part3/chapter15.md +1625 -0
  22. package/lib/assets/docs/article/practical-database-design/part3/chapter16.md +1915 -0
  23. package/lib/assets/docs/article/practical-database-design/part3/chapter17.md +1708 -0
  24. package/lib/assets/docs/article/practical-database-design/part3/chapter18.md +2095 -0
  25. package/lib/assets/docs/article/practical-database-design/part3/chapter19.md +1123 -0
  26. package/lib/assets/docs/article/practical-database-design/part3/chapter20.md +1031 -0
  27. package/lib/assets/docs/article/practical-database-design/part3/chapter21.md +1382 -0
  28. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter14-orm.md +991 -0
  29. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter15-orm.md +1300 -0
  30. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter16-orm.md +1166 -0
  31. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter17-orm.md +1584 -0
  32. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter18-orm.md +1183 -0
  33. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter19-orm.md +1016 -0
  34. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter20-orm.md +1753 -0
  35. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter21-orm.md +1447 -0
  36. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter22-orm.md +1878 -0
  37. package/lib/assets/docs/article/practical-database-design/part4/chapter22.md +965 -0
  38. package/lib/assets/docs/article/practical-database-design/part4/chapter23.md +2069 -0
  39. package/lib/assets/docs/article/practical-database-design/part4/chapter24.md +2439 -0
  40. package/lib/assets/docs/article/practical-database-design/part4/chapter25.md +3661 -0
  41. package/lib/assets/docs/article/practical-database-design/part4/chapter26.md +2916 -0
  42. package/lib/assets/docs/article/practical-database-design/part4/chapter27.md +3105 -0
  43. package/lib/assets/docs/article/practical-database-design/part4/chapter28.md +2697 -0
  44. package/lib/assets/docs/article/practical-database-design/part4/chapter29.md +2544 -0
  45. package/lib/assets/docs/article/practical-database-design/part4/chapter30.md +2180 -0
  46. package/lib/assets/docs/article/practical-database-design/part4/chapter31.md +1192 -0
  47. package/lib/assets/docs/article/practical-database-design/part4/chapter32.md +2101 -0
  48. package/lib/assets/docs/article/practical-database-design/part5/chapter33.md +1032 -0
  49. package/lib/assets/docs/article/practical-database-design/part5/chapter34.md +1609 -0
  50. package/lib/assets/docs/article/practical-database-design/part5/chapter35.md +1453 -0
  51. package/lib/assets/docs/article/practical-database-design/part5/chapter36.md +1292 -0
  52. package/lib/assets/docs/article/practical-database-design/part5/chapter37.md +1470 -0
  53. package/lib/assets/docs/article/practical-database-design/part5/chapter38.md +1698 -0
  54. package/lib/assets/docs/article/practical-database-design/part5/chapter39.md +2334 -0
  55. package/lib/assets/docs/article/practical-database-design/study/study2-1.md +1693 -0
  56. package/lib/assets/docs/article/practical-database-design/study/study2-2.md +1347 -0
  57. package/lib/assets/docs/article/practical-database-design/study/study2-3.md +2044 -0
  58. package/lib/assets/docs/article/practical-database-design/study/study2-4.md +2229 -0
  59. package/lib/assets/docs/article/practical-database-design/study/study2-5.md +2418 -0
  60. package/lib/assets/docs/article/practical-database-design/study/study3-1.md +2205 -0
  61. package/lib/assets/docs/article/practical-database-design/study/study3-2.md +2221 -0
  62. package/lib/assets/docs/article/practical-database-design/study/study3-3.md +2253 -0
  63. package/lib/assets/docs/article/practical-database-design/study/study3-4.md +2106 -0
  64. package/lib/assets/docs/article/practical-database-design/study/study3-5.md +2507 -0
  65. package/lib/assets/docs/article/practical-database-design/study/study4-1.md +2587 -0
  66. package/lib/assets/docs/article/practical-database-design/study/study4-2.md +2075 -0
  67. package/lib/assets/docs/article/practical-database-design/study/study4-3.md +1805 -0
  68. package/lib/assets/docs/article/practical-database-design/study/study4-4.md +1895 -0
  69. package/lib/assets/docs/article/practical-database-design/study/study4-5.md +2878 -0
  70. package/package.json +1 -1
@@ -0,0 +1,2878 @@
1
+ # 実践データベース設計:生産管理システム 研究 5 - Axon CQRS/ES の実装
2
+
3
+ ## はじめに
4
+
5
+ 本研究では、REST API(第32章)、gRPC(研究 3)、GraphQL(研究 4)とは異なるアプローチとして、**CQRS(Command Query Responsibility Segregation)** と **Event Sourcing** による生産管理システムを実装します。Axon Framework を使用し、コマンド(書き込み)とクエリ(読み取り)を分離し、すべての状態変更をイベントとして記録するアーキテクチャを構築します。
6
+
7
+ 研究 1 で構築したヘキサゴナルアーキテクチャの考え方を踏襲しつつ、**ドメインモデルをフレームワークから分離**し、Axon Aggregate Adapter を介して連携させます。Read Model の永続化には **MyBatis** を使用します。
8
+
9
+ ---
10
+
11
+ ## 第38章:Axon CQRS/ES アーキテクチャの基礎
12
+
13
+ ### 38.1 CQRS とは
14
+
15
+ CQRS(Command Query Responsibility Segregation)は、コマンド(書き込み)とクエリ(読み取り)の責務を分離するアーキテクチャパターンです。生産管理システムでは、製造指図の発行・進捗管理と、在庫照会・生産実績照会を分離することで、それぞれに最適化された設計が可能になります。
16
+
17
+ ```plantuml
18
+ @startuml cqrs_architecture
19
+ skinparam componentStyle rectangle
20
+ skinparam backgroundColor #FEFEFE
21
+
22
+ package "Command Side (Write)" {
23
+ [REST Controller] --> [CommandGateway]
24
+ [CommandGateway] --> [Aggregate]
25
+ [Aggregate] --> [Event Store]
26
+ }
27
+
28
+ package "Query Side (Read)" {
29
+ [Event Store] --> [Projection]
30
+ [Projection] --> [Read Model DB]
31
+ [REST Controller] --> [MyBatis Mapper]
32
+ [MyBatis Mapper] --> [Read Model DB]
33
+ }
34
+
35
+ @enduml
36
+ ```
37
+
38
+ **CQRS の利点:**
39
+
40
+ | 観点 | 説明 |
41
+ |------|------|
42
+ | **スケーラビリティ** | 読み取りと書き込みを独立してスケール可能 |
43
+ | **パフォーマンス** | 読み取りに最適化されたモデルで高速クエリ |
44
+ | **複雑性の分離** | 書き込みロジックと読み取りロジックを独立して開発 |
45
+ | **監査対応** | Event Sourcing と組み合わせて完全な履歴を保持 |
46
+
47
+ ---
48
+
49
+ ### 38.2 Event Sourcing とは
50
+
51
+ Event Sourcing は、アプリケーションの状態をイベントの連続として保存するパターンです。生産管理システムでは、製造指図の作成、着手、完成などの状態変更をイベントとして記録し、必要に応じてイベントを再生して現在の状態を再構築します。
52
+
53
+ ```plantuml
54
+ @startuml event_sourcing
55
+ skinparam componentStyle rectangle
56
+ skinparam backgroundColor #FEFEFE
57
+
58
+ [Command] --> [Aggregate] : 1. コマンド受信
59
+ [Aggregate] --> [Domain Model] : 2. ビジネスロジック実行
60
+ [Domain Model] --> [Event] : 3. イベント生成
61
+ [Event] --> [Event Store] : 4. イベント永続化
62
+ [Event Store] --> [Aggregate] : 5. イベント再生(状態復元)
63
+
64
+ @enduml
65
+ ```
66
+
67
+ **Event Sourcing の利点:**
68
+
69
+ | 観点 | 説明 |
70
+ |------|------|
71
+ | **完全な履歴** | すべての状態変更が記録される |
72
+ | **監査証跡** | いつ、誰が、何を変更したかが明確 |
73
+ | **時間旅行** | 過去の任意の時点の状態を再構築可能 |
74
+ | **イベント駆動** | 他システムとの連携が容易 |
75
+ | **デバッグ** | 問題発生時にイベントを追跡可能 |
76
+
77
+ ---
78
+
79
+ ### 38.3 ヘキサゴナルアーキテクチャとの統合
80
+
81
+ 本実装では、ヘキサゴナルアーキテクチャ(Ports & Adapters)を採用し、ビジネスロジックを外部依存から分離します。
82
+
83
+ ```plantuml
84
+ @startuml hexagonal_cqrs
85
+ !define RECTANGLE class
86
+ skinparam backgroundColor #FEFEFE
87
+
88
+ package "Hexagonal Architecture (CQRS/ES 版)" {
89
+
90
+ package "Inbound Adapters" {
91
+ [REST Controller]
92
+ }
93
+
94
+ package "Application Core" {
95
+ package "Application Layer" {
96
+ [Aggregate Adapter]
97
+ [Projection]
98
+ [Policy Handler]
99
+ }
100
+
101
+ package "Domain Layer" {
102
+ [Domain Model]
103
+ [Commands]
104
+ [Value Objects]
105
+ }
106
+
107
+ package "API Layer" {
108
+ [Events]
109
+ }
110
+ }
111
+
112
+ package "Outbound Adapters" {
113
+ [MyBatis Mapper]
114
+ [Event Store]
115
+ }
116
+ }
117
+
118
+ [REST Controller] --> [Aggregate Adapter]
119
+ [Aggregate Adapter] --> [Domain Model]
120
+ [Projection] --> [MyBatis Mapper]
121
+
122
+ note top of [Domain Model]
123
+ 純粋なドメインモデル
124
+ Axon Framework に依存しない
125
+ ビジネスロジックのみ
126
+ end note
127
+
128
+ note left of [Aggregate Adapter]
129
+ Axon 用アダプター
130
+ @Aggregate, @CommandHandler
131
+ フレームワーク依存を吸収
132
+ end note
133
+
134
+ @enduml
135
+ ```
136
+
137
+ **設計原則:**
138
+
139
+ 1. **ドメインモデルの純粋性**: ドメインモデルは Axon に依存しない純粋な Java コード
140
+ 2. **Aggregate Adapter**: Axon Framework 用のアダプターを Application Layer に配置
141
+ 3. **イベントは公開 API**: イベントは他の Context から参照される公開 API として定義
142
+
143
+ ---
144
+
145
+ ### 38.4 技術スタック
146
+
147
+ | カテゴリ | 技術 | バージョン |
148
+ |---------|------|-----------|
149
+ | 言語 | Java | 21 |
150
+ | フレームワーク | Spring Boot | 3.4.1 |
151
+ | CQRS/ES | Axon Framework | 4.10.3 |
152
+ | ORM | MyBatis | 3.0.4 |
153
+ | API ドキュメント | SpringDoc OpenAPI | 2.7.0 |
154
+ | データベース | H2 (開発) / PostgreSQL (本番) | - |
155
+
156
+ #### build.gradle.kts
157
+
158
+ <details>
159
+ <summary>コード例: build.gradle.kts</summary>
160
+
161
+ ```kotlin
162
+ dependencies {
163
+ // Spring Boot
164
+ implementation("org.springframework.boot:spring-boot-starter-web")
165
+ implementation("org.springframework.boot:spring-boot-starter-validation")
166
+
167
+ // Axon Framework
168
+ implementation("org.axonframework:axon-spring-boot-starter:4.10.3") {
169
+ exclude(group = "org.axonframework", module = "axon-server-connector")
170
+ }
171
+
172
+ // MyBatis
173
+ implementation("org.mybatis.spring.boot:mybatis-spring-boot-starter:3.0.4")
174
+
175
+ // OpenAPI
176
+ implementation("org.springdoc:springdoc-openapi-starter-webmvc-ui:2.7.0")
177
+
178
+ // Database
179
+ runtimeOnly("com.h2database:h2")
180
+ runtimeOnly("org.postgresql:postgresql")
181
+
182
+ // Test
183
+ testImplementation("org.springframework.boot:spring-boot-starter-test")
184
+ testImplementation("org.axonframework:axon-test:4.10.3")
185
+ testImplementation("org.mybatis.spring.boot:mybatis-spring-boot-starter-test:3.0.4")
186
+ }
187
+ ```
188
+
189
+ </details>
190
+
191
+ ---
192
+
193
+ ### 38.5 ディレクトリ構造
194
+
195
+ <details>
196
+ <summary>コード例: ディレクトリ構成</summary>
197
+
198
+ ```
199
+ src/main/java/com/example/production/
200
+ ├── app/ # アプリケーション共通
201
+ │ └── RootController.java
202
+ ├── config/ # 設定クラス
203
+ │ ├── AxonConfig.java
204
+ │ ├── MyBatisConfig.java
205
+ │ └── OpenApiConfig.java
206
+ ├── workorder/ # WorkOrder Bounded Context(製造指図)
207
+ │ ├── api/
208
+ │ │ └── events/ # 公開イベント API
209
+ │ │ ├── WorkOrderEvent.java # sealed interface
210
+ │ │ ├── WorkOrderCreatedEvent.java
211
+ │ │ ├── WorkOrderReleasedEvent.java
212
+ │ │ ├── WorkOrderStartedEvent.java
213
+ │ │ ├── WorkOrderCompletedEvent.java
214
+ │ │ └── WorkOrderCancelledEvent.java
215
+ │ ├── adapter/
216
+ │ │ ├── inbound/rest/workorders/ # Inbound Adapter (REST)
217
+ │ │ │ ├── WorkOrdersController.java
218
+ │ │ │ └── protocol/ # Request/Response DTO
219
+ │ │ └── outbound/persistence/ # Outbound Adapter (MyBatis)
220
+ │ │ ├── entity/
221
+ │ │ │ └── WorkOrderEntity.java
222
+ │ │ └── mapper/
223
+ │ │ └── WorkOrderMapper.java
224
+ │ ├── application/
225
+ │ │ ├── aggregate/ # Axon Aggregate Adapter
226
+ │ │ │ └── WorkOrderAggregateAdapter.java
227
+ │ │ ├── policy/ # イベントハンドラー(Choreography)
228
+ │ │ │ └── MaterialEventHandler.java
229
+ │ │ └── query/ # Projection
230
+ │ │ └── WorkOrderProjection.java
231
+ │ └── domain/
232
+ │ └── model/aggregate/workorder/ # 純粋なドメインモデル
233
+ │ ├── WorkOrder.java
234
+ │ ├── WorkOrderCommands.java
235
+ │ └── WorkOrderStatus.java
236
+ └── material/ # Material Bounded Context(資材・在庫)
237
+ ├── api/events/
238
+ │ ├── MaterialEvent.java
239
+ │ ├── MaterialReservedEvent.java
240
+ │ ├── MaterialReservationFailedEvent.java
241
+ │ ├── MaterialIssuedEvent.java
242
+ │ └── MaterialReturnedEvent.java
243
+ ├── application/
244
+ │ ├── aggregate/
245
+ │ │ └── MaterialAggregateAdapter.java
246
+ │ └── policy/
247
+ │ └── WorkOrderEventHandler.java
248
+ └── domain/model/aggregate/material/
249
+ ├── Material.java
250
+ └── MaterialCommands.java
251
+
252
+ src/main/resources/
253
+ ├── application.yml
254
+ ├── schema.sql # Read Model スキーマ
255
+ └── mapper/
256
+ └── WorkOrderMapper.xml # MyBatis マッパー XML
257
+ ```
258
+
259
+ </details>
260
+
261
+ ---
262
+
263
+ ### 38.6 Axon 設定クラス
264
+
265
+ <details>
266
+ <summary>コード例: AxonConfig.java</summary>
267
+
268
+ ```java
269
+ package com.example.production.config;
270
+
271
+ import org.axonframework.eventsourcing.eventstore.EmbeddedEventStore;
272
+ import org.axonframework.eventsourcing.eventstore.EventStorageEngine;
273
+ import org.axonframework.eventsourcing.eventstore.EventStore;
274
+ import org.axonframework.eventsourcing.eventstore.jdbc.JdbcEventStorageEngine;
275
+ import org.axonframework.serialization.Serializer;
276
+ import org.axonframework.serialization.json.JacksonSerializer;
277
+ import org.springframework.context.annotation.Bean;
278
+ import org.springframework.context.annotation.Configuration;
279
+
280
+ import javax.sql.DataSource;
281
+
282
+ /**
283
+ * Axon Framework 設定
284
+ */
285
+ @Configuration
286
+ public class AxonConfig {
287
+
288
+ /**
289
+ * JDBC ベースの Event Storage Engine
290
+ */
291
+ @Bean
292
+ public EventStorageEngine eventStorageEngine(
293
+ DataSource dataSource,
294
+ Serializer serializer) {
295
+ return JdbcEventStorageEngine.builder()
296
+ .snapshotSerializer(serializer)
297
+ .eventSerializer(serializer)
298
+ .dataSource(dataSource)
299
+ .build();
300
+ }
301
+
302
+ /**
303
+ * Event Store
304
+ */
305
+ @Bean
306
+ public EventStore eventStore(EventStorageEngine eventStorageEngine) {
307
+ return EmbeddedEventStore.builder()
308
+ .storageEngine(eventStorageEngine)
309
+ .build();
310
+ }
311
+
312
+ /**
313
+ * JSON シリアライザー
314
+ */
315
+ @Bean
316
+ public Serializer eventSerializer() {
317
+ return JacksonSerializer.defaultSerializer();
318
+ }
319
+ }
320
+ ```
321
+
322
+ </details>
323
+
324
+ ### 38.7 MyBatis 設定クラス
325
+
326
+ <details>
327
+ <summary>コード例: MyBatisConfig.java</summary>
328
+
329
+ ```java
330
+ package com.example.production.config;
331
+
332
+ import org.apache.ibatis.session.SqlSessionFactory;
333
+ import org.mybatis.spring.SqlSessionFactoryBean;
334
+ import org.mybatis.spring.annotation.MapperScan;
335
+ import org.springframework.context.annotation.Bean;
336
+ import org.springframework.context.annotation.Configuration;
337
+ import org.springframework.core.io.support.PathMatchingResourcePatternResolver;
338
+
339
+ import javax.sql.DataSource;
340
+
341
+ /**
342
+ * MyBatis 設定
343
+ */
344
+ @Configuration
345
+ @MapperScan(basePackages = {
346
+ "com.example.production.workorder.adapter.outbound.persistence.mapper",
347
+ "com.example.production.material.adapter.outbound.persistence.mapper"
348
+ })
349
+ public class MyBatisConfig {
350
+
351
+ @Bean
352
+ public SqlSessionFactory sqlSessionFactory(DataSource dataSource) throws Exception {
353
+ SqlSessionFactoryBean factoryBean = new SqlSessionFactoryBean();
354
+ factoryBean.setDataSource(dataSource);
355
+ factoryBean.setMapperLocations(
356
+ new PathMatchingResourcePatternResolver()
357
+ .getResources("classpath:mapper/**/*.xml")
358
+ );
359
+ factoryBean.setTypeAliasesPackage(
360
+ "com.example.production.workorder.adapter.outbound.persistence.entity," +
361
+ "com.example.production.material.adapter.outbound.persistence.entity"
362
+ );
363
+
364
+ org.apache.ibatis.session.Configuration configuration =
365
+ new org.apache.ibatis.session.Configuration();
366
+ configuration.setMapUnderscoreToCamelCase(true);
367
+ factoryBean.setConfiguration(configuration);
368
+
369
+ return factoryBean.getObject();
370
+ }
371
+ }
372
+ ```
373
+
374
+ </details>
375
+
376
+ ---
377
+
378
+ ## 第39章:ドメインモデルとイベント設計
379
+
380
+ ### 39.1 WorkOrder Bounded Context(製造指図)
381
+
382
+ 製造指図の作成から完成までのライフサイクルを管理します。
383
+
384
+ #### 状態遷移図
385
+
386
+ ```plantuml
387
+ @startuml workorder_state
388
+ skinparam backgroundColor #FEFEFE
389
+
390
+ [*] --> CREATED : WorkOrderCreated
391
+ CREATED --> RELEASED : WorkOrderReleased
392
+ CREATED --> CANCELLED : WorkOrderCancelled
393
+ RELEASED --> IN_PROGRESS : WorkOrderStarted
394
+ RELEASED --> CANCELLED : WorkOrderCancelled
395
+ IN_PROGRESS --> COMPLETED : WorkOrderCompleted
396
+ IN_PROGRESS --> SUSPENDED : WorkOrderSuspended
397
+ SUSPENDED --> IN_PROGRESS : WorkOrderResumed
398
+ IN_PROGRESS --> CANCELLED : WorkOrderCancelled
399
+
400
+ @enduml
401
+ ```
402
+
403
+ #### イベント一覧
404
+
405
+ | イベント | 説明 |
406
+ |---------|------|
407
+ | `WorkOrderCreatedEvent` | 製造指図が作成された |
408
+ | `WorkOrderReleasedEvent` | 製造指図がリリースされた(資材引当成功) |
409
+ | `WorkOrderRejectedEvent` | 製造指図が却下された(資材引当失敗) |
410
+ | `WorkOrderStartedEvent` | 製造が開始された |
411
+ | `WorkOrderSuspendedEvent` | 製造が中断された |
412
+ | `WorkOrderResumedEvent` | 製造が再開された |
413
+ | `WorkOrderCompletedEvent` | 製造が完了した |
414
+ | `WorkOrderCancelledEvent` | 製造指図がキャンセルされた |
415
+
416
+ ---
417
+
418
+ ### 39.2 イベント定義(API Layer)
419
+
420
+ #### sealed interface によるイベントの型安全性
421
+
422
+ <details>
423
+ <summary>コード例: WorkOrderEvent.java</summary>
424
+
425
+ ```java
426
+ package com.example.production.workorder.api.events;
427
+
428
+ /**
429
+ * WorkOrder イベントの sealed interface
430
+ * すべての WorkOrder イベントの親インターフェース
431
+ */
432
+ public sealed interface WorkOrderEvent permits
433
+ WorkOrderCreatedEvent,
434
+ WorkOrderReleasedEvent,
435
+ WorkOrderRejectedEvent,
436
+ WorkOrderStartedEvent,
437
+ WorkOrderSuspendedEvent,
438
+ WorkOrderResumedEvent,
439
+ WorkOrderCompletedEvent,
440
+ WorkOrderCancelledEvent {
441
+
442
+ String workOrderId();
443
+ }
444
+ ```
445
+
446
+ </details>
447
+
448
+ **sealed interface の利点:**
449
+
450
+ - **網羅性チェック**: switch 式でコンパイル時に全ケースをチェック
451
+ - **型安全性**: 許可されたクラスのみが実装可能
452
+ - **ドキュメント**: 存在するイベントが一目でわかる
453
+
454
+ #### イベント record の実装
455
+
456
+ <details>
457
+ <summary>コード例: WorkOrderCreatedEvent.java</summary>
458
+
459
+ ```java
460
+ package com.example.production.workorder.api.events;
461
+
462
+ import java.math.BigDecimal;
463
+ import java.time.LocalDate;
464
+ import java.time.LocalDateTime;
465
+
466
+ /**
467
+ * 製造指図作成イベント
468
+ */
469
+ public record WorkOrderCreatedEvent(
470
+ String workOrderId,
471
+ String itemId,
472
+ String itemName,
473
+ BigDecimal orderQuantity,
474
+ String unitOfMeasure,
475
+ LocalDate plannedStartDate,
476
+ LocalDate plannedEndDate,
477
+ String workCenterId,
478
+ String createdBy,
479
+ LocalDateTime createdAt
480
+ ) implements WorkOrderEvent {
481
+ }
482
+ ```
483
+
484
+ </details>
485
+
486
+ <details>
487
+ <summary>コード例: WorkOrderReleasedEvent.java</summary>
488
+
489
+ ```java
490
+ package com.example.production.workorder.api.events;
491
+
492
+ import java.time.LocalDateTime;
493
+
494
+ /**
495
+ * 製造指図リリースイベント
496
+ */
497
+ public record WorkOrderReleasedEvent(
498
+ String workOrderId,
499
+ String releasedBy,
500
+ LocalDateTime releasedAt
501
+ ) implements WorkOrderEvent {
502
+ }
503
+ ```
504
+
505
+ </details>
506
+
507
+ <details>
508
+ <summary>コード例: WorkOrderStartedEvent.java</summary>
509
+
510
+ ```java
511
+ package com.example.production.workorder.api.events;
512
+
513
+ import java.time.LocalDateTime;
514
+
515
+ /**
516
+ * 製造開始イベント
517
+ */
518
+ public record WorkOrderStartedEvent(
519
+ String workOrderId,
520
+ String operatorId,
521
+ LocalDateTime actualStartTime
522
+ ) implements WorkOrderEvent {
523
+ }
524
+ ```
525
+
526
+ </details>
527
+
528
+ <details>
529
+ <summary>コード例: WorkOrderCompletedEvent.java</summary>
530
+
531
+ ```java
532
+ package com.example.production.workorder.api.events;
533
+
534
+ import java.math.BigDecimal;
535
+ import java.time.LocalDateTime;
536
+
537
+ /**
538
+ * 製造完了イベント
539
+ */
540
+ public record WorkOrderCompletedEvent(
541
+ String workOrderId,
542
+ String itemId,
543
+ BigDecimal completedQuantity,
544
+ BigDecimal defectQuantity,
545
+ String operatorId,
546
+ LocalDateTime actualEndTime
547
+ ) implements WorkOrderEvent {
548
+ }
549
+ ```
550
+
551
+ </details>
552
+
553
+ <details>
554
+ <summary>コード例: WorkOrderCancelledEvent.java</summary>
555
+
556
+ ```java
557
+ package com.example.production.workorder.api.events;
558
+
559
+ import java.time.LocalDateTime;
560
+
561
+ /**
562
+ * 製造指図キャンセルイベント
563
+ */
564
+ public record WorkOrderCancelledEvent(
565
+ String workOrderId,
566
+ String itemId,
567
+ String cancelledBy,
568
+ String reason,
569
+ LocalDateTime cancelledAt
570
+ ) implements WorkOrderEvent {
571
+ }
572
+ ```
573
+
574
+ </details>
575
+
576
+ ---
577
+
578
+ ### 39.3 コマンド定義(Domain Layer)
579
+
580
+ <details>
581
+ <summary>コード例: WorkOrderCommands.java</summary>
582
+
583
+ ```java
584
+ package com.example.production.workorder.domain.model.aggregate.workorder;
585
+
586
+ import org.axonframework.modelling.command.TargetAggregateIdentifier;
587
+ import java.math.BigDecimal;
588
+ import java.time.LocalDate;
589
+
590
+ /**
591
+ * WorkOrder 集約へのコマンド定義
592
+ */
593
+ public final class WorkOrderCommands {
594
+
595
+ private WorkOrderCommands() {
596
+ }
597
+
598
+ /**
599
+ * 製造指図作成コマンド
600
+ */
601
+ public record CreateWorkOrderCommand(
602
+ @TargetAggregateIdentifier
603
+ String workOrderId,
604
+ String itemId,
605
+ String itemName,
606
+ BigDecimal orderQuantity,
607
+ String unitOfMeasure,
608
+ LocalDate plannedStartDate,
609
+ LocalDate plannedEndDate,
610
+ String workCenterId,
611
+ String createdBy
612
+ ) {
613
+ }
614
+
615
+ /**
616
+ * 製造指図リリースコマンド
617
+ */
618
+ public record ReleaseWorkOrderCommand(
619
+ @TargetAggregateIdentifier
620
+ String workOrderId,
621
+ String releasedBy
622
+ ) {
623
+ }
624
+
625
+ /**
626
+ * 製造指図却下コマンド
627
+ */
628
+ public record RejectWorkOrderCommand(
629
+ @TargetAggregateIdentifier
630
+ String workOrderId,
631
+ String reason
632
+ ) {
633
+ }
634
+
635
+ /**
636
+ * 製造開始コマンド
637
+ */
638
+ public record StartWorkOrderCommand(
639
+ @TargetAggregateIdentifier
640
+ String workOrderId,
641
+ String operatorId
642
+ ) {
643
+ }
644
+
645
+ /**
646
+ * 製造中断コマンド
647
+ */
648
+ public record SuspendWorkOrderCommand(
649
+ @TargetAggregateIdentifier
650
+ String workOrderId,
651
+ String reason,
652
+ String suspendedBy
653
+ ) {
654
+ }
655
+
656
+ /**
657
+ * 製造再開コマンド
658
+ */
659
+ public record ResumeWorkOrderCommand(
660
+ @TargetAggregateIdentifier
661
+ String workOrderId,
662
+ String resumedBy
663
+ ) {
664
+ }
665
+
666
+ /**
667
+ * 製造完了コマンド
668
+ */
669
+ public record CompleteWorkOrderCommand(
670
+ @TargetAggregateIdentifier
671
+ String workOrderId,
672
+ BigDecimal completedQuantity,
673
+ BigDecimal defectQuantity,
674
+ String operatorId
675
+ ) {
676
+ }
677
+
678
+ /**
679
+ * キャンセルコマンド
680
+ */
681
+ public record CancelWorkOrderCommand(
682
+ @TargetAggregateIdentifier
683
+ String workOrderId,
684
+ String cancelledBy,
685
+ String reason
686
+ ) {
687
+ }
688
+ }
689
+ ```
690
+
691
+ </details>
692
+
693
+ **@TargetAggregateIdentifier の役割:**
694
+
695
+ - Axon がコマンドをどの集約インスタンスにルーティングするかを決定
696
+ - 集約の一意識別子となるフィールドに付与
697
+
698
+ ---
699
+
700
+ ### 39.4 ドメインモデル(純粋な Java)
701
+
702
+ <details>
703
+ <summary>コード例: WorkOrder.java</summary>
704
+
705
+ ```java
706
+ package com.example.production.workorder.domain.model.aggregate.workorder;
707
+
708
+ import com.example.production.workorder.api.events.*;
709
+ import java.math.BigDecimal;
710
+ import java.time.LocalDate;
711
+ import java.time.LocalDateTime;
712
+
713
+ /**
714
+ * WorkOrder ドメインモデル(Axon 非依存)
715
+ * 純粋なビジネスロジックのみを含む
716
+ */
717
+ public record WorkOrder(
718
+ String workOrderId,
719
+ String itemId,
720
+ String itemName,
721
+ BigDecimal orderQuantity,
722
+ String unitOfMeasure,
723
+ LocalDate plannedStartDate,
724
+ LocalDate plannedEndDate,
725
+ String workCenterId,
726
+ WorkOrderStatus status
727
+ ) {
728
+
729
+ // ======== ファクトリメソッド ========
730
+
731
+ /**
732
+ * 製造指図作成
733
+ */
734
+ public static WorkOrderCreatedEvent create(
735
+ String workOrderId,
736
+ String itemId,
737
+ String itemName,
738
+ BigDecimal orderQuantity,
739
+ String unitOfMeasure,
740
+ LocalDate plannedStartDate,
741
+ LocalDate plannedEndDate,
742
+ String workCenterId,
743
+ String createdBy
744
+ ) {
745
+ // バリデーション
746
+ if (orderQuantity.compareTo(BigDecimal.ZERO) <= 0) {
747
+ throw new IllegalArgumentException("Order quantity must be positive");
748
+ }
749
+ if (plannedEndDate.isBefore(plannedStartDate)) {
750
+ throw new IllegalArgumentException("Planned end date must be after start date");
751
+ }
752
+
753
+ return new WorkOrderCreatedEvent(
754
+ workOrderId,
755
+ itemId,
756
+ itemName,
757
+ orderQuantity,
758
+ unitOfMeasure,
759
+ plannedStartDate,
760
+ plannedEndDate,
761
+ workCenterId,
762
+ createdBy,
763
+ LocalDateTime.now()
764
+ );
765
+ }
766
+
767
+ /**
768
+ * イベントからの再構築
769
+ */
770
+ public static WorkOrder from(WorkOrderCreatedEvent event) {
771
+ return new WorkOrder(
772
+ event.workOrderId(),
773
+ event.itemId(),
774
+ event.itemName(),
775
+ event.orderQuantity(),
776
+ event.unitOfMeasure(),
777
+ event.plannedStartDate(),
778
+ event.plannedEndDate(),
779
+ event.workCenterId(),
780
+ WorkOrderStatus.CREATED
781
+ );
782
+ }
783
+
784
+ // ======== 状態遷移メソッド ========
785
+
786
+ /**
787
+ * 製造指図リリース
788
+ */
789
+ public WorkOrderReleasedEvent release(String releasedBy) {
790
+ if (status != WorkOrderStatus.CREATED) {
791
+ throw new IllegalStateException("Only created work orders can be released");
792
+ }
793
+ return new WorkOrderReleasedEvent(workOrderId, releasedBy, LocalDateTime.now());
794
+ }
795
+
796
+ /**
797
+ * 製造指図却下
798
+ */
799
+ public WorkOrderRejectedEvent reject(String reason) {
800
+ if (status != WorkOrderStatus.CREATED) {
801
+ throw new IllegalStateException("Only created work orders can be rejected");
802
+ }
803
+ return new WorkOrderRejectedEvent(workOrderId, reason, LocalDateTime.now());
804
+ }
805
+
806
+ /**
807
+ * 製造開始
808
+ */
809
+ public WorkOrderStartedEvent start(String operatorId) {
810
+ if (status != WorkOrderStatus.RELEASED) {
811
+ throw new IllegalStateException("Only released work orders can be started");
812
+ }
813
+ return new WorkOrderStartedEvent(workOrderId, operatorId, LocalDateTime.now());
814
+ }
815
+
816
+ /**
817
+ * 製造中断
818
+ */
819
+ public WorkOrderSuspendedEvent suspend(String reason, String suspendedBy) {
820
+ if (status != WorkOrderStatus.IN_PROGRESS) {
821
+ throw new IllegalStateException("Only in-progress work orders can be suspended");
822
+ }
823
+ return new WorkOrderSuspendedEvent(workOrderId, reason, suspendedBy, LocalDateTime.now());
824
+ }
825
+
826
+ /**
827
+ * 製造再開
828
+ */
829
+ public WorkOrderResumedEvent resume(String resumedBy) {
830
+ if (status != WorkOrderStatus.SUSPENDED) {
831
+ throw new IllegalStateException("Only suspended work orders can be resumed");
832
+ }
833
+ return new WorkOrderResumedEvent(workOrderId, resumedBy, LocalDateTime.now());
834
+ }
835
+
836
+ /**
837
+ * 製造完了
838
+ */
839
+ public WorkOrderCompletedEvent complete(
840
+ BigDecimal completedQuantity,
841
+ BigDecimal defectQuantity,
842
+ String operatorId
843
+ ) {
844
+ if (status != WorkOrderStatus.IN_PROGRESS) {
845
+ throw new IllegalStateException("Only in-progress work orders can be completed");
846
+ }
847
+ if (completedQuantity.compareTo(BigDecimal.ZERO) < 0) {
848
+ throw new IllegalArgumentException("Completed quantity cannot be negative");
849
+ }
850
+ if (defectQuantity.compareTo(BigDecimal.ZERO) < 0) {
851
+ throw new IllegalArgumentException("Defect quantity cannot be negative");
852
+ }
853
+ return new WorkOrderCompletedEvent(
854
+ workOrderId,
855
+ itemId,
856
+ completedQuantity,
857
+ defectQuantity,
858
+ operatorId,
859
+ LocalDateTime.now()
860
+ );
861
+ }
862
+
863
+ /**
864
+ * キャンセル
865
+ */
866
+ public WorkOrderCancelledEvent cancel(String cancelledBy, String reason) {
867
+ if (status == WorkOrderStatus.COMPLETED) {
868
+ throw new IllegalStateException("Completed work orders cannot be cancelled");
869
+ }
870
+ if (status == WorkOrderStatus.CANCELLED) {
871
+ throw new IllegalStateException("Work order is already cancelled");
872
+ }
873
+ return new WorkOrderCancelledEvent(workOrderId, itemId, cancelledBy, reason, LocalDateTime.now());
874
+ }
875
+
876
+ // ======== イベント適用メソッド ========
877
+
878
+ /**
879
+ * イベントを適用して新しい状態を生成
880
+ */
881
+ public WorkOrder apply(WorkOrderEvent event) {
882
+ return switch (event) {
883
+ case WorkOrderCreatedEvent e -> from(e);
884
+ case WorkOrderReleasedEvent e -> withStatus(WorkOrderStatus.RELEASED);
885
+ case WorkOrderRejectedEvent e -> withStatus(WorkOrderStatus.REJECTED);
886
+ case WorkOrderStartedEvent e -> withStatus(WorkOrderStatus.IN_PROGRESS);
887
+ case WorkOrderSuspendedEvent e -> withStatus(WorkOrderStatus.SUSPENDED);
888
+ case WorkOrderResumedEvent e -> withStatus(WorkOrderStatus.IN_PROGRESS);
889
+ case WorkOrderCompletedEvent e -> withStatus(WorkOrderStatus.COMPLETED);
890
+ case WorkOrderCancelledEvent e -> withStatus(WorkOrderStatus.CANCELLED);
891
+ };
892
+ }
893
+
894
+ /**
895
+ * 状態更新ヘルパー
896
+ */
897
+ private WorkOrder withStatus(WorkOrderStatus newStatus) {
898
+ return new WorkOrder(
899
+ workOrderId, itemId, itemName, orderQuantity, unitOfMeasure,
900
+ plannedStartDate, plannedEndDate, workCenterId, newStatus
901
+ );
902
+ }
903
+ }
904
+ ```
905
+
906
+ </details>
907
+
908
+ <details>
909
+ <summary>コード例: WorkOrderStatus.java</summary>
910
+
911
+ ```java
912
+ package com.example.production.workorder.domain.model.aggregate.workorder;
913
+
914
+ /**
915
+ * 製造指図ステータス
916
+ */
917
+ public enum WorkOrderStatus {
918
+ CREATED, // 作成済み
919
+ RELEASED, // リリース済み(資材引当完了)
920
+ REJECTED, // 却下(資材引当失敗)
921
+ IN_PROGRESS, // 製造中
922
+ SUSPENDED, // 中断
923
+ COMPLETED, // 完了
924
+ CANCELLED // キャンセル
925
+ }
926
+ ```
927
+
928
+ </details>
929
+
930
+ **ドメインモデル設計原則:**
931
+
932
+ | 原則 | 説明 |
933
+ |------|------|
934
+ | **Axon 非依存** | ドメインモデルにフレームワーク依存を持たせない |
935
+ | **イミュータブル** | record でイミュータブルに設計 |
936
+ | **イベントを返す** | 状態遷移メソッドはイベントを返す |
937
+ | **最小限のフィールド** | 状態遷移の判定に必要な最小限のみ保持 |
938
+
939
+ ---
940
+
941
+ ### 39.5 Material Bounded Context(資材・在庫)
942
+
943
+ 部品・原材料の引当・払出・返却を管理します。
944
+
945
+ #### イベント定義
946
+
947
+ <details>
948
+ <summary>コード例: MaterialEvent.java</summary>
949
+
950
+ ```java
951
+ package com.example.production.material.api.events;
952
+
953
+ /**
954
+ * Material イベントの sealed interface
955
+ */
956
+ public sealed interface MaterialEvent permits
957
+ MaterialInitializedEvent,
958
+ MaterialReservedEvent,
959
+ MaterialReservationFailedEvent,
960
+ MaterialIssuedEvent,
961
+ MaterialReturnedEvent {
962
+
963
+ String materialId();
964
+ }
965
+ ```
966
+
967
+ </details>
968
+
969
+ <details>
970
+ <summary>コード例: MaterialReservedEvent.java</summary>
971
+
972
+ ```java
973
+ package com.example.production.material.api.events;
974
+
975
+ import java.math.BigDecimal;
976
+ import java.time.LocalDateTime;
977
+
978
+ /**
979
+ * 資材引当成功イベント
980
+ */
981
+ public record MaterialReservedEvent(
982
+ String materialId,
983
+ String workOrderId,
984
+ BigDecimal quantity,
985
+ LocalDateTime reservedAt
986
+ ) implements MaterialEvent {
987
+ }
988
+ ```
989
+
990
+ </details>
991
+
992
+ <details>
993
+ <summary>コード例: MaterialReservationFailedEvent.java</summary>
994
+
995
+ ```java
996
+ package com.example.production.material.api.events;
997
+
998
+ import java.math.BigDecimal;
999
+ import java.time.LocalDateTime;
1000
+
1001
+ /**
1002
+ * 資材引当失敗イベント
1003
+ */
1004
+ public record MaterialReservationFailedEvent(
1005
+ String materialId,
1006
+ String workOrderId,
1007
+ BigDecimal requestedQuantity,
1008
+ BigDecimal availableQuantity,
1009
+ String reason,
1010
+ LocalDateTime failedAt
1011
+ ) implements MaterialEvent {
1012
+ }
1013
+ ```
1014
+
1015
+ </details>
1016
+
1017
+ <details>
1018
+ <summary>コード例: MaterialIssuedEvent.java</summary>
1019
+
1020
+ ```java
1021
+ package com.example.production.material.api.events;
1022
+
1023
+ import java.math.BigDecimal;
1024
+ import java.time.LocalDateTime;
1025
+
1026
+ /**
1027
+ * 資材払出イベント
1028
+ */
1029
+ public record MaterialIssuedEvent(
1030
+ String materialId,
1031
+ String workOrderId,
1032
+ BigDecimal quantity,
1033
+ String issuedBy,
1034
+ LocalDateTime issuedAt
1035
+ ) implements MaterialEvent {
1036
+ }
1037
+ ```
1038
+
1039
+ </details>
1040
+
1041
+ #### ドメインモデル
1042
+
1043
+ <details>
1044
+ <summary>コード例: Material.java</summary>
1045
+
1046
+ ```java
1047
+ package com.example.production.material.domain.model.aggregate.material;
1048
+
1049
+ import com.example.production.material.api.events.*;
1050
+ import java.math.BigDecimal;
1051
+ import java.time.LocalDateTime;
1052
+ import java.util.HashMap;
1053
+ import java.util.Map;
1054
+
1055
+ /**
1056
+ * Material ドメインモデル(資材・在庫管理)
1057
+ */
1058
+ public record Material(
1059
+ String materialId,
1060
+ String itemId,
1061
+ String locationId,
1062
+ BigDecimal totalQuantity,
1063
+ BigDecimal availableQuantity,
1064
+ Map<String, BigDecimal> reservations // workOrderId -> quantity
1065
+ ) {
1066
+
1067
+ public Material {
1068
+ reservations = reservations != null ? new HashMap<>(reservations) : new HashMap<>();
1069
+ }
1070
+
1071
+ /**
1072
+ * 初期化ファクトリ
1073
+ */
1074
+ public static Material initial(String materialId, String itemId, String locationId, BigDecimal initialQuantity) {
1075
+ return new Material(materialId, itemId, locationId, initialQuantity, initialQuantity, new HashMap<>());
1076
+ }
1077
+
1078
+ /**
1079
+ * 資材引当(成功または失敗イベントを返す)
1080
+ */
1081
+ public MaterialEvent reserve(String workOrderId, BigDecimal quantity) {
1082
+ if (reservations.containsKey(workOrderId)) {
1083
+ return new MaterialReservationFailedEvent(
1084
+ materialId, workOrderId, quantity, availableQuantity,
1085
+ "Work order already has a reservation",
1086
+ LocalDateTime.now()
1087
+ );
1088
+ }
1089
+
1090
+ if (availableQuantity.compareTo(quantity) < 0) {
1091
+ return new MaterialReservationFailedEvent(
1092
+ materialId, workOrderId, quantity, availableQuantity,
1093
+ "Insufficient material",
1094
+ LocalDateTime.now()
1095
+ );
1096
+ }
1097
+
1098
+ return new MaterialReservedEvent(materialId, workOrderId, quantity, LocalDateTime.now());
1099
+ }
1100
+
1101
+ /**
1102
+ * 資材払出(製造開始時)
1103
+ */
1104
+ public MaterialIssuedEvent issue(String workOrderId, BigDecimal quantity, String issuedBy) {
1105
+ if (!reservations.containsKey(workOrderId)) {
1106
+ throw new IllegalStateException("No reservation found for work order: " + workOrderId);
1107
+ }
1108
+ return new MaterialIssuedEvent(materialId, workOrderId, quantity, issuedBy, LocalDateTime.now());
1109
+ }
1110
+
1111
+ /**
1112
+ * 資材返却(キャンセル時)
1113
+ */
1114
+ public MaterialReturnedEvent returnMaterial(String workOrderId, BigDecimal quantity, String returnedBy) {
1115
+ return new MaterialReturnedEvent(materialId, workOrderId, quantity, returnedBy, LocalDateTime.now());
1116
+ }
1117
+
1118
+ /**
1119
+ * イベント適用
1120
+ */
1121
+ public Material apply(MaterialEvent event) {
1122
+ return switch (event) {
1123
+ case MaterialInitializedEvent e -> initial(e.materialId(), e.itemId(), e.locationId(), e.initialQuantity());
1124
+ case MaterialReservedEvent e -> {
1125
+ var newReservations = new HashMap<>(reservations);
1126
+ newReservations.put(e.workOrderId(), e.quantity());
1127
+ yield new Material(
1128
+ materialId, itemId, locationId,
1129
+ totalQuantity,
1130
+ availableQuantity.subtract(e.quantity()),
1131
+ newReservations
1132
+ );
1133
+ }
1134
+ case MaterialReservationFailedEvent e -> this; // 状態変更なし
1135
+ case MaterialIssuedEvent e -> {
1136
+ var newReservations = new HashMap<>(reservations);
1137
+ newReservations.remove(e.workOrderId());
1138
+ yield new Material(
1139
+ materialId, itemId, locationId,
1140
+ totalQuantity.subtract(e.quantity()),
1141
+ availableQuantity,
1142
+ newReservations
1143
+ );
1144
+ }
1145
+ case MaterialReturnedEvent e -> new Material(
1146
+ materialId, itemId, locationId,
1147
+ totalQuantity.add(e.quantity()),
1148
+ availableQuantity.add(e.quantity()),
1149
+ reservations
1150
+ );
1151
+ };
1152
+ }
1153
+ }
1154
+ ```
1155
+
1156
+ </details>
1157
+
1158
+ ---
1159
+
1160
+ ## 第40章:Aggregate Adapter と Policy Handler
1161
+
1162
+ ### 40.1 Aggregate Adapter パターン
1163
+
1164
+ Aggregate Adapter は、純粋なドメインモデルと Axon Framework を繋ぐアダプター層です。フレームワーク固有のアノテーションやライフサイクル処理をドメインモデルから分離します。
1165
+
1166
+ ```plantuml
1167
+ @startuml aggregate_adapter
1168
+ skinparam backgroundColor #FEFEFE
1169
+
1170
+ package "Application Layer" {
1171
+ class WorkOrderAggregateAdapter << @Aggregate >> {
1172
+ - workOrderId: String
1173
+ - workOrder: WorkOrder
1174
+ + @CommandHandler: handle(CreateWorkOrderCommand)
1175
+ + @CommandHandler: handle(ReleaseWorkOrderCommand)
1176
+ + @EventSourcingHandler: on(WorkOrderCreatedEvent)
1177
+ + @EventSourcingHandler: on(WorkOrderReleasedEvent)
1178
+ }
1179
+ }
1180
+
1181
+ package "Domain Layer" {
1182
+ class WorkOrder << record >> {
1183
+ + create(): WorkOrderCreatedEvent
1184
+ + release(): WorkOrderReleasedEvent
1185
+ + apply(event): WorkOrder
1186
+ }
1187
+ }
1188
+
1189
+ WorkOrderAggregateAdapter --> WorkOrder : "委譲"
1190
+
1191
+ note bottom of WorkOrderAggregateAdapter
1192
+ Axon Framework 用アダプター
1193
+ フレームワーク依存を吸収
1194
+ end note
1195
+
1196
+ note bottom of WorkOrder
1197
+ 純粋なドメインモデル
1198
+ Axon に依存しない
1199
+ end note
1200
+
1201
+ @enduml
1202
+ ```
1203
+
1204
+ ---
1205
+
1206
+ ### 40.2 WorkOrder Aggregate Adapter
1207
+
1208
+ <details>
1209
+ <summary>コード例: WorkOrderAggregateAdapter.java</summary>
1210
+
1211
+ ```java
1212
+ package com.example.production.workorder.application.aggregate;
1213
+
1214
+ import com.example.production.workorder.api.events.*;
1215
+ import com.example.production.workorder.domain.model.aggregate.workorder.WorkOrder;
1216
+ import com.example.production.workorder.domain.model.aggregate.workorder.WorkOrderCommands.*;
1217
+ import org.axonframework.commandhandling.CommandHandler;
1218
+ import org.axonframework.eventsourcing.EventSourcingHandler;
1219
+ import org.axonframework.modelling.command.AggregateIdentifier;
1220
+ import org.axonframework.modelling.command.AggregateLifecycle;
1221
+ import org.axonframework.spring.stereotype.Aggregate;
1222
+
1223
+ /**
1224
+ * WorkOrder Aggregate Adapter(Axon Framework 用)
1225
+ */
1226
+ @Aggregate
1227
+ public class WorkOrderAggregateAdapter {
1228
+
1229
+ @AggregateIdentifier
1230
+ private String workOrderId;
1231
+
1232
+ private WorkOrder workOrder;
1233
+
1234
+ /**
1235
+ * Axon が使用するデフォルトコンストラクタ
1236
+ */
1237
+ protected WorkOrderAggregateAdapter() {
1238
+ }
1239
+
1240
+ // ======== Command Handlers ========
1241
+
1242
+ /**
1243
+ * 集約作成コマンドハンドラー(コンストラクタ)
1244
+ */
1245
+ @CommandHandler
1246
+ public WorkOrderAggregateAdapter(CreateWorkOrderCommand command) {
1247
+ // ドメインモデルのファクトリメソッドを呼び出し
1248
+ var event = WorkOrder.create(
1249
+ command.workOrderId(),
1250
+ command.itemId(),
1251
+ command.itemName(),
1252
+ command.orderQuantity(),
1253
+ command.unitOfMeasure(),
1254
+ command.plannedStartDate(),
1255
+ command.plannedEndDate(),
1256
+ command.workCenterId(),
1257
+ command.createdBy()
1258
+ );
1259
+ // イベントを発行
1260
+ AggregateLifecycle.apply(event);
1261
+ }
1262
+
1263
+ /**
1264
+ * リリースコマンドハンドラー
1265
+ */
1266
+ @CommandHandler
1267
+ public void handle(ReleaseWorkOrderCommand command) {
1268
+ var event = workOrder.release(command.releasedBy());
1269
+ AggregateLifecycle.apply(event);
1270
+ }
1271
+
1272
+ /**
1273
+ * 却下コマンドハンドラー
1274
+ */
1275
+ @CommandHandler
1276
+ public void handle(RejectWorkOrderCommand command) {
1277
+ var event = workOrder.reject(command.reason());
1278
+ AggregateLifecycle.apply(event);
1279
+ }
1280
+
1281
+ /**
1282
+ * 製造開始コマンドハンドラー
1283
+ */
1284
+ @CommandHandler
1285
+ public void handle(StartWorkOrderCommand command) {
1286
+ var event = workOrder.start(command.operatorId());
1287
+ AggregateLifecycle.apply(event);
1288
+ }
1289
+
1290
+ /**
1291
+ * 中断コマンドハンドラー
1292
+ */
1293
+ @CommandHandler
1294
+ public void handle(SuspendWorkOrderCommand command) {
1295
+ var event = workOrder.suspend(command.reason(), command.suspendedBy());
1296
+ AggregateLifecycle.apply(event);
1297
+ }
1298
+
1299
+ /**
1300
+ * 再開コマンドハンドラー
1301
+ */
1302
+ @CommandHandler
1303
+ public void handle(ResumeWorkOrderCommand command) {
1304
+ var event = workOrder.resume(command.resumedBy());
1305
+ AggregateLifecycle.apply(event);
1306
+ }
1307
+
1308
+ /**
1309
+ * 完了コマンドハンドラー
1310
+ */
1311
+ @CommandHandler
1312
+ public void handle(CompleteWorkOrderCommand command) {
1313
+ var event = workOrder.complete(
1314
+ command.completedQuantity(),
1315
+ command.defectQuantity(),
1316
+ command.operatorId()
1317
+ );
1318
+ AggregateLifecycle.apply(event);
1319
+ }
1320
+
1321
+ /**
1322
+ * キャンセルコマンドハンドラー
1323
+ */
1324
+ @CommandHandler
1325
+ public void handle(CancelWorkOrderCommand command) {
1326
+ var event = workOrder.cancel(command.cancelledBy(), command.reason());
1327
+ AggregateLifecycle.apply(event);
1328
+ }
1329
+
1330
+ // ======== Event Sourcing Handlers(状態復元)========
1331
+
1332
+ @EventSourcingHandler
1333
+ public void on(WorkOrderCreatedEvent event) {
1334
+ this.workOrderId = event.workOrderId();
1335
+ this.workOrder = WorkOrder.from(event);
1336
+ }
1337
+
1338
+ @EventSourcingHandler
1339
+ public void on(WorkOrderReleasedEvent event) {
1340
+ this.workOrder = workOrder.apply(event);
1341
+ }
1342
+
1343
+ @EventSourcingHandler
1344
+ public void on(WorkOrderRejectedEvent event) {
1345
+ this.workOrder = workOrder.apply(event);
1346
+ }
1347
+
1348
+ @EventSourcingHandler
1349
+ public void on(WorkOrderStartedEvent event) {
1350
+ this.workOrder = workOrder.apply(event);
1351
+ }
1352
+
1353
+ @EventSourcingHandler
1354
+ public void on(WorkOrderSuspendedEvent event) {
1355
+ this.workOrder = workOrder.apply(event);
1356
+ }
1357
+
1358
+ @EventSourcingHandler
1359
+ public void on(WorkOrderResumedEvent event) {
1360
+ this.workOrder = workOrder.apply(event);
1361
+ }
1362
+
1363
+ @EventSourcingHandler
1364
+ public void on(WorkOrderCompletedEvent event) {
1365
+ this.workOrder = workOrder.apply(event);
1366
+ }
1367
+
1368
+ @EventSourcingHandler
1369
+ public void on(WorkOrderCancelledEvent event) {
1370
+ this.workOrder = workOrder.apply(event);
1371
+ }
1372
+ }
1373
+ ```
1374
+
1375
+ </details>
1376
+
1377
+ **Axon アノテーション解説:**
1378
+
1379
+ | アノテーション | 説明 |
1380
+ |--------------|------|
1381
+ | `@Aggregate` | Event Sourcing 集約であることを宣言 |
1382
+ | `@AggregateIdentifier` | 集約の識別子フィールド |
1383
+ | `@CommandHandler` | コマンドを処理するメソッド |
1384
+ | `@EventSourcingHandler` | イベントから状態を復元するメソッド |
1385
+ | `AggregateLifecycle.apply()` | イベントを発行するメソッド |
1386
+
1387
+ ---
1388
+
1389
+ ### 40.3 Material Aggregate Adapter
1390
+
1391
+ <details>
1392
+ <summary>コード例: MaterialAggregateAdapter.java</summary>
1393
+
1394
+ ```java
1395
+ package com.example.production.material.application.aggregate;
1396
+
1397
+ import com.example.production.material.api.events.*;
1398
+ import com.example.production.material.domain.model.aggregate.material.Material;
1399
+ import com.example.production.material.domain.model.aggregate.material.MaterialCommands.*;
1400
+ import org.axonframework.commandhandling.CommandHandler;
1401
+ import org.axonframework.eventsourcing.EventSourcingHandler;
1402
+ import org.axonframework.modelling.command.AggregateIdentifier;
1403
+ import org.axonframework.modelling.command.AggregateLifecycle;
1404
+ import org.axonframework.spring.stereotype.Aggregate;
1405
+
1406
+ import java.time.LocalDateTime;
1407
+
1408
+ /**
1409
+ * Material Aggregate Adapter
1410
+ */
1411
+ @Aggregate
1412
+ public class MaterialAggregateAdapter {
1413
+
1414
+ @AggregateIdentifier
1415
+ private String materialId;
1416
+
1417
+ private Material material;
1418
+
1419
+ protected MaterialAggregateAdapter() {
1420
+ }
1421
+
1422
+ @CommandHandler
1423
+ public MaterialAggregateAdapter(InitializeMaterialCommand command) {
1424
+ AggregateLifecycle.apply(new MaterialInitializedEvent(
1425
+ command.materialId(),
1426
+ command.itemId(),
1427
+ command.locationId(),
1428
+ command.initialQuantity(),
1429
+ LocalDateTime.now()
1430
+ ));
1431
+ }
1432
+
1433
+ @CommandHandler
1434
+ public void handle(ReserveMaterialCommand command) {
1435
+ var event = material.reserve(command.workOrderId(), command.quantity());
1436
+ AggregateLifecycle.apply(event);
1437
+ }
1438
+
1439
+ @CommandHandler
1440
+ public void handle(IssueMaterialCommand command) {
1441
+ var event = material.issue(command.workOrderId(), command.quantity(), command.issuedBy());
1442
+ AggregateLifecycle.apply(event);
1443
+ }
1444
+
1445
+ @CommandHandler
1446
+ public void handle(ReturnMaterialCommand command) {
1447
+ var event = material.returnMaterial(command.workOrderId(), command.quantity(), command.returnedBy());
1448
+ AggregateLifecycle.apply(event);
1449
+ }
1450
+
1451
+ @EventSourcingHandler
1452
+ public void on(MaterialInitializedEvent event) {
1453
+ this.materialId = event.materialId();
1454
+ this.material = Material.initial(
1455
+ event.materialId(),
1456
+ event.itemId(),
1457
+ event.locationId(),
1458
+ event.initialQuantity()
1459
+ );
1460
+ }
1461
+
1462
+ @EventSourcingHandler
1463
+ public void on(MaterialReservedEvent event) {
1464
+ this.material = material.apply(event);
1465
+ }
1466
+
1467
+ @EventSourcingHandler
1468
+ public void on(MaterialReservationFailedEvent event) {
1469
+ // 状態変更なし
1470
+ }
1471
+
1472
+ @EventSourcingHandler
1473
+ public void on(MaterialIssuedEvent event) {
1474
+ this.material = material.apply(event);
1475
+ }
1476
+
1477
+ @EventSourcingHandler
1478
+ public void on(MaterialReturnedEvent event) {
1479
+ this.material = material.apply(event);
1480
+ }
1481
+ }
1482
+ ```
1483
+
1484
+ </details>
1485
+
1486
+ ---
1487
+
1488
+ ### 40.4 Policy Handler(Choreography パターン)
1489
+
1490
+ Policy Handler は、他の Bounded Context から発行されたイベントを購読し、自律的に反応する処理を実装します。Choreography パターンでは、各 Context が独立して動作し、イベントを介して連携します。
1491
+
1492
+ ```plantuml
1493
+ @startuml choreography
1494
+ skinparam backgroundColor #FEFEFE
1495
+
1496
+ participant "WorkOrder Context" as WorkOrder
1497
+ participant "Event Bus" as Bus
1498
+ participant "Material Context" as Material
1499
+
1500
+ == 製造指図作成フロー ==
1501
+ WorkOrder -> Bus : WorkOrderCreatedEvent
1502
+ Bus -> Material : WorkOrderCreatedEvent
1503
+ Material -> Material : reserve()
1504
+ alt 資材あり
1505
+ Material -> Bus : MaterialReservedEvent
1506
+ Bus -> WorkOrder : MaterialReservedEvent
1507
+ WorkOrder -> WorkOrder : release()
1508
+ else 資材不足
1509
+ Material -> Bus : MaterialReservationFailedEvent
1510
+ Bus -> WorkOrder : MaterialReservationFailedEvent
1511
+ WorkOrder -> WorkOrder : reject()
1512
+ end
1513
+
1514
+ == 製造開始フロー ==
1515
+ WorkOrder -> Bus : WorkOrderStartedEvent
1516
+ Bus -> Material : WorkOrderStartedEvent
1517
+ Material -> Material : issue()
1518
+ Material -> Bus : MaterialIssuedEvent
1519
+
1520
+ == キャンセルフロー ==
1521
+ WorkOrder -> Bus : WorkOrderCancelledEvent
1522
+ Bus -> Material : WorkOrderCancelledEvent
1523
+ Material -> Material : returnMaterial()
1524
+ Material -> Bus : MaterialReturnedEvent
1525
+
1526
+ @enduml
1527
+ ```
1528
+
1529
+ ---
1530
+
1531
+ ### 40.5 WorkOrderEventHandler(Material Context 内)
1532
+
1533
+ <details>
1534
+ <summary>コード例: WorkOrderEventHandler.java</summary>
1535
+
1536
+ ```java
1537
+ package com.example.production.material.application.policy;
1538
+
1539
+ import com.example.production.material.domain.model.aggregate.material.MaterialCommands.*;
1540
+ import com.example.production.workorder.api.events.WorkOrderCancelledEvent;
1541
+ import com.example.production.workorder.api.events.WorkOrderCreatedEvent;
1542
+ import com.example.production.workorder.api.events.WorkOrderStartedEvent;
1543
+ import org.axonframework.commandhandling.gateway.CommandGateway;
1544
+ import org.axonframework.eventhandling.EventHandler;
1545
+ import org.springframework.stereotype.Component;
1546
+
1547
+ import java.math.BigDecimal;
1548
+
1549
+ /**
1550
+ * WorkOrder イベントに反応して資材操作を実行する Policy Handler
1551
+ */
1552
+ @Component
1553
+ public class WorkOrderEventHandler {
1554
+
1555
+ private final CommandGateway commandGateway;
1556
+
1557
+ public WorkOrderEventHandler(CommandGateway commandGateway) {
1558
+ this.commandGateway = commandGateway;
1559
+ }
1560
+
1561
+ /**
1562
+ * Policy: "When a work order is created, reserve material"
1563
+ */
1564
+ @EventHandler
1565
+ public void on(WorkOrderCreatedEvent event) {
1566
+ // BOM に基づいて必要な資材を引当
1567
+ // 簡略化のため、単一の資材を引当
1568
+ var command = new ReserveMaterialCommand(
1569
+ "material-" + event.itemId(), // 資材ID(簡略化)
1570
+ event.workOrderId(),
1571
+ event.orderQuantity()
1572
+ );
1573
+ commandGateway.send(command);
1574
+ }
1575
+
1576
+ /**
1577
+ * Policy: "When a work order is started, issue material"
1578
+ */
1579
+ @EventHandler
1580
+ public void on(WorkOrderStartedEvent event) {
1581
+ // 製造開始時に資材を払出
1582
+ // 実際には workOrderId から必要な資材を取得
1583
+ var command = new IssueMaterialCommand(
1584
+ "material-default", // 資材ID(簡略化)
1585
+ event.workOrderId(),
1586
+ BigDecimal.ONE, // 数量(簡略化)
1587
+ event.operatorId()
1588
+ );
1589
+ commandGateway.send(command);
1590
+ }
1591
+
1592
+ /**
1593
+ * Policy: "When a work order is cancelled, return material"
1594
+ */
1595
+ @EventHandler
1596
+ public void on(WorkOrderCancelledEvent event) {
1597
+ var command = new ReturnMaterialCommand(
1598
+ "material-" + event.itemId(),
1599
+ event.workOrderId(),
1600
+ BigDecimal.ONE, // 数量(簡略化)
1601
+ event.cancelledBy()
1602
+ );
1603
+ commandGateway.send(command);
1604
+ }
1605
+ }
1606
+ ```
1607
+
1608
+ </details>
1609
+
1610
+ ---
1611
+
1612
+ ### 40.6 MaterialEventHandler(WorkOrder Context 内)
1613
+
1614
+ <details>
1615
+ <summary>コード例: MaterialEventHandler.java</summary>
1616
+
1617
+ ```java
1618
+ package com.example.production.workorder.application.policy;
1619
+
1620
+ import com.example.production.material.api.events.MaterialReservationFailedEvent;
1621
+ import com.example.production.material.api.events.MaterialReservedEvent;
1622
+ import com.example.production.workorder.domain.model.aggregate.workorder.WorkOrderCommands.*;
1623
+ import org.axonframework.commandhandling.gateway.CommandGateway;
1624
+ import org.axonframework.eventhandling.EventHandler;
1625
+ import org.springframework.stereotype.Component;
1626
+
1627
+ /**
1628
+ * Material イベントに反応して製造指図状態を更新する Policy Handler
1629
+ */
1630
+ @Component
1631
+ public class MaterialEventHandler {
1632
+
1633
+ private final CommandGateway commandGateway;
1634
+
1635
+ public MaterialEventHandler(CommandGateway commandGateway) {
1636
+ this.commandGateway = commandGateway;
1637
+ }
1638
+
1639
+ /**
1640
+ * Policy: "When material is reserved, release the work order"
1641
+ */
1642
+ @EventHandler
1643
+ public void on(MaterialReservedEvent event) {
1644
+ var command = new ReleaseWorkOrderCommand(event.workOrderId(), "system");
1645
+ commandGateway.send(command);
1646
+ }
1647
+
1648
+ /**
1649
+ * Policy: "When material reservation fails, reject the work order"
1650
+ */
1651
+ @EventHandler
1652
+ public void on(MaterialReservationFailedEvent event) {
1653
+ var command = new RejectWorkOrderCommand(event.workOrderId(), event.reason());
1654
+ commandGateway.send(command);
1655
+ }
1656
+ }
1657
+ ```
1658
+
1659
+ </details>
1660
+
1661
+ **Policy Handler の設計原則:**
1662
+
1663
+ | 原則 | 説明 |
1664
+ |------|------|
1665
+ | **単一責任** | 1 つの Policy Handler は 1 つの関心事のみを扱う |
1666
+ | **疎結合** | 他の Context のイベントを購読し、自 Context のコマンドを発行 |
1667
+ | **自律性** | 他のサービスに依存せず独立して動作 |
1668
+ | **冪等性** | 同じイベントを複数回受信しても問題ないよう設計 |
1669
+
1670
+ ---
1671
+
1672
+ ## 第41章:Projection と Read Model
1673
+
1674
+ ### 41.1 Projection の役割
1675
+
1676
+ Projection は、イベントを購読して Read Model(クエリ用のデータモデル)を更新するコンポーネントです。CQRS では、書き込みモデル(Event Store)と読み取りモデル(Read Model DB)を分離し、それぞれに最適化されたデータ構造を使用します。
1677
+
1678
+ ```plantuml
1679
+ @startuml projection
1680
+ skinparam backgroundColor #FEFEFE
1681
+
1682
+ participant "CommandHandler" as CH
1683
+ participant "Event Store" as ES
1684
+ participant "Projection" as P
1685
+ participant "MyBatis Mapper" as MB
1686
+ participant "Read Model DB" as DB
1687
+ participant "REST Controller" as RC
1688
+
1689
+ == 書き込み(Command Side)==
1690
+ CH -> ES : イベント保存
1691
+ ES -> P : イベント通知
1692
+
1693
+ == 読み取りモデル更新 ==
1694
+ P -> MB : insert/update
1695
+ MB -> DB : SQL 実行
1696
+
1697
+ == 読み取り(Query Side)==
1698
+ RC -> MB : findById
1699
+ MB -> DB : SELECT
1700
+ DB --> MB : 結果
1701
+ MB --> RC : Entity
1702
+
1703
+ note right of P
1704
+ Projection は
1705
+ イベントを購読して
1706
+ MyBatis 経由で
1707
+ Read Model を更新
1708
+ end note
1709
+
1710
+ @enduml
1711
+ ```
1712
+
1713
+ ---
1714
+
1715
+ ### 41.2 WorkOrder Projection(MyBatis 版)
1716
+
1717
+ <details>
1718
+ <summary>コード例: WorkOrderProjection.java</summary>
1719
+
1720
+ ```java
1721
+ package com.example.production.workorder.application.query;
1722
+
1723
+ import com.example.production.workorder.adapter.outbound.persistence.entity.WorkOrderEntity;
1724
+ import com.example.production.workorder.adapter.outbound.persistence.mapper.WorkOrderMapper;
1725
+ import com.example.production.workorder.api.events.*;
1726
+ import org.axonframework.eventhandling.EventHandler;
1727
+ import org.springframework.stereotype.Component;
1728
+
1729
+ /**
1730
+ * WorkOrder Projection(MyBatis を使用した Read Model の更新)
1731
+ */
1732
+ @Component
1733
+ public class WorkOrderProjection {
1734
+
1735
+ private final WorkOrderMapper workOrderMapper;
1736
+
1737
+ public WorkOrderProjection(WorkOrderMapper workOrderMapper) {
1738
+ this.workOrderMapper = workOrderMapper;
1739
+ }
1740
+
1741
+ @EventHandler
1742
+ public void on(WorkOrderCreatedEvent event) {
1743
+ var entity = new WorkOrderEntity();
1744
+ entity.setWorkOrderId(event.workOrderId());
1745
+ entity.setItemId(event.itemId());
1746
+ entity.setItemName(event.itemName());
1747
+ entity.setOrderQuantity(event.orderQuantity());
1748
+ entity.setUnitOfMeasure(event.unitOfMeasure());
1749
+ entity.setPlannedStartDate(event.plannedStartDate());
1750
+ entity.setPlannedEndDate(event.plannedEndDate());
1751
+ entity.setWorkCenterId(event.workCenterId());
1752
+ entity.setStatus("CREATED");
1753
+ entity.setCreatedBy(event.createdBy());
1754
+ entity.setCreatedAt(event.createdAt());
1755
+
1756
+ workOrderMapper.insert(entity);
1757
+ }
1758
+
1759
+ @EventHandler
1760
+ public void on(WorkOrderReleasedEvent event) {
1761
+ workOrderMapper.updateStatus(event.workOrderId(), "RELEASED");
1762
+ workOrderMapper.updateReleasedAt(event.workOrderId(), event.releasedBy(), event.releasedAt());
1763
+ }
1764
+
1765
+ @EventHandler
1766
+ public void on(WorkOrderRejectedEvent event) {
1767
+ workOrderMapper.updateStatus(event.workOrderId(), "REJECTED");
1768
+ workOrderMapper.updateRejection(event.workOrderId(), event.reason(), event.rejectedAt());
1769
+ }
1770
+
1771
+ @EventHandler
1772
+ public void on(WorkOrderStartedEvent event) {
1773
+ workOrderMapper.updateStatus(event.workOrderId(), "IN_PROGRESS");
1774
+ workOrderMapper.updateStarted(event.workOrderId(), event.operatorId(), event.actualStartTime());
1775
+ }
1776
+
1777
+ @EventHandler
1778
+ public void on(WorkOrderSuspendedEvent event) {
1779
+ workOrderMapper.updateStatus(event.workOrderId(), "SUSPENDED");
1780
+ workOrderMapper.updateSuspended(event.workOrderId(), event.reason(), event.suspendedBy(), event.suspendedAt());
1781
+ }
1782
+
1783
+ @EventHandler
1784
+ public void on(WorkOrderResumedEvent event) {
1785
+ workOrderMapper.updateStatus(event.workOrderId(), "IN_PROGRESS");
1786
+ workOrderMapper.updateResumed(event.workOrderId(), event.resumedBy(), event.resumedAt());
1787
+ }
1788
+
1789
+ @EventHandler
1790
+ public void on(WorkOrderCompletedEvent event) {
1791
+ workOrderMapper.updateStatus(event.workOrderId(), "COMPLETED");
1792
+ workOrderMapper.updateCompleted(
1793
+ event.workOrderId(),
1794
+ event.completedQuantity(),
1795
+ event.defectQuantity(),
1796
+ event.operatorId(),
1797
+ event.actualEndTime()
1798
+ );
1799
+ }
1800
+
1801
+ @EventHandler
1802
+ public void on(WorkOrderCancelledEvent event) {
1803
+ workOrderMapper.updateStatus(event.workOrderId(), "CANCELLED");
1804
+ workOrderMapper.updateCancellation(
1805
+ event.workOrderId(),
1806
+ event.cancelledBy(),
1807
+ event.reason(),
1808
+ event.cancelledAt()
1809
+ );
1810
+ }
1811
+ }
1812
+ ```
1813
+
1814
+ </details>
1815
+
1816
+ ---
1817
+
1818
+ ### 41.3 Read Model Entity
1819
+
1820
+ <details>
1821
+ <summary>コード例: WorkOrderEntity.java</summary>
1822
+
1823
+ ```java
1824
+ package com.example.production.workorder.adapter.outbound.persistence.entity;
1825
+
1826
+ import java.math.BigDecimal;
1827
+ import java.time.LocalDate;
1828
+ import java.time.LocalDateTime;
1829
+
1830
+ /**
1831
+ * WorkOrder Read Model(表示用のすべてのフィールドを保持)
1832
+ */
1833
+ public class WorkOrderEntity {
1834
+
1835
+ private String workOrderId;
1836
+ private String itemId;
1837
+ private String itemName;
1838
+ private BigDecimal orderQuantity;
1839
+ private String unitOfMeasure;
1840
+ private LocalDate plannedStartDate;
1841
+ private LocalDate plannedEndDate;
1842
+ private String workCenterId;
1843
+ private String status;
1844
+ private String createdBy;
1845
+ private LocalDateTime createdAt;
1846
+ private String releasedBy;
1847
+ private LocalDateTime releasedAt;
1848
+ private String rejectionReason;
1849
+ private LocalDateTime rejectedAt;
1850
+ private String operatorId;
1851
+ private LocalDateTime actualStartTime;
1852
+ private LocalDateTime actualEndTime;
1853
+ private BigDecimal completedQuantity;
1854
+ private BigDecimal defectQuantity;
1855
+ private String suspendReason;
1856
+ private String suspendedBy;
1857
+ private LocalDateTime suspendedAt;
1858
+ private String resumedBy;
1859
+ private LocalDateTime resumedAt;
1860
+ private String cancelledBy;
1861
+ private String cancellationReason;
1862
+ private LocalDateTime cancelledAt;
1863
+
1864
+ // Getters and Setters
1865
+ public String getWorkOrderId() { return workOrderId; }
1866
+ public void setWorkOrderId(String workOrderId) { this.workOrderId = workOrderId; }
1867
+
1868
+ public String getItemId() { return itemId; }
1869
+ public void setItemId(String itemId) { this.itemId = itemId; }
1870
+
1871
+ public String getItemName() { return itemName; }
1872
+ public void setItemName(String itemName) { this.itemName = itemName; }
1873
+
1874
+ public BigDecimal getOrderQuantity() { return orderQuantity; }
1875
+ public void setOrderQuantity(BigDecimal orderQuantity) { this.orderQuantity = orderQuantity; }
1876
+
1877
+ public String getUnitOfMeasure() { return unitOfMeasure; }
1878
+ public void setUnitOfMeasure(String unitOfMeasure) { this.unitOfMeasure = unitOfMeasure; }
1879
+
1880
+ public LocalDate getPlannedStartDate() { return plannedStartDate; }
1881
+ public void setPlannedStartDate(LocalDate plannedStartDate) { this.plannedStartDate = plannedStartDate; }
1882
+
1883
+ public LocalDate getPlannedEndDate() { return plannedEndDate; }
1884
+ public void setPlannedEndDate(LocalDate plannedEndDate) { this.plannedEndDate = plannedEndDate; }
1885
+
1886
+ public String getWorkCenterId() { return workCenterId; }
1887
+ public void setWorkCenterId(String workCenterId) { this.workCenterId = workCenterId; }
1888
+
1889
+ public String getStatus() { return status; }
1890
+ public void setStatus(String status) { this.status = status; }
1891
+
1892
+ public String getCreatedBy() { return createdBy; }
1893
+ public void setCreatedBy(String createdBy) { this.createdBy = createdBy; }
1894
+
1895
+ public LocalDateTime getCreatedAt() { return createdAt; }
1896
+ public void setCreatedAt(LocalDateTime createdAt) { this.createdAt = createdAt; }
1897
+
1898
+ public String getReleasedBy() { return releasedBy; }
1899
+ public void setReleasedBy(String releasedBy) { this.releasedBy = releasedBy; }
1900
+
1901
+ public LocalDateTime getReleasedAt() { return releasedAt; }
1902
+ public void setReleasedAt(LocalDateTime releasedAt) { this.releasedAt = releasedAt; }
1903
+
1904
+ public String getRejectionReason() { return rejectionReason; }
1905
+ public void setRejectionReason(String rejectionReason) { this.rejectionReason = rejectionReason; }
1906
+
1907
+ public LocalDateTime getRejectedAt() { return rejectedAt; }
1908
+ public void setRejectedAt(LocalDateTime rejectedAt) { this.rejectedAt = rejectedAt; }
1909
+
1910
+ public String getOperatorId() { return operatorId; }
1911
+ public void setOperatorId(String operatorId) { this.operatorId = operatorId; }
1912
+
1913
+ public LocalDateTime getActualStartTime() { return actualStartTime; }
1914
+ public void setActualStartTime(LocalDateTime actualStartTime) { this.actualStartTime = actualStartTime; }
1915
+
1916
+ public LocalDateTime getActualEndTime() { return actualEndTime; }
1917
+ public void setActualEndTime(LocalDateTime actualEndTime) { this.actualEndTime = actualEndTime; }
1918
+
1919
+ public BigDecimal getCompletedQuantity() { return completedQuantity; }
1920
+ public void setCompletedQuantity(BigDecimal completedQuantity) { this.completedQuantity = completedQuantity; }
1921
+
1922
+ public BigDecimal getDefectQuantity() { return defectQuantity; }
1923
+ public void setDefectQuantity(BigDecimal defectQuantity) { this.defectQuantity = defectQuantity; }
1924
+
1925
+ public String getSuspendReason() { return suspendReason; }
1926
+ public void setSuspendReason(String suspendReason) { this.suspendReason = suspendReason; }
1927
+
1928
+ public String getSuspendedBy() { return suspendedBy; }
1929
+ public void setSuspendedBy(String suspendedBy) { this.suspendedBy = suspendedBy; }
1930
+
1931
+ public LocalDateTime getSuspendedAt() { return suspendedAt; }
1932
+ public void setSuspendedAt(LocalDateTime suspendedAt) { this.suspendedAt = suspendedAt; }
1933
+
1934
+ public String getResumedBy() { return resumedBy; }
1935
+ public void setResumedBy(String resumedBy) { this.resumedBy = resumedBy; }
1936
+
1937
+ public LocalDateTime getResumedAt() { return resumedAt; }
1938
+ public void setResumedAt(LocalDateTime resumedAt) { this.resumedAt = resumedAt; }
1939
+
1940
+ public String getCancelledBy() { return cancelledBy; }
1941
+ public void setCancelledBy(String cancelledBy) { this.cancelledBy = cancelledBy; }
1942
+
1943
+ public String getCancellationReason() { return cancellationReason; }
1944
+ public void setCancellationReason(String cancellationReason) { this.cancellationReason = cancellationReason; }
1945
+
1946
+ public LocalDateTime getCancelledAt() { return cancelledAt; }
1947
+ public void setCancelledAt(LocalDateTime cancelledAt) { this.cancelledAt = cancelledAt; }
1948
+ }
1949
+ ```
1950
+
1951
+ </details>
1952
+
1953
+ ---
1954
+
1955
+ ### 41.4 MyBatis Mapper インターフェース
1956
+
1957
+ <details>
1958
+ <summary>コード例: WorkOrderMapper.java</summary>
1959
+
1960
+ ```java
1961
+ package com.example.production.workorder.adapter.outbound.persistence.mapper;
1962
+
1963
+ import com.example.production.workorder.adapter.outbound.persistence.entity.WorkOrderEntity;
1964
+ import org.apache.ibatis.annotations.Mapper;
1965
+ import org.apache.ibatis.annotations.Param;
1966
+
1967
+ import java.math.BigDecimal;
1968
+ import java.time.LocalDateTime;
1969
+ import java.util.List;
1970
+ import java.util.Optional;
1971
+
1972
+ /**
1973
+ * WorkOrder MyBatis Mapper
1974
+ */
1975
+ @Mapper
1976
+ public interface WorkOrderMapper {
1977
+
1978
+ // 挿入
1979
+ void insert(WorkOrderEntity entity);
1980
+
1981
+ // 検索
1982
+ Optional<WorkOrderEntity> findById(@Param("workOrderId") String workOrderId);
1983
+
1984
+ List<WorkOrderEntity> findAll();
1985
+
1986
+ List<WorkOrderEntity> findByItemId(@Param("itemId") String itemId);
1987
+
1988
+ List<WorkOrderEntity> findByStatus(@Param("status") String status);
1989
+
1990
+ List<WorkOrderEntity> findByWorkCenterId(@Param("workCenterId") String workCenterId);
1991
+
1992
+ // 更新
1993
+ void updateStatus(@Param("workOrderId") String workOrderId, @Param("status") String status);
1994
+
1995
+ void updateReleasedAt(
1996
+ @Param("workOrderId") String workOrderId,
1997
+ @Param("releasedBy") String releasedBy,
1998
+ @Param("releasedAt") LocalDateTime releasedAt
1999
+ );
2000
+
2001
+ void updateRejection(
2002
+ @Param("workOrderId") String workOrderId,
2003
+ @Param("rejectionReason") String rejectionReason,
2004
+ @Param("rejectedAt") LocalDateTime rejectedAt
2005
+ );
2006
+
2007
+ void updateStarted(
2008
+ @Param("workOrderId") String workOrderId,
2009
+ @Param("operatorId") String operatorId,
2010
+ @Param("actualStartTime") LocalDateTime actualStartTime
2011
+ );
2012
+
2013
+ void updateSuspended(
2014
+ @Param("workOrderId") String workOrderId,
2015
+ @Param("suspendReason") String suspendReason,
2016
+ @Param("suspendedBy") String suspendedBy,
2017
+ @Param("suspendedAt") LocalDateTime suspendedAt
2018
+ );
2019
+
2020
+ void updateResumed(
2021
+ @Param("workOrderId") String workOrderId,
2022
+ @Param("resumedBy") String resumedBy,
2023
+ @Param("resumedAt") LocalDateTime resumedAt
2024
+ );
2025
+
2026
+ void updateCompleted(
2027
+ @Param("workOrderId") String workOrderId,
2028
+ @Param("completedQuantity") BigDecimal completedQuantity,
2029
+ @Param("defectQuantity") BigDecimal defectQuantity,
2030
+ @Param("operatorId") String operatorId,
2031
+ @Param("actualEndTime") LocalDateTime actualEndTime
2032
+ );
2033
+
2034
+ void updateCancellation(
2035
+ @Param("workOrderId") String workOrderId,
2036
+ @Param("cancelledBy") String cancelledBy,
2037
+ @Param("cancellationReason") String cancellationReason,
2038
+ @Param("cancelledAt") LocalDateTime cancelledAt
2039
+ );
2040
+ }
2041
+ ```
2042
+
2043
+ </details>
2044
+
2045
+ ---
2046
+
2047
+ ### 41.5 MyBatis Mapper XML
2048
+
2049
+ <details>
2050
+ <summary>コード例: WorkOrderMapper.xml</summary>
2051
+
2052
+ ```xml
2053
+ <?xml version="1.0" encoding="UTF-8" ?>
2054
+ <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
2055
+ "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
2056
+
2057
+ <mapper namespace="com.example.production.workorder.adapter.outbound.persistence.mapper.WorkOrderMapper">
2058
+
2059
+ <resultMap id="workOrderResultMap" type="WorkOrderEntity">
2060
+ <id property="workOrderId" column="work_order_id"/>
2061
+ <result property="itemId" column="item_id"/>
2062
+ <result property="itemName" column="item_name"/>
2063
+ <result property="orderQuantity" column="order_quantity"/>
2064
+ <result property="unitOfMeasure" column="unit_of_measure"/>
2065
+ <result property="plannedStartDate" column="planned_start_date"/>
2066
+ <result property="plannedEndDate" column="planned_end_date"/>
2067
+ <result property="workCenterId" column="work_center_id"/>
2068
+ <result property="status" column="status"/>
2069
+ <result property="createdBy" column="created_by"/>
2070
+ <result property="createdAt" column="created_at"/>
2071
+ <result property="releasedBy" column="released_by"/>
2072
+ <result property="releasedAt" column="released_at"/>
2073
+ <result property="rejectionReason" column="rejection_reason"/>
2074
+ <result property="rejectedAt" column="rejected_at"/>
2075
+ <result property="operatorId" column="operator_id"/>
2076
+ <result property="actualStartTime" column="actual_start_time"/>
2077
+ <result property="actualEndTime" column="actual_end_time"/>
2078
+ <result property="completedQuantity" column="completed_quantity"/>
2079
+ <result property="defectQuantity" column="defect_quantity"/>
2080
+ <result property="suspendReason" column="suspend_reason"/>
2081
+ <result property="suspendedBy" column="suspended_by"/>
2082
+ <result property="suspendedAt" column="suspended_at"/>
2083
+ <result property="resumedBy" column="resumed_by"/>
2084
+ <result property="resumedAt" column="resumed_at"/>
2085
+ <result property="cancelledBy" column="cancelled_by"/>
2086
+ <result property="cancellationReason" column="cancellation_reason"/>
2087
+ <result property="cancelledAt" column="cancelled_at"/>
2088
+ </resultMap>
2089
+
2090
+ <insert id="insert" parameterType="WorkOrderEntity">
2091
+ INSERT INTO work_orders (
2092
+ work_order_id, item_id, item_name, order_quantity, unit_of_measure,
2093
+ planned_start_date, planned_end_date, work_center_id, status,
2094
+ created_by, created_at
2095
+ ) VALUES (
2096
+ #{workOrderId}, #{itemId}, #{itemName}, #{orderQuantity}, #{unitOfMeasure},
2097
+ #{plannedStartDate}, #{plannedEndDate}, #{workCenterId}, #{status},
2098
+ #{createdBy}, #{createdAt}
2099
+ )
2100
+ </insert>
2101
+
2102
+ <select id="findById" resultMap="workOrderResultMap">
2103
+ SELECT * FROM work_orders WHERE work_order_id = #{workOrderId}
2104
+ </select>
2105
+
2106
+ <select id="findAll" resultMap="workOrderResultMap">
2107
+ SELECT * FROM work_orders ORDER BY created_at DESC
2108
+ </select>
2109
+
2110
+ <select id="findByItemId" resultMap="workOrderResultMap">
2111
+ SELECT * FROM work_orders WHERE item_id = #{itemId} ORDER BY created_at DESC
2112
+ </select>
2113
+
2114
+ <select id="findByStatus" resultMap="workOrderResultMap">
2115
+ SELECT * FROM work_orders WHERE status = #{status} ORDER BY created_at DESC
2116
+ </select>
2117
+
2118
+ <select id="findByWorkCenterId" resultMap="workOrderResultMap">
2119
+ SELECT * FROM work_orders WHERE work_center_id = #{workCenterId} ORDER BY created_at DESC
2120
+ </select>
2121
+
2122
+ <update id="updateStatus">
2123
+ UPDATE work_orders SET status = #{status} WHERE work_order_id = #{workOrderId}
2124
+ </update>
2125
+
2126
+ <update id="updateReleasedAt">
2127
+ UPDATE work_orders
2128
+ SET released_by = #{releasedBy}, released_at = #{releasedAt}
2129
+ WHERE work_order_id = #{workOrderId}
2130
+ </update>
2131
+
2132
+ <update id="updateRejection">
2133
+ UPDATE work_orders
2134
+ SET rejection_reason = #{rejectionReason}, rejected_at = #{rejectedAt}
2135
+ WHERE work_order_id = #{workOrderId}
2136
+ </update>
2137
+
2138
+ <update id="updateStarted">
2139
+ UPDATE work_orders
2140
+ SET operator_id = #{operatorId}, actual_start_time = #{actualStartTime}
2141
+ WHERE work_order_id = #{workOrderId}
2142
+ </update>
2143
+
2144
+ <update id="updateSuspended">
2145
+ UPDATE work_orders
2146
+ SET suspend_reason = #{suspendReason}, suspended_by = #{suspendedBy}, suspended_at = #{suspendedAt}
2147
+ WHERE work_order_id = #{workOrderId}
2148
+ </update>
2149
+
2150
+ <update id="updateResumed">
2151
+ UPDATE work_orders
2152
+ SET resumed_by = #{resumedBy}, resumed_at = #{resumedAt}
2153
+ WHERE work_order_id = #{workOrderId}
2154
+ </update>
2155
+
2156
+ <update id="updateCompleted">
2157
+ UPDATE work_orders
2158
+ SET completed_quantity = #{completedQuantity},
2159
+ defect_quantity = #{defectQuantity},
2160
+ operator_id = #{operatorId},
2161
+ actual_end_time = #{actualEndTime}
2162
+ WHERE work_order_id = #{workOrderId}
2163
+ </update>
2164
+
2165
+ <update id="updateCancellation">
2166
+ UPDATE work_orders
2167
+ SET cancelled_by = #{cancelledBy},
2168
+ cancellation_reason = #{cancellationReason},
2169
+ cancelled_at = #{cancelledAt}
2170
+ WHERE work_order_id = #{workOrderId}
2171
+ </update>
2172
+
2173
+ </mapper>
2174
+ ```
2175
+
2176
+ </details>
2177
+
2178
+ ---
2179
+
2180
+ ### 41.6 Read Model スキーマ
2181
+
2182
+ <details>
2183
+ <summary>コード例: schema.sql</summary>
2184
+
2185
+ ```sql
2186
+ -- Read Model: WorkOrders テーブル
2187
+ CREATE TABLE IF NOT EXISTS work_orders (
2188
+ work_order_id VARCHAR(36) PRIMARY KEY,
2189
+ item_id VARCHAR(36) NOT NULL,
2190
+ item_name VARCHAR(100) NOT NULL,
2191
+ order_quantity DECIMAL(12, 3) NOT NULL,
2192
+ unit_of_measure VARCHAR(10) NOT NULL,
2193
+ planned_start_date DATE NOT NULL,
2194
+ planned_end_date DATE NOT NULL,
2195
+ work_center_id VARCHAR(36) NOT NULL,
2196
+ status VARCHAR(20) NOT NULL,
2197
+ created_by VARCHAR(100) NOT NULL,
2198
+ created_at TIMESTAMP NOT NULL,
2199
+ released_by VARCHAR(100),
2200
+ released_at TIMESTAMP,
2201
+ rejection_reason TEXT,
2202
+ rejected_at TIMESTAMP,
2203
+ operator_id VARCHAR(100),
2204
+ actual_start_time TIMESTAMP,
2205
+ actual_end_time TIMESTAMP,
2206
+ completed_quantity DECIMAL(12, 3),
2207
+ defect_quantity DECIMAL(12, 3),
2208
+ suspend_reason TEXT,
2209
+ suspended_by VARCHAR(100),
2210
+ suspended_at TIMESTAMP,
2211
+ resumed_by VARCHAR(100),
2212
+ resumed_at TIMESTAMP,
2213
+ cancelled_by VARCHAR(100),
2214
+ cancellation_reason TEXT,
2215
+ cancelled_at TIMESTAMP
2216
+ );
2217
+
2218
+ -- インデックス
2219
+ CREATE INDEX IF NOT EXISTS idx_work_orders_item_id ON work_orders(item_id);
2220
+ CREATE INDEX IF NOT EXISTS idx_work_orders_status ON work_orders(status);
2221
+ CREATE INDEX IF NOT EXISTS idx_work_orders_work_center_id ON work_orders(work_center_id);
2222
+ CREATE INDEX IF NOT EXISTS idx_work_orders_created_at ON work_orders(created_at);
2223
+ CREATE INDEX IF NOT EXISTS idx_work_orders_planned_start_date ON work_orders(planned_start_date);
2224
+ ```
2225
+
2226
+ </details>
2227
+
2228
+ ---
2229
+
2230
+ ### 41.7 最小限フィールドの原則
2231
+
2232
+ Event Sourcing では、ドメインモデルは「次のコマンドを処理するために必要な最小限の状態」のみを保持します。一方、Read Model は表示に必要なすべてのフィールドを保持します。
2233
+
2234
+ | フィールド | ドメインモデル | Read Model | 理由 |
2235
+ |-----------|:-------------:|:----------:|------|
2236
+ | workOrderId | O | O | 識別子として必要 |
2237
+ | status | O | O | 状態遷移の判定に必要 |
2238
+ | itemId | O | O | 資材引当に必要 |
2239
+ | orderQuantity | O | O | 完了数量の検証に必要 |
2240
+ | actualStartTime | X | O | 判定に不要、表示のみ |
2241
+ | completedQuantity | X | O | 判定に不要、表示のみ |
2242
+ | cancelledBy | X | O | 判定に不要、表示のみ |
2243
+ | createdAt | X | O | 判定に不要、表示のみ |
2244
+
2245
+ ---
2246
+
2247
+ ## 第42章:REST API と統合テスト
2248
+
2249
+ ### 42.1 REST Controller(Inbound Adapter)
2250
+
2251
+ <details>
2252
+ <summary>コード例: WorkOrdersController.java</summary>
2253
+
2254
+ ```java
2255
+ package com.example.production.workorder.adapter.inbound.rest.workorders;
2256
+
2257
+ import com.example.production.workorder.adapter.inbound.rest.workorders.protocol.*;
2258
+ import com.example.production.workorder.adapter.outbound.persistence.mapper.WorkOrderMapper;
2259
+ import com.example.production.workorder.domain.model.aggregate.workorder.WorkOrderCommands.*;
2260
+ import io.swagger.v3.oas.annotations.Operation;
2261
+ import io.swagger.v3.oas.annotations.tags.Tag;
2262
+ import jakarta.validation.Valid;
2263
+ import org.axonframework.commandhandling.gateway.CommandGateway;
2264
+ import org.springframework.http.HttpStatus;
2265
+ import org.springframework.http.ResponseEntity;
2266
+ import org.springframework.web.bind.annotation.*;
2267
+
2268
+ import java.util.UUID;
2269
+ import java.util.concurrent.CompletableFuture;
2270
+
2271
+ /**
2272
+ * WorkOrder REST Controller(Inbound Adapter)
2273
+ */
2274
+ @RestController
2275
+ @RequestMapping("/api/work-orders")
2276
+ @Tag(name = "WorkOrders", description = "Work Order management API")
2277
+ public class WorkOrdersController {
2278
+
2279
+ private final CommandGateway commandGateway;
2280
+ private final WorkOrderMapper workOrderMapper;
2281
+
2282
+ public WorkOrdersController(CommandGateway commandGateway, WorkOrderMapper workOrderMapper) {
2283
+ this.commandGateway = commandGateway;
2284
+ this.workOrderMapper = workOrderMapper;
2285
+ }
2286
+
2287
+ // ========== Command Side (Write) ==========
2288
+
2289
+ @PostMapping
2290
+ @Operation(summary = "Create a new work order")
2291
+ public CompletableFuture<ResponseEntity<WorkOrderCreateResponse>> create(
2292
+ @Valid @RequestBody WorkOrderCreateRequest request
2293
+ ) {
2294
+ var workOrderId = UUID.randomUUID().toString();
2295
+ var command = new CreateWorkOrderCommand(
2296
+ workOrderId,
2297
+ request.itemId(),
2298
+ request.itemName(),
2299
+ request.orderQuantity(),
2300
+ request.unitOfMeasure(),
2301
+ request.plannedStartDate(),
2302
+ request.plannedEndDate(),
2303
+ request.workCenterId(),
2304
+ request.createdBy()
2305
+ );
2306
+
2307
+ return commandGateway.send(command)
2308
+ .thenApply(result -> ResponseEntity
2309
+ .status(HttpStatus.CREATED)
2310
+ .body(new WorkOrderCreateResponse(workOrderId)));
2311
+ }
2312
+
2313
+ @PostMapping("/{id}/start")
2314
+ @Operation(summary = "Start manufacturing")
2315
+ public CompletableFuture<ResponseEntity<Void>> start(
2316
+ @PathVariable String id,
2317
+ @Valid @RequestBody WorkOrderStartRequest request
2318
+ ) {
2319
+ var command = new StartWorkOrderCommand(id, request.operatorId());
2320
+
2321
+ return commandGateway.send(command)
2322
+ .thenApply(result -> ResponseEntity.ok().<Void>build());
2323
+ }
2324
+
2325
+ @PostMapping("/{id}/suspend")
2326
+ @Operation(summary = "Suspend manufacturing")
2327
+ public CompletableFuture<ResponseEntity<Void>> suspend(
2328
+ @PathVariable String id,
2329
+ @Valid @RequestBody WorkOrderSuspendRequest request
2330
+ ) {
2331
+ var command = new SuspendWorkOrderCommand(id, request.reason(), request.suspendedBy());
2332
+
2333
+ return commandGateway.send(command)
2334
+ .thenApply(result -> ResponseEntity.ok().<Void>build());
2335
+ }
2336
+
2337
+ @PostMapping("/{id}/resume")
2338
+ @Operation(summary = "Resume manufacturing")
2339
+ public CompletableFuture<ResponseEntity<Void>> resume(
2340
+ @PathVariable String id,
2341
+ @Valid @RequestBody WorkOrderResumeRequest request
2342
+ ) {
2343
+ var command = new ResumeWorkOrderCommand(id, request.resumedBy());
2344
+
2345
+ return commandGateway.send(command)
2346
+ .thenApply(result -> ResponseEntity.ok().<Void>build());
2347
+ }
2348
+
2349
+ @PostMapping("/{id}/complete")
2350
+ @Operation(summary = "Complete manufacturing")
2351
+ public CompletableFuture<ResponseEntity<Void>> complete(
2352
+ @PathVariable String id,
2353
+ @Valid @RequestBody WorkOrderCompleteRequest request
2354
+ ) {
2355
+ var command = new CompleteWorkOrderCommand(
2356
+ id,
2357
+ request.completedQuantity(),
2358
+ request.defectQuantity(),
2359
+ request.operatorId()
2360
+ );
2361
+
2362
+ return commandGateway.send(command)
2363
+ .thenApply(result -> ResponseEntity.ok().<Void>build());
2364
+ }
2365
+
2366
+ @PostMapping("/{id}/cancel")
2367
+ @Operation(summary = "Cancel work order")
2368
+ public CompletableFuture<ResponseEntity<Void>> cancel(
2369
+ @PathVariable String id,
2370
+ @Valid @RequestBody WorkOrderCancelRequest request
2371
+ ) {
2372
+ var command = new CancelWorkOrderCommand(id, request.cancelledBy(), request.reason());
2373
+
2374
+ return commandGateway.send(command)
2375
+ .thenApply(result -> ResponseEntity.ok().<Void>build());
2376
+ }
2377
+
2378
+ // ========== Query Side (Read) ==========
2379
+
2380
+ @GetMapping("/{id}")
2381
+ @Operation(summary = "Get a work order by ID")
2382
+ public ResponseEntity<WorkOrderGetResponse> get(@PathVariable String id) {
2383
+ return workOrderMapper.findById(id)
2384
+ .map(entity -> ResponseEntity.ok(WorkOrderGetResponse.from(entity)))
2385
+ .orElse(ResponseEntity.notFound().build());
2386
+ }
2387
+
2388
+ @GetMapping
2389
+ @Operation(summary = "Get all work orders")
2390
+ public ResponseEntity<WorkOrderListResponse> getAll() {
2391
+ var entities = workOrderMapper.findAll();
2392
+ var items = entities.stream()
2393
+ .map(WorkOrderGetResponse::from)
2394
+ .toList();
2395
+ return ResponseEntity.ok(new WorkOrderListResponse(items));
2396
+ }
2397
+
2398
+ @GetMapping("/status/{status}")
2399
+ @Operation(summary = "Get work orders by status")
2400
+ public ResponseEntity<WorkOrderListResponse> getByStatus(@PathVariable String status) {
2401
+ var entities = workOrderMapper.findByStatus(status);
2402
+ var items = entities.stream()
2403
+ .map(WorkOrderGetResponse::from)
2404
+ .toList();
2405
+ return ResponseEntity.ok(new WorkOrderListResponse(items));
2406
+ }
2407
+ }
2408
+ ```
2409
+
2410
+ </details>
2411
+
2412
+ ---
2413
+
2414
+ ### 42.2 Request/Response DTO
2415
+
2416
+ <details>
2417
+ <summary>コード例: WorkOrderCreateRequest.java</summary>
2418
+
2419
+ ```java
2420
+ package com.example.production.workorder.adapter.inbound.rest.workorders.protocol;
2421
+
2422
+ import jakarta.validation.constraints.NotBlank;
2423
+ import jakarta.validation.constraints.NotNull;
2424
+ import jakarta.validation.constraints.Positive;
2425
+ import java.math.BigDecimal;
2426
+ import java.time.LocalDate;
2427
+
2428
+ /**
2429
+ * 製造指図作成リクエスト
2430
+ */
2431
+ public record WorkOrderCreateRequest(
2432
+ @NotBlank
2433
+ String itemId,
2434
+
2435
+ @NotBlank
2436
+ String itemName,
2437
+
2438
+ @NotNull @Positive
2439
+ BigDecimal orderQuantity,
2440
+
2441
+ @NotBlank
2442
+ String unitOfMeasure,
2443
+
2444
+ @NotNull
2445
+ LocalDate plannedStartDate,
2446
+
2447
+ @NotNull
2448
+ LocalDate plannedEndDate,
2449
+
2450
+ @NotBlank
2451
+ String workCenterId,
2452
+
2453
+ @NotBlank
2454
+ String createdBy
2455
+ ) {
2456
+ }
2457
+ ```
2458
+
2459
+ </details>
2460
+
2461
+ <details>
2462
+ <summary>コード例: WorkOrderGetResponse.java</summary>
2463
+
2464
+ ```java
2465
+ package com.example.production.workorder.adapter.inbound.rest.workorders.protocol;
2466
+
2467
+ import com.example.production.workorder.adapter.outbound.persistence.entity.WorkOrderEntity;
2468
+ import java.math.BigDecimal;
2469
+ import java.time.LocalDate;
2470
+ import java.time.LocalDateTime;
2471
+
2472
+ /**
2473
+ * 製造指図取得レスポンス
2474
+ */
2475
+ public record WorkOrderGetResponse(
2476
+ String workOrderId,
2477
+ String itemId,
2478
+ String itemName,
2479
+ BigDecimal orderQuantity,
2480
+ String unitOfMeasure,
2481
+ LocalDate plannedStartDate,
2482
+ LocalDate plannedEndDate,
2483
+ String workCenterId,
2484
+ String status,
2485
+ String createdBy,
2486
+ LocalDateTime createdAt,
2487
+ String releasedBy,
2488
+ LocalDateTime releasedAt,
2489
+ String rejectionReason,
2490
+ LocalDateTime rejectedAt,
2491
+ String operatorId,
2492
+ LocalDateTime actualStartTime,
2493
+ LocalDateTime actualEndTime,
2494
+ BigDecimal completedQuantity,
2495
+ BigDecimal defectQuantity,
2496
+ String suspendReason,
2497
+ String suspendedBy,
2498
+ LocalDateTime suspendedAt,
2499
+ String resumedBy,
2500
+ LocalDateTime resumedAt,
2501
+ String cancelledBy,
2502
+ String cancellationReason,
2503
+ LocalDateTime cancelledAt
2504
+ ) {
2505
+
2506
+ public static WorkOrderGetResponse from(WorkOrderEntity entity) {
2507
+ return new WorkOrderGetResponse(
2508
+ entity.getWorkOrderId(),
2509
+ entity.getItemId(),
2510
+ entity.getItemName(),
2511
+ entity.getOrderQuantity(),
2512
+ entity.getUnitOfMeasure(),
2513
+ entity.getPlannedStartDate(),
2514
+ entity.getPlannedEndDate(),
2515
+ entity.getWorkCenterId(),
2516
+ entity.getStatus(),
2517
+ entity.getCreatedBy(),
2518
+ entity.getCreatedAt(),
2519
+ entity.getReleasedBy(),
2520
+ entity.getReleasedAt(),
2521
+ entity.getRejectionReason(),
2522
+ entity.getRejectedAt(),
2523
+ entity.getOperatorId(),
2524
+ entity.getActualStartTime(),
2525
+ entity.getActualEndTime(),
2526
+ entity.getCompletedQuantity(),
2527
+ entity.getDefectQuantity(),
2528
+ entity.getSuspendReason(),
2529
+ entity.getSuspendedBy(),
2530
+ entity.getSuspendedAt(),
2531
+ entity.getResumedBy(),
2532
+ entity.getResumedAt(),
2533
+ entity.getCancelledBy(),
2534
+ entity.getCancellationReason(),
2535
+ entity.getCancelledAt()
2536
+ );
2537
+ }
2538
+ }
2539
+ ```
2540
+
2541
+ </details>
2542
+
2543
+ ---
2544
+
2545
+ ### 42.3 API エンドポイント一覧
2546
+
2547
+ | メソッド | パス | 説明 | 種別 |
2548
+ |---------|------|------|------|
2549
+ | POST | `/api/work-orders` | 製造指図作成 | Command |
2550
+ | POST | `/api/work-orders/{id}/start` | 製造開始 | Command |
2551
+ | POST | `/api/work-orders/{id}/suspend` | 製造中断 | Command |
2552
+ | POST | `/api/work-orders/{id}/resume` | 製造再開 | Command |
2553
+ | POST | `/api/work-orders/{id}/complete` | 製造完了 | Command |
2554
+ | POST | `/api/work-orders/{id}/cancel` | キャンセル | Command |
2555
+ | GET | `/api/work-orders` | 製造指図一覧取得 | Query |
2556
+ | GET | `/api/work-orders/{id}` | 製造指図詳細取得 | Query |
2557
+ | GET | `/api/work-orders/status/{status}` | ステータス別取得 | Query |
2558
+
2559
+ ---
2560
+
2561
+ ### 42.4 Axon Test による単体テスト
2562
+
2563
+ <details>
2564
+ <summary>コード例: WorkOrderAggregateTest.java</summary>
2565
+
2566
+ ```java
2567
+ package com.example.production.workorder.application.aggregate;
2568
+
2569
+ import com.example.production.workorder.api.events.*;
2570
+ import com.example.production.workorder.domain.model.aggregate.workorder.WorkOrderCommands.*;
2571
+ import org.axonframework.test.aggregate.AggregateTestFixture;
2572
+ import org.axonframework.test.aggregate.FixtureConfiguration;
2573
+ import org.junit.jupiter.api.BeforeEach;
2574
+ import org.junit.jupiter.api.DisplayName;
2575
+ import org.junit.jupiter.api.Nested;
2576
+ import org.junit.jupiter.api.Test;
2577
+
2578
+ import java.math.BigDecimal;
2579
+ import java.time.LocalDate;
2580
+ import java.time.LocalDateTime;
2581
+
2582
+ class WorkOrderAggregateTest {
2583
+
2584
+ private FixtureConfiguration<WorkOrderAggregateAdapter> fixture;
2585
+
2586
+ @BeforeEach
2587
+ void setUp() {
2588
+ fixture = new AggregateTestFixture<>(WorkOrderAggregateAdapter.class);
2589
+ }
2590
+
2591
+ @Nested
2592
+ @DisplayName("製造指図作成")
2593
+ class CreateWorkOrder {
2594
+
2595
+ @Test
2596
+ @DisplayName("正常な製造指図を作成できる")
2597
+ void shouldCreateWorkOrder() {
2598
+ var command = new CreateWorkOrderCommand(
2599
+ "wo-1",
2600
+ "item-1",
2601
+ "製品A",
2602
+ BigDecimal.valueOf(100),
2603
+ "EA",
2604
+ LocalDate.now(),
2605
+ LocalDate.now().plusDays(7),
2606
+ "wc-1",
2607
+ "user-1"
2608
+ );
2609
+
2610
+ fixture.givenNoPriorActivity()
2611
+ .when(command)
2612
+ .expectSuccessfulHandlerExecution()
2613
+ .expectEventsMatching(events ->
2614
+ events.getPayload() instanceof WorkOrderCreatedEvent
2615
+ );
2616
+ }
2617
+
2618
+ @Test
2619
+ @DisplayName("数量が0以下の場合はエラー")
2620
+ void shouldRejectZeroQuantity() {
2621
+ var command = new CreateWorkOrderCommand(
2622
+ "wo-1",
2623
+ "item-1",
2624
+ "製品A",
2625
+ BigDecimal.ZERO,
2626
+ "EA",
2627
+ LocalDate.now(),
2628
+ LocalDate.now().plusDays(7),
2629
+ "wc-1",
2630
+ "user-1"
2631
+ );
2632
+
2633
+ fixture.givenNoPriorActivity()
2634
+ .when(command)
2635
+ .expectException(IllegalArgumentException.class);
2636
+ }
2637
+
2638
+ @Test
2639
+ @DisplayName("終了日が開始日より前の場合はエラー")
2640
+ void shouldRejectInvalidDateRange() {
2641
+ var command = new CreateWorkOrderCommand(
2642
+ "wo-1",
2643
+ "item-1",
2644
+ "製品A",
2645
+ BigDecimal.valueOf(100),
2646
+ "EA",
2647
+ LocalDate.now(),
2648
+ LocalDate.now().minusDays(1),
2649
+ "wc-1",
2650
+ "user-1"
2651
+ );
2652
+
2653
+ fixture.givenNoPriorActivity()
2654
+ .when(command)
2655
+ .expectException(IllegalArgumentException.class);
2656
+ }
2657
+ }
2658
+
2659
+ @Nested
2660
+ @DisplayName("製造開始")
2661
+ class StartWorkOrder {
2662
+
2663
+ @Test
2664
+ @DisplayName("リリース済みの製造指図を開始できる")
2665
+ void shouldStartReleasedWorkOrder() {
2666
+ var createdEvent = new WorkOrderCreatedEvent(
2667
+ "wo-1", "item-1", "製品A", BigDecimal.valueOf(100), "EA",
2668
+ LocalDate.now(), LocalDate.now().plusDays(7), "wc-1",
2669
+ "user-1", LocalDateTime.now()
2670
+ );
2671
+ var releasedEvent = new WorkOrderReleasedEvent("wo-1", "system", LocalDateTime.now());
2672
+
2673
+ fixture.given(createdEvent, releasedEvent)
2674
+ .when(new StartWorkOrderCommand("wo-1", "operator-1"))
2675
+ .expectSuccessfulHandlerExecution()
2676
+ .expectEventsMatching(events ->
2677
+ events.getPayload() instanceof WorkOrderStartedEvent
2678
+ );
2679
+ }
2680
+
2681
+ @Test
2682
+ @DisplayName("作成済みの製造指図は開始できない")
2683
+ void shouldNotStartCreatedWorkOrder() {
2684
+ var createdEvent = new WorkOrderCreatedEvent(
2685
+ "wo-1", "item-1", "製品A", BigDecimal.valueOf(100), "EA",
2686
+ LocalDate.now(), LocalDate.now().plusDays(7), "wc-1",
2687
+ "user-1", LocalDateTime.now()
2688
+ );
2689
+
2690
+ fixture.given(createdEvent)
2691
+ .when(new StartWorkOrderCommand("wo-1", "operator-1"))
2692
+ .expectException(IllegalStateException.class);
2693
+ }
2694
+ }
2695
+
2696
+ @Nested
2697
+ @DisplayName("製造完了")
2698
+ class CompleteWorkOrder {
2699
+
2700
+ @Test
2701
+ @DisplayName("製造中の製造指図を完了できる")
2702
+ void shouldCompleteInProgressWorkOrder() {
2703
+ var createdEvent = new WorkOrderCreatedEvent(
2704
+ "wo-1", "item-1", "製品A", BigDecimal.valueOf(100), "EA",
2705
+ LocalDate.now(), LocalDate.now().plusDays(7), "wc-1",
2706
+ "user-1", LocalDateTime.now()
2707
+ );
2708
+ var releasedEvent = new WorkOrderReleasedEvent("wo-1", "system", LocalDateTime.now());
2709
+ var startedEvent = new WorkOrderStartedEvent("wo-1", "operator-1", LocalDateTime.now());
2710
+
2711
+ fixture.given(createdEvent, releasedEvent, startedEvent)
2712
+ .when(new CompleteWorkOrderCommand("wo-1", BigDecimal.valueOf(98), BigDecimal.valueOf(2), "operator-1"))
2713
+ .expectSuccessfulHandlerExecution()
2714
+ .expectEventsMatching(events ->
2715
+ events.getPayload() instanceof WorkOrderCompletedEvent
2716
+ );
2717
+ }
2718
+ }
2719
+
2720
+ @Nested
2721
+ @DisplayName("製造指図キャンセル")
2722
+ class CancelWorkOrder {
2723
+
2724
+ @Test
2725
+ @DisplayName("作成済みの製造指図をキャンセルできる")
2726
+ void shouldCancelCreatedWorkOrder() {
2727
+ var createdEvent = new WorkOrderCreatedEvent(
2728
+ "wo-1", "item-1", "製品A", BigDecimal.valueOf(100), "EA",
2729
+ LocalDate.now(), LocalDate.now().plusDays(7), "wc-1",
2730
+ "user-1", LocalDateTime.now()
2731
+ );
2732
+
2733
+ fixture.given(createdEvent)
2734
+ .when(new CancelWorkOrderCommand("wo-1", "user-1", "Customer request"))
2735
+ .expectSuccessfulHandlerExecution()
2736
+ .expectEventsMatching(events ->
2737
+ events.getPayload() instanceof WorkOrderCancelledEvent
2738
+ );
2739
+ }
2740
+
2741
+ @Test
2742
+ @DisplayName("完了済みの製造指図はキャンセルできない")
2743
+ void shouldNotCancelCompletedWorkOrder() {
2744
+ var createdEvent = new WorkOrderCreatedEvent(
2745
+ "wo-1", "item-1", "製品A", BigDecimal.valueOf(100), "EA",
2746
+ LocalDate.now(), LocalDate.now().plusDays(7), "wc-1",
2747
+ "user-1", LocalDateTime.now()
2748
+ );
2749
+ var releasedEvent = new WorkOrderReleasedEvent("wo-1", "system", LocalDateTime.now());
2750
+ var startedEvent = new WorkOrderStartedEvent("wo-1", "operator-1", LocalDateTime.now());
2751
+ var completedEvent = new WorkOrderCompletedEvent(
2752
+ "wo-1", "item-1", BigDecimal.valueOf(100), BigDecimal.ZERO,
2753
+ "operator-1", LocalDateTime.now()
2754
+ );
2755
+
2756
+ fixture.given(createdEvent, releasedEvent, startedEvent, completedEvent)
2757
+ .when(new CancelWorkOrderCommand("wo-1", "user-1", "Customer request"))
2758
+ .expectException(IllegalStateException.class);
2759
+ }
2760
+ }
2761
+ }
2762
+ ```
2763
+
2764
+ </details>
2765
+
2766
+ ---
2767
+
2768
+ ## 研究 5 のまとめ
2769
+
2770
+ ### 実装した機能一覧
2771
+
2772
+ | 章 | 内容 | 主要技術 |
2773
+ |---|---|---|
2774
+ | **第38章: 基礎** | CQRS/ES アーキテクチャの基礎 | Axon Framework, Event Sourcing |
2775
+ | **第39章: ドメイン設計** | イベント、コマンド、ドメインモデル | sealed interface, record |
2776
+ | **第40章: Aggregate/Policy** | Aggregate Adapter, Choreography | @Aggregate, @EventHandler |
2777
+ | **第41章: Projection** | Read Model の更新、最小限フィールド | @EventHandler, MyBatis |
2778
+ | **第42章: REST API** | Inbound Adapter, テスト | CommandGateway, Axon Test |
2779
+
2780
+ ### アーキテクチャの特徴
2781
+
2782
+ ```plantuml
2783
+ @startuml cqrs_summary
2784
+ skinparam backgroundColor #FEFEFE
2785
+
2786
+ package "Command Side" {
2787
+ [REST Controller] as RC1
2788
+ [CommandGateway] as CG
2789
+ [Aggregate Adapter] as AA
2790
+ [Domain Model] as DM
2791
+ [Event Store] as ES
2792
+ }
2793
+
2794
+ package "Query Side" {
2795
+ [REST Controller] as RC2
2796
+ [MyBatis Mapper] as MB
2797
+ [Read Model DB] as DB
2798
+ [Projection] as P
2799
+ }
2800
+
2801
+ RC1 --> CG : send(command)
2802
+ CG --> AA : @CommandHandler
2803
+ AA --> DM : ビジネスロジック
2804
+ DM --> AA : event
2805
+ AA --> ES : apply(event)
2806
+ ES --> P : @EventHandler
2807
+ P --> MB : insert/update
2808
+ MB --> DB : SQL
2809
+
2810
+ RC2 --> MB : query
2811
+ MB --> DB : SELECT
2812
+
2813
+ note right of DM
2814
+ 純粋なドメインモデル
2815
+ Axon に依存しない
2816
+ end note
2817
+
2818
+ note right of P
2819
+ イベントを購読して
2820
+ MyBatis 経由で
2821
+ Read Model を更新
2822
+ end note
2823
+
2824
+ @enduml
2825
+ ```
2826
+
2827
+ ### 設計上の特徴
2828
+
2829
+ | 特徴 | 説明 |
2830
+ |------|------|
2831
+ | **ドメインモデルとフレームワークの分離** | `WorkOrder.java` は純粋なドメインモデル、`WorkOrderAggregateAdapter.java` は Axon 用アダプター |
2832
+ | **Choreography による疎結合** | 各 Context は独立して動作、イベントを介した非同期連携 |
2833
+ | **sealed interface による型安全性** | イベントの網羅性チェック、switch 式でコンパイル時検証 |
2834
+ | **record によるイミュータブル設計** | 状態変更は常に新しいインスタンスを生成 |
2835
+ | **最小限フィールドの原則** | ドメインモデルは状態遷移判定に必要な最小限のみ保持 |
2836
+ | **MyBatis による柔軟な SQL 制御** | Read Model の更新に MyBatis Mapper を使用 |
2837
+
2838
+ ### 技術スタック
2839
+
2840
+ | カテゴリ | 技術 |
2841
+ |---------|------|
2842
+ | **言語** | Java 21 |
2843
+ | **フレームワーク** | Spring Boot 3.4.1 |
2844
+ | **CQRS/ES** | Axon Framework 4.10.3 |
2845
+ | **ORM** | MyBatis 3.0.4 |
2846
+ | **データベース** | H2 / PostgreSQL |
2847
+ | **テスト** | JUnit 5, Axon Test |
2848
+
2849
+ ### 生産管理システムにおける CQRS/ES の利点
2850
+
2851
+ | 観点 | 説明 |
2852
+ |------|------|
2853
+ | **製造履歴の完全追跡** | 製造指図の作成から完了までの全履歴を保持 |
2854
+ | **品質トレーサビリティ** | 不良発生時に製造工程を遡及調査可能 |
2855
+ | **資材引当の厳密管理** | 引当・払出・返却を厳密にイベントで追跡 |
2856
+ | **工程別進捗管理** | 開始・中断・再開・完了を正確に記録 |
2857
+ | **原価計算への活用** | イベントから製造原価を正確に計算可能 |
2858
+
2859
+ ### API 形式の比較
2860
+
2861
+ | 観点 | REST API | gRPC | GraphQL | CQRS/ES |
2862
+ |------|----------|------|---------|---------|
2863
+ | **書き込みモデル** | 同一 | 同一 | 同一 | 分離(Command) |
2864
+ | **読み取りモデル** | 同一 | 同一 | 同一 | 分離(Query) |
2865
+ | **履歴保持** | 別途実装 | 別途実装 | 別途実装 | 標準搭載 |
2866
+ | **監査証跡** | 別途実装 | 別途実装 | 別途実装 | 標準搭載 |
2867
+ | **スケーラビリティ** | 中 | 高 | 中 | 非常に高 |
2868
+ | **複雑性** | 低 | 中 | 中 | 高 |
2869
+
2870
+ ### CQRS/ES を選択する場面
2871
+
2872
+ 1. **製造履歴要件**: すべての製造工程の履歴が必要
2873
+ 2. **品質トレーサビリティ**: 製品の製造履歴を追跡したい
2874
+ 3. **複雑な業務フロー**: 製造指図の多様な状態遷移を管理
2875
+ 4. **高いスケーラビリティ**: MRP 計算と照会を独立してスケール
2876
+ 5. **原価計算**: イベントから正確な製造原価を算出
2877
+
2878
+ CQRS/ES は導入コストが高いですが、製造履歴の追跡や品質トレーサビリティが重要な生産管理システムでは、その恩恵は大きくなります。Axon Framework を使用することで、イベントソーシングと CQRS の複雑さを軽減し、ドメインロジックに集中できます。MyBatis を使用することで、Read Model の更新において柔軟な SQL 制御が可能になります。