@k2works/claude-code-booster 3.2.1 → 3.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (66) hide show
  1. package/lib/assets/docs/article/index.md +4 -1
  2. package/lib/assets/docs/article/practical-database-design/index.md +121 -0
  3. package/lib/assets/docs/article/practical-database-design/part1/chapter01.md +288 -0
  4. package/lib/assets/docs/article/practical-database-design/part1/chapter02.md +518 -0
  5. package/lib/assets/docs/article/practical-database-design/part1/chapter03.md +557 -0
  6. package/lib/assets/docs/article/practical-database-design/part2/chapter04.md +924 -0
  7. package/lib/assets/docs/article/practical-database-design/part2/chapter05.md +1627 -0
  8. package/lib/assets/docs/article/practical-database-design/part2/chapter06.md +2716 -0
  9. package/lib/assets/docs/article/practical-database-design/part2/chapter07.md +2082 -0
  10. package/lib/assets/docs/article/practical-database-design/part2/chapter08.md +2105 -0
  11. package/lib/assets/docs/article/practical-database-design/part2/chapter09.md +2031 -0
  12. package/lib/assets/docs/article/practical-database-design/part2/chapter10.md +1387 -0
  13. package/lib/assets/docs/article/practical-database-design/part2/chapter11.md +1677 -0
  14. package/lib/assets/docs/article/practical-database-design/part2/chapter12.md +1417 -0
  15. package/lib/assets/docs/article/practical-database-design/part2/chapter13.md +1434 -0
  16. package/lib/assets/docs/article/practical-database-design/part3/chapter14.md +667 -0
  17. package/lib/assets/docs/article/practical-database-design/part3/chapter15.md +1625 -0
  18. package/lib/assets/docs/article/practical-database-design/part3/chapter16.md +1915 -0
  19. package/lib/assets/docs/article/practical-database-design/part3/chapter17.md +1708 -0
  20. package/lib/assets/docs/article/practical-database-design/part3/chapter18.md +2095 -0
  21. package/lib/assets/docs/article/practical-database-design/part3/chapter19.md +1123 -0
  22. package/lib/assets/docs/article/practical-database-design/part3/chapter20.md +1031 -0
  23. package/lib/assets/docs/article/practical-database-design/part3/chapter21.md +1382 -0
  24. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter14-orm.md +991 -0
  25. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter15-orm.md +1300 -0
  26. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter16-orm.md +1166 -0
  27. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter17-orm.md +1584 -0
  28. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter18-orm.md +1183 -0
  29. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter19-orm.md +1016 -0
  30. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter20-orm.md +1753 -0
  31. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter21-orm.md +1447 -0
  32. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter22-orm.md +1878 -0
  33. package/lib/assets/docs/article/practical-database-design/part4/chapter22.md +965 -0
  34. package/lib/assets/docs/article/practical-database-design/part4/chapter23.md +2069 -0
  35. package/lib/assets/docs/article/practical-database-design/part4/chapter24.md +2439 -0
  36. package/lib/assets/docs/article/practical-database-design/part4/chapter25.md +3661 -0
  37. package/lib/assets/docs/article/practical-database-design/part4/chapter26.md +2916 -0
  38. package/lib/assets/docs/article/practical-database-design/part4/chapter27.md +3105 -0
  39. package/lib/assets/docs/article/practical-database-design/part4/chapter28.md +2697 -0
  40. package/lib/assets/docs/article/practical-database-design/part4/chapter29.md +2544 -0
  41. package/lib/assets/docs/article/practical-database-design/part4/chapter30.md +2180 -0
  42. package/lib/assets/docs/article/practical-database-design/part4/chapter31.md +1192 -0
  43. package/lib/assets/docs/article/practical-database-design/part4/chapter32.md +2101 -0
  44. package/lib/assets/docs/article/practical-database-design/part5/chapter33.md +1032 -0
  45. package/lib/assets/docs/article/practical-database-design/part5/chapter34.md +1609 -0
  46. package/lib/assets/docs/article/practical-database-design/part5/chapter35.md +1453 -0
  47. package/lib/assets/docs/article/practical-database-design/part5/chapter36.md +1292 -0
  48. package/lib/assets/docs/article/practical-database-design/part5/chapter37.md +1470 -0
  49. package/lib/assets/docs/article/practical-database-design/part5/chapter38.md +1698 -0
  50. package/lib/assets/docs/article/practical-database-design/part5/chapter39.md +2334 -0
  51. package/lib/assets/docs/article/practical-database-design/study/study2-1.md +1693 -0
  52. package/lib/assets/docs/article/practical-database-design/study/study2-2.md +1347 -0
  53. package/lib/assets/docs/article/practical-database-design/study/study2-3.md +2044 -0
  54. package/lib/assets/docs/article/practical-database-design/study/study2-4.md +2229 -0
  55. package/lib/assets/docs/article/practical-database-design/study/study2-5.md +2418 -0
  56. package/lib/assets/docs/article/practical-database-design/study/study3-1.md +2205 -0
  57. package/lib/assets/docs/article/practical-database-design/study/study3-2.md +2221 -0
  58. package/lib/assets/docs/article/practical-database-design/study/study3-3.md +2253 -0
  59. package/lib/assets/docs/article/practical-database-design/study/study3-4.md +2106 -0
  60. package/lib/assets/docs/article/practical-database-design/study/study3-5.md +2507 -0
  61. package/lib/assets/docs/article/practical-database-design/study/study4-1.md +2587 -0
  62. package/lib/assets/docs/article/practical-database-design/study/study4-2.md +2075 -0
  63. package/lib/assets/docs/article/practical-database-design/study/study4-3.md +1805 -0
  64. package/lib/assets/docs/article/practical-database-design/study/study4-4.md +1895 -0
  65. package/lib/assets/docs/article/practical-database-design/study/study4-5.md +2878 -0
  66. package/package.json +1 -1
