@k2works/claude-code-booster 3.2.1 → 3.3.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (70) hide show
  1. package/lib/assets/.claude/skills/analyzing-business/SKILL.md +2 -2
  2. package/lib/assets/.claude/skills/analyzing-inception-deck/SKILL.md +5 -5
  3. package/lib/assets/.claude/skills/analyzing-requirements/SKILL.md +2 -2
  4. package/lib/assets/.claude/skills/generating-slides/SKILL.md +7 -7
  5. package/lib/assets/docs/article/index.md +4 -1
  6. package/lib/assets/docs/article/practical-database-design/index.md +121 -0
  7. package/lib/assets/docs/article/practical-database-design/part1/chapter01.md +288 -0
  8. package/lib/assets/docs/article/practical-database-design/part1/chapter02.md +518 -0
  9. package/lib/assets/docs/article/practical-database-design/part1/chapter03.md +557 -0
  10. package/lib/assets/docs/article/practical-database-design/part2/chapter04.md +924 -0
  11. package/lib/assets/docs/article/practical-database-design/part2/chapter05.md +1627 -0
  12. package/lib/assets/docs/article/practical-database-design/part2/chapter06.md +2716 -0
  13. package/lib/assets/docs/article/practical-database-design/part2/chapter07.md +2082 -0
  14. package/lib/assets/docs/article/practical-database-design/part2/chapter08.md +2105 -0
  15. package/lib/assets/docs/article/practical-database-design/part2/chapter09.md +2031 -0
  16. package/lib/assets/docs/article/practical-database-design/part2/chapter10.md +1387 -0
  17. package/lib/assets/docs/article/practical-database-design/part2/chapter11.md +1677 -0
  18. package/lib/assets/docs/article/practical-database-design/part2/chapter12.md +1417 -0
  19. package/lib/assets/docs/article/practical-database-design/part2/chapter13.md +1434 -0
  20. package/lib/assets/docs/article/practical-database-design/part3/chapter14.md +667 -0
  21. package/lib/assets/docs/article/practical-database-design/part3/chapter15.md +1625 -0
  22. package/lib/assets/docs/article/practical-database-design/part3/chapter16.md +1915 -0
  23. package/lib/assets/docs/article/practical-database-design/part3/chapter17.md +1708 -0
  24. package/lib/assets/docs/article/practical-database-design/part3/chapter18.md +2095 -0
  25. package/lib/assets/docs/article/practical-database-design/part3/chapter19.md +1123 -0
  26. package/lib/assets/docs/article/practical-database-design/part3/chapter20.md +1031 -0
  27. package/lib/assets/docs/article/practical-database-design/part3/chapter21.md +1382 -0
  28. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter14-orm.md +991 -0
  29. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter15-orm.md +1300 -0
  30. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter16-orm.md +1166 -0
  31. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter17-orm.md +1584 -0
  32. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter18-orm.md +1183 -0
  33. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter19-orm.md +1016 -0
  34. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter20-orm.md +1753 -0
  35. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter21-orm.md +1447 -0
  36. package/lib/assets/docs/article/practical-database-design/part3-orm/chapter22-orm.md +1878 -0
  37. package/lib/assets/docs/article/practical-database-design/part4/chapter22.md +965 -0
  38. package/lib/assets/docs/article/practical-database-design/part4/chapter23.md +2069 -0
  39. package/lib/assets/docs/article/practical-database-design/part4/chapter24.md +2439 -0
  40. package/lib/assets/docs/article/practical-database-design/part4/chapter25.md +3661 -0
  41. package/lib/assets/docs/article/practical-database-design/part4/chapter26.md +2916 -0
  42. package/lib/assets/docs/article/practical-database-design/part4/chapter27.md +3105 -0
  43. package/lib/assets/docs/article/practical-database-design/part4/chapter28.md +2697 -0
  44. package/lib/assets/docs/article/practical-database-design/part4/chapter29.md +2544 -0
  45. package/lib/assets/docs/article/practical-database-design/part4/chapter30.md +2180 -0
  46. package/lib/assets/docs/article/practical-database-design/part4/chapter31.md +1192 -0
  47. package/lib/assets/docs/article/practical-database-design/part4/chapter32.md +2101 -0
  48. package/lib/assets/docs/article/practical-database-design/part5/chapter33.md +1032 -0
  49. package/lib/assets/docs/article/practical-database-design/part5/chapter34.md +1609 -0
  50. package/lib/assets/docs/article/practical-database-design/part5/chapter35.md +1453 -0
  51. package/lib/assets/docs/article/practical-database-design/part5/chapter36.md +1292 -0
  52. package/lib/assets/docs/article/practical-database-design/part5/chapter37.md +1470 -0
  53. package/lib/assets/docs/article/practical-database-design/part5/chapter38.md +1698 -0
  54. package/lib/assets/docs/article/practical-database-design/part5/chapter39.md +2334 -0
  55. package/lib/assets/docs/article/practical-database-design/study/study2-1.md +1693 -0
  56. package/lib/assets/docs/article/practical-database-design/study/study2-2.md +1347 -0
  57. package/lib/assets/docs/article/practical-database-design/study/study2-3.md +2044 -0
  58. package/lib/assets/docs/article/practical-database-design/study/study2-4.md +2229 -0
  59. package/lib/assets/docs/article/practical-database-design/study/study2-5.md +2418 -0
  60. package/lib/assets/docs/article/practical-database-design/study/study3-1.md +2205 -0
  61. package/lib/assets/docs/article/practical-database-design/study/study3-2.md +2221 -0
  62. package/lib/assets/docs/article/practical-database-design/study/study3-3.md +2253 -0
  63. package/lib/assets/docs/article/practical-database-design/study/study3-4.md +2106 -0
  64. package/lib/assets/docs/article/practical-database-design/study/study3-5.md +2507 -0
  65. package/lib/assets/docs/article/practical-database-design/study/study4-1.md +2587 -0
  66. package/lib/assets/docs/article/practical-database-design/study/study4-2.md +2075 -0
  67. package/lib/assets/docs/article/practical-database-design/study/study4-3.md +1805 -0
  68. package/lib/assets/docs/article/practical-database-design/study/study4-4.md +1895 -0
  69. package/lib/assets/docs/article/practical-database-design/study/study4-5.md +2878 -0
  70. package/package.json +1 -1