@@ -0,0 +1,3105 @@
1
+ # 第27章:工程管理の設計
2
+
3
+ ## 概要
4
+
5
+ 本章では、生産管理システムにおける工程管理の設計を解説します。製造オーダから作業指示を生成し、完成実績・工数実績を記録する仕組みを構築します。
6
+
7
+ ```plantuml
8
+ @startuml
9
+
10
+ title 工程管理のデータ構造
11
+
12
+ package "製造指示" {
13
+ entity "作業指示データ" as work_order
14
+ entity "作業指示明細データ" as work_order_detail
15
+ }
16
+
17
+ package "製造実績" {
18
+ entity "完成実績データ" as completion_result
19
+ entity "完成検査結果データ" as inspection_result
20
+ entity "工数実績データ" as labor_hours
21
+ }
22
+
23
+ package "マスタ" {
24
+ entity "オーダ情報" as order
25
+ entity "工程マスタ" as process
26
+ entity "工程表" as routing
27
+ entity "品目マスタ" as item
28
+ entity "欠点マスタ" as defect
29
+ entity "担当者マスタ" as employee
30
+ }
31
+
32
+ order ||--o{ work_order : "展開"
33
+ work_order ||--|{ work_order_detail : "contains"
34
+ routing ||--o{ work_order_detail : "参照"
35
+ process ||--o{ work_order_detail : "工程"
36
+ item ||--o{ work_order : "品目"
37
+
38
+ work_order ||--o{ completion_result : "報告"
39
+ completion_result ||--o{ inspection_result : "検査"
40
+ defect ||--o{ inspection_result : "欠点"
41
+
42
+ work_order ||--o{ labor_hours : "工数"
43
+ work_order_detail ||--o{ labor_hours : "工順"
44
+ employee ||--o{ labor_hours : "担当"
45
+
46
+ @enduml
47
+ ```
48
+
49
+ ---
50
+
51
+ ## 27.1 製造指示の DB 設計
52
+
53
+ ### 工程管理業務の流れ
54
+
55
+ 製造オーダが確定すると、作業指示に展開されます。作業指示は工程表に基づいて工程ごとの明細を持ち、現場での製造作業を管理します。
56
+
57
+ ```plantuml
58
+ @startwbs
59
+
60
+ title 部品構成と作業工程の例
61
+
62
+ * 製品
63
+ ** 組立
64
+ *** 部品B
65
+ **** 組立
66
+ ***** 部品A
67
+ *** 部品C
68
+
69
+ @endwbs
70
+ ```
71
+
72
+ ```plantuml
73
+ @startuml
74
+
75
+ title 工程管理の業務フロー
76
+
77
+ |生産管理部|
78
+ start
79
+ :オーダ情報;
80
+ note right: 製造オーダを\n作業指示に展開
81
+
82
+ |製造部|
83
+ :作業計画作成;
84
+ :作業指示発行;
85
+ :作業指示書印刷;
86
+
87
+ |資材倉庫|
88
+ :材料払出;
89
+
90
+ |製造ライン|
91
+ :作業実施;
92
+ fork
93
+ :完成実績報告;
94
+ fork again
95
+ :工数実績報告;
96
+ fork again
97
+ :消費実績報告;
98
+ fork end
99
+
100
+ |品質管理|
101
+ :完成検査;
102
+
103
+ if (合格?) then (yes)
104
+ |製品倉庫|
105
+ :製品入庫;
106
+ else (no)
107
+ |製造ライン|
108
+ :手直し/廃棄;
109
+ endif
110
+
111
+ |製造部|
112
+ :作業完了処理;
113
+
114
+ stop
115
+
116
+ @enduml
117
+ ```
118
+
119
+ ### 工程マスタ・工程表のスキーマ設計
120
+
121
+ 工程マスタは製造工程を定義し、工程表は品目ごとの製造工順を管理します。
122
+
123
+ <details>
124
+ <summary>DDL: 工程マスタ・工程表</summary>
125
+
126
+ ```sql
127
+ -- V013__create_process_master_tables.sql
128
+
129
+ -- 工程マスタ
130
+ CREATE TABLE "工程マスタ" (
131
+ "工程コード" VARCHAR(20) PRIMARY KEY,
132
+ "工程名" VARCHAR(100) NOT NULL,
133
+ "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
134
+ "作成者" VARCHAR(50),
135
+ "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
136
+ "更新者" VARCHAR(50)
137
+ );
138
+
139
+ -- 工程表
140
+ CREATE TABLE "工程表" (
141
+ "ID" SERIAL PRIMARY KEY,
142
+ "品目コード" VARCHAR(20) NOT NULL,
143
+ "工順" INTEGER NOT NULL,
144
+ "工程コード" VARCHAR(20) NOT NULL,
145
+ "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
146
+ "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
147
+ CONSTRAINT "fk_工程表_品目"
148
+ FOREIGN KEY ("品目コード") REFERENCES "品目マスタ"("品目コード"),
149
+ CONSTRAINT "fk_工程表_工程"
150
+ FOREIGN KEY ("工程コード") REFERENCES "工程マスタ"("工程コード"),
151
+ UNIQUE ("品目コード", "工順")
152
+ );
153
+
154
+ -- インデックス
155
+ CREATE INDEX "idx_工程表_品目コード" ON "工程表"("品目コード");
156
+ CREATE INDEX "idx_工程表_工程コード" ON "工程表"("工程コード");
157
+ ```
158
+
159
+ </details>
160
+
161
+ ### 作業指示のスキーマ設計
162
+
163
+ 作業指示データはオーダ情報から展開され、作業指示明細データは工程表に基づいて生成されます。
164
+
165
+ ```plantuml
166
+ @startuml
167
+
168
+ title 作業指示の構造
169
+
170
+ entity "作業指示データ" as work_order {
171
+ * ID [PK]
172
+ --
173
+ * 作業指示番号 [UNIQUE]
174
+ * オーダ番号 [FK]
175
+ * 作業指示日
176
+ * 品目コード [FK]
177
+ * 作業指示数
178
+ * 場所コード [FK]
179
+ * 開始予定日
180
+ * 完成予定日
181
+ 実績開始日
182
+ 実績完了日
183
+ * 完成済数
184
+ * 総良品数
185
+ * 総不良品数
186
+ * ステータス
187
+ * 完了フラグ
188
+ 備考
189
+ }
190
+
191
+ entity "作業指示明細データ" as work_order_detail {
192
+ * ID [PK]
193
+ --
194
+ * 作業指示番号 [FK]
195
+ * 工順
196
+ * 工程コード [FK]
197
+ }
198
+
199
+ work_order ||--|{ work_order_detail : "contains"
200
+
201
+ @enduml
202
+ ```
203
+
204
+ <details>
205
+ <summary>DDL: 作業指示データ・作業指示明細データ</summary>
206
+
207
+ ```sql
208
+ -- V014__create_work_order_tables.sql
209
+
210
+ -- 作業指示ステータス
211
+ CREATE TYPE 作業指示ステータス AS ENUM ('未着手', '作業中', '完了', '中断');
212
+
213
+ -- 作業指示データ
214
+ CREATE TABLE "作業指示データ" (
215
+ "ID" SERIAL PRIMARY KEY,
216
+ "作業指示番号" VARCHAR(20) UNIQUE NOT NULL,
217
+ "オーダ番号" VARCHAR(20) NOT NULL,
218
+ "作業指示日" DATE NOT NULL,
219
+ "品目コード" VARCHAR(20) NOT NULL,
220
+ "作業指示数" DECIMAL(15, 2) NOT NULL,
221
+ "場所コード" VARCHAR(20) NOT NULL,
222
+ "開始予定日" DATE NOT NULL,
223
+ "完成予定日" DATE NOT NULL,
224
+ "実績開始日" DATE,
225
+ "実績完了日" DATE,
226
+ "完成済数" DECIMAL(15, 2) DEFAULT 0 NOT NULL,
227
+ "総良品数" DECIMAL(15, 2) DEFAULT 0 NOT NULL,
228
+ "総不良品数" DECIMAL(15, 2) DEFAULT 0 NOT NULL,
229
+ "ステータス" 作業指示ステータス DEFAULT '未着手' NOT NULL,
230
+ "完了フラグ" BOOLEAN DEFAULT false NOT NULL,
231
+ "備考" TEXT,
232
+ "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
233
+ "作成者" VARCHAR(50),
234
+ "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
235
+ "更新者" VARCHAR(50),
236
+ CONSTRAINT "fk_作業指示_オーダ"
237
+ FOREIGN KEY ("オーダ番号") REFERENCES "オーダ情報"("オーダNO"),
238
+ CONSTRAINT "fk_作業指示_品目"
239
+ FOREIGN KEY ("品目コード") REFERENCES "品目マスタ"("品目コード"),
240
+ CONSTRAINT "fk_作業指示_場所"
241
+ FOREIGN KEY ("場所コード") REFERENCES "場所マスタ"("場所コード")
242
+ );
243
+
244
+ -- 作業指示明細データ
245
+ CREATE TABLE "作業指示明細データ" (
246
+ "ID" SERIAL PRIMARY KEY,
247
+ "作業指示番号" VARCHAR(20) NOT NULL,
248
+ "工順" INTEGER NOT NULL,
249
+ "工程コード" VARCHAR(20) NOT NULL,
250
+ "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
251
+ "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
252
+ CONSTRAINT "fk_作業指示明細_作業指示"
253
+ FOREIGN KEY ("作業指示番号") REFERENCES "作業指示データ"("作業指示番号"),
254
+ CONSTRAINT "fk_作業指示明細_工程"
255
+ FOREIGN KEY ("工程コード") REFERENCES "工程マスタ"("工程コード"),
256
+ UNIQUE ("作業指示番号", "工順")
257
+ );
258
+
259
+ -- インデックス
260
+ CREATE INDEX "idx_作業指示_オーダ番号" ON "作業指示データ"("オーダ番号");
261
+ CREATE INDEX "idx_作業指示_品目コード" ON "作業指示データ"("品目コード");
262
+ CREATE INDEX "idx_作業指示_ステータス" ON "作業指示データ"("ステータス");
263
+ CREATE INDEX "idx_作業指示明細_工程コード" ON "作業指示明細データ"("工程コード");
264
+ ```
265
+
266
+ </details>
267
+
268
+ ### Java エンティティの定義
269
+
270
+ #### 工程マスタ・工程表エンティティ
271
+
272
+ <details>
273
+ <summary>Entity: Process(工程マスタ)</summary>
274
+
275
+ ```java
276
+ // src/main/java/com/example/pms/domain/model/process/Process.java
277
+ package com.example.pms.domain.model.process;
278
+
279
+ import lombok.AllArgsConstructor;
280
+ import lombok.Builder;
281
+ import lombok.Data;
282
+ import lombok.NoArgsConstructor;
283
+
284
+ import java.time.LocalDateTime;
285
+
286
+ @Data
287
+ @Builder
288
+ @NoArgsConstructor
289
+ @AllArgsConstructor
290
+ public class Process {
291
+ private String processCode;
292
+ private String processName;
293
+ private LocalDateTime createdAt;
294
+ private String createdBy;
295
+ private LocalDateTime updatedAt;
296
+ private String updatedBy;
297
+ }
298
+ ```
299
+
300
+ </details>
301
+
302
+ <details>
303
+ <summary>Entity: Routing(工程表)</summary>
304
+
305
+ ```java
306
+ // src/main/java/com/example/pms/domain/model/process/Routing.java
307
+ package com.example.pms.domain.model.process;
308
+
309
+ import com.example.pms.domain.model.item.Item;
310
+ import lombok.AllArgsConstructor;
311
+ import lombok.Builder;
312
+ import lombok.Data;
313
+ import lombok.NoArgsConstructor;
314
+
315
+ import java.time.LocalDateTime;
316
+
317
+ @Data
318
+ @Builder
319
+ @NoArgsConstructor
320
+ @AllArgsConstructor
321
+ public class Routing {
322
+ private Integer id;
323
+ private String itemCode;
324
+ private Integer sequence;
325
+ private String processCode;
326
+ private LocalDateTime createdAt;
327
+ private LocalDateTime updatedAt;
328
+
329
+ // リレーション
330
+ private Item item;
331
+ private Process process;
332
+ }
333
+ ```
334
+
335
+ </details>
336
+
337
+ #### 作業指示ステータス Enum
338
+
339
+ <details>
340
+ <summary>Enum: WorkOrderStatus(作業指示ステータス)</summary>
341
+
342
+ ```java
343
+ // src/main/java/com/example/pms/domain/model/process/WorkOrderStatus.java
344
+ package com.example.pms.domain.model.process;
345
+
346
+ import lombok.Getter;
347
+ import lombok.RequiredArgsConstructor;
348
+
349
+ @Getter
350
+ @RequiredArgsConstructor
351
+ public enum WorkOrderStatus {
352
+ NOT_STARTED("未着手"),
353
+ IN_PROGRESS("作業中"),
354
+ COMPLETED("完了"),
355
+ SUSPENDED("中断");
356
+
357
+ private final String displayName;
358
+
359
+ public static WorkOrderStatus fromDisplayName(String displayName) {
360
+ for (WorkOrderStatus status : values()) {
361
+ if (status.displayName.equals(displayName)) {
362
+ return status;
363
+ }
364
+ }
365
+ throw new IllegalArgumentException("不正な作業指示ステータス: " + displayName);
366
+ }
367
+ }
368
+ ```
369
+
370
+ </details>
371
+
372
+ #### 作業指示エンティティ
373
+
374
+ <details>
375
+ <summary>Entity: WorkOrder(作業指示データ)</summary>
376
+
377
+ ```java
378
+ // src/main/java/com/example/pms/domain/model/process/WorkOrder.java
379
+ package com.example.pms.domain.model.process;
380
+
381
+ import com.example.pms.domain.model.item.Item;
382
+ import com.example.pms.domain.master.Location;
383
+ import com.example.pms.domain.model.planning.Order;
384
+ import lombok.AllArgsConstructor;
385
+ import lombok.Builder;
386
+ import lombok.Data;
387
+ import lombok.NoArgsConstructor;
388
+
389
+ import java.math.BigDecimal;
390
+ import java.time.LocalDate;
391
+ import java.time.LocalDateTime;
392
+ import java.util.List;
393
+
394
+ @Data
395
+ @Builder
396
+ @NoArgsConstructor
397
+ @AllArgsConstructor
398
+ public class WorkOrder {
399
+ private Integer id;
400
+ private String workOrderNumber;
401
+ private String orderNumber;
402
+ private LocalDate workOrderDate;
403
+ private String itemCode;
404
+ private BigDecimal orderQuantity;
405
+ private String locationCode;
406
+ private LocalDate plannedStartDate;
407
+ private LocalDate plannedEndDate;
408
+ private LocalDate actualStartDate;
409
+ private LocalDate actualEndDate;
410
+ private BigDecimal completedQuantity;
411
+ private BigDecimal totalGoodQuantity;
412
+ private BigDecimal totalDefectQuantity;
413
+ private WorkOrderStatus status;
414
+ private Boolean completedFlag;
415
+ private String remarks;
416
+ private LocalDateTime createdAt;
417
+ private String createdBy;
418
+ private LocalDateTime updatedAt;
419
+ private String updatedBy;
420
+
421
+ // リレーション
422
+ private Order order;
423
+ private Item item;
424
+ private Location location;
425
+ private List<WorkOrderDetail> details;
426
+ }
427
+ ```
428
+
429
+ </details>
430
+
431
+ <details>
432
+ <summary>Entity: WorkOrderDetail(作業指示明細データ)</summary>
433
+
434
+ ```java
435
+ // src/main/java/com/example/pms/domain/model/process/WorkOrderDetail.java
436
+ package com.example.pms.domain.model.process;
437
+
438
+ import lombok.AllArgsConstructor;
439
+ import lombok.Builder;
440
+ import lombok.Data;
441
+ import lombok.NoArgsConstructor;
442
+
443
+ import java.time.LocalDateTime;
444
+
445
+ @Data
446
+ @Builder
447
+ @NoArgsConstructor
448
+ @AllArgsConstructor
449
+ public class WorkOrderDetail {
450
+ private Integer id;
451
+ private String workOrderNumber;
452
+ private Integer sequence;
453
+ private String processCode;
454
+ private LocalDateTime createdAt;
455
+ private String createdBy;
456
+ private LocalDateTime updatedAt;
457
+ private String updatedBy;
458
+
459
+ // リレーション
460
+ private WorkOrder workOrder;
461
+ private Process process;
462
+ }
463
+ ```
464
+
465
+ </details>
466
+
467
+ ### TypeHandler の実装
468
+
469
+ <details>
470
+ <summary>TypeHandler: WorkOrderStatusTypeHandler</summary>
471
+
472
+ ```java
473
+ // src/main/java/com/example/pms/infrastructure/out/persistence/typehandler/WorkOrderStatusTypeHandler.java
474
+ package com.example.pms.infrastructure.out.persistence.typehandler;
475
+
476
+ import com.example.pms.domain.model.process.WorkOrderStatus;
477
+ import org.apache.ibatis.type.BaseTypeHandler;
478
+ import org.apache.ibatis.type.JdbcType;
479
+ import org.apache.ibatis.type.MappedTypes;
480
+
481
+ import java.sql.CallableStatement;
482
+ import java.sql.PreparedStatement;
483
+ import java.sql.ResultSet;
484
+ import java.sql.SQLException;
485
+
486
+ @MappedTypes(WorkOrderStatus.class)
487
+ public class WorkOrderStatusTypeHandler extends BaseTypeHandler<WorkOrderStatus> {
488
+
489
+ @Override
490
+ public void setNonNullParameter(PreparedStatement ps, int i, WorkOrderStatus parameter, JdbcType jdbcType)
491
+ throws SQLException {
492
+ ps.setString(i, parameter.getDisplayName());
493
+ }
494
+
495
+ @Override
496
+ public WorkOrderStatus getNullableResult(ResultSet rs, String columnName) throws SQLException {
497
+ String value = rs.getString(columnName);
498
+ return value == null ? null : WorkOrderStatus.fromDisplayName(value);
499
+ }
500
+
501
+ @Override
502
+ public WorkOrderStatus getNullableResult(ResultSet rs, int columnIndex) throws SQLException {
503
+ String value = rs.getString(columnIndex);
504
+ return value == null ? null : WorkOrderStatus.fromDisplayName(value);
505
+ }
506
+
507
+ @Override
508
+ public WorkOrderStatus getNullableResult(CallableStatement cs, int columnIndex) throws SQLException {
509
+ String value = cs.getString(columnIndex);
510
+ return value == null ? null : WorkOrderStatus.fromDisplayName(value);
511
+ }
512
+ }
513
+ ```
514
+
515
+ </details>
516
+
517
+ ### MyBatis Mapper
518
+
519
+ <details>
520
+ <summary>Mapper XML: WorkOrderMapper.xml</summary>
521
+
522
+ ```xml
523
+ <!-- src/main/resources/com/example/pms/infrastructure/out/persistence/mapper/WorkOrderMapper.xml -->
524
+ <?xml version="1.0" encoding="UTF-8" ?>
525
+ <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
526
+ "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
527
+ <mapper namespace="com.example.pms.infrastructure.out.persistence.mapper.WorkOrderMapper">
528
+
529
+ <resultMap id="WorkOrderResultMap" type="com.example.pms.domain.model.process.WorkOrder">
530
+ <id property="id" column="ID"/>
531
+ <result property="workOrderNumber" column="作業指示番号"/>
532
+ <result property="orderNumber" column="オーダ番号"/>
533
+ <result property="workOrderDate" column="作業指示日"/>
534
+ <result property="itemCode" column="品目コード"/>
535
+ <result property="orderQuantity" column="作業指示数"/>
536
+ <result property="locationCode" column="場所コード"/>
537
+ <result property="plannedStartDate" column="開始予定日"/>
538
+ <result property="plannedEndDate" column="完成予定日"/>
539
+ <result property="actualStartDate" column="実績開始日"/>
540
+ <result property="actualEndDate" column="実績完了日"/>
541
+ <result property="completedQuantity" column="完成済数"/>
542
+ <result property="totalGoodQuantity" column="総良品数"/>
543
+ <result property="totalDefectQuantity" column="総不良品数"/>
544
+ <result property="status" column="ステータス"
545
+ typeHandler="com.example.pms.infrastructure.out.persistence.typehandler.WorkOrderStatusTypeHandler"/>
546
+ <result property="completedFlag" column="完了フラグ"/>
547
+ <result property="remarks" column="備考"/>
548
+ <result property="createdAt" column="作成日時"/>
549
+ <result property="createdBy" column="作成者"/>
550
+ <result property="updatedAt" column="更新日時"/>
551
+ <result property="updatedBy" column="更新者"/>
552
+ </resultMap>
553
+
554
+ <!-- PostgreSQL用 INSERT (ENUM キャスト必須) -->
555
+ <insert id="insert" useGeneratedKeys="true" keyProperty="id" keyColumn="ID" databaseId="postgresql">
556
+ INSERT INTO "作業指示データ" (
557
+ "作業指示番号", "オーダ番号", "作業指示日", "品目コード", "作業指示数",
558
+ "場所コード", "開始予定日", "完成予定日", "ステータス", "完了フラグ", "備考", "作成者"
559
+ ) VALUES (
560
+ #{workOrderNumber},
561
+ #{orderNumber},
562
+ #{workOrderDate},
563
+ #{itemCode},
564
+ #{orderQuantity},
565
+ #{locationCode},
566
+ #{plannedStartDate},
567
+ #{plannedEndDate},
568
+ #{status, typeHandler=com.example.pms.infrastructure.out.persistence.typehandler.WorkOrderStatusTypeHandler}::作業指示ステータス,
569
+ #{completedFlag},
570
+ #{remarks},
571
+ #{createdBy}
572
+ )
573
+ </insert>
574
+
575
+ <!-- H2用 INSERT (ENUM キャスト不要) -->
576
+ <insert id="insert" useGeneratedKeys="true" keyProperty="id" keyColumn="ID" databaseId="h2">
577
+ INSERT INTO "作業指示データ" (
578
+ "作業指示番号", "オーダ番号", "作業指示日", "品目コード", "作業指示数",
579
+ "場所コード", "開始予定日", "完成予定日", "ステータス", "完了フラグ", "備考", "作成者"
580
+ ) VALUES (
581
+ #{workOrderNumber},
582
+ #{orderNumber},
583
+ #{workOrderDate},
584
+ #{itemCode},
585
+ #{orderQuantity},
586
+ #{locationCode},
587
+ #{plannedStartDate},
588
+ #{plannedEndDate},
589
+ #{status, typeHandler=com.example.pms.infrastructure.out.persistence.typehandler.WorkOrderStatusTypeHandler},
590
+ #{completedFlag},
591
+ #{remarks},
592
+ #{createdBy}
593
+ )
594
+ </insert>
595
+
596
+ <select id="findByWorkOrderNumber" resultMap="WorkOrderResultMap">
597
+ SELECT * FROM "作業指示データ" WHERE "作業指示番号" = #{workOrderNumber}
598
+ </select>
599
+
600
+ <select id="findLatestWorkOrderNumber" resultType="string">
601
+ SELECT "作業指示番号" FROM "作業指示データ"
602
+ WHERE "作業指示番号" LIKE #{prefix}
603
+ ORDER BY "作業指示番号" DESC
604
+ LIMIT 1
605
+ </select>
606
+
607
+ <!-- PostgreSQL用 UPDATE (ENUM キャスト必須) -->
608
+ <update id="startWork" databaseId="postgresql">
609
+ UPDATE "作業指示データ"
610
+ SET "ステータス" = '作業中'::作業指示ステータス,
611
+ "実績開始日" = #{actualStartDate},
612
+ "更新日時" = CURRENT_TIMESTAMP
613
+ WHERE "作業指示番号" = #{workOrderNumber}
614
+ </update>
615
+
616
+ <!-- H2用 UPDATE (ENUM キャスト不要) -->
617
+ <update id="startWork" databaseId="h2">
618
+ UPDATE "作業指示データ"
619
+ SET "ステータス" = '作業中',
620
+ "実績開始日" = #{actualStartDate},
621
+ "更新日時" = CURRENT_TIMESTAMP
622
+ WHERE "作業指示番号" = #{workOrderNumber}
623
+ </update>
624
+
625
+ <!-- PostgreSQL用 completeWork -->
626
+ <update id="completeWork" databaseId="postgresql">
627
+ UPDATE "作業指示データ"
628
+ SET "ステータス" = '完了'::作業指示ステータス,
629
+ "完了フラグ" = true,
630
+ "実績完了日" = #{actualEndDate},
631
+ "更新日時" = CURRENT_TIMESTAMP
632
+ WHERE "作業指示番号" = #{workOrderNumber}
633
+ </update>
634
+
635
+ <!-- H2用 completeWork -->
636
+ <update id="completeWork" databaseId="h2">
637
+ UPDATE "作業指示データ"
638
+ SET "ステータス" = '完了',
639
+ "完了フラグ" = true,
640
+ "実績完了日" = #{actualEndDate},
641
+ "更新日時" = CURRENT_TIMESTAMP
642
+ WHERE "作業指示番号" = #{workOrderNumber}
643
+ </update>
644
+
645
+ <update id="updateCompletionQuantities">
646
+ UPDATE "作業指示データ"
647
+ SET "完成済数" = "完成済数" + #{completedQuantity},
648
+ "総良品数" = "総良品数" + #{goodQuantity},
649
+ "総不良品数" = "総不良品数" + #{defectQuantity},
650
+ "更新日時" = CURRENT_TIMESTAMP
651
+ WHERE "作業指示番号" = #{workOrderNumber}
652
+ </update>
653
+
654
+ <delete id="deleteAll">
655
+ DELETE FROM "作業指示データ"
656
+ </delete>
657
+ </mapper>
658
+ ```
659
+
660
+ </details>
661
+
662
+ <details>
663
+ <summary>Mapper XML: WorkOrderDetailMapper.xml</summary>
664
+
665
+ ```xml
666
+ <!-- src/main/resources/com/example/pms/infrastructure/out/persistence/mapper/WorkOrderDetailMapper.xml -->
667
+ <?xml version="1.0" encoding="UTF-8" ?>
668
+ <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
669
+ "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
670
+ <mapper namespace="com.example.pms.infrastructure.out.persistence.mapper.WorkOrderDetailMapper">
671
+
672
+ <resultMap id="WorkOrderDetailResultMap" type="com.example.pms.domain.model.process.WorkOrderDetail">
673
+ <id property="id" column="ID"/>
674
+ <result property="workOrderNumber" column="作業指示番号"/>
675
+ <result property="sequence" column="工順"/>
676
+ <result property="processCode" column="工程コード"/>
677
+ <result property="createdAt" column="作成日時"/>
678
+ <result property="createdBy" column="作成者"/>
679
+ <result property="updatedAt" column="更新日時"/>
680
+ <result property="updatedBy" column="更新者"/>
681
+ </resultMap>
682
+
683
+ <insert id="insert" useGeneratedKeys="true" keyProperty="id" keyColumn="ID">
684
+ INSERT INTO "作業指示明細データ" (
685
+ "作業指示番号", "工順", "工程コード", "作成者", "更新者"
686
+ ) VALUES (
687
+ #{workOrderNumber},
688
+ #{sequence},
689
+ #{processCode},
690
+ #{createdBy},
691
+ #{updatedBy}
692
+ )
693
+ </insert>
694
+
695
+ <select id="findByWorkOrderNumber" resultMap="WorkOrderDetailResultMap">
696
+ SELECT * FROM "作業指示明細データ"
697
+ WHERE "作業指示番号" = #{workOrderNumber}
698
+ ORDER BY "工順"
699
+ </select>
700
+
701
+ <select id="findByWorkOrderNumberAndSequence" resultMap="WorkOrderDetailResultMap">
702
+ SELECT * FROM "作業指示明細データ"
703
+ WHERE "作業指示番号" = #{workOrderNumber} AND "工順" = #{sequence}
704
+ </select>
705
+
706
+ <delete id="deleteAll">
707
+ DELETE FROM "作業指示明細データ"
708
+ </delete>
709
+ </mapper>
710
+ ```
711
+
712
+ </details>
713
+
714
+ ### Mapper インターフェース
715
+
716
+ <details>
717
+ <summary>Mapper Interface: WorkOrderMapper</summary>
718
+
719
+ ```java
720
+ // src/main/java/com/example/pms/infrastructure/out/persistence/mapper/WorkOrderMapper.java
721
+ package com.example.pms.infrastructure.out.persistence.mapper;
722
+
723
+ import com.example.pms.domain.model.process.WorkOrder;
724
+ import com.example.pms.domain.model.process.WorkOrderStatus;
725
+ import org.apache.ibatis.annotations.Mapper;
726
+ import org.apache.ibatis.annotations.Param;
727
+
728
+ import java.math.BigDecimal;
729
+ import java.time.LocalDate;
730
+ import java.util.List;
731
+
732
+ @Mapper
733
+ public interface WorkOrderMapper {
734
+ void insert(WorkOrder workOrder);
735
+ void update(WorkOrder workOrder);
736
+ WorkOrder findById(Integer id);
737
+ WorkOrder findByWorkOrderNumber(String workOrderNumber);
738
+ List<WorkOrder> findByOrderNumber(String orderNumber);
739
+ List<WorkOrder> findByStatus(WorkOrderStatus status);
740
+ List<WorkOrder> findAll();
741
+ String findLatestWorkOrderNumber(String prefix);
742
+ void startWork(@Param("workOrderNumber") String workOrderNumber,
743
+ @Param("actualStartDate") LocalDate actualStartDate);
744
+ void completeWork(@Param("workOrderNumber") String workOrderNumber,
745
+ @Param("actualEndDate") LocalDate actualEndDate);
746
+ void updateCompletionQuantities(@Param("workOrderNumber") String workOrderNumber,
747
+ @Param("completedQuantity") BigDecimal completedQuantity,
748
+ @Param("goodQuantity") BigDecimal goodQuantity,
749
+ @Param("defectQuantity") BigDecimal defectQuantity);
750
+ void deleteAll();
751
+ }
752
+ ```
753
+
754
+ </details>
755
+
756
+ <details>
757
+ <summary>Mapper Interface: WorkOrderDetailMapper</summary>
758
+
759
+ ```java
760
+ // src/main/java/com/example/pms/infrastructure/out/persistence/mapper/WorkOrderDetailMapper.java
761
+ package com.example.pms.infrastructure.out.persistence.mapper;
762
+
763
+ import com.example.pms.domain.model.process.WorkOrderDetail;
764
+ import org.apache.ibatis.annotations.Mapper;
765
+ import org.apache.ibatis.annotations.Param;
766
+
767
+ import java.util.List;
768
+
769
+ @Mapper
770
+ public interface WorkOrderDetailMapper {
771
+ void insert(WorkOrderDetail detail);
772
+ void update(WorkOrderDetail detail);
773
+ WorkOrderDetail findById(Integer id);
774
+ List<WorkOrderDetail> findByWorkOrderNumber(String workOrderNumber);
775
+ WorkOrderDetail findByWorkOrderNumberAndSequence(@Param("workOrderNumber") String workOrderNumber,
776
+ @Param("sequence") Integer sequence);
777
+ List<WorkOrderDetail> findAll();
778
+ void deleteAll();
779
+ }
780
+ ```
781
+
782
+ </details>
783
+
784
+ ### 作業指示サービスの実装
785
+
786
+ <details>
787
+ <summary>Service: WorkOrderService</summary>
788
+
789
+ ```java
790
+ // src/main/java/com/example/pms/application/service/WorkOrderService.java
791
+ package com.example.pms.application.service;
792
+
793
+ import com.example.pms.domain.model.planning.Order;
794
+ import com.example.pms.domain.model.process.*;
795
+ import com.example.pms.infrastructure.out.persistence.mapper.*;
796
+ import lombok.RequiredArgsConstructor;
797
+ import org.springframework.stereotype.Service;
798
+ import org.springframework.transaction.annotation.Transactional;
799
+
800
+ import java.math.BigDecimal;
801
+ import java.time.LocalDate;
802
+ import java.time.format.DateTimeFormatter;
803
+ import java.util.ArrayList;
804
+ import java.util.List;
805
+
806
+ @Service
807
+ @RequiredArgsConstructor
808
+ public class WorkOrderService {
809
+
810
+ private final WorkOrderMapper workOrderMapper;
811
+ private final WorkOrderDetailMapper workOrderDetailMapper;
812
+ private final OrderMapper orderMapper;
813
+ private final RoutingMapper routingMapper;
814
+
815
+ /**
816
+ * 作業指示番号を生成する
817
+ */
818
+ private String generateWorkOrderNumber(LocalDate workOrderDate) {
819
+ String prefix = "WO-" + workOrderDate.format(DateTimeFormatter.ofPattern("yyyyMM")) + "-";
820
+ String latestNumber = workOrderMapper.findLatestWorkOrderNumber(prefix + "%");
821
+
822
+ int sequence = 1;
823
+ if (latestNumber != null) {
824
+ int currentSequence = Integer.parseInt(latestNumber.substring(latestNumber.length() - 4));
825
+ sequence = currentSequence + 1;
826
+ }
827
+
828
+ return prefix + String.format("%04d", sequence);
829
+ }
830
+
831
+ /**
832
+ * 作業指示を作成する
833
+ */
834
+ @Transactional
835
+ public WorkOrder createWorkOrder(WorkOrderCreateInput input) {
836
+ // オーダ情報を取得
837
+ Order order = orderMapper.findByOrderNumber(input.getOrderNumber());
838
+ if (order == null) {
839
+ throw new IllegalArgumentException("Order not found: " + input.getOrderNumber());
840
+ }
841
+
842
+ // 工程表を取得
843
+ List<Routing> routings = routingMapper.findByItemCode(order.getItemCode());
844
+ if (routings.isEmpty()) {
845
+ throw new IllegalArgumentException("Routing not found for item: " + order.getItemCode());
846
+ }
847
+
848
+ String workOrderNumber = generateWorkOrderNumber(input.getWorkOrderDate());
849
+
850
+ // 作業指示ヘッダを作成
851
+ WorkOrder workOrder = WorkOrder.builder()
852
+ .workOrderNumber(workOrderNumber)
853
+ .orderNumber(input.getOrderNumber())
854
+ .workOrderDate(input.getWorkOrderDate())
855
+ .itemCode(order.getItemCode())
856
+ .orderQuantity(order.getPlannedQuantity())
857
+ .locationCode(input.getLocationCode())
858
+ .plannedStartDate(input.getPlannedStartDate())
859
+ .plannedEndDate(input.getPlannedEndDate())
860
+ .completedQuantity(BigDecimal.ZERO)
861
+ .totalGoodQuantity(BigDecimal.ZERO)
862
+ .totalDefectQuantity(BigDecimal.ZERO)
863
+ .status(WorkOrderStatus.NOT_STARTED)
864
+ .completedFlag(false)
865
+ .remarks(input.getRemarks())
866
+ .build();
867
+ workOrderMapper.insert(workOrder);
868
+
869
+ // 作業指示明細を作成(工程表から自動展開)
870
+ List<WorkOrderDetail> details = new ArrayList<>();
871
+ for (Routing routing : routings) {
872
+ WorkOrderDetail detail = WorkOrderDetail.builder()
873
+ .workOrderNumber(workOrderNumber)
874
+ .sequence(routing.getSequence())
875
+ .processCode(routing.getProcessCode())
876
+ .build();
877
+ workOrderDetailMapper.insert(detail);
878
+ details.add(detail);
879
+ }
880
+
881
+ workOrder.setDetails(details);
882
+ return workOrder;
883
+ }
884
+
885
+ /**
886
+ * 作業を開始する
887
+ */
888
+ @Transactional
889
+ public WorkOrder startWork(String workOrderNumber) {
890
+ WorkOrder workOrder = workOrderMapper.findByWorkOrderNumber(workOrderNumber);
891
+ if (workOrder == null) {
892
+ throw new IllegalArgumentException("Work order not found: " + workOrderNumber);
893
+ }
894
+
895
+ if (workOrder.getStatus() != WorkOrderStatus.NOT_STARTED) {
896
+ throw new IllegalStateException("Only NOT_STARTED work orders can be started");
897
+ }
898
+
899
+ workOrderMapper.startWork(workOrderNumber, LocalDate.now());
900
+ return workOrderMapper.findByWorkOrderNumber(workOrderNumber);
901
+ }
902
+
903
+ /**
904
+ * 作業を完了する
905
+ */
906
+ @Transactional
907
+ public WorkOrder completeWork(String workOrderNumber) {
908
+ WorkOrder workOrder = workOrderMapper.findByWorkOrderNumber(workOrderNumber);
909
+ if (workOrder == null) {
910
+ throw new IllegalArgumentException("Work order not found: " + workOrderNumber);
911
+ }
912
+
913
+ if (workOrder.getStatus() != WorkOrderStatus.IN_PROGRESS) {
914
+ throw new IllegalStateException("Only IN_PROGRESS work orders can be completed");
915
+ }
916
+
917
+ workOrderMapper.completeWork(workOrderNumber, LocalDate.now());
918
+ return workOrderMapper.findByWorkOrderNumber(workOrderNumber);
919
+ }
920
+
921
+ /**
922
+ * 作業指示を検索する
923
+ */
924
+ public WorkOrder findByWorkOrderNumber(String workOrderNumber) {
925
+ WorkOrder workOrder = workOrderMapper.findByWorkOrderNumber(workOrderNumber);
926
+ if (workOrder != null) {
927
+ workOrder.setDetails(workOrderDetailMapper.findByWorkOrderNumber(workOrderNumber));
928
+ }
929
+ return workOrder;
930
+ }
931
+ }
932
+ ```
933
+
934
+ </details>
935
+
936
+ <details>
937
+ <summary>Input DTO: WorkOrderCreateInput</summary>
938
+
939
+ ```java
940
+ // src/main/java/com/example/pms/application/service/WorkOrderCreateInput.java
941
+ package com.example.pms.application.service;
942
+
943
+ import lombok.Builder;
944
+ import lombok.Data;
945
+
946
+ import java.time.LocalDate;
947
+
948
+ @Data
949
+ @Builder
950
+ public class WorkOrderCreateInput {
951
+ private String orderNumber;
952
+ private LocalDate workOrderDate;
953
+ private String locationCode;
954
+ private LocalDate plannedStartDate;
955
+ private LocalDate plannedEndDate;
956
+ private String remarks;
957
+ }
958
+ ```
959
+
960
+ </details>
961
+
962
+ ### TDD: 作業指示のテスト
963
+
964
+ <details>
965
+ <summary>Test: WorkOrderServiceTest</summary>
966
+
967
+ ```java
968
+ // src/test/java/com/example/pms/application/service/WorkOrderServiceTest.java
969
+ package com.example.pms.application.service;
970
+
971
+ import com.example.pms.domain.model.item.Item;
972
+ import com.example.pms.domain.model.item.ItemCategory;
973
+ import com.example.pms.domain.master.Location;
974
+ import com.example.pms.domain.model.planning.Order;
975
+ import com.example.pms.domain.model.planning.OrderType;
976
+ import com.example.pms.domain.model.planning.OrderStatus;
977
+ import com.example.pms.domain.model.process.*;
978
+ import com.example.pms.infrastructure.out.persistence.mapper.*;
979
+ import org.junit.jupiter.api.*;
980
+ import org.mybatis.spring.boot.test.autoconfigure.MybatisTest;
981
+ import org.springframework.beans.factory.annotation.Autowired;
982
+ import org.springframework.boot.test.autoconfigure.jdbc.AutoConfigureTestDatabase;
983
+ import org.springframework.context.annotation.Import;
984
+ import org.springframework.test.context.DynamicPropertyRegistry;
985
+ import org.springframework.test.context.DynamicPropertySource;
986
+ import org.testcontainers.containers.PostgreSQLContainer;
987
+ import org.testcontainers.junit.jupiter.Container;
988
+ import org.testcontainers.junit.jupiter.Testcontainers;
989
+
990
+ import java.math.BigDecimal;
991
+ import java.time.LocalDate;
992
+ import java.util.List;
993
+
994
+ import static org.assertj.core.api.Assertions.*;
995
+
996
+ @MybatisTest
997
+ @AutoConfigureTestDatabase(replace = AutoConfigureTestDatabase.Replace.NONE)
998
+ @Import(WorkOrderService.class)
999
+ @Testcontainers
1000
+ @DisplayName("作業指示")
1001
+ class WorkOrderServiceTest {
1002
+
1003
+ @Container
1004
+ static PostgreSQLContainer<?> postgres = new PostgreSQLContainer<>("postgres:15")
1005
+ .withDatabaseName("testdb")
1006
+ .withUsername("testuser")
1007
+ .withPassword("testpass");
1008
+
1009
+ @DynamicPropertySource
1010
+ static void configureProperties(DynamicPropertyRegistry registry) {
1011
+ registry.add("spring.datasource.url", postgres::getJdbcUrl);
1012
+ registry.add("spring.datasource.username", postgres::getUsername);
1013
+ registry.add("spring.datasource.password", postgres::getPassword);
1014
+ }
1015
+
1016
+ @Autowired
1017
+ private WorkOrderService workOrderService;
1018
+
1019
+ @Autowired
1020
+ private ItemMapper itemMapper;
1021
+
1022
+ @Autowired
1023
+ private ProcessMapper processMapper;
1024
+
1025
+ @Autowired
1026
+ private RoutingMapper routingMapper;
1027
+
1028
+ @Autowired
1029
+ private LocationMapper locationMapper;
1030
+
1031
+ @Autowired
1032
+ private OrderMapper orderMapper;
1033
+
1034
+ @Autowired
1035
+ private WorkOrderMapper workOrderMapper;
1036
+
1037
+ @Autowired
1038
+ private WorkOrderDetailMapper workOrderDetailMapper;
1039
+
1040
+ @BeforeEach
1041
+ void setUp() {
1042
+ workOrderDetailMapper.deleteAll();
1043
+ workOrderMapper.deleteAll();
1044
+ routingMapper.deleteAll();
1045
+ orderMapper.deleteAll();
1046
+ processMapper.deleteAll();
1047
+ locationMapper.deleteAll();
1048
+ itemMapper.deleteAll();
1049
+
1050
+ setupMasterData();
1051
+ }
1052
+
1053
+ void setupMasterData() {
1054
+ // 品目マスタ
1055
+ itemMapper.insert(Item.builder()
1056
+ .itemCode("PROD-001")
1057
+ .effectiveFrom(LocalDate.of(2025, 1, 1))
1058
+ .itemName("製品A")
1059
+ .itemCategory(ItemCategory.PRODUCT)
1060
+ .build());
1061
+
1062
+ // 工程マスタ
1063
+ processMapper.insert(Process.builder()
1064
+ .processCode("PRESS")
1065
+ .processName("プレス加工")
1066
+ .build());
1067
+ processMapper.insert(Process.builder()
1068
+ .processCode("ASSEMBLY")
1069
+ .processName("組立")
1070
+ .build());
1071
+ processMapper.insert(Process.builder()
1072
+ .processCode("INSPECT")
1073
+ .processName("検査")
1074
+ .build());
1075
+
1076
+ // 工程表
1077
+ routingMapper.insert(Routing.builder()
1078
+ .itemCode("PROD-001")
1079
+ .sequence(1)
1080
+ .processCode("PRESS")
1081
+ .build());
1082
+ routingMapper.insert(Routing.builder()
1083
+ .itemCode("PROD-001")
1084
+ .sequence(2)
1085
+ .processCode("ASSEMBLY")
1086
+ .build());
1087
+ routingMapper.insert(Routing.builder()
1088
+ .itemCode("PROD-001")
1089
+ .sequence(3)
1090
+ .processCode("INSPECT")
1091
+ .build());
1092
+
1093
+ // 場所マスタ
1094
+ locationMapper.insert(Location.builder()
1095
+ .locationCode("LINE001")
1096
+ .locationName("製造ライン1")
1097
+ .locationTypeCode("製造")
1098
+ .build());
1099
+
1100
+ // オーダ情報
1101
+ orderMapper.insert(Order.builder()
1102
+ .orderNumber("MO-2025-001")
1103
+ .orderType(OrderType.MANUFACTURING)
1104
+ .itemCode("PROD-001")
1105
+ .plannedStartDate(LocalDate.of(2025, 1, 21))
1106
+ .dueDate(LocalDate.of(2025, 1, 25))
1107
+ .plannedQuantity(new BigDecimal("100"))
1108
+ .status(OrderStatus.CONFIRMED)
1109
+ .build());
1110
+ }
1111
+
1112
+ @Nested
1113
+ @DisplayName("作業指示の作成")
1114
+ class WorkOrderCreation {
1115
+
1116
+ @Test
1117
+ @DisplayName("オーダ情報から作業指示を作成できる")
1118
+ void canCreateWorkOrderFromOrder() {
1119
+ // Act
1120
+ WorkOrderCreateInput input = WorkOrderCreateInput.builder()
1121
+ .orderNumber("MO-2025-001")
1122
+ .workOrderDate(LocalDate.of(2025, 1, 20))
1123
+ .locationCode("LINE001")
1124
+ .plannedStartDate(LocalDate.of(2025, 1, 21))
1125
+ .plannedEndDate(LocalDate.of(2025, 1, 25))
1126
+ .build();
1127
+
1128
+ WorkOrder workOrder = workOrderService.createWorkOrder(input);
1129
+
1130
+ // Assert
1131
+ assertThat(workOrder).isNotNull();
1132
+ assertThat(workOrder.getWorkOrderNumber()).startsWith("WO-");
1133
+ assertThat(workOrder.getItemCode()).isEqualTo("PROD-001");
1134
+ assertThat(workOrder.getOrderQuantity()).isEqualByComparingTo(new BigDecimal("100"));
1135
+ assertThat(workOrder.getDetails()).hasSize(3); // 工程表から3工程
1136
+ }
1137
+
1138
+ @Test
1139
+ @DisplayName("工程表の工順に従って明細が作成される")
1140
+ void detailsAreCreatedAccordingToRouting() {
1141
+ // Act
1142
+ WorkOrderCreateInput input = WorkOrderCreateInput.builder()
1143
+ .orderNumber("MO-2025-001")
1144
+ .workOrderDate(LocalDate.of(2025, 1, 20))
1145
+ .locationCode("LINE001")
1146
+ .plannedStartDate(LocalDate.of(2025, 1, 21))
1147
+ .plannedEndDate(LocalDate.of(2025, 1, 25))
1148
+ .build();
1149
+
1150
+ WorkOrder workOrder = workOrderService.createWorkOrder(input);
1151
+
1152
+ // Assert: 工順の順序を確認
1153
+ List<Integer> sequences = workOrder.getDetails().stream()
1154
+ .map(WorkOrderDetail::getSequence)
1155
+ .sorted()
1156
+ .toList();
1157
+ assertThat(sequences).containsExactly(1, 2, 3);
1158
+
1159
+ // 各工程の確認
1160
+ List<String> processCodes = workOrder.getDetails().stream()
1161
+ .map(WorkOrderDetail::getProcessCode)
1162
+ .toList();
1163
+ assertThat(processCodes).containsExactlyInAnyOrder("PRESS", "ASSEMBLY", "INSPECT");
1164
+ }
1165
+ }
1166
+
1167
+ @Nested
1168
+ @DisplayName("作業指示のステータス管理")
1169
+ class WorkOrderStatusManagement {
1170
+
1171
+ @Test
1172
+ @DisplayName("作業指示を開始できる")
1173
+ void canStartWorkOrder() {
1174
+ // Arrange
1175
+ WorkOrder workOrder = workOrderService.createWorkOrder(WorkOrderCreateInput.builder()
1176
+ .orderNumber("MO-2025-001")
1177
+ .workOrderDate(LocalDate.of(2025, 1, 20))
1178
+ .locationCode("LINE001")
1179
+ .plannedStartDate(LocalDate.of(2025, 1, 21))
1180
+ .plannedEndDate(LocalDate.of(2025, 1, 25))
1181
+ .build());
1182
+
1183
+ // Act
1184
+ WorkOrder updated = workOrderService.startWork(workOrder.getWorkOrderNumber());
1185
+
1186
+ // Assert
1187
+ assertThat(updated.getStatus()).isEqualTo(WorkOrderStatus.IN_PROGRESS);
1188
+ assertThat(updated.getActualStartDate()).isNotNull();
1189
+ }
1190
+
1191
+ @Test
1192
+ @DisplayName("作業指示を完了できる")
1193
+ void canCompleteWorkOrder() {
1194
+ // Arrange
1195
+ WorkOrder workOrder = workOrderService.createWorkOrder(WorkOrderCreateInput.builder()
1196
+ .orderNumber("MO-2025-001")
1197
+ .workOrderDate(LocalDate.of(2025, 1, 20))
1198
+ .locationCode("LINE001")
1199
+ .plannedStartDate(LocalDate.of(2025, 1, 21))
1200
+ .plannedEndDate(LocalDate.of(2025, 1, 25))
1201
+ .build());
1202
+ workOrderService.startWork(workOrder.getWorkOrderNumber());
1203
+
1204
+ // Act
1205
+ WorkOrder updated = workOrderService.completeWork(workOrder.getWorkOrderNumber());
1206
+
1207
+ // Assert
1208
+ assertThat(updated.getStatus()).isEqualTo(WorkOrderStatus.COMPLETED);
1209
+ assertThat(updated.getCompletedFlag()).isTrue();
1210
+ assertThat(updated.getActualEndDate()).isNotNull();
1211
+ }
1212
+ }
1213
+ }
1214
+ ```
1215
+
1216
+ </details>
1217
+
1218
+ ---
1219
+
1220
+ ## 27.2 製造実績の DB 設計
1221
+
1222
+ ### 完成実績データの構造
1223
+
1224
+ 製造現場からの完成報告を記録します。良品数・不良品数を管理し、作業指示の完成済数に反映します。
1225
+
1226
+ ```plantuml
1227
+ @startuml
1228
+
1229
+ title 完成実績の構造
1230
+
1231
+ entity "完成実績データ" as completion_result {
1232
+ * ID [PK]
1233
+ --
1234
+ * 完成実績番号 [UNIQUE]
1235
+ * 作業指示番号 [FK]
1236
+ * 品目コード [FK]
1237
+ * 完成日
1238
+ * 完成数量
1239
+ * 良品数
1240
+ * 不良品数
1241
+ 備考
1242
+ }
1243
+
1244
+ entity "完成検査結果データ" as inspection_result {
1245
+ * ID [PK]
1246
+ --
1247
+ * 完成実績番号 [FK]
1248
+ * 欠点コード [FK]
1249
+ * 数量
1250
+ }
1251
+
1252
+ entity "作業指示データ" as work_order
1253
+
1254
+ completion_result }o--|| work_order : "報告"
1255
+ completion_result ||--o{ inspection_result : "検査"
1256
+
1257
+ @enduml
1258
+ ```
1259
+
1260
+ <details>
1261
+ <summary>DDL: 完成実績データ・完成検査結果データ</summary>
1262
+
1263
+ ```sql
1264
+ -- V015__create_completion_result_tables.sql
1265
+
1266
+ -- 完成実績データ
1267
+ CREATE TABLE "完成実績データ" (
1268
+ "ID" SERIAL PRIMARY KEY,
1269
+ "完成実績番号" VARCHAR(20) UNIQUE NOT NULL,
1270
+ "作業指示番号" VARCHAR(20) NOT NULL,
1271
+ "品目コード" VARCHAR(20) NOT NULL,
1272
+ "完成日" DATE NOT NULL,
1273
+ "完成数量" DECIMAL(15, 2) NOT NULL,
1274
+ "良品数" DECIMAL(15, 2) NOT NULL,
1275
+ "不良品数" DECIMAL(15, 2) NOT NULL,
1276
+ "備考" TEXT,
1277
+ "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
1278
+ "作成者" VARCHAR(50),
1279
+ "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
1280
+ "更新者" VARCHAR(50),
1281
+ CONSTRAINT "fk_完成実績_作業指示"
1282
+ FOREIGN KEY ("作業指示番号") REFERENCES "作業指示データ"("作業指示番号"),
1283
+ CONSTRAINT "fk_完成実績_品目"
1284
+ FOREIGN KEY ("品目コード") REFERENCES "品目マスタ"("品目コード")
1285
+ );
1286
+
1287
+ -- 完成検査結果データ
1288
+ CREATE TABLE "完成検査結果データ" (
1289
+ "ID" SERIAL PRIMARY KEY,
1290
+ "完成実績番号" VARCHAR(20) NOT NULL,
1291
+ "欠点コード" VARCHAR(20) NOT NULL,
1292
+ "数量" DECIMAL(15, 2) NOT NULL,
1293
+ "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
1294
+ "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
1295
+ CONSTRAINT "fk_完成検査結果_完成実績"
1296
+ FOREIGN KEY ("完成実績番号") REFERENCES "完成実績データ"("完成実績番号"),
1297
+ CONSTRAINT "fk_完成検査結果_欠点"
1298
+ FOREIGN KEY ("欠点コード") REFERENCES "欠点マスタ"("欠点コード"),
1299
+ UNIQUE ("完成実績番号", "欠点コード")
1300
+ );
1301
+
1302
+ -- インデックス
1303
+ CREATE INDEX "idx_完成実績_作業指示番号" ON "完成実績データ"("作業指示番号");
1304
+ CREATE INDEX "idx_完成実績_品目コード" ON "完成実績データ"("品目コード");
1305
+ CREATE INDEX "idx_完成実績_完成日" ON "完成実績データ"("完成日");
1306
+ ```
1307
+
1308
+ </details>
1309
+
1310
+ ### Java エンティティの定義(完成実績)
1311
+
1312
+ <details>
1313
+ <summary>Entity: CompletionResult(完成実績データ)</summary>
1314
+
1315
+ ```java
1316
+ // src/main/java/com/example/pms/domain/model/process/CompletionResult.java
1317
+ package com.example.pms.domain.model.process;
1318
+
1319
+ import com.example.pms.domain.model.item.Item;
1320
+ import lombok.AllArgsConstructor;
1321
+ import lombok.Builder;
1322
+ import lombok.Data;
1323
+ import lombok.NoArgsConstructor;
1324
+
1325
+ import java.math.BigDecimal;
1326
+ import java.time.LocalDate;
1327
+ import java.time.LocalDateTime;
1328
+ import java.util.List;
1329
+
1330
+ @Data
1331
+ @Builder
1332
+ @NoArgsConstructor
1333
+ @AllArgsConstructor
1334
+ public class CompletionResult {
1335
+ private Integer id;
1336
+ private String completionResultNumber;
1337
+ private String workOrderNumber;
1338
+ private String itemCode;
1339
+ private LocalDate completionDate;
1340
+ private BigDecimal completedQuantity;
1341
+ private BigDecimal goodQuantity;
1342
+ private BigDecimal defectQuantity;
1343
+ private String remarks;
1344
+ private LocalDateTime createdAt;
1345
+ private String createdBy;
1346
+ private LocalDateTime updatedAt;
1347
+ private String updatedBy;
1348
+
1349
+ // リレーション
1350
+ private WorkOrder workOrder;
1351
+ private Item item;
1352
+ private List<InspectionResult> inspectionResults;
1353
+ }
1354
+ ```
1355
+
1356
+ </details>
1357
+
1358
+ <details>
1359
+ <summary>Entity: InspectionResult(完成検査結果データ)</summary>
1360
+
1361
+ ```java
1362
+ // src/main/java/com/example/pms/domain/model/process/InspectionResult.java
1363
+ package com.example.pms.domain.model.process;
1364
+
1365
+ import com.example.pms.domain.master.Defect;
1366
+ import lombok.AllArgsConstructor;
1367
+ import lombok.Builder;
1368
+ import lombok.Data;
1369
+ import lombok.NoArgsConstructor;
1370
+
1371
+ import java.math.BigDecimal;
1372
+ import java.time.LocalDateTime;
1373
+
1374
+ @Data
1375
+ @Builder
1376
+ @NoArgsConstructor
1377
+ @AllArgsConstructor
1378
+ public class InspectionResult {
1379
+ private Integer id;
1380
+ private String completionResultNumber;
1381
+ private String defectCode;
1382
+ private BigDecimal quantity;
1383
+ private LocalDateTime createdAt;
1384
+ private String createdBy;
1385
+ private LocalDateTime updatedAt;
1386
+ private String updatedBy;
1387
+
1388
+ // リレーション
1389
+ private CompletionResult completionResult;
1390
+ private Defect defect;
1391
+ }
1392
+ ```
1393
+
1394
+ </details>
1395
+
1396
+ ### MyBatis Mapper(完成実績)
1397
+
1398
+ <details>
1399
+ <summary>Mapper XML: CompletionResultMapper.xml</summary>
1400
+
1401
+ ```xml
1402
+ <!-- src/main/resources/com/example/pms/infrastructure/out/persistence/mapper/CompletionResultMapper.xml -->
1403
+ <?xml version="1.0" encoding="UTF-8" ?>
1404
+ <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
1405
+ "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
1406
+ <mapper namespace="com.example.pms.infrastructure.out.persistence.mapper.CompletionResultMapper">
1407
+
1408
+ <resultMap id="CompletionResultResultMap" type="com.example.pms.domain.model.process.CompletionResult">
1409
+ <id property="id" column="ID"/>
1410
+ <result property="completionResultNumber" column="完成実績番号"/>
1411
+ <result property="workOrderNumber" column="作業指示番号"/>
1412
+ <result property="itemCode" column="品目コード"/>
1413
+ <result property="completionDate" column="完成日"/>
1414
+ <result property="completedQuantity" column="完成数量"/>
1415
+ <result property="goodQuantity" column="良品数"/>
1416
+ <result property="defectQuantity" column="不良品数"/>
1417
+ <result property="remarks" column="備考"/>
1418
+ <result property="createdAt" column="作成日時"/>
1419
+ <result property="createdBy" column="作成者"/>
1420
+ <result property="updatedAt" column="更新日時"/>
1421
+ <result property="updatedBy" column="更新者"/>
1422
+ </resultMap>
1423
+
1424
+ <insert id="insert" useGeneratedKeys="true" keyProperty="id" keyColumn="ID">
1425
+ INSERT INTO "完成実績データ" (
1426
+ "完成実績番号", "作業指示番号", "品目コード", "完成日",
1427
+ "完成数量", "良品数", "不良品数", "備考", "作成者"
1428
+ ) VALUES (
1429
+ #{completionResultNumber},
1430
+ #{workOrderNumber},
1431
+ #{itemCode},
1432
+ #{completionDate},
1433
+ #{completedQuantity},
1434
+ #{goodQuantity},
1435
+ #{defectQuantity},
1436
+ #{remarks},
1437
+ #{createdBy}
1438
+ )
1439
+ </insert>
1440
+
1441
+ <select id="findByCompletionResultNumber" resultMap="CompletionResultResultMap">
1442
+ SELECT * FROM "完成実績データ" WHERE "完成実績番号" = #{completionResultNumber}
1443
+ </select>
1444
+
1445
+ <select id="findByWorkOrderNumber" resultMap="CompletionResultResultMap">
1446
+ SELECT * FROM "完成実績データ"
1447
+ WHERE "作業指示番号" = #{workOrderNumber}
1448
+ ORDER BY "完成日"
1449
+ </select>
1450
+
1451
+ <select id="findLatestCompletionResultNumber" resultType="string">
1452
+ SELECT "完成実績番号" FROM "完成実績データ"
1453
+ WHERE "完成実績番号" LIKE #{prefix}
1454
+ ORDER BY "完成実績番号" DESC
1455
+ LIMIT 1
1456
+ </select>
1457
+
1458
+ <delete id="deleteAll">
1459
+ DELETE FROM "完成実績データ"
1460
+ </delete>
1461
+ </mapper>
1462
+ ```
1463
+
1464
+ </details>
1465
+
1466
+ ### 完成実績サービスの実装
1467
+
1468
+ <details>
1469
+ <summary>Service: CompletionResultService</summary>
1470
+
1471
+ ```java
1472
+ // src/main/java/com/example/pms/application/service/CompletionResultService.java
1473
+ package com.example.pms.application.service;
1474
+
1475
+ import com.example.pms.domain.model.process.*;
1476
+ import com.example.pms.infrastructure.out.persistence.mapper.*;
1477
+ import lombok.RequiredArgsConstructor;
1478
+ import org.springframework.stereotype.Service;
1479
+ import org.springframework.transaction.annotation.Transactional;
1480
+
1481
+ import java.time.LocalDate;
1482
+ import java.time.format.DateTimeFormatter;
1483
+ import java.util.ArrayList;
1484
+ import java.util.List;
1485
+
1486
+ @Service
1487
+ @RequiredArgsConstructor
1488
+ public class CompletionResultService {
1489
+
1490
+ private final CompletionResultMapper completionResultMapper;
1491
+ private final InspectionResultMapper inspectionResultMapper;
1492
+ private final WorkOrderMapper workOrderMapper;
1493
+
1494
+ /**
1495
+ * 完成実績番号を生成する
1496
+ */
1497
+ private String generateCompletionResultNumber(LocalDate completionDate) {
1498
+ String prefix = "CR-" + completionDate.format(DateTimeFormatter.ofPattern("yyyyMM")) + "-";
1499
+ String latestNumber = completionResultMapper.findLatestCompletionResultNumber(prefix + "%");
1500
+
1501
+ int sequence = 1;
1502
+ if (latestNumber != null) {
1503
+ int currentSequence = Integer.parseInt(latestNumber.substring(latestNumber.length() - 4));
1504
+ sequence = currentSequence + 1;
1505
+ }
1506
+
1507
+ return prefix + String.format("%04d", sequence);
1508
+ }
1509
+
1510
+ /**
1511
+ * 完成実績を報告する
1512
+ */
1513
+ @Transactional
1514
+ public CompletionResult reportCompletion(CompletionResultInput input) {
1515
+ // 作業指示を取得
1516
+ WorkOrder workOrder = workOrderMapper.findByWorkOrderNumber(input.getWorkOrderNumber());
1517
+ if (workOrder == null) {
1518
+ throw new IllegalArgumentException("Work order not found: " + input.getWorkOrderNumber());
1519
+ }
1520
+
1521
+ String completionResultNumber = generateCompletionResultNumber(input.getCompletionDate());
1522
+
1523
+ // 完成実績を作成
1524
+ CompletionResult completionResult = CompletionResult.builder()
1525
+ .completionResultNumber(completionResultNumber)
1526
+ .workOrderNumber(input.getWorkOrderNumber())
1527
+ .itemCode(workOrder.getItemCode())
1528
+ .completionDate(input.getCompletionDate())
1529
+ .completedQuantity(input.getCompletedQuantity())
1530
+ .goodQuantity(input.getGoodQuantity())
1531
+ .defectQuantity(input.getDefectQuantity())
1532
+ .remarks(input.getRemarks())
1533
+ .build();
1534
+ completionResultMapper.insert(completionResult);
1535
+
1536
+ // 検査結果を作成
1537
+ List<InspectionResult> inspectionResults = new ArrayList<>();
1538
+ if (input.getInspectionResults() != null) {
1539
+ for (InspectionResultInput irInput : input.getInspectionResults()) {
1540
+ InspectionResult ir = InspectionResult.builder()
1541
+ .completionResultNumber(completionResultNumber)
1542
+ .defectCode(irInput.getDefectCode())
1543
+ .quantity(irInput.getQuantity())
1544
+ .build();
1545
+ inspectionResultMapper.insert(ir);
1546
+ inspectionResults.add(ir);
1547
+ }
1548
+ }
1549
+ completionResult.setInspectionResults(inspectionResults);
1550
+
1551
+ // 作業指示の累計を更新
1552
+ workOrderMapper.updateCompletionQuantities(
1553
+ input.getWorkOrderNumber(),
1554
+ input.getCompletedQuantity(),
1555
+ input.getGoodQuantity(),
1556
+ input.getDefectQuantity()
1557
+ );
1558
+
1559
+ return completionResult;
1560
+ }
1561
+ }
1562
+ ```
1563
+
1564
+ </details>
1565
+
1566
+ ### 工数実績データの構造
1567
+
1568
+ 工数実績は、作業指示明細(工順)ごとに担当者の作業時間を記録します。
1569
+
1570
+ ```plantuml
1571
+ @startuml
1572
+
1573
+ title 工数実績の構造
1574
+
1575
+ entity "工数実績データ" as labor_hours {
1576
+ * ID [PK]
1577
+ --
1578
+ * 工数実績番号 [UNIQUE]
1579
+ * 作業指示番号 [FK]
1580
+ * 品目コード [FK]
1581
+ * 工順
1582
+ * 工程コード [FK]
1583
+ * 部門コード [FK]
1584
+ * 担当者コード [FK]
1585
+ * 作業日
1586
+ * 工数(時間)
1587
+ 備考
1588
+ }
1589
+
1590
+ entity "作業指示明細データ" as work_order_detail
1591
+
1592
+ labor_hours }o--|| work_order_detail : "報告"
1593
+
1594
+ @enduml
1595
+ ```
1596
+
1597
+ <details>
1598
+ <summary>DDL: 工数実績データ</summary>
1599
+
1600
+ ```sql
1601
+ -- V016__create_labor_hours_tables.sql
1602
+
1603
+ -- 工数実績データ
1604
+ CREATE TABLE "工数実績データ" (
1605
+ "ID" SERIAL PRIMARY KEY,
1606
+ "工数実績番号" VARCHAR(20) UNIQUE NOT NULL,
1607
+ "作業指示番号" VARCHAR(20) NOT NULL,
1608
+ "品目コード" VARCHAR(20) NOT NULL,
1609
+ "工順" INTEGER NOT NULL,
1610
+ "工程コード" VARCHAR(20) NOT NULL,
1611
+ "部門コード" VARCHAR(20) NOT NULL,
1612
+ "担当者コード" VARCHAR(20) NOT NULL,
1613
+ "作業日" DATE NOT NULL,
1614
+ "工数" DECIMAL(10, 2) NOT NULL,
1615
+ "備考" TEXT,
1616
+ "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
1617
+ "作成者" VARCHAR(50),
1618
+ "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
1619
+ "更新者" VARCHAR(50),
1620
+ CONSTRAINT "fk_工数実績_作業指示"
1621
+ FOREIGN KEY ("作業指示番号") REFERENCES "作業指示データ"("作業指示番号"),
1622
+ CONSTRAINT "fk_工数実績_品目"
1623
+ FOREIGN KEY ("品目コード") REFERENCES "品目マスタ"("品目コード"),
1624
+ CONSTRAINT "fk_工数実績_工程"
1625
+ FOREIGN KEY ("工程コード") REFERENCES "工程マスタ"("工程コード"),
1626
+ CONSTRAINT "fk_工数実績_部門"
1627
+ FOREIGN KEY ("部門コード") REFERENCES "部門マスタ"("部門コード"),
1628
+ CONSTRAINT "fk_工数実績_担当者"
1629
+ FOREIGN KEY ("担当者コード") REFERENCES "担当者マスタ"("担当者コード")
1630
+ );
1631
+
1632
+ -- インデックス
1633
+ CREATE INDEX "idx_工数実績_作業指示番号" ON "工数実績データ"("作業指示番号");
1634
+ CREATE INDEX "idx_工数実績_品目コード" ON "工数実績データ"("品目コード");
1635
+ CREATE INDEX "idx_工数実績_工程コード" ON "工数実績データ"("工程コード");
1636
+ CREATE INDEX "idx_工数実績_担当者コード" ON "工数実績データ"("担当者コード");
1637
+ CREATE INDEX "idx_工数実績_作業日" ON "工数実績データ"("作業日");
1638
+ ```
1639
+
1640
+ </details>
1641
+
1642
+ ### Java エンティティの定義(工数実績)
1643
+
1644
+ <details>
1645
+ <summary>Entity: LaborHours(工数実績データ)</summary>
1646
+
1647
+ ```java
1648
+ // src/main/java/com/example/pms/domain/model/process/LaborHours.java
1649
+ package com.example.pms.domain.model.process;
1650
+
1651
+ import com.example.pms.domain.model.item.Item;
1652
+ import com.example.pms.domain.master.Department;
1653
+ import com.example.pms.domain.master.Employee;
1654
+ import lombok.AllArgsConstructor;
1655
+ import lombok.Builder;
1656
+ import lombok.Data;
1657
+ import lombok.NoArgsConstructor;
1658
+
1659
+ import java.math.BigDecimal;
1660
+ import java.time.LocalDate;
1661
+ import java.time.LocalDateTime;
1662
+
1663
+ @Data
1664
+ @Builder
1665
+ @NoArgsConstructor
1666
+ @AllArgsConstructor
1667
+ public class LaborHours {
1668
+ private Integer id;
1669
+ private String laborHoursNumber;
1670
+ private String workOrderNumber;
1671
+ private String itemCode;
1672
+ private Integer sequence;
1673
+ private String processCode;
1674
+ private String departmentCode;
1675
+ private String employeeCode;
1676
+ private LocalDate workDate;
1677
+ private BigDecimal hours;
1678
+ private String remarks;
1679
+ private LocalDateTime createdAt;
1680
+ private String createdBy;
1681
+ private LocalDateTime updatedAt;
1682
+ private String updatedBy;
1683
+
1684
+ // リレーション
1685
+ private WorkOrder workOrder;
1686
+ private Item item;
1687
+ private Process process;
1688
+ private Department department;
1689
+ private Employee employee;
1690
+ }
1691
+ ```
1692
+
1693
+ </details>
1694
+
1695
+ ### MyBatis Mapper(工数実績)
1696
+
1697
+ <details>
1698
+ <summary>Mapper XML: LaborHoursMapper.xml</summary>
1699
+
1700
+ ```xml
1701
+ <!-- src/main/resources/com/example/pms/infrastructure/out/persistence/mapper/LaborHoursMapper.xml -->
1702
+ <?xml version="1.0" encoding="UTF-8" ?>
1703
+ <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
1704
+ "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
1705
+ <mapper namespace="com.example.pms.infrastructure.out.persistence.mapper.LaborHoursMapper">
1706
+
1707
+ <resultMap id="LaborHoursResultMap" type="com.example.pms.domain.model.process.LaborHours">
1708
+ <id property="id" column="ID"/>
1709
+ <result property="laborHoursNumber" column="工数実績番号"/>
1710
+ <result property="workOrderNumber" column="作業指示番号"/>
1711
+ <result property="itemCode" column="品目コード"/>
1712
+ <result property="sequence" column="工順"/>
1713
+ <result property="processCode" column="工程コード"/>
1714
+ <result property="departmentCode" column="部門コード"/>
1715
+ <result property="employeeCode" column="担当者コード"/>
1716
+ <result property="workDate" column="作業日"/>
1717
+ <result property="hours" column="工数"/>
1718
+ <result property="remarks" column="備考"/>
1719
+ <result property="createdAt" column="作成日時"/>
1720
+ <result property="createdBy" column="作成者"/>
1721
+ <result property="updatedAt" column="更新日時"/>
1722
+ <result property="updatedBy" column="更新者"/>
1723
+ </resultMap>
1724
+
1725
+ <insert id="insert" useGeneratedKeys="true" keyProperty="id" keyColumn="ID">
1726
+ INSERT INTO "工数実績データ" (
1727
+ "工数実績番号", "作業指示番号", "品目コード", "工順", "工程コード",
1728
+ "部門コード", "担当者コード", "作業日", "工数", "備考", "作成者"
1729
+ ) VALUES (
1730
+ #{laborHoursNumber},
1731
+ #{workOrderNumber},
1732
+ #{itemCode},
1733
+ #{sequence},
1734
+ #{processCode},
1735
+ #{departmentCode},
1736
+ #{employeeCode},
1737
+ #{workDate},
1738
+ #{hours},
1739
+ #{remarks},
1740
+ #{createdBy}
1741
+ )
1742
+ </insert>
1743
+
1744
+ <select id="findByLaborHoursNumber" resultMap="LaborHoursResultMap">
1745
+ SELECT * FROM "工数実績データ" WHERE "工数実績番号" = #{laborHoursNumber}
1746
+ </select>
1747
+
1748
+ <select id="sumByWorkOrderAndSequence" resultType="java.math.BigDecimal">
1749
+ SELECT COALESCE(SUM("工数"), 0)
1750
+ FROM "工数実績データ"
1751
+ WHERE "作業指示番号" = #{workOrderNumber} AND "工順" = #{sequence}
1752
+ </select>
1753
+
1754
+ <select id="sumByWorkOrder" resultType="java.math.BigDecimal">
1755
+ SELECT COALESCE(SUM("工数"), 0)
1756
+ FROM "工数実績データ"
1757
+ WHERE "作業指示番号" = #{workOrderNumber}
1758
+ </select>
1759
+
1760
+ <select id="sumByEmployee" resultType="java.math.BigDecimal">
1761
+ SELECT COALESCE(SUM("工数"), 0)
1762
+ FROM "工数実績データ"
1763
+ WHERE "担当者コード" = #{employeeCode}
1764
+ AND "作業日" BETWEEN #{startDate} AND #{endDate}
1765
+ </select>
1766
+
1767
+ <select id="findLatestLaborHoursNumber" resultType="string">
1768
+ SELECT "工数実績番号" FROM "工数実績データ"
1769
+ WHERE "工数実績番号" LIKE #{prefix}
1770
+ ORDER BY "工数実績番号" DESC
1771
+ LIMIT 1
1772
+ </select>
1773
+
1774
+ <delete id="deleteAll">
1775
+ DELETE FROM "工数実績データ"
1776
+ </delete>
1777
+ </mapper>
1778
+ ```
1779
+
1780
+ </details>
1781
+
1782
+ ### 工数実績サービスの実装
1783
+
1784
+ <details>
1785
+ <summary>Service: LaborHoursService</summary>
1786
+
1787
+ ```java
1788
+ // src/main/java/com/example/pms/application/service/LaborHoursService.java
1789
+ package com.example.pms.application.service;
1790
+
1791
+ import com.example.pms.domain.model.process.*;
1792
+ import com.example.pms.infrastructure.out.persistence.mapper.*;
1793
+ import lombok.RequiredArgsConstructor;
1794
+ import org.springframework.stereotype.Service;
1795
+ import org.springframework.transaction.annotation.Transactional;
1796
+
1797
+ import java.math.BigDecimal;
1798
+ import java.time.LocalDate;
1799
+ import java.time.format.DateTimeFormatter;
1800
+ import java.util.ArrayList;
1801
+ import java.util.List;
1802
+
1803
+ @Service
1804
+ @RequiredArgsConstructor
1805
+ public class LaborHoursService {
1806
+
1807
+ private final LaborHoursMapper laborHoursMapper;
1808
+ private final WorkOrderDetailMapper workOrderDetailMapper;
1809
+ private final WorkOrderMapper workOrderMapper;
1810
+ private final ProcessMapper processMapper;
1811
+
1812
+ /**
1813
+ * 工数実績番号を生成する
1814
+ */
1815
+ private String generateLaborHoursNumber(LocalDate workDate) {
1816
+ String prefix = "LH-" + workDate.format(DateTimeFormatter.ofPattern("yyyyMM")) + "-";
1817
+ String latestNumber = laborHoursMapper.findLatestLaborHoursNumber(prefix + "%");
1818
+
1819
+ int sequence = 1;
1820
+ if (latestNumber != null) {
1821
+ int currentSequence = Integer.parseInt(latestNumber.substring(latestNumber.length() - 4));
1822
+ sequence = currentSequence + 1;
1823
+ }
1824
+
1825
+ return prefix + String.format("%04d", sequence);
1826
+ }
1827
+
1828
+ /**
1829
+ * 工数実績を報告する
1830
+ */
1831
+ @Transactional
1832
+ public LaborHours reportLaborHours(LaborHoursInput input) {
1833
+ // 作業指示明細を取得
1834
+ WorkOrderDetail detail = workOrderDetailMapper.findByWorkOrderAndSequence(
1835
+ input.getWorkOrderNumber(), input.getSequence());
1836
+ if (detail == null) {
1837
+ throw new IllegalArgumentException("Work order detail not found");
1838
+ }
1839
+
1840
+ // 作業指示を取得
1841
+ WorkOrder workOrder = workOrderMapper.findByWorkOrderNumber(input.getWorkOrderNumber());
1842
+ if (workOrder == null) {
1843
+ throw new IllegalArgumentException("Work order not found");
1844
+ }
1845
+
1846
+ String laborHoursNumber = generateLaborHoursNumber(input.getWorkDate());
1847
+
1848
+ // 工数実績を作成
1849
+ LaborHours laborHours = LaborHours.builder()
1850
+ .laborHoursNumber(laborHoursNumber)
1851
+ .workOrderNumber(input.getWorkOrderNumber())
1852
+ .itemCode(workOrder.getItemCode())
1853
+ .sequence(input.getSequence())
1854
+ .processCode(detail.getProcessCode())
1855
+ .departmentCode(input.getDepartmentCode())
1856
+ .employeeCode(input.getEmployeeCode())
1857
+ .workDate(input.getWorkDate())
1858
+ .hours(input.getHours())
1859
+ .remarks(input.getRemarks())
1860
+ .build();
1861
+ laborHoursMapper.insert(laborHours);
1862
+
1863
+ return laborHours;
1864
+ }
1865
+
1866
+ /**
1867
+ * 工順別の工数合計を取得する
1868
+ */
1869
+ public BigDecimal getTotalHoursBySequence(String workOrderNumber, Integer sequence) {
1870
+ return laborHoursMapper.sumByWorkOrderAndSequence(workOrderNumber, sequence);
1871
+ }
1872
+
1873
+ /**
1874
+ * 作業指示の工数サマリを取得する
1875
+ */
1876
+ public LaborHoursSummary getSummary(String workOrderNumber) {
1877
+ WorkOrder workOrder = workOrderMapper.findByWorkOrderNumber(workOrderNumber);
1878
+ if (workOrder == null) {
1879
+ throw new IllegalArgumentException("Work order not found");
1880
+ }
1881
+
1882
+ List<WorkOrderDetail> details = workOrderDetailMapper.findByWorkOrderNumber(workOrderNumber);
1883
+ List<ProcessLaborHours> processHours = new ArrayList<>();
1884
+ BigDecimal totalHours = BigDecimal.ZERO;
1885
+
1886
+ for (WorkOrderDetail detail : details) {
1887
+ BigDecimal hours = laborHoursMapper.sumByWorkOrderAndSequence(workOrderNumber, detail.getSequence());
1888
+ Process process = processMapper.findByProcessCode(detail.getProcessCode());
1889
+
1890
+ processHours.add(ProcessLaborHours.builder()
1891
+ .processCode(detail.getProcessCode())
1892
+ .processName(process != null ? process.getProcessName() : "")
1893
+ .hours(hours)
1894
+ .build());
1895
+
1896
+ totalHours = totalHours.add(hours);
1897
+ }
1898
+
1899
+ return LaborHoursSummary.builder()
1900
+ .totalHours(totalHours)
1901
+ .processHours(processHours)
1902
+ .build();
1903
+ }
1904
+
1905
+ /**
1906
+ * 担当者別の工数を取得する
1907
+ */
1908
+ public BigDecimal getTotalHoursByEmployee(String employeeCode, LocalDate startDate, LocalDate endDate) {
1909
+ return laborHoursMapper.sumByEmployee(employeeCode, startDate, endDate);
1910
+ }
1911
+ }
1912
+ ```
1913
+
1914
+ </details>
1915
+
1916
+ <details>
1917
+ <summary>Input DTO: LaborHoursInput</summary>
1918
+
1919
+ ```java
1920
+ // src/main/java/com/example/pms/application/service/LaborHoursInput.java
1921
+ package com.example.pms.application.service;
1922
+
1923
+ import lombok.Builder;
1924
+ import lombok.Data;
1925
+
1926
+ import java.math.BigDecimal;
1927
+ import java.time.LocalDate;
1928
+
1929
+ @Data
1930
+ @Builder
1931
+ public class LaborHoursInput {
1932
+ private String workOrderNumber;
1933
+ private Integer sequence;
1934
+ private LocalDate workDate;
1935
+ private String employeeCode;
1936
+ private String departmentCode;
1937
+ private BigDecimal hours;
1938
+ private String remarks;
1939
+ }
1940
+ ```
1941
+
1942
+ </details>
1943
+
1944
+ <details>
1945
+ <summary>DTO: LaborHoursSummary・ProcessLaborHours</summary>
1946
+
1947
+ ```java
1948
+ // src/main/java/com/example/pms/application/service/LaborHoursSummary.java
1949
+ package com.example.pms.application.service;
1950
+
1951
+ import lombok.Builder;
1952
+ import lombok.Data;
1953
+
1954
+ import java.math.BigDecimal;
1955
+ import java.util.List;
1956
+
1957
+ @Data
1958
+ @Builder
1959
+ public class LaborHoursSummary {
1960
+ private BigDecimal totalHours;
1961
+ private List<ProcessLaborHours> processHours;
1962
+ }
1963
+ ```
1964
+
1965
+ ```java
1966
+ // src/main/java/com/example/pms/application/service/ProcessLaborHours.java
1967
+ package com.example.pms.application.service;
1968
+
1969
+ import lombok.Builder;
1970
+ import lombok.Data;
1971
+
1972
+ import java.math.BigDecimal;
1973
+
1974
+ @Data
1975
+ @Builder
1976
+ public class ProcessLaborHours {
1977
+ private String processCode;
1978
+ private String processName;
1979
+ private BigDecimal hours;
1980
+ }
1981
+ ```
1982
+
1983
+ </details>
1984
+
1985
+ ---
1986
+
1987
+ ## 27.3 進捗管理
1988
+
1989
+ ### 工程別進捗の管理
1990
+
1991
+ 作業指示の進捗状況を工程別に把握するための設計を行います。
1992
+
1993
+ ```plantuml
1994
+ @startuml
1995
+
1996
+ title 進捗管理の概念
1997
+
1998
+ entity "作業指示データ" as work_order {
1999
+ * 作業指示番号
2000
+ --
2001
+ * ステータス
2002
+ * 開始予定日
2003
+ * 完成予定日
2004
+ * 実績開始日
2005
+ * 実績完了日
2006
+ * 完成済数
2007
+ * 作業指示数
2008
+ }
2009
+
2010
+ entity "作業指示明細データ" as work_order_detail {
2011
+ * 作業指示番号
2012
+ * 工順
2013
+ * 工程コード
2014
+ }
2015
+
2016
+ entity "工数実績データ" as labor_hours {
2017
+ * 作業指示番号
2018
+ * 工順
2019
+ * 工数
2020
+ }
2021
+
2022
+ work_order ||--|{ work_order_detail
2023
+ work_order_detail ||--o{ labor_hours
2024
+
2025
+ note right of work_order
2026
+ 進捗率 = 完成済数 / 作業指示数
2027
+ 遅延判定 = 実績日 > 予定日
2028
+ end note
2029
+
2030
+ @enduml
2031
+ ```
2032
+
2033
+ ### 進捗状況DTO
2034
+
2035
+ <details>
2036
+ <summary>DTO: WorkOrderProgress</summary>
2037
+
2038
+ ```java
2039
+ // src/main/java/com/example/pms/application/service/WorkOrderProgress.java
2040
+ package com.example.pms.application.service;
2041
+
2042
+ import com.example.pms.domain.model.process.WorkOrderStatus;
2043
+ import lombok.Builder;
2044
+ import lombok.Data;
2045
+
2046
+ import java.math.BigDecimal;
2047
+ import java.math.RoundingMode;
2048
+ import java.time.LocalDate;
2049
+ import java.util.List;
2050
+
2051
+ @Data
2052
+ @Builder
2053
+ public class WorkOrderProgress {
2054
+ private String workOrderNumber;
2055
+ private String itemCode;
2056
+ private WorkOrderStatus status;
2057
+ private BigDecimal orderQuantity;
2058
+ private BigDecimal completedQuantity;
2059
+ private LocalDate plannedStartDate;
2060
+ private LocalDate plannedEndDate;
2061
+ private LocalDate actualStartDate;
2062
+ private LocalDate actualEndDate;
2063
+ private List<ProcessProgress> processProgresses;
2064
+
2065
+ /**
2066
+ * 進捗率を計算する(0-100%)
2067
+ */
2068
+ public BigDecimal getProgressRate() {
2069
+ if (orderQuantity == null || orderQuantity.compareTo(BigDecimal.ZERO) == 0) {
2070
+ return BigDecimal.ZERO;
2071
+ }
2072
+ return completedQuantity
2073
+ .multiply(new BigDecimal("100"))
2074
+ .divide(orderQuantity, 2, RoundingMode.HALF_UP);
2075
+ }
2076
+
2077
+ /**
2078
+ * 遅延判定
2079
+ */
2080
+ public boolean isDelayed() {
2081
+ if (status == WorkOrderStatus.COMPLETED) {
2082
+ return false;
2083
+ }
2084
+ LocalDate today = LocalDate.now();
2085
+ return today.isAfter(plannedEndDate);
2086
+ }
2087
+
2088
+ /**
2089
+ * 遅延日数を計算
2090
+ */
2091
+ public long getDelayDays() {
2092
+ if (!isDelayed()) {
2093
+ return 0;
2094
+ }
2095
+ LocalDate today = LocalDate.now();
2096
+ return java.time.temporal.ChronoUnit.DAYS.between(plannedEndDate, today);
2097
+ }
2098
+ }
2099
+ ```
2100
+
2101
+ </details>
2102
+
2103
+ <details>
2104
+ <summary>DTO: ProcessProgress</summary>
2105
+
2106
+ ```java
2107
+ // src/main/java/com/example/pms/application/service/ProcessProgress.java
2108
+ package com.example.pms.application.service;
2109
+
2110
+ import lombok.Builder;
2111
+ import lombok.Data;
2112
+
2113
+ import java.math.BigDecimal;
2114
+
2115
+ @Data
2116
+ @Builder
2117
+ public class ProcessProgress {
2118
+ private Integer sequence;
2119
+ private String processCode;
2120
+ private String processName;
2121
+ private BigDecimal plannedHours;
2122
+ private BigDecimal actualHours;
2123
+
2124
+ /**
2125
+ * 工数消化率を計算
2126
+ */
2127
+ public BigDecimal getHoursRate() {
2128
+ if (plannedHours == null || plannedHours.compareTo(BigDecimal.ZERO) == 0) {
2129
+ return BigDecimal.ZERO;
2130
+ }
2131
+ return actualHours
2132
+ .multiply(new BigDecimal("100"))
2133
+ .divide(plannedHours, 2, java.math.RoundingMode.HALF_UP);
2134
+ }
2135
+ }
2136
+ ```
2137
+
2138
+ </details>
2139
+
2140
+ ### 進捗管理サービス
2141
+
2142
+ <details>
2143
+ <summary>Service: ProgressManagementService</summary>
2144
+
2145
+ ```java
2146
+ // src/main/java/com/example/pms/application/service/ProgressManagementService.java
2147
+ package com.example.pms.application.service;
2148
+
2149
+ import com.example.pms.domain.model.process.*;
2150
+ import com.example.pms.infrastructure.out.persistence.mapper.*;
2151
+ import lombok.RequiredArgsConstructor;
2152
+ import org.springframework.stereotype.Service;
2153
+
2154
+ import java.time.LocalDate;
2155
+ import java.util.ArrayList;
2156
+ import java.util.List;
2157
+ import java.util.stream.Collectors;
2158
+
2159
+ @Service
2160
+ @RequiredArgsConstructor
2161
+ public class ProgressManagementService {
2162
+
2163
+ private final WorkOrderMapper workOrderMapper;
2164
+ private final WorkOrderDetailMapper workOrderDetailMapper;
2165
+ private final LaborHoursMapper laborHoursMapper;
2166
+ private final ProcessMapper processMapper;
2167
+
2168
+ /**
2169
+ * 作業指示の進捗を取得する
2170
+ */
2171
+ public WorkOrderProgress getProgress(String workOrderNumber) {
2172
+ WorkOrder workOrder = workOrderMapper.findByWorkOrderNumber(workOrderNumber);
2173
+ if (workOrder == null) {
2174
+ throw new IllegalArgumentException("Work order not found: " + workOrderNumber);
2175
+ }
2176
+
2177
+ List<WorkOrderDetail> details = workOrderDetailMapper.findByWorkOrderNumber(workOrderNumber);
2178
+ List<ProcessProgress> processProgresses = new ArrayList<>();
2179
+
2180
+ for (WorkOrderDetail detail : details) {
2181
+ Process process = processMapper.findByProcessCode(detail.getProcessCode());
2182
+ var actualHours = laborHoursMapper.sumByWorkOrderAndSequence(
2183
+ workOrderNumber, detail.getSequence());
2184
+
2185
+ processProgresses.add(ProcessProgress.builder()
2186
+ .sequence(detail.getSequence())
2187
+ .processCode(detail.getProcessCode())
2188
+ .processName(process != null ? process.getProcessName() : "")
2189
+ .actualHours(actualHours)
2190
+ .build());
2191
+ }
2192
+
2193
+ return WorkOrderProgress.builder()
2194
+ .workOrderNumber(workOrder.getWorkOrderNumber())
2195
+ .itemCode(workOrder.getItemCode())
2196
+ .status(workOrder.getStatus())
2197
+ .orderQuantity(workOrder.getOrderQuantity())
2198
+ .completedQuantity(workOrder.getCompletedQuantity())
2199
+ .plannedStartDate(workOrder.getPlannedStartDate())
2200
+ .plannedEndDate(workOrder.getPlannedEndDate())
2201
+ .actualStartDate(workOrder.getActualStartDate())
2202
+ .actualEndDate(workOrder.getActualEndDate())
2203
+ .processProgresses(processProgresses)
2204
+ .build();
2205
+ }
2206
+
2207
+ /**
2208
+ * 遅延している作業指示を取得する
2209
+ */
2210
+ public List<WorkOrderProgress> getDelayedWorkOrders() {
2211
+ // 未完了の作業指示を取得
2212
+ List<WorkOrder> workOrders = workOrderMapper.findByStatus(
2213
+ List.of(WorkOrderStatus.NOT_STARTED, WorkOrderStatus.IN_PROGRESS));
2214
+
2215
+ LocalDate today = LocalDate.now();
2216
+
2217
+ return workOrders.stream()
2218
+ .filter(wo -> today.isAfter(wo.getPlannedEndDate()))
2219
+ .map(wo -> getProgress(wo.getWorkOrderNumber()))
2220
+ .collect(Collectors.toList());
2221
+ }
2222
+
2223
+ /**
2224
+ * 本日開始予定の作業指示を取得する
2225
+ */
2226
+ public List<WorkOrderProgress> getTodayStartWorkOrders() {
2227
+ LocalDate today = LocalDate.now();
2228
+ List<WorkOrder> workOrders = workOrderMapper.findByPlannedStartDate(today);
2229
+
2230
+ return workOrders.stream()
2231
+ .map(wo -> getProgress(wo.getWorkOrderNumber()))
2232
+ .collect(Collectors.toList());
2233
+ }
2234
+
2235
+ /**
2236
+ * 本日完成予定の作業指示を取得する
2237
+ */
2238
+ public List<WorkOrderProgress> getTodayDueWorkOrders() {
2239
+ LocalDate today = LocalDate.now();
2240
+ List<WorkOrder> workOrders = workOrderMapper.findByPlannedEndDate(today);
2241
+
2242
+ return workOrders.stream()
2243
+ .map(wo -> getProgress(wo.getWorkOrderNumber()))
2244
+ .collect(Collectors.toList());
2245
+ }
2246
+ }
2247
+ ```
2248
+
2249
+ </details>
2250
+
2251
+ ### 遅延アラートの設計
2252
+
2253
+ ```plantuml
2254
+ @startuml
2255
+
2256
+ title 遅延アラートの判定ロジック
2257
+
2258
+ start
2259
+
2260
+ :作業指示を取得;
2261
+
2262
+ if (ステータス == 完了?) then (yes)
2263
+ :アラート不要;
2264
+ stop
2265
+ else (no)
2266
+ :今日の日付を取得;
2267
+ endif
2268
+
2269
+ if (今日 > 完成予定日?) then (yes)
2270
+ :【アラート】遅延発生;
2271
+ :遅延日数を計算;
2272
+ note right
2273
+ 遅延日数 = 今日 - 完成予定日
2274
+ end note
2275
+ else (no)
2276
+ if (進捗率 < 期待進捗率?) then (yes)
2277
+ :【警告】進捗遅れ;
2278
+ note right
2279
+ 期待進捗率 =
2280
+ (今日 - 開始予定日) /
2281
+ (完成予定日 - 開始予定日) × 100
2282
+ end note
2283
+ else (no)
2284
+ :正常進捗;
2285
+ endif
2286
+ endif
2287
+
2288
+ stop
2289
+
2290
+ @enduml
2291
+ ```
2292
+
2293
+ ### 進捗状況の可視化
2294
+
2295
+ 進捗状況を可視化するためのビュー定義例を示します。
2296
+
2297
+ <details>
2298
+ <summary>DDL: 進捗管理ビュー</summary>
2299
+
2300
+ ```sql
2301
+ -- 作業指示進捗ビュー
2302
+ CREATE OR REPLACE VIEW "作業指示進捗ビュー" AS
2303
+ SELECT
2304
+ wo."作業指示番号",
2305
+ wo."オーダ番号",
2306
+ wo."品目コード",
2307
+ i."品目名",
2308
+ wo."作業指示数",
2309
+ wo."完成済数",
2310
+ CASE
2311
+ WHEN wo."作業指示数" = 0 THEN 0
2312
+ ELSE ROUND((wo."完成済数" / wo."作業指示数") * 100, 2)
2313
+ END AS "進捗率",
2314
+ wo."開始予定日",
2315
+ wo."完成予定日",
2316
+ wo."実績開始日",
2317
+ wo."実績完了日",
2318
+ wo."ステータス",
2319
+ CASE
2320
+ WHEN wo."ステータス" = '完了' THEN '完了'
2321
+ WHEN CURRENT_DATE > wo."完成予定日" THEN '遅延'
2322
+ WHEN CURRENT_DATE >= wo."開始予定日" AND wo."ステータス" = '未着手' THEN '未着手遅延'
2323
+ ELSE '正常'
2324
+ END AS "進捗状態",
2325
+ CASE
2326
+ WHEN wo."ステータス" = '完了' THEN 0
2327
+ WHEN CURRENT_DATE > wo."完成予定日" THEN CURRENT_DATE - wo."完成予定日"
2328
+ ELSE 0
2329
+ END AS "遅延日数"
2330
+ FROM "作業指示データ" wo
2331
+ LEFT JOIN "品目マスタ" i ON wo."品目コード" = i."品目コード";
2332
+
2333
+ -- 工程別工数実績ビュー
2334
+ CREATE OR REPLACE VIEW "工程別工数実績ビュー" AS
2335
+ SELECT
2336
+ lh."作業指示番号",
2337
+ lh."工順",
2338
+ lh."工程コード",
2339
+ p."工程名",
2340
+ SUM(lh."工数") AS "合計工数",
2341
+ COUNT(*) AS "記録件数"
2342
+ FROM "工数実績データ" lh
2343
+ LEFT JOIN "工程マスタ" p ON lh."工程コード" = p."工程コード"
2344
+ GROUP BY
2345
+ lh."作業指示番号",
2346
+ lh."工順",
2347
+ lh."工程コード",
2348
+ p."工程名"
2349
+ ORDER BY
2350
+ lh."作業指示番号",
2351
+ lh."工順";
2352
+ ```
2353
+
2354
+ </details>
2355
+
2356
+ ---
2357
+
2358
+ ## 27.4 リレーションと楽観ロックの設計
2359
+
2360
+ ### MyBatis ネストした ResultMap によるリレーション設定
2361
+
2362
+ 工程管理では、作業指示→作業指示明細、完成実績→検査結果といった親子関係があります。MyBatis でこれらの関係を効率的に取得するためのリレーション設定を実装します。
2363
+
2364
+ #### 作業指示のネスト ResultMap(明細・オーダ・品目を含む)
2365
+
2366
+ <details>
2367
+ <summary>WorkOrderMapper.xml(リレーション設定)</summary>
2368
+
2369
+ ```xml
2370
+ <?xml version="1.0" encoding="UTF-8" ?>
2371
+ <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
2372
+ "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
2373
+
2374
+ <!-- src/main/resources/mapper/WorkOrderMapper.xml -->
2375
+ <mapper namespace="com.example.pms.infrastructure.out.persistence.mapper.WorkOrderMapper">
2376
+
2377
+ <!-- 作業指示(ヘッダ)with 明細・オーダ・品目 ResultMap -->
2378
+ <resultMap id="workOrderWithDetailsResultMap" type="com.example.pms.domain.model.process.WorkOrder">
2379
+ <id property="id" column="wo_id"/>
2380
+ <result property="workOrderNumber" column="wo_作業指示番号"/>
2381
+ <result property="orderNumber" column="wo_オーダ番号"/>
2382
+ <result property="workOrderDate" column="wo_作業指示日"/>
2383
+ <result property="itemCode" column="wo_品目コード"/>
2384
+ <result property="orderQuantity" column="wo_作業指示数"/>
2385
+ <result property="locationCode" column="wo_場所コード"/>
2386
+ <result property="plannedStartDate" column="wo_開始予定日"/>
2387
+ <result property="plannedEndDate" column="wo_完成予定日"/>
2388
+ <result property="actualStartDate" column="wo_実績開始日"/>
2389
+ <result property="actualEndDate" column="wo_実績完了日"/>
2390
+ <result property="completedQuantity" column="wo_完成済数"/>
2391
+ <result property="totalGoodQuantity" column="wo_総良品数"/>
2392
+ <result property="totalDefectQuantity" column="wo_総不良品数"/>
2393
+ <result property="status" column="wo_ステータス"
2394
+ typeHandler="com.example.pms.infrastructure.persistence.WorkOrderStatusTypeHandler"/>
2395
+ <result property="completedFlag" column="wo_完了フラグ"/>
2396
+ <result property="remarks" column="wo_備考"/>
2397
+ <result property="version" column="wo_バージョン"/>
2398
+ <result property="createdAt" column="wo_作成日時"/>
2399
+ <result property="updatedAt" column="wo_更新日時"/>
2400
+
2401
+ <!-- オーダ情報との N:1 関連 -->
2402
+ <association property="order" javaType="com.example.pms.domain.model.planning.Order">
2403
+ <id property="id" column="o_id"/>
2404
+ <result property="orderNumber" column="o_オーダNO"/>
2405
+ <result property="itemCode" column="o_品目コード"/>
2406
+ <result property="plannedQuantity" column="o_計画数量"/>
2407
+ <result property="dueDate" column="o_完了日"/>
2408
+ </association>
2409
+
2410
+ <!-- 品目マスタとの N:1 関連 -->
2411
+ <association property="item" javaType="com.example.pms.domain.model.item.Item">
2412
+ <id property="itemCode" column="i_品目コード"/>
2413
+ <result property="itemName" column="i_品目名"/>
2414
+ <result property="itemCategory" column="i_品目カテゴリ"
2415
+ typeHandler="com.example.pms.infrastructure.persistence.ItemCategoryTypeHandler"/>
2416
+ </association>
2417
+
2418
+ <!-- 作業指示明細との 1:N 関連 -->
2419
+ <collection property="details" ofType="com.example.pms.domain.model.process.WorkOrderDetail"
2420
+ resultMap="workOrderDetailNestedResultMap"/>
2421
+ </resultMap>
2422
+
2423
+ <!-- 作業指示明細のネスト ResultMap(工程マスタを含む) -->
2424
+ <resultMap id="workOrderDetailNestedResultMap" type="com.example.pms.domain.model.process.WorkOrderDetail">
2425
+ <id property="id" column="wd_id"/>
2426
+ <result property="workOrderNumber" column="wd_作業指示番号"/>
2427
+ <result property="sequence" column="wd_工順"/>
2428
+ <result property="processCode" column="wd_工程コード"/>
2429
+ <result property="createdAt" column="wd_作成日時"/>
2430
+ <result property="updatedAt" column="wd_更新日時"/>
2431
+
2432
+ <!-- 工程マスタとの N:1 関連 -->
2433
+ <association property="process" javaType="com.example.pms.domain.model.process.Process">
2434
+ <id property="processCode" column="p_工程コード"/>
2435
+ <result property="processName" column="p_工程名"/>
2436
+ </association>
2437
+ </resultMap>
2438
+
2439
+ <!-- JOIN による一括取得クエリ -->
2440
+ <select id="findWithDetailsByWorkOrderNumber" resultMap="workOrderWithDetailsResultMap">
2441
+ SELECT
2442
+ wo."ID" AS wo_id,
2443
+ wo."作業指示番号" AS wo_作業指示番号,
2444
+ wo."オーダ番号" AS wo_オーダ番号,
2445
+ wo."作業指示日" AS wo_作業指示日,
2446
+ wo."品目コード" AS wo_品目コード,
2447
+ wo."作業指示数" AS wo_作業指示数,
2448
+ wo."場所コード" AS wo_場所コード,
2449
+ wo."開始予定日" AS wo_開始予定日,
2450
+ wo."完成予定日" AS wo_完成予定日,
2451
+ wo."実績開始日" AS wo_実績開始日,
2452
+ wo."実績完了日" AS wo_実績完了日,
2453
+ wo."完成済数" AS wo_完成済数,
2454
+ wo."総良品数" AS wo_総良品数,
2455
+ wo."総不良品数" AS wo_総不良品数,
2456
+ wo."ステータス" AS wo_ステータス,
2457
+ wo."完了フラグ" AS wo_完了フラグ,
2458
+ wo."備考" AS wo_備考,
2459
+ wo."バージョン" AS wo_バージョン,
2460
+ wo."作成日時" AS wo_作成日時,
2461
+ wo."更新日時" AS wo_更新日時,
2462
+ o."ID" AS o_id,
2463
+ o."オーダNO" AS o_オーダNO,
2464
+ o."品目コード" AS o_品目コード,
2465
+ o."計画数量" AS o_計画数量,
2466
+ o."完了日" AS o_完了日,
2467
+ i."品目コード" AS i_品目コード,
2468
+ i."品目名" AS i_品目名,
2469
+ i."品目カテゴリ" AS i_品目カテゴリ,
2470
+ wd."ID" AS wd_id,
2471
+ wd."作業指示番号" AS wd_作業指示番号,
2472
+ wd."工順" AS wd_工順,
2473
+ wd."工程コード" AS wd_工程コード,
2474
+ wd."作成日時" AS wd_作成日時,
2475
+ wd."更新日時" AS wd_更新日時,
2476
+ p."工程コード" AS p_工程コード,
2477
+ p."工程名" AS p_工程名
2478
+ FROM "作業指示データ" wo
2479
+ LEFT JOIN "オーダ情報" o ON wo."オーダ番号" = o."オーダNO"
2480
+ LEFT JOIN "品目マスタ" i ON wo."品目コード" = i."品目コード"
2481
+ LEFT JOIN "作業指示明細データ" wd ON wo."作業指示番号" = wd."作業指示番号"
2482
+ LEFT JOIN "工程マスタ" p ON wd."工程コード" = p."工程コード"
2483
+ WHERE wo."作業指示番号" = #{workOrderNumber}
2484
+ ORDER BY wd."工順"
2485
+ </select>
2486
+
2487
+ </mapper>
2488
+ ```
2489
+
2490
+ </details>
2491
+
2492
+ #### 完成実績のネスト ResultMap(検査結果・作業指示を含む)
2493
+
2494
+ <details>
2495
+ <summary>CompletionResultMapper.xml(リレーション設定)</summary>
2496
+
2497
+ ```xml
2498
+ <?xml version="1.0" encoding="UTF-8" ?>
2499
+ <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
2500
+ "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
2501
+
2502
+ <!-- src/main/resources/mapper/CompletionResultMapper.xml -->
2503
+ <mapper namespace="com.example.pms.infrastructure.out.persistence.mapper.CompletionResultMapper">
2504
+
2505
+ <!-- 完成実績 with 検査結果・作業指示 ResultMap -->
2506
+ <resultMap id="completionResultWithInspectionsResultMap"
2507
+ type="com.example.pms.domain.model.process.CompletionResult">
2508
+ <id property="id" column="cr_id"/>
2509
+ <result property="completionResultNumber" column="cr_完成実績番号"/>
2510
+ <result property="workOrderNumber" column="cr_作業指示番号"/>
2511
+ <result property="itemCode" column="cr_品目コード"/>
2512
+ <result property="completionDate" column="cr_完成日"/>
2513
+ <result property="completedQuantity" column="cr_完成数量"/>
2514
+ <result property="goodQuantity" column="cr_良品数"/>
2515
+ <result property="defectQuantity" column="cr_不良品数"/>
2516
+ <result property="remarks" column="cr_備考"/>
2517
+ <result property="version" column="cr_バージョン"/>
2518
+ <result property="createdAt" column="cr_作成日時"/>
2519
+ <result property="updatedAt" column="cr_更新日時"/>
2520
+
2521
+ <!-- 作業指示との N:1 関連 -->
2522
+ <association property="workOrder" javaType="com.example.pms.domain.model.process.WorkOrder">
2523
+ <id property="id" column="wo_id"/>
2524
+ <result property="workOrderNumber" column="wo_作業指示番号"/>
2525
+ <result property="itemCode" column="wo_品目コード"/>
2526
+ <result property="orderQuantity" column="wo_作業指示数"/>
2527
+ <result property="completedQuantity" column="wo_完成済数"/>
2528
+ <result property="status" column="wo_ステータス"
2529
+ typeHandler="com.example.pms.infrastructure.persistence.WorkOrderStatusTypeHandler"/>
2530
+ </association>
2531
+
2532
+ <!-- 検査結果との 1:N 関連 -->
2533
+ <collection property="inspectionResults"
2534
+ ofType="com.example.pms.domain.model.process.InspectionResult"
2535
+ resultMap="inspectionResultNestedResultMap"/>
2536
+ </resultMap>
2537
+
2538
+ <!-- 検査結果のネスト ResultMap(欠点マスタを含む) -->
2539
+ <resultMap id="inspectionResultNestedResultMap"
2540
+ type="com.example.pms.domain.model.process.InspectionResult">
2541
+ <id property="id" column="ir_id"/>
2542
+ <result property="completionResultNumber" column="ir_完成実績番号"/>
2543
+ <result property="defectCode" column="ir_欠点コード"/>
2544
+ <result property="quantity" column="ir_数量"/>
2545
+ <result property="createdAt" column="ir_作成日時"/>
2546
+ <result property="updatedAt" column="ir_更新日時"/>
2547
+
2548
+ <!-- 欠点マスタとの N:1 関連 -->
2549
+ <association property="defect" javaType="com.example.pms.domain.master.Defect">
2550
+ <id property="defectCode" column="d_欠点コード"/>
2551
+ <result property="defectName" column="d_欠点名"/>
2552
+ <result property="defectCategory" column="d_欠点区分"/>
2553
+ </association>
2554
+ </resultMap>
2555
+
2556
+ <!-- JOIN による一括取得クエリ -->
2557
+ <select id="findWithInspectionsByCompletionResultNumber"
2558
+ resultMap="completionResultWithInspectionsResultMap">
2559
+ SELECT
2560
+ cr."ID" AS cr_id,
2561
+ cr."完成実績番号" AS cr_完成実績番号,
2562
+ cr."作業指示番号" AS cr_作業指示番号,
2563
+ cr."品目コード" AS cr_品目コード,
2564
+ cr."完成日" AS cr_完成日,
2565
+ cr."完成数量" AS cr_完成数量,
2566
+ cr."良品数" AS cr_良品数,
2567
+ cr."不良品数" AS cr_不良品数,
2568
+ cr."備考" AS cr_備考,
2569
+ cr."バージョン" AS cr_バージョン,
2570
+ cr."作成日時" AS cr_作成日時,
2571
+ cr."更新日時" AS cr_更新日時,
2572
+ wo."ID" AS wo_id,
2573
+ wo."作業指示番号" AS wo_作業指示番号,
2574
+ wo."品目コード" AS wo_品目コード,
2575
+ wo."作業指示数" AS wo_作業指示数,
2576
+ wo."完成済数" AS wo_完成済数,
2577
+ wo."ステータス" AS wo_ステータス,
2578
+ ir."ID" AS ir_id,
2579
+ ir."完成実績番号" AS ir_完成実績番号,
2580
+ ir."欠点コード" AS ir_欠点コード,
2581
+ ir."数量" AS ir_数量,
2582
+ ir."作成日時" AS ir_作成日時,
2583
+ ir."更新日時" AS ir_更新日時,
2584
+ d."欠点コード" AS d_欠点コード,
2585
+ d."欠点名" AS d_欠点名,
2586
+ d."欠点区分" AS d_欠点区分
2587
+ FROM "完成実績データ" cr
2588
+ LEFT JOIN "作業指示データ" wo ON cr."作業指示番号" = wo."作業指示番号"
2589
+ LEFT JOIN "完成検査結果データ" ir ON cr."完成実績番号" = ir."完成実績番号"
2590
+ LEFT JOIN "欠点マスタ" d ON ir."欠点コード" = d."欠点コード"
2591
+ WHERE cr."完成実績番号" = #{completionResultNumber}
2592
+ ORDER BY ir."欠点コード"
2593
+ </select>
2594
+
2595
+ </mapper>
2596
+ ```
2597
+
2598
+ </details>
2599
+
2600
+ #### リレーション設定のポイント
2601
+
2602
+ | 設定項目 | 説明 |
2603
+ |---------|------|
2604
+ | `<collection>` | 1:N 関連のマッピング(作業指示→明細、完成実績→検査結果) |
2605
+ | `<association>` | N:1 関連のマッピング(作業指示→オーダ、明細→工程マスタ) |
2606
+ | `<id>` | 主キーの識別(MyBatis が重複排除に使用) |
2607
+ | エイリアス(AS) | カラム名の重複を避けるプレフィックス(`wo_`, `wd_`, `p_` など) |
2608
+ | `ORDER BY` | コレクションの順序を保証(工順順、欠点コード順) |
2609
+
2610
+ ### 楽観ロックの実装
2611
+
2612
+ 工程管理では、複数の作業者が同時に完成実績を報告したり、工数を入力したりする状況が発生します。作業指示の完成数量を正しく更新するために楽観ロックを実装します。
2613
+
2614
+ #### Flyway マイグレーション: バージョンカラム追加
2615
+
2616
+ <details>
2617
+ <summary>V017__add_process_version_columns.sql</summary>
2618
+
2619
+ ```sql
2620
+ -- src/main/resources/db/migration/V017__add_process_version_columns.sql
2621
+
2622
+ -- 作業指示データテーブルにバージョンカラムを追加
2623
+ ALTER TABLE "作業指示データ" ADD COLUMN "バージョン" INTEGER DEFAULT 1 NOT NULL;
2624
+
2625
+ -- 完成実績データテーブルにバージョンカラムを追加
2626
+ ALTER TABLE "完成実績データ" ADD COLUMN "バージョン" INTEGER DEFAULT 1 NOT NULL;
2627
+
2628
+ -- 工数実績データテーブルにバージョンカラムを追加
2629
+ ALTER TABLE "工数実績データ" ADD COLUMN "バージョン" INTEGER DEFAULT 1 NOT NULL;
2630
+
2631
+ -- コメント追加
2632
+ COMMENT ON COLUMN "作業指示データ"."バージョン" IS '楽観ロック用バージョン番号';
2633
+ COMMENT ON COLUMN "完成実績データ"."バージョン" IS '楽観ロック用バージョン番号';
2634
+ COMMENT ON COLUMN "工数実績データ"."バージョン" IS '楽観ロック用バージョン番号';
2635
+ ```
2636
+
2637
+ </details>
2638
+
2639
+ #### エンティティへのバージョンフィールド追加
2640
+
2641
+ <details>
2642
+ <summary>WorkOrder.java(バージョンフィールド追加)</summary>
2643
+
2644
+ ```java
2645
+ // src/main/java/com/example/production/domain/model/process/WorkOrder.java
2646
+ package com.example.pms.domain.model.process;
2647
+
2648
+ import com.example.pms.domain.model.item.Item;
2649
+ import com.example.pms.domain.master.Location;
2650
+ import com.example.pms.domain.model.planning.Order;
2651
+ import lombok.Builder;
2652
+ import lombok.Data;
2653
+
2654
+ import java.math.BigDecimal;
2655
+ import java.time.LocalDate;
2656
+ import java.time.LocalDateTime;
2657
+ import java.util.ArrayList;
2658
+ import java.util.List;
2659
+
2660
+ @Data
2661
+ @Builder
2662
+ public class WorkOrder {
2663
+ private Integer id;
2664
+ private String workOrderNumber;
2665
+ private String orderNumber;
2666
+ private LocalDate workOrderDate;
2667
+ private String itemCode;
2668
+ private BigDecimal orderQuantity;
2669
+ private String locationCode;
2670
+ private LocalDate plannedStartDate;
2671
+ private LocalDate plannedEndDate;
2672
+ private LocalDate actualStartDate;
2673
+ private LocalDate actualEndDate;
2674
+ private BigDecimal completedQuantity;
2675
+ private BigDecimal totalGoodQuantity;
2676
+ private BigDecimal totalDefectQuantity;
2677
+ private WorkOrderStatus status;
2678
+ private Boolean completedFlag;
2679
+ private String remarks;
2680
+ private LocalDateTime createdAt;
2681
+ private String createdBy;
2682
+ private LocalDateTime updatedAt;
2683
+ private String updatedBy;
2684
+
2685
+ // 楽観ロック用バージョン
2686
+ @Builder.Default
2687
+ private Integer version = 1;
2688
+
2689
+ // リレーション
2690
+ private Order order;
2691
+ private Item item;
2692
+ private Location location;
2693
+ @Builder.Default
2694
+ private List<WorkOrderDetail> details = new ArrayList<>();
2695
+
2696
+ /**
2697
+ * 完成可能かどうかをチェック
2698
+ */
2699
+ public boolean canComplete() {
2700
+ return status == WorkOrderStatus.IN_PROGRESS;
2701
+ }
2702
+
2703
+ /**
2704
+ * 残数量を計算
2705
+ */
2706
+ public BigDecimal getRemainingQuantity() {
2707
+ return orderQuantity.subtract(completedQuantity);
2708
+ }
2709
+ }
2710
+ ```
2711
+
2712
+ </details>
2713
+
2714
+ #### MyBatis Mapper: 楽観ロック対応の更新
2715
+
2716
+ 工程管理では、複数作業者が同時に完成報告を行う可能性があるため、完成数量の更新時に楽観ロックを適用します。
2717
+
2718
+ <details>
2719
+ <summary>WorkOrderMapper.xml(楽観ロック対応 UPDATE)</summary>
2720
+
2721
+ ```xml
2722
+ <!-- 完成数量更新(楽観ロック対応) -->
2723
+ <update id="updateCompletionQuantitiesWithOptimisticLock">
2724
+ UPDATE "作業指示データ"
2725
+ SET
2726
+ "完成済数" = "完成済数" + #{completedQuantity},
2727
+ "総良品数" = "総良品数" + #{goodQuantity},
2728
+ "総不良品数" = "総不良品数" + #{defectQuantity},
2729
+ "更新日時" = CURRENT_TIMESTAMP,
2730
+ "バージョン" = "バージョン" + 1
2731
+ WHERE "作業指示番号" = #{workOrderNumber}
2732
+ AND "バージョン" = #{version}
2733
+ </update>
2734
+
2735
+ <!-- ステータス更新(楽観ロック対応) -->
2736
+ <update id="updateStatusWithOptimisticLock">
2737
+ UPDATE "作業指示データ"
2738
+ SET
2739
+ "ステータス" = #{newStatus, typeHandler=com.example.pms.infrastructure.persistence.WorkOrderStatusTypeHandler}::作業指示ステータス,
2740
+ "実績開始日" = COALESCE("実績開始日", #{actualStartDate}),
2741
+ "実績完了日" = #{actualEndDate},
2742
+ "完了フラグ" = #{completedFlag},
2743
+ "更新日時" = CURRENT_TIMESTAMP,
2744
+ "バージョン" = "バージョン" + 1
2745
+ WHERE "作業指示番号" = #{workOrderNumber}
2746
+ AND "バージョン" = #{version}
2747
+ </update>
2748
+
2749
+ <!-- バージョン取得 -->
2750
+ <select id="findVersionByWorkOrderNumber" resultType="java.lang.Integer">
2751
+ SELECT "バージョン" FROM "作業指示データ"
2752
+ WHERE "作業指示番号" = #{workOrderNumber}
2753
+ </select>
2754
+ ```
2755
+
2756
+ </details>
2757
+
2758
+ #### Repository 実装: 楽観ロック対応
2759
+
2760
+ <details>
2761
+ <summary>WorkOrderRepositoryImpl.java(楽観ロック対応)</summary>
2762
+
2763
+ ```java
2764
+ // src/main/java/com/example/production/infrastructure/persistence/repository/WorkOrderRepositoryImpl.java
2765
+ package com.example.pms.infrastructure.persistence.repository;
2766
+
2767
+ import com.example.pms.application.port.out.WorkOrderRepository;
2768
+ import com.example.pms.domain.exception.OptimisticLockException;
2769
+ import com.example.pms.domain.model.process.WorkOrder;
2770
+ import com.example.pms.domain.model.process.WorkOrderStatus;
2771
+ import com.example.pms.infrastructure.out.persistence.mapper.WorkOrderMapper;
2772
+ import lombok.RequiredArgsConstructor;
2773
+ import org.springframework.stereotype.Repository;
2774
+ import org.springframework.transaction.annotation.Transactional;
2775
+
2776
+ import java.math.BigDecimal;
2777
+ import java.time.LocalDate;
2778
+ import java.util.Optional;
2779
+
2780
+ @Repository
2781
+ @RequiredArgsConstructor
2782
+ public class WorkOrderRepositoryImpl implements WorkOrderRepository {
2783
+
2784
+ private final WorkOrderMapper mapper;
2785
+
2786
+ @Override
2787
+ @Transactional
2788
+ public void updateCompletionQuantities(String workOrderNumber, Integer version,
2789
+ BigDecimal completedQuantity,
2790
+ BigDecimal goodQuantity,
2791
+ BigDecimal defectQuantity) {
2792
+ int updatedCount = mapper.updateCompletionQuantitiesWithOptimisticLock(
2793
+ workOrderNumber, version, completedQuantity, goodQuantity, defectQuantity);
2794
+
2795
+ if (updatedCount == 0) {
2796
+ handleOptimisticLockFailure(workOrderNumber, version);
2797
+ }
2798
+ }
2799
+
2800
+ @Override
2801
+ @Transactional
2802
+ public void updateStatus(String workOrderNumber, Integer version,
2803
+ WorkOrderStatus newStatus, LocalDate actualStartDate,
2804
+ LocalDate actualEndDate, Boolean completedFlag) {
2805
+ int updatedCount = mapper.updateStatusWithOptimisticLock(
2806
+ workOrderNumber, version, newStatus, actualStartDate, actualEndDate, completedFlag);
2807
+
2808
+ if (updatedCount == 0) {
2809
+ handleOptimisticLockFailure(workOrderNumber, version);
2810
+ }
2811
+ }
2812
+
2813
+ private void handleOptimisticLockFailure(String workOrderNumber, Integer expectedVersion) {
2814
+ Integer currentVersion = mapper.findVersionByWorkOrderNumber(workOrderNumber);
2815
+ if (currentVersion == null) {
2816
+ throw new OptimisticLockException("作業指示", workOrderNumber);
2817
+ } else {
2818
+ throw new OptimisticLockException("作業指示", workOrderNumber,
2819
+ expectedVersion, currentVersion);
2820
+ }
2821
+ }
2822
+
2823
+ @Override
2824
+ public Optional<WorkOrder> findWithDetailsByWorkOrderNumber(String workOrderNumber) {
2825
+ return Optional.ofNullable(mapper.findWithDetailsByWorkOrderNumber(workOrderNumber));
2826
+ }
2827
+ }
2828
+ ```
2829
+
2830
+ </details>
2831
+
2832
+ #### TDD: 楽観ロックのテスト
2833
+
2834
+ <details>
2835
+ <summary>WorkOrderRepositoryOptimisticLockTest.java</summary>
2836
+
2837
+ ```java
2838
+ // src/test/java/com/example/production/infrastructure/persistence/repository/WorkOrderRepositoryOptimisticLockTest.java
2839
+ package com.example.pms.infrastructure.persistence.repository;
2840
+
2841
+ import com.example.pms.application.port.out.WorkOrderRepository;
2842
+ import com.example.pms.domain.exception.OptimisticLockException;
2843
+ import com.example.pms.domain.model.process.WorkOrder;
2844
+ import com.example.pms.domain.model.process.WorkOrderStatus;
2845
+ import com.example.pms.testsetup.BaseIntegrationTest;
2846
+ import org.junit.jupiter.api.*;
2847
+ import org.springframework.beans.factory.annotation.Autowired;
2848
+
2849
+ import java.math.BigDecimal;
2850
+ import java.time.LocalDate;
2851
+
2852
+ import static org.assertj.core.api.Assertions.*;
2853
+
2854
+ @DisplayName("作業指示リポジトリ - 楽観ロック")
2855
+ class WorkOrderRepositoryOptimisticLockTest extends BaseIntegrationTest {
2856
+
2857
+ @Autowired
2858
+ private WorkOrderRepository workOrderRepository;
2859
+
2860
+ @BeforeEach
2861
+ void setUp() {
2862
+ // テストデータのセットアップ
2863
+ }
2864
+
2865
+ @Nested
2866
+ @DisplayName("完成数量更新の楽観ロック")
2867
+ class CompletionQuantityOptimisticLocking {
2868
+
2869
+ @Test
2870
+ @DisplayName("同じバージョンで完成数量を更新できる")
2871
+ void canUpdateCompletionQuantityWithSameVersion() {
2872
+ // Arrange
2873
+ WorkOrder workOrder = createTestWorkOrder("WO-TEST-001");
2874
+ Integer initialVersion = workOrder.getVersion();
2875
+
2876
+ // Act
2877
+ workOrderRepository.updateCompletionQuantities(
2878
+ workOrder.getWorkOrderNumber(),
2879
+ initialVersion,
2880
+ new BigDecimal("10"),
2881
+ new BigDecimal("9"),
2882
+ new BigDecimal("1"));
2883
+
2884
+ // Assert
2885
+ var updated = workOrderRepository.findWithDetailsByWorkOrderNumber("WO-TEST-001").get();
2886
+ assertThat(updated.getCompletedQuantity()).isEqualByComparingTo(new BigDecimal("10"));
2887
+ assertThat(updated.getTotalGoodQuantity()).isEqualByComparingTo(new BigDecimal("9"));
2888
+ assertThat(updated.getTotalDefectQuantity()).isEqualByComparingTo(new BigDecimal("1"));
2889
+ assertThat(updated.getVersion()).isEqualTo(initialVersion + 1);
2890
+ }
2891
+
2892
+ @Test
2893
+ @DisplayName("異なるバージョンで更新すると楽観ロック例外が発生する")
2894
+ void throwsExceptionWhenVersionMismatch() {
2895
+ // Arrange
2896
+ WorkOrder workOrder = createTestWorkOrder("WO-TEST-002");
2897
+ Integer initialVersion = workOrder.getVersion();
2898
+
2899
+ // 作業者Aが完成報告(成功)
2900
+ workOrderRepository.updateCompletionQuantities(
2901
+ workOrder.getWorkOrderNumber(),
2902
+ initialVersion,
2903
+ new BigDecimal("10"),
2904
+ new BigDecimal("10"),
2905
+ BigDecimal.ZERO);
2906
+
2907
+ // Act & Assert: 作業者Bが古いバージョンで完成報告(失敗)
2908
+ assertThatThrownBy(() -> workOrderRepository.updateCompletionQuantities(
2909
+ workOrder.getWorkOrderNumber(),
2910
+ initialVersion, // 古いバージョン
2911
+ new BigDecimal("5"),
2912
+ new BigDecimal("5"),
2913
+ BigDecimal.ZERO))
2914
+ .isInstanceOf(OptimisticLockException.class)
2915
+ .hasMessageContaining("他のユーザーによって更新されています");
2916
+ }
2917
+
2918
+ @Test
2919
+ @DisplayName("複数回の完成報告で累計が正しく更新される")
2920
+ void correctlyAccumulatesMultipleCompletionReports() {
2921
+ // Arrange
2922
+ WorkOrder workOrder = createTestWorkOrder("WO-TEST-003");
2923
+
2924
+ // Act: 最新バージョンを取得しながら3回報告
2925
+ for (int i = 0; i < 3; i++) {
2926
+ var current = workOrderRepository.findWithDetailsByWorkOrderNumber("WO-TEST-003").get();
2927
+ workOrderRepository.updateCompletionQuantities(
2928
+ current.getWorkOrderNumber(),
2929
+ current.getVersion(),
2930
+ new BigDecimal("10"),
2931
+ new BigDecimal("9"),
2932
+ new BigDecimal("1"));
2933
+ }
2934
+
2935
+ // Assert
2936
+ var result = workOrderRepository.findWithDetailsByWorkOrderNumber("WO-TEST-003").get();
2937
+ assertThat(result.getCompletedQuantity()).isEqualByComparingTo(new BigDecimal("30"));
2938
+ assertThat(result.getTotalGoodQuantity()).isEqualByComparingTo(new BigDecimal("27"));
2939
+ assertThat(result.getTotalDefectQuantity()).isEqualByComparingTo(new BigDecimal("3"));
2940
+ assertThat(result.getVersion()).isEqualTo(4); // 初期1 + 3回更新
2941
+ }
2942
+ }
2943
+
2944
+ private WorkOrder createTestWorkOrder(String workOrderNumber) {
2945
+ // テスト用作業指示の作成
2946
+ return WorkOrder.builder()
2947
+ .workOrderNumber(workOrderNumber)
2948
+ .orderNumber("MO-TEST-001")
2949
+ .workOrderDate(LocalDate.now())
2950
+ .itemCode("PROD-001")
2951
+ .orderQuantity(new BigDecimal("100"))
2952
+ .locationCode("LINE001")
2953
+ .plannedStartDate(LocalDate.now())
2954
+ .plannedEndDate(LocalDate.now().plusDays(5))
2955
+ .completedQuantity(BigDecimal.ZERO)
2956
+ .totalGoodQuantity(BigDecimal.ZERO)
2957
+ .totalDefectQuantity(BigDecimal.ZERO)
2958
+ .status(WorkOrderStatus.IN_PROGRESS)
2959
+ .completedFlag(false)
2960
+ .build();
2961
+ }
2962
+ }
2963
+ ```
2964
+
2965
+ </details>
2966
+
2967
+ ### 完成報告処理のシーケンス図
2968
+
2969
+ 完成報告では、作業指示の完成数量を楽観ロックで安全に更新します。
2970
+
2971
+ ```plantuml
2972
+ @startuml
2973
+
2974
+ title 完成報告処理シーケンス(楽観ロック対応)
2975
+
2976
+ actor 作業者A
2977
+ actor 作業者B
2978
+ participant "CompletionResultService" as Service
2979
+ participant "WorkOrderRepository" as WORepo
2980
+ participant "CompletionResultMapper" as CRMapper
2981
+ database "作業指示データ" as WOTable
2982
+
2983
+ == 同時完成報告シナリオ ==
2984
+
2985
+ 作業者A -> Service: 完成報告(作業指示番号, 完成数=10)
2986
+ activate Service
2987
+ Service -> WORepo: findWithDetails(作業指示番号)
2988
+ WORepo -> WOTable: SELECT ... WHERE 作業指示番号 = ?
2989
+ WOTable --> WORepo: 作業指示(version=1)
2990
+ WORepo --> Service: 作業指示(version=1)
2991
+
2992
+ 作業者B -> Service: 完成報告(作業指示番号, 完成数=5)
2993
+ activate Service
2994
+ Service -> WORepo: findWithDetails(作業指示番号)
2995
+ WORepo -> WOTable: SELECT ... WHERE 作業指示番号 = ?
2996
+ WOTable --> WORepo: 作業指示(version=1)
2997
+ WORepo --> Service: 作業指示(version=1)
2998
+
2999
+ note over 作業者A,WOTable: 作業者Aが先に更新
3000
+
3001
+ Service -> CRMapper: insert(完成実績)
3002
+ Service -> WORepo: updateCompletionQuantities(version=1)
3003
+ WORepo -> WOTable: UPDATE SET 完成済数 += 10, バージョン += 1\nWHERE バージョン = 1
3004
+ WOTable --> WORepo: 1 row updated
3005
+ WORepo --> Service: 成功
3006
+ Service --> 作業者A: 完成報告完了
3007
+ deactivate Service
3008
+
3009
+ note over 作業者A,WOTable: 作業者Bの更新(楽観ロック失敗)
3010
+
3011
+ Service -> CRMapper: insert(完成実績)
3012
+ Service -> WORepo: updateCompletionQuantities(version=1)
3013
+ WORepo -> WOTable: UPDATE SET 完成済数 += 5, バージョン += 1\nWHERE バージョン = 1
3014
+ WOTable --> WORepo: 0 rows updated
3015
+ WORepo -> WOTable: SELECT バージョン WHERE 作業指示番号 = ?
3016
+ WOTable --> WORepo: version=2
3017
+ WORepo --> Service: OptimisticLockException
3018
+ Service --> 作業者B: エラー: 他のユーザーが更新
3019
+ deactivate Service
3020
+
3021
+ note over 作業者B: 作業者Bはリトライ
3022
+
3023
+ 作業者B -> Service: 完成報告(作業指示番号, 完成数=5)
3024
+ activate Service
3025
+ Service -> WORepo: findWithDetails(作業指示番号)
3026
+ WORepo -> WOTable: SELECT
3027
+ WOTable --> WORepo: 作業指示(version=2, 完成済数=10)
3028
+ WORepo --> Service: 作業指示(version=2)
3029
+ Service -> CRMapper: insert(完成実績)
3030
+ Service -> WORepo: updateCompletionQuantities(version=2)
3031
+ WORepo -> WOTable: UPDATE SET 完成済数 += 5, バージョン += 1\nWHERE バージョン = 2
3032
+ WOTable --> WORepo: 1 row updated
3033
+ WORepo --> Service: 成功
3034
+ Service --> 作業者B: 完成報告完了(累計: 15)
3035
+ deactivate Service
3036
+
3037
+ @enduml
3038
+ ```
3039
+
3040
+ ### 工程管理向け楽観ロックのベストプラクティス
3041
+
3042
+ | ポイント | 説明 |
3043
+ |---------|------|
3044
+ | **累計更新** | `完成済数 = 完成済数 + #{completedQuantity}` でアトミックに加算 |
3045
+ | **バージョンチェック** | 必ず `AND "バージョン" = #{version}` を WHERE 条件に含める |
3046
+ | **リトライ戦略** | 楽観ロック失敗時は最新データを再取得して再試行 |
3047
+ | **UI 通知** | 競合発生時はユーザーに分かりやすいメッセージを表示 |
3048
+ | **完成上限チェック** | `completedQuantity + 報告数 <= orderQuantity` のビジネスチェックも併用 |
3049
+ | **トランザクション境界** | 完成実績 INSERT と作業指示 UPDATE を同一トランザクションで実行 |
3050
+
3051
+ ---
3052
+
3053
+ ## 27.5 まとめ
3054
+
3055
+ 本章では、工程管理の設計について学びました。
3056
+
3057
+ ### 学んだこと
3058
+
3059
+ 1. **製造指示の設計**
3060
+ - 作業指示データと作業指示明細データの親子関係
3061
+ - 工程表からの自動展開
3062
+ - ステータス管理(未着手→作業中→完了)
3063
+
3064
+ 2. **製造実績の設計**
3065
+ - 完成実績と検査結果の記録
3066
+ - 作業指示の累計更新
3067
+ - 不良品の欠点別管理
3068
+
3069
+ 3. **工数実績の設計**
3070
+ - 工順ごとの工数記録
3071
+ - 担当者別・工程別の集計
3072
+
3073
+ 4. **進捗管理の設計**
3074
+ - 進捗率の計算
3075
+ - 遅延アラートの判定
3076
+ - 進捗状況の可視化
3077
+
3078
+ ### テーブル一覧(第27章)
3079
+
3080
+ | テーブル名(日本語) | Java エンティティ | 説明 |
3081
+ |---|---|---|
3082
+ | 工程マスタ | Process | 工程の定義 |
3083
+ | 工程表 | Routing | 品目の製造工程定義 |
3084
+ | 作業指示データ | WorkOrder | 製造指示情報 |
3085
+ | 作業指示明細データ | WorkOrderDetail | 工程ごとの指示明細 |
3086
+ | 完成実績データ | CompletionResult | 完成報告 |
3087
+ | 完成検査結果データ | InspectionResult | 検査での不良内訳 |
3088
+ | 工数実績データ | LaborHours | 工程ごとの作業時間 |
3089
+
3090
+ ### ENUM 一覧(第27章)
3091
+
3092
+ | DB ENUM 型(日本語) | Java Enum | 値 |
3093
+ |---|---|---|
3094
+ | 作業指示ステータス | WorkOrderStatus | 未着手→NOT_STARTED, 作業中→IN_PROGRESS, 完了→COMPLETED, 中断→SUSPENDED |
3095
+
3096
+ ### 命名規則のパターン
3097
+
3098
+ - **DB(日本語)**: テーブル名、カラム名、ENUM 型・値は日本語
3099
+ - **Java(英語)**: クラス名、フィールド名、enum 値は英語
3100
+ - **MyBatis resultMap**: 日本語カラムと英語プロパティのマッピング
3101
+ - **TypeHandler**: 日本語 ENUM 値と英語 enum の相互変換
3102
+
3103
+ ---
3104
+
3105
+ [← 第26章:外注委託管理の設計](chapter26.md) | [第28章:在庫管理の設計 →](chapter28.md)