@@ -0,0 +1,1453 @@
1
+ # 第35章:システム間連携パターン
2
+
3
+ 本章では、第33章で解説した境界づけられたコンテキストと、第34章で解説したメッセージングパターンを活用して、販売管理・財務会計・生産管理の3つのシステム間の具体的な連携パターンを解説します。
4
+
5
+ ---
6
+
7
+ ## 35.1 販売管理と財務会計の連携
8
+
9
+ 販売管理システムと財務会計システムの連携は、基幹業務システム統合の中核となる部分です。売上、請求、入金などの取引データを正確に会計仕訳へ変換することが求められます。
10
+
11
+ ### 連携の全体像
12
+
13
+ ```plantuml
14
+ @startuml
15
+ title 販売管理と財務会計の連携全体像
16
+
17
+ package "販売管理システム" as sales {
18
+ rectangle "受注管理" as order
19
+ rectangle "出荷管理" as shipment
20
+ rectangle "売上管理" as sales_mgmt
21
+ rectangle "請求管理" as billing
22
+ rectangle "入金管理" as receipt
23
+ }
24
+
25
+ package "イベントバス" as events {
26
+ collections "売上イベント" as sales_event
27
+ collections "請求イベント" as billing_event
28
+ collections "入金イベント" as receipt_event
29
+ }
30
+
31
+ package "財務会計システム" as accounting {
32
+ rectangle "自動仕訳" as auto_journal
33
+ rectangle "仕訳管理" as journal
34
+ rectangle "残高管理" as balance
35
+ rectangle "決算処理" as closing
36
+ }
37
+
38
+ order --> shipment
39
+ shipment --> sales_mgmt
40
+ sales_mgmt --> billing
41
+ billing --> receipt
42
+
43
+ sales_mgmt --> sales_event : 売上計上
44
+ billing --> billing_event : 請求確定
45
+ receipt --> receipt_event : 入金確認
46
+
47
+ sales_event --> auto_journal
48
+ billing_event --> auto_journal
49
+ receipt_event --> auto_journal
50
+
51
+ auto_journal --> journal
52
+ journal --> balance
53
+
54
+ @enduml
55
+ ```
56
+
57
+ ### 売上データから仕訳データへの変換
58
+
59
+ 売上計上時に発生する仕訳は、売上の種類や取引条件によって異なります。
60
+
61
+ ```plantuml
62
+ @startuml
63
+ title 売上から仕訳への変換フロー
64
+
65
+ |販売管理|
66
+ start
67
+ :売上計上;
68
+ :売上イベント発行;
69
+
70
+ |イベントバス|
71
+ :売上イベント受信;
72
+
73
+ |財務会計|
74
+ :仕訳パターン判定;
75
+ note right
76
+ ・商品グループ
77
+ ・顧客グループ
78
+ ・取引条件
79
+ end note
80
+
81
+ :仕訳明細生成;
82
+ note right
83
+ 借方:売掛金
84
+ 貸方:売上高
85
+ 貸方:仮受消費税
86
+ end note
87
+
88
+ :貸借一致検証;
89
+ :仕訳登録;
90
+ :残高更新;
91
+
92
+ stop
93
+
94
+ @enduml
95
+ ```
96
+
97
+ #### 売上仕訳の基本パターン
98
+
99
+ ```plantuml
100
+ @startuml
101
+ title 売上仕訳の基本パターン
102
+
103
+ object "売上データ" as sales {
104
+ 売上番号 = "SLS-2024-001"
105
+ 売上日 = "2024/01/15"
106
+ 顧客コード = "CUS-001"
107
+ 売上金額 = 110,000円
108
+ 消費税額 = 10,000円
109
+ 税抜金額 = 100,000円
110
+ }
111
+
112
+ object "仕訳伝票" as journal {
113
+ 伝票番号 = "JRN-2024-001"
114
+ 起票日 = "2024/01/15"
115
+ 伝票区分 = "売上仕訳"
116
+ }
117
+
118
+ object "借方明細" as debit {
119
+ 勘定科目 = "売掛金"
120
+ 金額 = 110,000円
121
+ 摘要 = "CUS-001 売上"
122
+ }
123
+
124
+ object "貸方明細1" as credit1 {
125
+ 勘定科目 = "売上高"
126
+ 金額 = 100,000円
127
+ 摘要 = "商品売上"
128
+ }
129
+
130
+ object "貸方明細2" as credit2 {
131
+ 勘定科目 = "仮受消費税"
132
+ 金額 = 10,000円
133
+ 摘要 = "消費税10%"
134
+ }
135
+
136
+ sales --> journal : 変換
137
+ journal --> debit
138
+ journal --> credit1
139
+ journal --> credit2
140
+
141
+ note bottom of journal
142
+ 【貸借一致の検証】
143
+ 借方合計:110,000円
144
+ 貸方合計:110,000円
145
+ 差額:0円 ✓
146
+ end note
147
+
148
+ @enduml
149
+ ```
150
+
151
+ #### 売上仕訳パターンマスタ
152
+
153
+ ```plantuml
154
+ @startuml
155
+ title 自動仕訳パターンマスタの構造
156
+
157
+ entity "自動仕訳パターン" as pattern {
158
+ *パターンID : string
159
+ --
160
+ *取引種別 : string
161
+ 商品グループ : string
162
+ 顧客グループ : string
163
+ 借方科目コード : string
164
+ 貸方科目コード : string
165
+ 消費税科目コード : string
166
+ 適用開始日 : date
167
+ 適用終了日 : date
168
+ }
169
+
170
+ entity "勘定科目" as account {
171
+ *科目コード : string
172
+ --
173
+ 科目名 : string
174
+ BSPL区分 : string
175
+ 貸借区分 : string
176
+ }
177
+
178
+ pattern }o--|| account : 借方科目
179
+ pattern }o--|| account : 貸方科目
180
+ pattern }o--|| account : 消費税科目
181
+
182
+ @enduml
183
+ ```
184
+
185
+ <details>
186
+ <summary>仕訳パターンテーブル定義</summary>
187
+
188
+ ```sql
189
+ -- 自動仕訳パターンマスタ
190
+ CREATE TABLE 自動仕訳パターン (
191
+ パターンID VARCHAR(20) PRIMARY KEY,
192
+ 取引種別 VARCHAR(20) NOT NULL,
193
+ 商品グループ VARCHAR(10),
194
+ 顧客グループ VARCHAR(10),
195
+ 借方科目コード VARCHAR(10) NOT NULL,
196
+ 貸方科目コード VARCHAR(10) NOT NULL,
197
+ 消費税科目コード VARCHAR(10),
198
+ 適用開始日 DATE NOT NULL,
199
+ 適用終了日 DATE,
200
+ 作成日時 TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
201
+ 更新日時 TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
202
+ FOREIGN KEY (借方科目コード) REFERENCES 勘定科目(科目コード),
203
+ FOREIGN KEY (貸方科目コード) REFERENCES 勘定科目(科目コード),
204
+ FOREIGN KEY (消費税科目コード) REFERENCES 勘定科目(科目コード)
205
+ );
206
+
207
+ -- サンプルデータ
208
+ INSERT INTO 自動仕訳パターン VALUES
209
+ ('PTN-SALES-001', '売上', NULL, NULL, '1310', '4110', '2191', '2024-01-01', NULL),
210
+ ('PTN-SALES-002', '売上', 'FOOD', NULL, '1310', '4120', '2191', '2024-01-01', NULL),
211
+ ('PTN-RETURN-001', '売上返品', NULL, NULL, '4110', '1310', '2191', '2024-01-01', NULL);
212
+ ```
213
+
214
+ </details>
215
+
216
+ ### 自動仕訳パターンの適用
217
+
218
+ 商品グループや顧客グループに応じて、適切な仕訳パターンを自動選択します。
219
+
220
+ ```plantuml
221
+ @startuml
222
+ title 自動仕訳パターン判定ロジック
223
+
224
+ start
225
+
226
+ :売上イベント受信;
227
+
228
+ :商品グループ取得;
229
+ :顧客グループ取得;
230
+
231
+ if (特定商品グループ?) then (yes)
232
+ :商品グループ専用\nパターン検索;
233
+ else (no)
234
+ if (特定顧客グループ?) then (yes)
235
+ :顧客グループ専用\nパターン検索;
236
+ else (no)
237
+ :汎用パターン検索;
238
+ endif
239
+ endif
240
+
241
+ if (パターン見つかった?) then (yes)
242
+ :仕訳明細生成;
243
+ :仕訳登録;
244
+ else (no)
245
+ :エラー通知;
246
+ :手動仕訳待ち\nキューに投入;
247
+ endif
248
+
249
+ stop
250
+
251
+ @enduml
252
+ ```
253
+
254
+ <details>
255
+ <summary>Java 実装例</summary>
256
+
257
+ ```java
258
+ // 自動仕訳パターンリポジトリ
259
+ public interface JournalPatternRepository {
260
+ Optional<JournalPattern> findByTransactionTypeAndGroups(
261
+ TransactionType type,
262
+ String productGroup,
263
+ String customerGroup,
264
+ LocalDate effectiveDate
265
+ );
266
+ }
267
+
268
+ // 自動仕訳サービス
269
+ @Service
270
+ @Transactional
271
+ public class AutoJournalService {
272
+ private final JournalPatternRepository patternRepository;
273
+ private final JournalRepository journalRepository;
274
+ private final AccountRepository accountRepository;
275
+
276
+ @EventListener
277
+ public void handleSalesCompleted(SalesCompletedEvent event) {
278
+ // パターン検索(優先順位:商品グループ > 顧客グループ > 汎用)
279
+ JournalPattern pattern = findPattern(event);
280
+
281
+ // 仕訳伝票生成
282
+ JournalEntry journal = createJournalEntry(event, pattern);
283
+
284
+ // 貸借一致検証
285
+ validateBalance(journal);
286
+
287
+ // 仕訳登録
288
+ journalRepository.save(journal);
289
+
290
+ // 残高更新イベント発行
291
+ publishBalanceUpdateEvent(journal);
292
+ }
293
+
294
+ private JournalPattern findPattern(SalesCompletedEvent event) {
295
+ // 1. 商品グループ + 顧客グループで検索
296
+ Optional<JournalPattern> pattern = patternRepository
297
+ .findByTransactionTypeAndGroups(
298
+ TransactionType.SALES,
299
+ event.productGroup(),
300
+ event.customerGroup(),
301
+ event.salesDate()
302
+ );
303
+
304
+ if (pattern.isPresent()) return pattern.get();
305
+
306
+ // 2. 商品グループのみで検索
307
+ pattern = patternRepository.findByTransactionTypeAndGroups(
308
+ TransactionType.SALES,
309
+ event.productGroup(),
310
+ null,
311
+ event.salesDate()
312
+ );
313
+
314
+ if (pattern.isPresent()) return pattern.get();
315
+
316
+ // 3. 顧客グループのみで検索
317
+ pattern = patternRepository.findByTransactionTypeAndGroups(
318
+ TransactionType.SALES,
319
+ null,
320
+ event.customerGroup(),
321
+ event.salesDate()
322
+ );
323
+
324
+ if (pattern.isPresent()) return pattern.get();
325
+
326
+ // 4. 汎用パターンで検索
327
+ return patternRepository.findByTransactionTypeAndGroups(
328
+ TransactionType.SALES,
329
+ null,
330
+ null,
331
+ event.salesDate()
332
+ ).orElseThrow(() -> new PatternNotFoundException(
333
+ "仕訳パターンが見つかりません: " + event.salesId()
334
+ ));
335
+ }
336
+
337
+ private JournalEntry createJournalEntry(
338
+ SalesCompletedEvent event,
339
+ JournalPattern pattern) {
340
+
341
+ List<JournalLine> lines = new ArrayList<>();
342
+
343
+ // 借方:売掛金
344
+ lines.add(JournalLine.debit(
345
+ pattern.debitAccountCode(),
346
+ event.totalAmount(),
347
+ "売掛金計上 " + event.customerName()
348
+ ));
349
+
350
+ // 貸方:売上高
351
+ lines.add(JournalLine.credit(
352
+ pattern.creditAccountCode(),
353
+ event.netAmount(),
354
+ "売上計上 " + event.salesId()
355
+ ));
356
+
357
+ // 貸方:仮受消費税
358
+ if (event.taxAmount().compareTo(BigDecimal.ZERO) > 0) {
359
+ lines.add(JournalLine.credit(
360
+ pattern.taxAccountCode(),
361
+ event.taxAmount(),
362
+ "仮受消費税"
363
+ ));
364
+ }
365
+
366
+ return new JournalEntry(
367
+ generateJournalId(),
368
+ event.salesDate(),
369
+ JournalType.AUTO_SALES,
370
+ event.salesId(),
371
+ lines
372
+ );
373
+ }
374
+
375
+ private void validateBalance(JournalEntry journal) {
376
+ BigDecimal debitTotal = journal.lines().stream()
377
+ .filter(l -> l.debitCredit() == DebitCredit.DEBIT)
378
+ .map(JournalLine::amount)
379
+ .reduce(BigDecimal.ZERO, BigDecimal::add);
380
+
381
+ BigDecimal creditTotal = journal.lines().stream()
382
+ .filter(l -> l.debitCredit() == DebitCredit.CREDIT)
383
+ .map(JournalLine::amount)
384
+ .reduce(BigDecimal.ZERO, BigDecimal::add);
385
+
386
+ if (debitTotal.compareTo(creditTotal) != 0) {
387
+ throw new BalanceMismatchException(
388
+ "貸借不一致: 借方=" + debitTotal + ", 貸方=" + creditTotal
389
+ );
390
+ }
391
+ }
392
+ }
393
+ ```
394
+
395
+ </details>
396
+
397
+ ### イベント駆動による仕訳生成
398
+
399
+ 販売管理システムで発生する各種イベントに応じて、自動的に仕訳を生成します。
400
+
401
+ ```plantuml
402
+ @startuml
403
+ title 販売イベントと仕訳の対応
404
+
405
+ ' 配置を制御するための設定
406
+ skinparam nodesep 50
407
+ skinparam ranksep 50
408
+
409
+ rectangle "販売管理イベント" as sales_events {
410
+ rectangle "売上計上" as sales_completed
411
+ rectangle "売上返品" as sales_return
412
+ rectangle "請求確定" as billing_confirmed
413
+ rectangle "入金確認" as payment_received
414
+ rectangle "貸倒発生" as bad_debt
415
+ }
416
+
417
+ rectangle "自動仕訳" as auto_journal {
418
+ rectangle "売上仕訳\n生成" as sales_journal
419
+ rectangle "返品仕訳\n生成" as return_journal
420
+ rectangle "(請求時仕訳\nなし)" as billing_journal
421
+ rectangle "入金仕訳\n生成" as payment_journal
422
+ rectangle "貸倒仕訳\n生成" as bad_debt_journal
423
+ }
424
+
425
+ rectangle "仕訳パターン" as patterns {
426
+ rectangle "==売上仕訳\n\n借方: 売掛金\n貸方: 売上高\n貸方: 仮受消費税" as p1
427
+ rectangle "==返品仕訳\n\n借方: 売上高\n借方: 仮受消費税\n貸方: 売掛金" as p2
428
+ rectangle "==入金仕訳\n\n借方: 普通預金\n貸方: 売掛金" as p3
429
+ rectangle "==貸倒仕訳\n\n借方: 貸倒損失\n貸方: 売掛金" as p4
430
+ }
431
+
432
+ sales_completed --> sales_journal
433
+ sales_return --> return_journal
434
+ billing_confirmed --> billing_journal
435
+ payment_received --> payment_journal
436
+ bad_debt --> bad_debt_journal
437
+
438
+ sales_journal --> p1
439
+ return_journal --> p2
440
+ payment_journal --> p3
441
+ bad_debt_journal --> p4
442
+
443
+ @enduml
444
+ ```
445
+
446
+ #### 入金消込と仕訳生成
447
+
448
+ ```plantuml
449
+ @startuml
450
+ title 入金消込と仕訳生成フロー
451
+
452
+ |販売管理|
453
+ start
454
+ :入金データ受信;
455
+ :請求データ照合;
456
+
457
+ if (消込対象あり?) then (yes)
458
+ :入金消込処理;
459
+ :消込イベント発行;
460
+ else (no)
461
+ :前受金計上;
462
+ :前受金イベント発行;
463
+ endif
464
+
465
+ |財務会計|
466
+ if (消込イベント?) then (yes)
467
+ :入金仕訳生成;
468
+ note right
469
+ 借方: 普通預金
470
+ 貸方: 売掛金
471
+ end note
472
+ else (no)
473
+ :前受金仕訳生成;
474
+ note right
475
+ 借方: 普通預金
476
+ 貸方: 前受金
477
+ end note
478
+ endif
479
+
480
+ :仕訳登録;
481
+ :売掛金残高更新;
482
+
483
+ stop
484
+
485
+ @enduml
486
+ ```
487
+
488
+ ---
489
+
490
+ ## 35.2 販売管理と生産管理の連携
491
+
492
+ 販売管理システムと生産管理システムの連携は、需要と供給のバランスを取るために重要です。受注情報を生産計画に反映し、在庫情報を双方向で同期します。
493
+
494
+ ### 連携の全体像
495
+
496
+ ```plantuml
497
+ @startuml
498
+ title 販売管理と生産管理の連携全体像
499
+
500
+ package "販売管理システム" as sales {
501
+ rectangle "受注管理" as order
502
+ rectangle "出荷管理" as shipment
503
+ rectangle "在庫照会" as sales_inv
504
+ }
505
+
506
+ package "イベントバス" as events {
507
+ collections "受注イベント" as order_event
508
+ collections "在庫イベント" as inv_event
509
+ collections "完成イベント" as complete_event
510
+ }
511
+
512
+ package "生産管理システム" as production {
513
+ rectangle "生産計画" as plan
514
+ rectangle "MRP" as mrp
515
+ rectangle "製造管理" as manufacturing
516
+ rectangle "在庫管理" as prod_inv
517
+ }
518
+
519
+ order --> order_event : 受注確定
520
+ order_event --> plan : 需要情報
521
+
522
+ plan --> mrp : 所要量展開
523
+ mrp --> manufacturing : 製造指示
524
+
525
+ manufacturing --> complete_event : 完成報告
526
+ complete_event --> prod_inv : 在庫計上
527
+ prod_inv --> inv_event : 在庫更新
528
+
529
+ inv_event --> sales_inv : 在庫同期
530
+ sales_inv --> shipment : 引当可能数
531
+
532
+ @enduml
533
+ ```
534
+
535
+ ### 受注情報から生産計画への連携
536
+
537
+ ```plantuml
538
+ @startuml
539
+ title 受注から生産計画への連携フロー
540
+
541
+ |販売管理|
542
+ start
543
+ :受注登録;
544
+ :受注確定;
545
+ :受注イベント発行;
546
+
547
+ |生産計画|
548
+ :受注イベント受信;
549
+ :需要情報登録;
550
+
551
+ :在庫・発注残確認;
552
+ if (在庫で充足?) then (yes)
553
+ :引当処理;
554
+ else (no)
555
+ :正味所要量計算;
556
+ if (製品?) then (yes)
557
+ :製造オーダ作成;
558
+ else (no)
559
+ :発注提案作成;
560
+ endif
561
+ endif
562
+
563
+ :計画確定;
564
+ :計画確定イベント発行;
565
+
566
+ |販売管理|
567
+ :納期回答更新;
568
+
569
+ stop
570
+
571
+ @enduml
572
+ ```
573
+
574
+ #### 受注イベントの構造
575
+
576
+ ```plantuml
577
+ @startuml
578
+ title 受注イベントの構造
579
+
580
+ class "OrderConfirmedEvent" as event {
581
+ +eventId: String
582
+ +timestamp: Instant
583
+ +orderId: String
584
+ +customerId: String
585
+ +requestedDeliveryDate: LocalDate
586
+ +priority: Priority
587
+ +lines: List<OrderLineEvent>
588
+ }
589
+
590
+ class "OrderLineEvent" as line {
591
+ +lineNumber: int
592
+ +productId: String
593
+ +quantity: BigDecimal
594
+ +unit: String
595
+ +requestedDate: LocalDate
596
+ }
597
+
598
+ event "1" -- "*" line
599
+
600
+ note right of event
601
+ 【生産計画に必要な情報】
602
+ ・製品コード
603
+ ・必要数量
604
+ ・希望納期
605
+ ・優先度
606
+ end note
607
+
608
+ @enduml
609
+ ```
610
+
611
+ <details>
612
+ <summary>Java 実装例</summary>
613
+
614
+ ```java
615
+ // 受注確定イベント
616
+ public record OrderConfirmedEvent(
617
+ String eventId,
618
+ Instant timestamp,
619
+ String orderId,
620
+ String customerId,
621
+ LocalDate requestedDeliveryDate,
622
+ Priority priority,
623
+ List<OrderLineEvent> lines
624
+ ) {
625
+ public enum Priority {
626
+ URGENT, // 緊急
627
+ HIGH, // 高
628
+ NORMAL, // 通常
629
+ LOW // 低
630
+ }
631
+ }
632
+
633
+ // 生産計画サービス
634
+ @Service
635
+ public class ProductionPlanningService {
636
+ private final DemandRepository demandRepository;
637
+ private final InventoryRepository inventoryRepository;
638
+ private final MrpService mrpService;
639
+
640
+ @EventListener
641
+ public void handleOrderConfirmed(OrderConfirmedEvent event) {
642
+ // 需要情報として登録
643
+ for (OrderLineEvent line : event.lines()) {
644
+ Demand demand = Demand.fromOrder(
645
+ event.orderId(),
646
+ line.productId(),
647
+ line.quantity(),
648
+ line.requestedDate(),
649
+ event.priority()
650
+ );
651
+ demandRepository.save(demand);
652
+
653
+ // 在庫引当を試行
654
+ AllocationResult result = tryAllocate(demand);
655
+
656
+ if (!result.isFullyAllocated()) {
657
+ // MRP実行して製造/発注オーダを生成
658
+ mrpService.execute(demand);
659
+ }
660
+ }
661
+
662
+ // 納期回答イベント発行
663
+ publishDeliveryDateResponse(event.orderId());
664
+ }
665
+
666
+ private AllocationResult tryAllocate(Demand demand) {
667
+ BigDecimal available = inventoryRepository
668
+ .findAvailableQuantity(demand.productId());
669
+
670
+ if (available.compareTo(demand.quantity()) >= 0) {
671
+ // 引当実行
672
+ inventoryRepository.allocate(
673
+ demand.productId(),
674
+ demand.quantity(),
675
+ demand.demandId()
676
+ );
677
+ return AllocationResult.fullyAllocated(demand.quantity());
678
+ } else {
679
+ // 部分引当
680
+ if (available.compareTo(BigDecimal.ZERO) > 0) {
681
+ inventoryRepository.allocate(
682
+ demand.productId(),
683
+ available,
684
+ demand.demandId()
685
+ );
686
+ }
687
+ return AllocationResult.partiallyAllocated(
688
+ available,
689
+ demand.quantity().subtract(available)
690
+ );
691
+ }
692
+ }
693
+ }
694
+ ```
695
+
696
+ </details>
697
+
698
+ ### 需要予測データの共有
699
+
700
+ ```plantuml
701
+ @startuml
702
+ title 需要予測データの共有
703
+
704
+ package "販売管理" as sales {
705
+ rectangle "販売実績" as sales_history
706
+ rectangle "受注残" as backlog
707
+ rectangle "見積案件" as quotation
708
+ }
709
+
710
+ package "需要予測エンジン" as forecast {
711
+ rectangle "時系列分析" as time_series
712
+ rectangle "季節調整" as seasonal
713
+ rectangle "トレンド分析" as trend
714
+ }
715
+
716
+ package "生産計画" as planning {
717
+ rectangle "MPS\n(基準生産計画)" as mps
718
+ rectangle "MRP\n(所要量計画)" as mrp
719
+ }
720
+
721
+ sales_history --> forecast
722
+ backlog --> forecast
723
+ quotation --> forecast
724
+
725
+ forecast --> mps : 需要予測データ
726
+ mps --> mrp : 計画生産量
727
+
728
+ note right of forecast
729
+ 【予測データの内容】
730
+ ・製品別月次予測数量
731
+ ・予測精度(信頼区間)
732
+ ・季節変動係数
733
+ ・トレンド方向
734
+ end note
735
+
736
+ @enduml
737
+ ```
738
+
739
+ #### 需要予測イベント
740
+
741
+ ```plantuml
742
+ @startuml
743
+ title 需要予測の連携サイクル
744
+
745
+ |販売管理|
746
+ start
747
+ :月次販売実績集計;
748
+ :受注残・見積集計;
749
+ :需要予測更新;
750
+ :予測データ発行;
751
+
752
+ |生産計画|
753
+ :予測データ受信;
754
+ :MPS更新;
755
+ note right
756
+ ・月次生産計画
757
+ ・安全在庫調整
758
+ ・生産能力確認
759
+ end note
760
+
761
+ :MRP実行;
762
+ :発注提案生成;
763
+ :製造計画生成;
764
+
765
+ if (計画変更あり?) then (yes)
766
+ :計画変更通知;
767
+ endif
768
+
769
+ |販売管理|
770
+ :納期情報更新;
771
+
772
+ stop
773
+
774
+ @enduml
775
+ ```
776
+
777
+ ### 在庫情報の同期
778
+
779
+ 販売管理と生産管理の両方で在庫を管理する場合、整合性を保つ必要があります。
780
+
781
+ ```plantuml
782
+ @startuml
783
+ title 在庫情報の同期パターン
784
+
785
+ package "販売管理" as sales {
786
+ database "販売在庫ビュー\n----\n在庫数\n引当数\n有効在庫数" as sales_inv
787
+ }
788
+
789
+ package "生産管理" as production {
790
+ database "生産在庫(マスタ)\n----\n在庫数\n引当数\n品質状態\nロット情報" as prod_inv
791
+ }
792
+
793
+ package "同期メカニズム" as sync {
794
+ collections "在庫更新\nイベント" as inv_event
795
+ rectangle "在庫同期\nサービス" as sync_svc
796
+ }
797
+
798
+ prod_inv --> inv_event : 在庫変動時
799
+ inv_event --> sync_svc
800
+ sync_svc --> sales_inv : ビュー更新
801
+
802
+ note bottom of sync
803
+ 【同期タイミング】
804
+ ・完成時(入庫)
805
+ ・出荷時(出庫)
806
+ ・棚卸差異発生時
807
+ ・品質状態変更時
808
+
809
+ 【同期内容】
810
+ ・製品別在庫数
811
+ ・引当可能数
812
+ ・入庫予定
813
+ end note
814
+
815
+ @enduml
816
+ ```
817
+
818
+ <details>
819
+ <summary>Java 実装例</summary>
820
+
821
+ ```java
822
+ // 在庫更新イベント
823
+ public record InventoryUpdatedEvent(
824
+ String eventId,
825
+ Instant timestamp,
826
+ String productId,
827
+ String warehouseId,
828
+ BigDecimal previousQuantity,
829
+ BigDecimal currentQuantity,
830
+ BigDecimal allocatedQuantity,
831
+ UpdateReason reason
832
+ ) {
833
+ public enum UpdateReason {
834
+ COMPLETION, // 製造完成
835
+ SHIPMENT, // 出荷
836
+ RECEIPT, // 入荷
837
+ ADJUSTMENT, // 棚卸調整
838
+ QUALITY_CHANGE // 品質状態変更
839
+ }
840
+
841
+ public BigDecimal getAvailableQuantity() {
842
+ return currentQuantity.subtract(allocatedQuantity);
843
+ }
844
+ }
845
+
846
+ // 在庫同期サービス
847
+ @Service
848
+ public class InventorySyncService {
849
+ private final SalesInventoryRepository salesInventoryRepository;
850
+
851
+ @EventListener
852
+ public void handleInventoryUpdated(InventoryUpdatedEvent event) {
853
+ // 販売在庫ビューを更新
854
+ SalesInventory salesInv = salesInventoryRepository
855
+ .findByProductId(event.productId())
856
+ .orElse(new SalesInventory(event.productId()));
857
+
858
+ salesInv.updateQuantity(
859
+ event.currentQuantity(),
860
+ event.allocatedQuantity()
861
+ );
862
+
863
+ salesInventoryRepository.save(salesInv);
864
+
865
+ // 低在庫アラート
866
+ if (salesInv.getAvailableQuantity()
867
+ .compareTo(salesInv.getSafetyStock()) < 0) {
868
+ publishLowStockAlert(event.productId());
869
+ }
870
+ }
871
+ }
872
+ ```
873
+
874
+ </details>
875
+
876
+ ---
877
+
878
+ ## 35.3 生産管理と財務会計の連携
879
+
880
+ 生産管理システムと財務会計システムの連携は、製造原価の正確な把握と会計処理に不可欠です。
881
+
882
+ ### 連携の全体像
883
+
884
+ ```plantuml
885
+ @startuml
886
+ title 生産管理と財務会計の連携全体像
887
+
888
+ package "生産管理システム" as production {
889
+ rectangle "購買管理" as purchase
890
+ rectangle "製造管理" as manufacturing
891
+ rectangle "在庫管理" as inventory
892
+ rectangle "原価管理" as cost
893
+ }
894
+
895
+ package "イベントバス" as events {
896
+ collections "検収イベント" as receipt_event
897
+ collections "完成イベント" as complete_event
898
+ collections "棚卸イベント" as inventory_event
899
+ collections "原価イベント" as cost_event
900
+ }
901
+
902
+ package "財務会計システム" as accounting {
903
+ rectangle "自動仕訳" as auto_journal
904
+ rectangle "仕訳管理" as journal
905
+ rectangle "原価会計" as cost_acc
906
+ }
907
+
908
+ purchase --> receipt_event : 検収確定
909
+ manufacturing --> complete_event : 製造完成
910
+ inventory --> inventory_event : 棚卸差異
911
+ cost --> cost_event : 原価計算完了
912
+
913
+ receipt_event --> auto_journal
914
+ complete_event --> auto_journal
915
+ inventory_event --> auto_journal
916
+ cost_event --> cost_acc
917
+
918
+ auto_journal --> journal
919
+
920
+ @enduml
921
+ ```
922
+
923
+ ### 製造原価から仕訳への変換
924
+
925
+ 製造原価は、材料費・労務費・製造間接費の3要素で構成されます。
926
+
927
+ ```plantuml
928
+ @startuml
929
+ title 製造原価の構成と仕訳
930
+
931
+ object "製造原価" as cost {
932
+ 製造オーダ = "WO-2024-001"
933
+ 製品コード = "PRD-001"
934
+ 完成数量 = 100
935
+ }
936
+
937
+ object "材料費" as material {
938
+ 直接材料費 = 50,000円
939
+ 間接材料費 = 5,000円
940
+ }
941
+
942
+ object "労務費" as labor {
943
+ 直接労務費 = 30,000円
944
+ 間接労務費 = 10,000円
945
+ }
946
+
947
+ object "製造間接費" as overhead {
948
+ 配賦額 = 15,000円
949
+ }
950
+
951
+ object "原価仕訳" as journal {
952
+ 借方: 製品 110,000円
953
+ 貸方: 仕掛品 110,000円
954
+ }
955
+
956
+ cost --> material
957
+ cost --> labor
958
+ cost --> overhead
959
+
960
+ material --> journal : 55,000円
961
+ labor --> journal : 40,000円
962
+ overhead --> journal : 15,000円
963
+
964
+ note bottom of journal
965
+ 【製造完成時の仕訳】
966
+ 借方:製品(資産)
967
+ 貸方:仕掛品(資産)
968
+
969
+ 製品原価 = 材料費 + 労務費 + 製造間接費
970
+ = 55,000 + 40,000 + 15,000
971
+ = 110,000円
972
+ end note
973
+
974
+ @enduml
975
+ ```
976
+
977
+ #### 製造原価計算フロー
978
+
979
+ ```plantuml
980
+ @startuml
981
+ title 製造原価計算と仕訳生成フロー
982
+
983
+ |生産管理|
984
+ start
985
+ :製造完成報告;
986
+
987
+ :材料消費実績集計;
988
+ note right
989
+ ・直接材料費
990
+ ・間接材料費
991
+ end note
992
+
993
+ :工数実績集計;
994
+ note right
995
+ ・直接労務費
996
+ ・間接労務費
997
+ end note
998
+
999
+ :製造間接費配賦;
1000
+ note right
1001
+ ・配賦基準で按分
1002
+ (直接作業時間等)
1003
+ end note
1004
+
1005
+ :製品原価計算;
1006
+ :完成原価イベント発行;
1007
+
1008
+ |財務会計|
1009
+ :原価イベント受信;
1010
+ :製造完成仕訳生成;
1011
+ note right
1012
+ 借方: 製品
1013
+ 貸方: 仕掛品
1014
+ end note
1015
+
1016
+ :仕訳登録;
1017
+ :在庫評価額更新;
1018
+
1019
+ stop
1020
+
1021
+ @enduml
1022
+ ```
1023
+
1024
+ <details>
1025
+ <summary>Java 実装例</summary>
1026
+
1027
+ ```java
1028
+ // 製造完成原価イベント
1029
+ public record ManufacturingCompletedEvent(
1030
+ String eventId,
1031
+ Instant timestamp,
1032
+ String workOrderId,
1033
+ String productId,
1034
+ BigDecimal completedQuantity,
1035
+ CostBreakdown costBreakdown
1036
+ ) {
1037
+ public record CostBreakdown(
1038
+ BigDecimal directMaterialCost,
1039
+ BigDecimal indirectMaterialCost,
1040
+ BigDecimal directLaborCost,
1041
+ BigDecimal indirectLaborCost,
1042
+ BigDecimal manufacturingOverhead
1043
+ ) {
1044
+ public BigDecimal getTotalCost() {
1045
+ return directMaterialCost
1046
+ .add(indirectMaterialCost)
1047
+ .add(directLaborCost)
1048
+ .add(indirectLaborCost)
1049
+ .add(manufacturingOverhead);
1050
+ }
1051
+
1052
+ public BigDecimal getMaterialCost() {
1053
+ return directMaterialCost.add(indirectMaterialCost);
1054
+ }
1055
+
1056
+ public BigDecimal getLaborCost() {
1057
+ return directLaborCost.add(indirectLaborCost);
1058
+ }
1059
+ }
1060
+ }
1061
+
1062
+ // 製造原価仕訳サービス
1063
+ @Service
1064
+ public class ManufacturingJournalService {
1065
+ private final JournalRepository journalRepository;
1066
+
1067
+ @EventListener
1068
+ public void handleManufacturingCompleted(ManufacturingCompletedEvent event) {
1069
+ CostBreakdown cost = event.costBreakdown();
1070
+
1071
+ List<JournalLine> lines = new ArrayList<>();
1072
+
1073
+ // 借方:製品
1074
+ lines.add(JournalLine.debit(
1075
+ AccountCode.FINISHED_GOODS,
1076
+ cost.getTotalCost(),
1077
+ "製造完成 " + event.workOrderId()
1078
+ ));
1079
+
1080
+ // 貸方:仕掛品
1081
+ lines.add(JournalLine.credit(
1082
+ AccountCode.WORK_IN_PROCESS,
1083
+ cost.getTotalCost(),
1084
+ "仕掛品振替"
1085
+ ));
1086
+
1087
+ JournalEntry journal = new JournalEntry(
1088
+ generateJournalId(),
1089
+ LocalDate.now(),
1090
+ JournalType.MANUFACTURING_COMPLETION,
1091
+ event.workOrderId(),
1092
+ lines
1093
+ );
1094
+
1095
+ journalRepository.save(journal);
1096
+ }
1097
+ }
1098
+ ```
1099
+
1100
+ </details>
1101
+
1102
+ ### 検収データの会計連携
1103
+
1104
+ 仕入先からの購買品検収時に、買掛金を計上します。
1105
+
1106
+ ```plantuml
1107
+ @startuml
1108
+ title 検収から仕訳への連携
1109
+
1110
+ |生産管理|
1111
+ start
1112
+ :入荷受入;
1113
+ :受入検査;
1114
+
1115
+ if (検査合格?) then (yes)
1116
+ :検収処理;
1117
+ :在庫計上;
1118
+ :検収イベント発行;
1119
+ else (no)
1120
+ :返品処理;
1121
+ stop
1122
+ endif
1123
+
1124
+ |財務会計|
1125
+ :検収イベント受信;
1126
+ :仕入仕訳生成;
1127
+ note right
1128
+ 借方: 材料/仕掛品
1129
+ 借方: 仮払消費税
1130
+ 貸方: 買掛金
1131
+ end note
1132
+
1133
+ :仕訳登録;
1134
+ :買掛金残高更新;
1135
+
1136
+ stop
1137
+
1138
+ @enduml
1139
+ ```
1140
+
1141
+ #### 検収仕訳のパターン
1142
+
1143
+ ```plantuml
1144
+ @startuml
1145
+ title 検収仕訳のパターン
1146
+
1147
+ object "検収データ" as receipt {
1148
+ 検収番号 = "RCV-2024-001"
1149
+ 検収日 = "2024/01/15"
1150
+ 仕入先 = "SUP-001"
1151
+ 検収金額 = 55,000円
1152
+ 消費税額 = 5,000円
1153
+ }
1154
+
1155
+ object "材料検収仕訳" as material_journal {
1156
+ 借方: 材料 50,000円
1157
+ 借方: 仮払消費税 5,000円
1158
+ 貸方: 買掛金 55,000円
1159
+ }
1160
+
1161
+ object "外注検収仕訳" as subcontract_journal {
1162
+ 借方: 外注加工費 50,000円
1163
+ 借方: 仮払消費税 5,000円
1164
+ 貸方: 買掛金 55,000円
1165
+ }
1166
+
1167
+ receipt --> material_journal : 材料の場合
1168
+ receipt --> subcontract_journal : 外注の場合
1169
+
1170
+ @enduml
1171
+ ```
1172
+
1173
+ ### 棚卸差異の会計処理
1174
+
1175
+ 棚卸で発見された差異は、適切に会計処理する必要があります。
1176
+
1177
+ ```plantuml
1178
+ @startuml
1179
+ title 棚卸差異の会計処理フロー
1180
+
1181
+ |生産管理|
1182
+ start
1183
+ :棚卸実施;
1184
+ :帳簿在庫と\n実地在庫の比較;
1185
+
1186
+ if (差異あり?) then (yes)
1187
+ :差異原因分析;
1188
+
1189
+ if (帳簿 > 実地?) then (棚卸減耗)
1190
+ :減耗損計上;
1191
+ :棚卸減耗\nイベント発行;
1192
+ else (帳簿 < 実地)
1193
+ :在庫増加調整;
1194
+ :棚卸増加\nイベント発行;
1195
+ endif
1196
+ else (no)
1197
+ :差異なし\n処理終了;
1198
+ stop
1199
+ endif
1200
+
1201
+ |財務会計|
1202
+ :棚卸イベント受信;
1203
+
1204
+ if (減耗?) then (yes)
1205
+ :減耗仕訳生成;
1206
+ note right
1207
+ 借方: 棚卸減耗損
1208
+ 貸方: 材料/製品
1209
+ end note
1210
+ else (no)
1211
+ :増加仕訳生成;
1212
+ note right
1213
+ 借方: 材料/製品
1214
+ 貸方: 雑収入
1215
+ end note
1216
+ endif
1217
+
1218
+ :仕訳登録;
1219
+ :在庫評価額更新;
1220
+
1221
+ stop
1222
+
1223
+ @enduml
1224
+ ```
1225
+
1226
+ <details>
1227
+ <summary>Java 実装例</summary>
1228
+
1229
+ ```java
1230
+ // 棚卸差異イベント
1231
+ public record InventoryAdjustmentEvent(
1232
+ String eventId,
1233
+ Instant timestamp,
1234
+ String productId,
1235
+ String warehouseId,
1236
+ BigDecimal bookQuantity,
1237
+ BigDecimal actualQuantity,
1238
+ BigDecimal differenceQuantity,
1239
+ BigDecimal unitCost,
1240
+ AdjustmentType type,
1241
+ String reason
1242
+ ) {
1243
+ public enum AdjustmentType {
1244
+ SHRINKAGE, // 減耗(帳簿 > 実地)
1245
+ SURPLUS // 過剰(帳簿 < 実地)
1246
+ }
1247
+
1248
+ public BigDecimal getAdjustmentAmount() {
1249
+ return differenceQuantity.abs().multiply(unitCost);
1250
+ }
1251
+ }
1252
+
1253
+ // 棚卸仕訳サービス
1254
+ @Service
1255
+ public class InventoryAdjustmentJournalService {
1256
+ private final JournalRepository journalRepository;
1257
+
1258
+ @EventListener
1259
+ public void handleInventoryAdjustment(InventoryAdjustmentEvent event) {
1260
+ List<JournalLine> lines = new ArrayList<>();
1261
+
1262
+ if (event.type() == AdjustmentType.SHRINKAGE) {
1263
+ // 減耗の場合
1264
+ lines.add(JournalLine.debit(
1265
+ AccountCode.INVENTORY_SHRINKAGE_LOSS,
1266
+ event.getAdjustmentAmount(),
1267
+ "棚卸減耗 " + event.productId() + " " + event.reason()
1268
+ ));
1269
+ lines.add(JournalLine.credit(
1270
+ determineInventoryAccount(event.productId()),
1271
+ event.getAdjustmentAmount(),
1272
+ "在庫減少"
1273
+ ));
1274
+ } else {
1275
+ // 過剰の場合
1276
+ lines.add(JournalLine.debit(
1277
+ determineInventoryAccount(event.productId()),
1278
+ event.getAdjustmentAmount(),
1279
+ "在庫増加"
1280
+ ));
1281
+ lines.add(JournalLine.credit(
1282
+ AccountCode.MISCELLANEOUS_INCOME,
1283
+ event.getAdjustmentAmount(),
1284
+ "棚卸差益 " + event.productId()
1285
+ ));
1286
+ }
1287
+
1288
+ JournalEntry journal = new JournalEntry(
1289
+ generateJournalId(),
1290
+ LocalDate.now(),
1291
+ JournalType.INVENTORY_ADJUSTMENT,
1292
+ event.eventId(),
1293
+ lines
1294
+ );
1295
+
1296
+ journalRepository.save(journal);
1297
+ }
1298
+
1299
+ private String determineInventoryAccount(String productId) {
1300
+ // 製品か材料かで勘定科目を決定
1301
+ Product product = productRepository.findById(productId).orElseThrow();
1302
+ return switch (product.type()) {
1303
+ case FINISHED_GOODS -> AccountCode.FINISHED_GOODS;
1304
+ case WORK_IN_PROCESS -> AccountCode.WORK_IN_PROCESS;
1305
+ case RAW_MATERIAL -> AccountCode.RAW_MATERIALS;
1306
+ default -> AccountCode.SUPPLIES;
1307
+ };
1308
+ }
1309
+ }
1310
+ ```
1311
+
1312
+ </details>
1313
+
1314
+ ---
1315
+
1316
+ ## 35.4 三システム統合の全体像
1317
+
1318
+ 販売・生産・会計の3システムを統合した全体像を整理します。
1319
+
1320
+ ```plantuml
1321
+ @startuml
1322
+ title 基幹業務システム統合の全体像
1323
+
1324
+ package "販売管理" as sales {
1325
+ rectangle "受注" as order
1326
+ rectangle "出荷" as shipment
1327
+ rectangle "売上" as sales_tx
1328
+ rectangle "請求" as billing
1329
+ rectangle "入金" as receipt
1330
+ }
1331
+
1332
+ package "生産管理" as production {
1333
+ rectangle "生産計画" as plan
1334
+ rectangle "購買" as purchase
1335
+ rectangle "製造" as manufacturing
1336
+ rectangle "在庫" as inventory
1337
+ rectangle "原価" as cost
1338
+ }
1339
+
1340
+ package "財務会計" as accounting {
1341
+ rectangle "仕訳" as journal
1342
+ rectangle "売掛金" as ar
1343
+ rectangle "買掛金" as ap
1344
+ rectangle "在庫資産" as inv_asset
1345
+ }
1346
+
1347
+ ' 販売→生産
1348
+ order --> plan : 受注情報
1349
+ inventory --> shipment : 在庫引当
1350
+
1351
+ ' 販売→会計
1352
+ sales_tx --> journal : 売上仕訳
1353
+ billing --> ar : 請求
1354
+ receipt --> ar : 入金消込
1355
+
1356
+ ' 生産→会計
1357
+ purchase --> ap : 検収仕訳
1358
+ manufacturing --> inv_asset : 完成仕訳
1359
+ cost --> journal : 原価仕訳
1360
+
1361
+ ' 在庫
1362
+ inventory --> inv_asset : 在庫評価
1363
+
1364
+ @enduml
1365
+ ```
1366
+
1367
+ ### イベントカタログ
1368
+
1369
+ ```plantuml
1370
+ @startuml
1371
+ title システム間連携イベントカタログ
1372
+
1373
+ class "販売イベント" as sales_events {
1374
+ OrderConfirmed(受注確定)
1375
+ ShipmentCompleted(出荷完了)
1376
+ SalesCompleted(売上計上)
1377
+ BillingConfirmed(請求確定)
1378
+ PaymentReceived(入金確認)
1379
+ }
1380
+
1381
+ class "生産イベント" as prod_events {
1382
+ DemandRegistered(需要登録)
1383
+ WorkOrderReleased(製造指示)
1384
+ ProductionCompleted(製造完成)
1385
+ ReceiptConfirmed(検収確定)
1386
+ InventoryAdjusted(在庫調整)
1387
+ }
1388
+
1389
+ class "会計イベント" as acc_events {
1390
+ JournalPosted(仕訳計上)
1391
+ BalanceUpdated(残高更新)
1392
+ MonthEndClosed(月次締め)
1393
+ }
1394
+
1395
+ note right of sales_events
1396
+ 【発行元】販売管理
1397
+ 【購読者】生産計画、自動仕訳
1398
+ end note
1399
+
1400
+ note right of prod_events
1401
+ 【発行元】生産管理
1402
+ 【購読者】販売在庫、自動仕訳
1403
+ end note
1404
+
1405
+ note right of acc_events
1406
+ 【発行元】財務会計
1407
+ 【購読者】経営ダッシュボード
1408
+ end note
1409
+
1410
+ @enduml
1411
+ ```
1412
+
1413
+ ---
1414
+
1415
+ ## 35.5 まとめ
1416
+
1417
+ 本章では、基幹業務システムの具体的な連携パターンについて解説しました。
1418
+
1419
+ ### 学んだ連携パターン
1420
+
1421
+ | 連携パターン | 発生元 | 連携先 | 主な処理 |
1422
+ |------------|-------|-------|---------|
1423
+ | 売上→仕訳 | 販売管理 | 財務会計 | 売掛金・売上計上 |
1424
+ | 入金→消込 | 販売管理 | 財務会計 | 売掛金消込 |
1425
+ | 受注→計画 | 販売管理 | 生産管理 | 需要登録・MRP |
1426
+ | 完成→在庫 | 生産管理 | 販売管理 | 在庫同期 |
1427
+ | 検収→仕訳 | 生産管理 | 財務会計 | 買掛金計上 |
1428
+ | 完成→仕訳 | 生産管理 | 財務会計 | 製品・仕掛品振替 |
1429
+ | 棚卸→仕訳 | 生産管理 | 財務会計 | 棚卸損益計上 |
1430
+
1431
+ ### 連携設計のポイント
1432
+
1433
+ 1. **イベント駆動の採用**
1434
+
1435
+ - システム間の疎結合を実現
1436
+ - 非同期処理による可用性向上
1437
+ - 監査証跡の確保
1438
+
1439
+ 2. **自動仕訳パターンの標準化**
1440
+
1441
+ - 取引種別ごとの仕訳パターン定義
1442
+ - 柔軟な条件分岐(商品/顧客グループ)
1443
+ - 例外処理の明確化
1444
+
1445
+ 3. **データ整合性の確保**
1446
+
1447
+ - イベント順序の保証
1448
+ - 補償トランザクション
1449
+ - 定期的な突合処理
1450
+
1451
+ ### 次章の予告
1452
+
1453
+ 第36章では、マスタデータ管理(MDM)について解説します。複数システムで共有されるマスタデータの一元管理方法と、MDMパターンの選択基準を学びます。