@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,1584 @@
1
+ # 第17章:自動仕訳の設計(ORM版)
2
+
3
+ 販売管理システムなどの業務システムから会計システムへの自動仕訳処理を TDD で設計していきます。売上データから仕訳データへの自動変換ルールと、効率的なバッチ処理の設計を行います。
4
+
5
+ JPA 版では、@Entity によるリレーション定義、@EntityGraph による N+1 問題対策、Spring Data JPA Specification による動的クエリを活用します。
6
+
7
+ ---
8
+
9
+ ## 17.1 自動仕訳の概要
10
+
11
+ ### 自動仕訳とは
12
+
13
+ 自動仕訳は、業務システムのトランザクションデータを会計システムの仕訳データへ自動的に変換する機能です。
14
+
15
+ #### 従来のアナログ連携(手入力による連携)
16
+
17
+ 従来の方式では、販売管理システムで発行した売上伝票を紙で経理部門に回送し、経理担当者が仕訳入力を行っていました。
18
+
19
+ ```plantuml
20
+ @startuml
21
+ title 販売管理システムと会計システムの非連携(アナログ連携)
22
+
23
+ skinparam rectangle {
24
+ BackgroundColor White
25
+ BorderColor Black
26
+ }
27
+ skinparam database {
28
+ BackgroundColor #E8F5E9
29
+ BorderColor Black
30
+ }
31
+
32
+ package "営業部門" {
33
+ rectangle "売上入力" as SalesInput <<Screen>>
34
+ database "売上データ" as SalesDB
35
+ rectangle "売上伝票" as SalesVoucher <<Document>>
36
+ }
37
+
38
+ package "経理部門" {
39
+ rectangle "仕訳入力" as JournalInput <<Screen>>
40
+ database "仕訳データ" as JournalDB
41
+ rectangle "損益計算書\n貸借対照表" as FinancialReports <<Document>>
42
+ }
43
+
44
+ SalesInput -down-> SalesDB
45
+ SalesDB -down-> SalesVoucher
46
+
47
+ SalesVoucher -[#red,bold]right-> JournalInput : **紙伝票の回送**\n**(手入力による再登録)**
48
+
49
+ JournalInput -down-> JournalDB
50
+ JournalDB -down-> FinancialReports
51
+
52
+ @enduml
53
+ ```
54
+
55
+ **アナログ連携の問題点:**
56
+
57
+ | 問題 | 説明 |
58
+ |-----|------|
59
+ | 二重入力 | 営業部門と経理部門で同じ情報を入力する作業負荷 |
60
+ | 入力ミス | 手入力によるデータ不整合 |
61
+ | タイムラグ | 紙伝票の回送による情報遅延 |
62
+ | 消費税計算ミス | 手計算による誤り |
63
+ | 勘定科目選択ミス | 担当者の判断ばらつき |
64
+
65
+ #### 自動仕訳によるデジタル連携
66
+
67
+ 自動仕訳処理を導入することで、売上データから仕訳データへの変換を自動化します。
68
+
69
+ ```plantuml
70
+ @startuml
71
+ title 自動仕訳処理のデータの流れ
72
+
73
+ skinparam rectangle {
74
+ BackgroundColor White
75
+ BorderColor Black
76
+ }
77
+ skinparam database {
78
+ BackgroundColor #E8F5E9
79
+ BorderColor Black
80
+ }
81
+
82
+ package "営業部門" {
83
+ rectangle "売上入力" as SalesInput <<Screen>>
84
+ database "売上データ" as SalesDB
85
+ rectangle "売上\nチェックリスト" as SalesList <<Document>>
86
+ }
87
+
88
+ package "自動仕訳機能" #F1F8E9 {
89
+ database "自動仕訳\nパターンマスタ" as PatternMaster
90
+ rectangle "自動仕訳処理" as AutoJournalProcess <<Process>> #4CAF50
91
+ database "自動仕訳データ" as AutoJournalDB
92
+ database "エラーデータ" as ErrorDB
93
+ rectangle "エラーリスト" as ErrorList <<Document>>
94
+ rectangle "自動仕訳\nチェックリスト" as AutoJournalList <<Document>>
95
+
96
+ rectangle "転記処理" as PostingProcess <<Process>> #4CAF50
97
+ circle " " as Switch
98
+ }
99
+
100
+ package "経理部門" {
101
+ rectangle "仕訳入力" as JournalInput <<Screen>>
102
+ database "仕訳データ" as JournalDB
103
+ }
104
+
105
+ SalesInput -down-> SalesDB
106
+ SalesDB -down-> SalesList
107
+
108
+ SalesDB -right-> AutoJournalProcess
109
+ PatternMaster -down-> AutoJournalProcess
110
+ AutoJournalProcess -right-> AutoJournalDB
111
+ AutoJournalProcess -down-> ErrorDB
112
+ ErrorDB -down-> ErrorList
113
+ AutoJournalDB -down-> AutoJournalList
114
+
115
+ AutoJournalDB -right-> Switch
116
+ note top of Switch : (転記指示)
117
+ Switch -right-> PostingProcess
118
+ PostingProcess -right-> JournalDB
119
+
120
+ JournalInput -down-> JournalDB
121
+
122
+ @enduml
123
+ ```
124
+
125
+ ### 自動仕訳処理の流れ
126
+
127
+ ```plantuml
128
+ @startuml
129
+
130
+ title 自動仕訳処理フロー
131
+
132
+ |営業部門|
133
+ start
134
+ :売上登録;
135
+
136
+ |自動仕訳処理|
137
+ :売上データ抽出;
138
+ note right
139
+ 未処理の売上データを
140
+ 抽出条件に基づいて取得
141
+ end note
142
+
143
+ :パターンマッチング;
144
+ note right
145
+ 商品区分・顧客区分から
146
+ 適用する仕訳パターンを決定
147
+ end note
148
+
149
+ if (パターン適合?) then (はい)
150
+ :仕訳データ生成;
151
+ :自動仕訳データに保存;
152
+ else (いいえ)
153
+ :エラーデータに保存;
154
+ :エラーリスト出力;
155
+ stop
156
+ endif
157
+
158
+ :売上データに処理済フラグ設定;
159
+
160
+ |経理部門|
161
+ :自動仕訳チェックリスト確認;
162
+
163
+ if (確認OK?) then (はい)
164
+ :転記処理実行;
165
+ :仕訳データに登録;
166
+ else (いいえ)
167
+ :修正・再処理;
168
+ stop
169
+ endif
170
+
171
+ :仕訳完了;
172
+ stop
173
+
174
+ @enduml
175
+ ```
176
+
177
+ ### 売上伝票から仕訳伝票への変換例
178
+
179
+ #### 売上伝票(ビジネス上の事実)
180
+
181
+ | 項目 | 値 |
182
+ |-----|-----|
183
+ | 伝票日付 | 2024/04/01 |
184
+ | 顧客 | DBMフード新宿 |
185
+ | 明細1 | いちご蒸缶 1,000個 × 1,000円 = 1,000,000円 |
186
+ | 明細2 | アスパラ 200個 × 10,000円 = 2,000,000円 |
187
+ | 明細3 | さざえのエスカルゴ 1,500個 × 100円 = 150,000円 |
188
+ | 合計 | 3,150,000円(税抜)+ 消費税 315,000円 = 3,465,000円(税込) |
189
+
190
+ #### 仕訳伝票(会計上の記録)
191
+
192
+ | 借方科目 | 借方金額 | 貸方科目 | 貸方金額 | 摘要 |
193
+ |---------|---------|---------|---------|-----|
194
+ | 売掛金/DBMフード本社 | 3,465,000 | | | 売上計上 |
195
+ | | | 売上加工品/DBMフード新宿 | 1,150,000 | いちご蒸缶・さざえ |
196
+ | | | 売上生鮮品/DBMフード新宿 | 2,000,000 | アスパラ |
197
+ | | | 仮受消費税 | 315,000 | 消費税10% |
198
+
199
+ ---
200
+
201
+ ## 17.2 自動仕訳テーブルの設計
202
+
203
+ ### フラグ管理方式と日付管理方式
204
+
205
+ 売上データの処理状態を管理する方式には、2つのアプローチがあります。
206
+
207
+ #### フラグ管理方式
208
+
209
+ - **メリット**:シンプルな実装、処理状態が明確
210
+ - **デメリット**:再処理時にフラグリセットが必要、大量データでの更新負荷
211
+
212
+ #### 日付管理方式
213
+
214
+ - **メリット**:差分処理が容易、再処理時の柔軟性
215
+ - **デメリット**:管理テーブルが必要、更新日時の整合性管理が必要
216
+
217
+ ### セット中心のアプリケーション設計
218
+
219
+ 大量データを効率的に処理するには、ループ処理よりもセット中心処理が有効です。
220
+
221
+ | 方式 | 処理方法 | DB操作回数 |
222
+ |-----|---------|-----------|
223
+ | ループ処理 | 売上データを1件ずつ読み込み、各売上に対してパターンマスタを検索、1件ずつ仕訳データを挿入 | N×M回 |
224
+ | セット中心処理 | パターンマスタを1件読み込み、該当パターンの売上データを一括INSERT | M回(パターン数) |
225
+
226
+ ### 自動仕訳関連テーブルの ER 図
227
+
228
+ ```plantuml
229
+ @startuml
230
+
231
+ entity "自動仕訳パターンマスタ" as AutoJournalPattern {
232
+ * **パターンコード**: VARCHAR(10) <<PK>>
233
+ --
234
+ * **パターン名**: VARCHAR(50)
235
+ * **商品グループ**: VARCHAR(10)
236
+ * **顧客グループ**: VARCHAR(10)
237
+ * **売上区分**: VARCHAR(2)
238
+ * **借方勘定科目コード**: VARCHAR(5) <<FK>>
239
+ * **借方補助科目設定**: VARCHAR(20)
240
+ * **貸方勘定科目コード**: VARCHAR(5) <<FK>>
241
+ * **貸方補助科目設定**: VARCHAR(20)
242
+ * **返品時借方科目コード**: VARCHAR(5)
243
+ * **返品時貸方科目コード**: VARCHAR(5)
244
+ * **消費税処理区分**: VARCHAR(2)
245
+ * **有効開始日**: DATE
246
+ * **有効終了日**: DATE
247
+ * **優先順位**: INTEGER
248
+ }
249
+
250
+ entity "自動仕訳データ" as AutoJournal {
251
+ * **自動仕訳番号**: VARCHAR(15) <<PK>>
252
+ --
253
+ * **売上番号**: VARCHAR(10) <<FK>>
254
+ * **売上行番号**: SMALLINT
255
+ * **パターンコード**: VARCHAR(10) <<FK>>
256
+ * **起票日**: DATE
257
+ * **仕訳行貸借区分**: VARCHAR(1)
258
+ * **勘定科目コード**: VARCHAR(5) <<FK>>
259
+ * **補助科目コード**: VARCHAR(10)
260
+ * **部門コード**: VARCHAR(5)
261
+ * **仕訳金額**: DECIMAL(14,2)
262
+ * **消費税額**: DECIMAL(14,2)
263
+ * **処理ステータス**: 自動仕訳ステータス
264
+ * **転記済フラグ**: SMALLINT
265
+ }
266
+
267
+ entity "自動仕訳処理履歴" as AutoJournalHistory {
268
+ * **処理番号**: VARCHAR(15) <<PK>>
269
+ --
270
+ * **処理日時**: TIMESTAMP
271
+ * **処理対象開始日**: DATE
272
+ * **処理対象終了日**: DATE
273
+ * **処理件数**: INTEGER
274
+ * **成功件数**: INTEGER
275
+ * **エラー件数**: INTEGER
276
+ * **処理金額合計**: DECIMAL(15,2)
277
+ }
278
+
279
+ entity "勘定科目マスタ" as Account {
280
+ * **勘定科目コード** <<PK>>
281
+ --
282
+ 勘定科目名
283
+ ...
284
+ }
285
+
286
+ AutoJournalPattern }o--|| Account : 借方科目
287
+ AutoJournalPattern }o--|| Account : 貸方科目
288
+ AutoJournal }o--|| AutoJournalPattern
289
+ AutoJournal }o--|| Account
290
+
291
+ @enduml
292
+ ```
293
+
294
+ ---
295
+
296
+ ## 17.3 MyBatis 版との実装比較
297
+
298
+ ### データアクセス層の比較
299
+
300
+ | 機能 | MyBatis 版 | JPA 版 |
301
+ |-----|-----------|--------|
302
+ | ENUM 型変換 | TypeHandler | AttributeConverter |
303
+ | 関連エンティティ取得 | resultMap の association | @ManyToOne |
304
+ | N+1 問題対策 | JOIN を含む SQL | @EntityGraph |
305
+ | 動的クエリ | XML の `<if>` タグ | Specification パターン |
306
+ | バッチ INSERT | foreach 句 | saveAll() |
307
+ | 有効期間検索 | WHERE 句で条件指定 | JPQL で条件指定 |
308
+
309
+ ### Flyway マイグレーション
310
+
311
+ <details>
312
+ <summary>V007__create_auto_journal_tables.sql</summary>
313
+
314
+ ```sql
315
+ -- 自動仕訳処理ステータス
316
+ CREATE TYPE 自動仕訳ステータス AS ENUM ('処理待ち', '処理中', '処理完了', '転記済', 'エラー');
317
+
318
+ -- 自動仕訳パターンマスタ
319
+ CREATE TABLE "自動仕訳パターンマスタ" (
320
+ "パターンコード" VARCHAR(10) PRIMARY KEY,
321
+ "パターン名" VARCHAR(50) NOT NULL,
322
+ "商品グループ" VARCHAR(10) DEFAULT 'ALL',
323
+ "顧客グループ" VARCHAR(10) DEFAULT 'ALL',
324
+ "売上区分" VARCHAR(2) DEFAULT '01',
325
+ "借方勘定科目コード" VARCHAR(5) NOT NULL,
326
+ "借方補助科目設定" VARCHAR(20),
327
+ "貸方勘定科目コード" VARCHAR(5) NOT NULL,
328
+ "貸方補助科目設定" VARCHAR(20),
329
+ "返品時借方科目コード" VARCHAR(5),
330
+ "返品時貸方科目コード" VARCHAR(5),
331
+ "消費税処理区分" VARCHAR(2) DEFAULT '01',
332
+ "有効開始日" DATE DEFAULT CURRENT_DATE,
333
+ "有効終了日" DATE DEFAULT '9999-12-31',
334
+ "優先順位" INTEGER DEFAULT 100,
335
+ "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
336
+ "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
337
+ CONSTRAINT "fk_自動仕訳パターン_借方科目"
338
+ FOREIGN KEY ("借方勘定科目コード") REFERENCES "勘定科目マスタ"("勘定科目コード"),
339
+ CONSTRAINT "fk_自動仕訳パターン_貸方科目"
340
+ FOREIGN KEY ("貸方勘定科目コード") REFERENCES "勘定科目マスタ"("勘定科目コード")
341
+ );
342
+
343
+ -- 自動仕訳データ
344
+ CREATE TABLE "自動仕訳データ" (
345
+ "自動仕訳番号" VARCHAR(15) PRIMARY KEY,
346
+ "売上番号" VARCHAR(10) NOT NULL,
347
+ "売上行番号" SMALLINT NOT NULL,
348
+ "パターンコード" VARCHAR(10) NOT NULL,
349
+ "起票日" DATE NOT NULL,
350
+ "仕訳行貸借区分" 仕訳行貸借区分 NOT NULL,
351
+ "勘定科目コード" VARCHAR(5) NOT NULL,
352
+ "補助科目コード" VARCHAR(10),
353
+ "部門コード" VARCHAR(5),
354
+ "仕訳金額" DECIMAL(14,2) NOT NULL,
355
+ "消費税額" DECIMAL(14,2) DEFAULT 0,
356
+ "処理ステータス" 自動仕訳ステータス DEFAULT '処理待ち' NOT NULL,
357
+ "転記済フラグ" SMALLINT DEFAULT 0,
358
+ "転記日" DATE,
359
+ "仕訳伝票番号" VARCHAR(10),
360
+ "エラーコード" VARCHAR(10),
361
+ "エラーメッセージ" VARCHAR(200),
362
+ "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL,
363
+ "更新日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL
364
+ );
365
+
366
+ -- 自動仕訳処理履歴
367
+ CREATE TABLE "自動仕訳処理履歴" (
368
+ "処理番号" VARCHAR(15) PRIMARY KEY,
369
+ "処理日時" TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
370
+ "処理対象開始日" DATE NOT NULL,
371
+ "処理対象終了日" DATE NOT NULL,
372
+ "処理件数" INTEGER DEFAULT 0,
373
+ "成功件数" INTEGER DEFAULT 0,
374
+ "エラー件数" INTEGER DEFAULT 0,
375
+ "処理金額合計" DECIMAL(15,2) DEFAULT 0,
376
+ "処理者" VARCHAR(50),
377
+ "備考" TEXT,
378
+ "作成日時" TIMESTAMP DEFAULT CURRENT_TIMESTAMP NOT NULL
379
+ );
380
+
381
+ -- インデックス
382
+ CREATE INDEX "idx_自動仕訳パターン_優先順位" ON "自動仕訳パターンマスタ"("優先順位");
383
+ CREATE INDEX "idx_自動仕訳_売上番号" ON "自動仕訳データ"("売上番号");
384
+ CREATE INDEX "idx_自動仕訳_処理ステータス" ON "自動仕訳データ"("処理ステータス");
385
+ CREATE INDEX "idx_自動仕訳_転記済フラグ" ON "自動仕訳データ"("転記済フラグ");
386
+ ```
387
+
388
+ </details>
389
+
390
+ ---
391
+
392
+ ## 17.4 JPA エンティティの実装
393
+
394
+ ### 自動仕訳ステータス Enum と AttributeConverter
395
+
396
+ <details>
397
+ <summary>AutoJournalStatus.java</summary>
398
+
399
+ ```java
400
+ package com.example.accounting.domain.model.autojournal;
401
+
402
+ import lombok.Getter;
403
+ import lombok.RequiredArgsConstructor;
404
+
405
+ @Getter
406
+ @RequiredArgsConstructor
407
+ public enum AutoJournalStatus {
408
+ PENDING("処理待ち"),
409
+ PROCESSING("処理中"),
410
+ COMPLETED("処理完了"),
411
+ POSTED("転記済"),
412
+ ERROR("エラー");
413
+
414
+ private final String displayName;
415
+
416
+ public static AutoJournalStatus fromDisplayName(String displayName) {
417
+ for (AutoJournalStatus status : values()) {
418
+ if (status.displayName.equals(displayName)) {
419
+ return status;
420
+ }
421
+ }
422
+ throw new IllegalArgumentException("Unknown auto journal status: " + displayName);
423
+ }
424
+ }
425
+ ```
426
+
427
+ </details>
428
+
429
+ <details>
430
+ <summary>AutoJournalStatusConverter.java</summary>
431
+
432
+ ```java
433
+ package com.example.accounting.infrastructure.persistence.converter;
434
+
435
+ import com.example.accounting.domain.model.autojournal.AutoJournalStatus;
436
+ import jakarta.persistence.AttributeConverter;
437
+ import jakarta.persistence.Converter;
438
+
439
+ /**
440
+ * 自動仕訳ステータス ↔ PostgreSQL ENUM 変換
441
+ * JPA では AttributeConverter を使用して ENUM 型を変換する
442
+ */
443
+ @Converter(autoApply = true)
444
+ public class AutoJournalStatusConverter implements AttributeConverter<AutoJournalStatus, String> {
445
+
446
+ @Override
447
+ public String convertToDatabaseColumn(AutoJournalStatus attribute) {
448
+ return attribute != null ? attribute.getDisplayName() : null;
449
+ }
450
+
451
+ @Override
452
+ public AutoJournalStatus convertToEntityAttribute(String dbData) {
453
+ return dbData != null ? AutoJournalStatus.fromDisplayName(dbData) : null;
454
+ }
455
+ }
456
+ ```
457
+
458
+ </details>
459
+
460
+ ### 自動仕訳パターンマスタ Entity
461
+
462
+ <details>
463
+ <summary>AutoJournalPattern.java</summary>
464
+
465
+ ```java
466
+ package com.example.accounting.domain.model.autojournal;
467
+
468
+ import com.example.accounting.domain.model.account.Account;
469
+ import jakarta.persistence.*;
470
+ import lombok.*;
471
+
472
+ import java.time.LocalDate;
473
+ import java.time.LocalDateTime;
474
+
475
+ /**
476
+ * 自動仕訳パターンマスタ Entity
477
+ *
478
+ * JPA 版では @ManyToOne で勘定科目との関連を定義
479
+ * MyBatis 版では resultMap で手動マッピングしていたが、
480
+ * JPA では @JoinColumn で外部キー関連を自動的に解決
481
+ */
482
+ @Entity
483
+ @Table(name = "自動仕訳パターンマスタ")
484
+ @Data
485
+ @Builder
486
+ @NoArgsConstructor
487
+ @AllArgsConstructor
488
+ public class AutoJournalPattern {
489
+
490
+ @Id
491
+ @Column(name = "パターンコード", length = 10)
492
+ private String patternCode;
493
+
494
+ @Column(name = "パターン名", length = 50, nullable = false)
495
+ private String patternName;
496
+
497
+ @Column(name = "商品グループ", length = 10)
498
+ private String productGroup;
499
+
500
+ @Column(name = "顧客グループ", length = 10)
501
+ private String customerGroup;
502
+
503
+ @Column(name = "売上区分", length = 2)
504
+ private String salesType;
505
+
506
+ /**
507
+ * 借方勘定科目との関連
508
+ * @ManyToOne で遅延ロード(LAZY)を指定
509
+ * N+1 問題が発生する場合は @EntityGraph で対策
510
+ */
511
+ @ManyToOne(fetch = FetchType.LAZY)
512
+ @JoinColumn(name = "借方勘定科目コード", referencedColumnName = "勘定科目コード",
513
+ insertable = false, updatable = false)
514
+ private Account debitAccount;
515
+
516
+ @Column(name = "借方勘定科目コード", length = 5, nullable = false)
517
+ private String debitAccountCode;
518
+
519
+ @Column(name = "借方補助科目設定", length = 20)
520
+ private String debitSubAccountSetting;
521
+
522
+ /**
523
+ * 貸方勘定科目との関連
524
+ */
525
+ @ManyToOne(fetch = FetchType.LAZY)
526
+ @JoinColumn(name = "貸方勘定科目コード", referencedColumnName = "勘定科目コード",
527
+ insertable = false, updatable = false)
528
+ private Account creditAccount;
529
+
530
+ @Column(name = "貸方勘定科目コード", length = 5, nullable = false)
531
+ private String creditAccountCode;
532
+
533
+ @Column(name = "貸方補助科目設定", length = 20)
534
+ private String creditSubAccountSetting;
535
+
536
+ @Column(name = "返品時借方科目コード", length = 5)
537
+ private String returnDebitAccountCode;
538
+
539
+ @Column(name = "返品時貸方科目コード", length = 5)
540
+ private String returnCreditAccountCode;
541
+
542
+ @Column(name = "消費税処理区分", length = 2)
543
+ private String taxProcessingType;
544
+
545
+ @Column(name = "有効開始日")
546
+ private LocalDate validFrom;
547
+
548
+ @Column(name = "有効終了日")
549
+ private LocalDate validTo;
550
+
551
+ @Column(name = "優先順位")
552
+ private Integer priority;
553
+
554
+ @Column(name = "作成日時", nullable = false, updatable = false)
555
+ private LocalDateTime createdAt;
556
+
557
+ @Column(name = "更新日時", nullable = false)
558
+ private LocalDateTime updatedAt;
559
+
560
+ /**
561
+ * 指定日付に有効かどうかを判定
562
+ */
563
+ public boolean isValidAt(LocalDate date) {
564
+ return !date.isBefore(validFrom) && !date.isAfter(validTo);
565
+ }
566
+
567
+ /**
568
+ * 商品グループと顧客グループにマッチするか判定
569
+ */
570
+ public boolean matches(String productGroup, String customerGroup) {
571
+ boolean productMatch = "ALL".equals(this.productGroup) ||
572
+ this.productGroup.equals(productGroup);
573
+ boolean customerMatch = "ALL".equals(this.customerGroup) ||
574
+ this.customerGroup.equals(customerGroup);
575
+ return productMatch && customerMatch;
576
+ }
577
+
578
+ @PrePersist
579
+ protected void onCreate() {
580
+ LocalDateTime now = LocalDateTime.now();
581
+ this.createdAt = now;
582
+ this.updatedAt = now;
583
+ if (this.validFrom == null) {
584
+ this.validFrom = LocalDate.now();
585
+ }
586
+ if (this.validTo == null) {
587
+ this.validTo = LocalDate.of(9999, 12, 31);
588
+ }
589
+ if (this.priority == null) {
590
+ this.priority = 100;
591
+ }
592
+ }
593
+
594
+ @PreUpdate
595
+ protected void onUpdate() {
596
+ this.updatedAt = LocalDateTime.now();
597
+ }
598
+ }
599
+ ```
600
+
601
+ </details>
602
+
603
+ ### 自動仕訳データ Entity
604
+
605
+ <details>
606
+ <summary>AutoJournalEntry.java</summary>
607
+
608
+ ```java
609
+ package com.example.accounting.domain.model.autojournal;
610
+
611
+ import com.example.accounting.domain.model.account.Account;
612
+ import com.example.accounting.domain.model.department.Department;
613
+ import com.example.accounting.domain.model.journal.DebitCreditType;
614
+ import com.example.accounting.infrastructure.persistence.converter.AutoJournalStatusConverter;
615
+ import com.example.accounting.infrastructure.persistence.converter.DebitCreditTypeConverter;
616
+ import jakarta.persistence.*;
617
+ import lombok.*;
618
+
619
+ import java.math.BigDecimal;
620
+ import java.time.LocalDate;
621
+ import java.time.LocalDateTime;
622
+
623
+ /**
624
+ * 自動仕訳データ Entity
625
+ *
626
+ * JPA 版では @ManyToOne で関連エンティティを定義
627
+ * 状態管理には Enum + AttributeConverter を使用
628
+ */
629
+ @Entity
630
+ @Table(name = "自動仕訳データ")
631
+ @Data
632
+ @Builder
633
+ @NoArgsConstructor
634
+ @AllArgsConstructor
635
+ public class AutoJournalEntry {
636
+
637
+ @Id
638
+ @Column(name = "自動仕訳番号", length = 15)
639
+ private String autoJournalNumber;
640
+
641
+ @Column(name = "売上番号", length = 10, nullable = false)
642
+ private String salesNumber;
643
+
644
+ @Column(name = "売上行番号", nullable = false)
645
+ private Integer salesLineNumber;
646
+
647
+ /**
648
+ * パターンマスタとの関連
649
+ */
650
+ @ManyToOne(fetch = FetchType.LAZY)
651
+ @JoinColumn(name = "パターンコード", referencedColumnName = "パターンコード",
652
+ insertable = false, updatable = false)
653
+ private AutoJournalPattern pattern;
654
+
655
+ @Column(name = "パターンコード", length = 10, nullable = false)
656
+ private String patternCode;
657
+
658
+ @Column(name = "起票日", nullable = false)
659
+ private LocalDate postingDate;
660
+
661
+ /**
662
+ * 仕訳行貸借区分
663
+ * AttributeConverter で PostgreSQL ENUM と Java Enum を変換
664
+ */
665
+ @Convert(converter = DebitCreditTypeConverter.class)
666
+ @Column(name = "仕訳行貸借区分", nullable = false, columnDefinition = "仕訳行貸借区分")
667
+ private DebitCreditType debitCreditType;
668
+
669
+ /**
670
+ * 勘定科目との関連
671
+ */
672
+ @ManyToOne(fetch = FetchType.LAZY)
673
+ @JoinColumn(name = "勘定科目コード", referencedColumnName = "勘定科目コード",
674
+ insertable = false, updatable = false)
675
+ private Account account;
676
+
677
+ @Column(name = "勘定科目コード", length = 5, nullable = false)
678
+ private String accountCode;
679
+
680
+ @Column(name = "補助科目コード", length = 10)
681
+ private String subAccountCode;
682
+
683
+ /**
684
+ * 部門との関連
685
+ */
686
+ @ManyToOne(fetch = FetchType.LAZY)
687
+ @JoinColumn(name = "部門コード", referencedColumnName = "部門コード",
688
+ insertable = false, updatable = false)
689
+ private Department department;
690
+
691
+ @Column(name = "部門コード", length = 5)
692
+ private String departmentCode;
693
+
694
+ @Column(name = "仕訳金額", nullable = false, precision = 14, scale = 2)
695
+ private BigDecimal amount;
696
+
697
+ @Column(name = "消費税額", precision = 14, scale = 2)
698
+ private BigDecimal taxAmount;
699
+
700
+ /**
701
+ * 処理ステータス
702
+ * AttributeConverter で PostgreSQL ENUM と Java Enum を変換
703
+ */
704
+ @Convert(converter = AutoJournalStatusConverter.class)
705
+ @Column(name = "処理ステータス", nullable = false, columnDefinition = "自動仕訳ステータス")
706
+ private AutoJournalStatus status;
707
+
708
+ @Column(name = "転記済フラグ")
709
+ private Boolean postedFlag;
710
+
711
+ @Column(name = "転記日")
712
+ private LocalDate postedDate;
713
+
714
+ @Column(name = "仕訳伝票番号", length = 10)
715
+ private String journalVoucherNumber;
716
+
717
+ @Column(name = "エラーコード", length = 10)
718
+ private String errorCode;
719
+
720
+ @Column(name = "エラーメッセージ", length = 200)
721
+ private String errorMessage;
722
+
723
+ @Column(name = "作成日時", nullable = false, updatable = false)
724
+ private LocalDateTime createdAt;
725
+
726
+ @Column(name = "更新日時", nullable = false)
727
+ private LocalDateTime updatedAt;
728
+
729
+ @PrePersist
730
+ protected void onCreate() {
731
+ LocalDateTime now = LocalDateTime.now();
732
+ this.createdAt = now;
733
+ this.updatedAt = now;
734
+ if (this.status == null) {
735
+ this.status = AutoJournalStatus.PENDING;
736
+ }
737
+ if (this.postedFlag == null) {
738
+ this.postedFlag = false;
739
+ }
740
+ if (this.taxAmount == null) {
741
+ this.taxAmount = BigDecimal.ZERO;
742
+ }
743
+ }
744
+
745
+ @PreUpdate
746
+ protected void onUpdate() {
747
+ this.updatedAt = LocalDateTime.now();
748
+ }
749
+ }
750
+ ```
751
+
752
+ </details>
753
+
754
+ ### 自動仕訳処理履歴 Entity
755
+
756
+ <details>
757
+ <summary>AutoJournalHistory.java</summary>
758
+
759
+ ```java
760
+ package com.example.accounting.domain.model.autojournal;
761
+
762
+ import jakarta.persistence.*;
763
+ import lombok.*;
764
+
765
+ import java.math.BigDecimal;
766
+ import java.time.LocalDate;
767
+ import java.time.LocalDateTime;
768
+
769
+ /**
770
+ * 自動仕訳処理履歴 Entity
771
+ */
772
+ @Entity
773
+ @Table(name = "自動仕訳処理履歴")
774
+ @Data
775
+ @Builder
776
+ @NoArgsConstructor
777
+ @AllArgsConstructor
778
+ public class AutoJournalHistory {
779
+
780
+ @Id
781
+ @Column(name = "処理番号", length = 15)
782
+ private String processNumber;
783
+
784
+ @Column(name = "処理日時", nullable = false)
785
+ private LocalDateTime processDateTime;
786
+
787
+ @Column(name = "処理対象開始日", nullable = false)
788
+ private LocalDate targetFromDate;
789
+
790
+ @Column(name = "処理対象終了日", nullable = false)
791
+ private LocalDate targetToDate;
792
+
793
+ @Column(name = "処理件数")
794
+ private Integer totalCount;
795
+
796
+ @Column(name = "成功件数")
797
+ private Integer successCount;
798
+
799
+ @Column(name = "エラー件数")
800
+ private Integer errorCount;
801
+
802
+ @Column(name = "処理金額合計", precision = 15, scale = 2)
803
+ private BigDecimal totalAmount;
804
+
805
+ @Column(name = "処理者", length = 50)
806
+ private String processedBy;
807
+
808
+ @Column(name = "備考", columnDefinition = "TEXT")
809
+ private String remarks;
810
+
811
+ @Column(name = "作成日時", nullable = false, updatable = false)
812
+ private LocalDateTime createdAt;
813
+
814
+ @PrePersist
815
+ protected void onCreate() {
816
+ this.createdAt = LocalDateTime.now();
817
+ if (this.totalCount == null) {
818
+ this.totalCount = 0;
819
+ }
820
+ if (this.successCount == null) {
821
+ this.successCount = 0;
822
+ }
823
+ if (this.errorCount == null) {
824
+ this.errorCount = 0;
825
+ }
826
+ if (this.totalAmount == null) {
827
+ this.totalAmount = BigDecimal.ZERO;
828
+ }
829
+ }
830
+ }
831
+ ```
832
+
833
+ </details>
834
+
835
+ ---
836
+
837
+ ## 17.5 TDD によるパターンマッチングの実装
838
+
839
+ ### パターンマッチングのテスト
840
+
841
+ <details>
842
+ <summary>AutoJournalPatternTest.java</summary>
843
+
844
+ ```java
845
+ package com.example.accounting.domain.model.autojournal;
846
+
847
+ import org.junit.jupiter.api.DisplayName;
848
+ import org.junit.jupiter.api.Nested;
849
+ import org.junit.jupiter.api.Test;
850
+
851
+ import java.time.LocalDate;
852
+
853
+ import static org.assertj.core.api.Assertions.*;
854
+
855
+ @DisplayName("自動仕訳パターンのテスト")
856
+ class AutoJournalPatternTest {
857
+
858
+ @Nested
859
+ @DisplayName("パターンマッチング")
860
+ class PatternMatchingTest {
861
+
862
+ @Test
863
+ @DisplayName("商品グループALLは全ての商品グループにマッチする")
864
+ void shouldMatchAllProductGroups() {
865
+ // Given: 商品グループALLのパターン
866
+ var pattern = AutoJournalPattern.builder()
867
+ .patternCode("P001")
868
+ .productGroup("ALL")
869
+ .customerGroup("ALL")
870
+ .build();
871
+
872
+ // When & Then
873
+ assertThat(pattern.matches("加工品", "一般")).isTrue();
874
+ assertThat(pattern.matches("生鮮品", "一般")).isTrue();
875
+ assertThat(pattern.matches("雑貨", "特約店")).isTrue();
876
+ }
877
+
878
+ @Test
879
+ @DisplayName("特定の商品グループのみにマッチする")
880
+ void shouldMatchSpecificProductGroup() {
881
+ // Given: 加工品専用パターン
882
+ var pattern = AutoJournalPattern.builder()
883
+ .patternCode("P002")
884
+ .productGroup("加工品")
885
+ .customerGroup("ALL")
886
+ .build();
887
+
888
+ // When & Then
889
+ assertThat(pattern.matches("加工品", "一般")).isTrue();
890
+ assertThat(pattern.matches("生鮮品", "一般")).isFalse();
891
+ }
892
+
893
+ @Test
894
+ @DisplayName("商品グループと顧客グループの両方でマッチングする")
895
+ void shouldMatchBothProductAndCustomerGroup() {
896
+ // Given: 特定の組み合わせパターン
897
+ var pattern = AutoJournalPattern.builder()
898
+ .patternCode("P003")
899
+ .productGroup("加工品")
900
+ .customerGroup("特約店")
901
+ .build();
902
+
903
+ // When & Then
904
+ assertThat(pattern.matches("加工品", "特約店")).isTrue();
905
+ assertThat(pattern.matches("加工品", "一般")).isFalse();
906
+ assertThat(pattern.matches("生鮮品", "特約店")).isFalse();
907
+ }
908
+ }
909
+
910
+ @Nested
911
+ @DisplayName("有効期間チェック")
912
+ class ValidityCheckTest {
913
+
914
+ @Test
915
+ @DisplayName("有効期間内の日付でtrueを返す")
916
+ void shouldReturnTrueForValidDate() {
917
+ // Given
918
+ var pattern = AutoJournalPattern.builder()
919
+ .patternCode("P001")
920
+ .validFrom(LocalDate.of(2024, 1, 1))
921
+ .validTo(LocalDate.of(2024, 12, 31))
922
+ .build();
923
+
924
+ // When & Then
925
+ assertThat(pattern.isValidAt(LocalDate.of(2024, 6, 15))).isTrue();
926
+ assertThat(pattern.isValidAt(LocalDate.of(2024, 1, 1))).isTrue();
927
+ assertThat(pattern.isValidAt(LocalDate.of(2024, 12, 31))).isTrue();
928
+ }
929
+
930
+ @Test
931
+ @DisplayName("有効期間外の日付でfalseを返す")
932
+ void shouldReturnFalseForInvalidDate() {
933
+ // Given
934
+ var pattern = AutoJournalPattern.builder()
935
+ .patternCode("P001")
936
+ .validFrom(LocalDate.of(2024, 1, 1))
937
+ .validTo(LocalDate.of(2024, 12, 31))
938
+ .build();
939
+
940
+ // When & Then
941
+ assertThat(pattern.isValidAt(LocalDate.of(2023, 12, 31))).isFalse();
942
+ assertThat(pattern.isValidAt(LocalDate.of(2025, 1, 1))).isFalse();
943
+ }
944
+ }
945
+ }
946
+ ```
947
+
948
+ </details>
949
+
950
+ ### リポジトリのテスト
951
+
952
+ <details>
953
+ <summary>AutoJournalPatternRepositoryTest.java</summary>
954
+
955
+ ```java
956
+ package com.example.accounting.infrastructure.persistence;
957
+
958
+ import com.example.accounting.domain.model.autojournal.*;
959
+ import com.example.accounting.infrastructure.persistence.repository.AutoJournalPatternJpaRepository;
960
+ import org.junit.jupiter.api.DisplayName;
961
+ import org.junit.jupiter.api.Nested;
962
+ import org.junit.jupiter.api.Test;
963
+ import org.springframework.beans.factory.annotation.Autowired;
964
+ import org.springframework.boot.test.autoconfigure.jdbc.AutoConfigureTestDatabase;
965
+ import org.springframework.boot.test.autoconfigure.orm.jpa.DataJpaTest;
966
+ import org.springframework.test.context.DynamicPropertyRegistry;
967
+ import org.springframework.test.context.DynamicPropertySource;
968
+ import org.testcontainers.containers.PostgreSQLContainer;
969
+ import org.testcontainers.junit.jupiter.Container;
970
+ import org.testcontainers.junit.jupiter.Testcontainers;
971
+
972
+ import java.time.LocalDate;
973
+
974
+ import static org.assertj.core.api.Assertions.*;
975
+
976
+ /**
977
+ * 自動仕訳リポジトリのテスト
978
+ *
979
+ * JPA 版では @DataJpaTest を使用(MyBatis 版では @MybatisTest)
980
+ * TestContainers で PostgreSQL コンテナを起動し、実際のDBでテスト
981
+ */
982
+ @DataJpaTest
983
+ @AutoConfigureTestDatabase(replace = AutoConfigureTestDatabase.Replace.NONE)
984
+ @Testcontainers
985
+ @DisplayName("自動仕訳パターンリポジトリのテスト")
986
+ class AutoJournalPatternRepositoryTest {
987
+
988
+ @Container
989
+ static PostgreSQLContainer<?> postgres = new PostgreSQLContainer<>("postgres:16")
990
+ .withDatabaseName("testdb")
991
+ .withUsername("test")
992
+ .withPassword("test");
993
+
994
+ @DynamicPropertySource
995
+ static void configureProperties(DynamicPropertyRegistry registry) {
996
+ registry.add("spring.datasource.url", postgres::getJdbcUrl);
997
+ registry.add("spring.datasource.username", postgres::getUsername);
998
+ registry.add("spring.datasource.password", postgres::getPassword);
999
+ }
1000
+
1001
+ @Autowired
1002
+ private AutoJournalPatternJpaRepository patternRepository;
1003
+
1004
+ @Nested
1005
+ @DisplayName("パターンマスタの操作")
1006
+ class PatternMasterTest {
1007
+
1008
+ @Test
1009
+ @DisplayName("パターンマスタを登録・取得できる")
1010
+ void shouldSaveAndFindPattern() {
1011
+ // Given
1012
+ var pattern = AutoJournalPattern.builder()
1013
+ .patternCode("P001")
1014
+ .patternName("加工品売上")
1015
+ .productGroup("加工品")
1016
+ .customerGroup("ALL")
1017
+ .debitAccountCode("11300") // 売掛金
1018
+ .creditAccountCode("41110") // 売上加工品
1019
+ .validFrom(LocalDate.of(2024, 1, 1))
1020
+ .validTo(LocalDate.of(9999, 12, 31))
1021
+ .priority(100)
1022
+ .build();
1023
+
1024
+ // When
1025
+ patternRepository.save(pattern);
1026
+ var saved = patternRepository.findById("P001");
1027
+
1028
+ // Then
1029
+ assertThat(saved).isPresent();
1030
+ assertThat(saved.get().getPatternName()).isEqualTo("加工品売上");
1031
+ assertThat(saved.get().getProductGroup()).isEqualTo("加工品");
1032
+ }
1033
+
1034
+ @Test
1035
+ @DisplayName("有効なパターンを優先順位順で取得できる")
1036
+ void shouldFindValidPatternsSortedByPriority() {
1037
+ // Given: 複数のパターンを登録
1038
+ var pattern1 = AutoJournalPattern.builder()
1039
+ .patternCode("P001")
1040
+ .patternName("加工品売上")
1041
+ .productGroup("加工品")
1042
+ .customerGroup("ALL")
1043
+ .debitAccountCode("11300")
1044
+ .creditAccountCode("41110")
1045
+ .validFrom(LocalDate.of(2024, 1, 1))
1046
+ .validTo(LocalDate.of(9999, 12, 31))
1047
+ .priority(200)
1048
+ .build();
1049
+
1050
+ var pattern2 = AutoJournalPattern.builder()
1051
+ .patternCode("P002")
1052
+ .patternName("生鮮品売上")
1053
+ .productGroup("生鮮品")
1054
+ .customerGroup("ALL")
1055
+ .debitAccountCode("11300")
1056
+ .creditAccountCode("41120")
1057
+ .validFrom(LocalDate.of(2024, 1, 1))
1058
+ .validTo(LocalDate.of(9999, 12, 31))
1059
+ .priority(100)
1060
+ .build();
1061
+
1062
+ patternRepository.save(pattern1);
1063
+ patternRepository.save(pattern2);
1064
+
1065
+ // When
1066
+ var patterns = patternRepository.findValidPatterns(LocalDate.of(2024, 4, 1));
1067
+
1068
+ // Then
1069
+ assertThat(patterns).isSortedAccordingTo(
1070
+ (p1, p2) -> p1.getPriority().compareTo(p2.getPriority())
1071
+ );
1072
+ assertThat(patterns.get(0).getPatternCode()).isEqualTo("P002"); // priority 100
1073
+ }
1074
+ }
1075
+ }
1076
+ ```
1077
+
1078
+ </details>
1079
+
1080
+ ---
1081
+
1082
+ ## 17.6 JpaRepository インターフェース
1083
+
1084
+ ### 自動仕訳パターンマスタ Repository
1085
+
1086
+ <details>
1087
+ <summary>AutoJournalPatternJpaRepository.java</summary>
1088
+
1089
+ ```java
1090
+ package com.example.accounting.infrastructure.persistence.repository;
1091
+
1092
+ import com.example.accounting.domain.model.autojournal.AutoJournalPattern;
1093
+ import org.springframework.data.jpa.repository.EntityGraph;
1094
+ import org.springframework.data.jpa.repository.JpaRepository;
1095
+ import org.springframework.data.jpa.repository.Query;
1096
+ import org.springframework.data.repository.query.Param;
1097
+ import org.springframework.stereotype.Repository;
1098
+
1099
+ import java.time.LocalDate;
1100
+ import java.util.List;
1101
+ import java.util.Optional;
1102
+
1103
+ /**
1104
+ * 自動仕訳パターンマスタ JpaRepository
1105
+ *
1106
+ * JPA 版では JpaRepository を継承(MyBatis 版では Mapper インターフェース)
1107
+ * メソッド名規約で自動的にクエリを生成
1108
+ */
1109
+ @Repository
1110
+ public interface AutoJournalPatternJpaRepository extends JpaRepository<AutoJournalPattern, String> {
1111
+
1112
+ /**
1113
+ * 有効なパターンを優先順位順で取得
1114
+ * @Query で JPQL を明示的に記述
1115
+ */
1116
+ @Query("SELECT p FROM AutoJournalPattern p " +
1117
+ "WHERE p.validFrom <= :date AND p.validTo >= :date " +
1118
+ "ORDER BY p.priority, p.patternCode")
1119
+ List<AutoJournalPattern> findValidPatterns(@Param("date") LocalDate date);
1120
+
1121
+ /**
1122
+ * パターンコードで検索(勘定科目も一緒に取得)
1123
+ * @EntityGraph で N+1 問題を防止
1124
+ */
1125
+ @EntityGraph(attributePaths = {"debitAccount", "creditAccount"})
1126
+ Optional<AutoJournalPattern> findWithAccountsByPatternCode(String patternCode);
1127
+
1128
+ /**
1129
+ * 商品グループで検索
1130
+ */
1131
+ List<AutoJournalPattern> findByProductGroupOrderByPriority(String productGroup);
1132
+
1133
+ /**
1134
+ * 顧客グループで検索
1135
+ */
1136
+ List<AutoJournalPattern> findByCustomerGroupOrderByPriority(String customerGroup);
1137
+ }
1138
+ ```
1139
+
1140
+ </details>
1141
+
1142
+ ### 自動仕訳データ Repository
1143
+
1144
+ <details>
1145
+ <summary>AutoJournalEntryJpaRepository.java</summary>
1146
+
1147
+ ```java
1148
+ package com.example.accounting.infrastructure.persistence.repository;
1149
+
1150
+ import com.example.accounting.domain.model.autojournal.AutoJournalEntry;
1151
+ import com.example.accounting.domain.model.autojournal.AutoJournalStatus;
1152
+ import org.springframework.data.jpa.repository.EntityGraph;
1153
+ import org.springframework.data.jpa.repository.JpaRepository;
1154
+ import org.springframework.data.jpa.repository.JpaSpecificationExecutor;
1155
+ import org.springframework.data.jpa.repository.Query;
1156
+ import org.springframework.data.repository.query.Param;
1157
+ import org.springframework.stereotype.Repository;
1158
+
1159
+ import java.time.LocalDate;
1160
+ import java.util.List;
1161
+ import java.util.Optional;
1162
+
1163
+ /**
1164
+ * 自動仕訳データ JpaRepository
1165
+ *
1166
+ * JpaSpecificationExecutor を継承して動的クエリに対応
1167
+ * MyBatis 版では動的 SQL を XML で記述していたが、
1168
+ * JPA 版では Specification パターンを使用
1169
+ */
1170
+ @Repository
1171
+ public interface AutoJournalEntryJpaRepository
1172
+ extends JpaRepository<AutoJournalEntry, String>,
1173
+ JpaSpecificationExecutor<AutoJournalEntry> {
1174
+
1175
+ /**
1176
+ * 売上番号で検索
1177
+ */
1178
+ List<AutoJournalEntry> findBySalesNumberOrderBySalesLineNumberAscDebitCreditTypeAsc(
1179
+ String salesNumber);
1180
+
1181
+ /**
1182
+ * 未転記の自動仕訳を取得
1183
+ */
1184
+ @Query("SELECT e FROM AutoJournalEntry e " +
1185
+ "WHERE e.postedFlag = false " +
1186
+ "AND e.status = com.example.accounting.domain.model.autojournal.AutoJournalStatus.COMPLETED " +
1187
+ "ORDER BY e.postingDate, e.autoJournalNumber")
1188
+ List<AutoJournalEntry> findUnposted();
1189
+
1190
+ /**
1191
+ * 未転記の自動仕訳を日付で取得
1192
+ */
1193
+ @Query("SELECT e FROM AutoJournalEntry e " +
1194
+ "WHERE e.postedFlag = false " +
1195
+ "AND e.status = com.example.accounting.domain.model.autojournal.AutoJournalStatus.COMPLETED " +
1196
+ "AND e.postingDate = :date " +
1197
+ "ORDER BY e.autoJournalNumber")
1198
+ List<AutoJournalEntry> findUnpostedByDate(@Param("date") LocalDate date);
1199
+
1200
+ /**
1201
+ * ステータスで検索
1202
+ */
1203
+ List<AutoJournalEntry> findByStatusOrderByPostingDateAscAutoJournalNumberAsc(
1204
+ AutoJournalStatus status);
1205
+
1206
+ /**
1207
+ * パターンと関連エンティティを一緒に取得
1208
+ * @EntityGraph で N+1 問題を防止
1209
+ */
1210
+ @EntityGraph(attributePaths = {"pattern", "account", "department"})
1211
+ Optional<AutoJournalEntry> findWithRelationsById(String autoJournalNumber);
1212
+ }
1213
+ ```
1214
+
1215
+ </details>
1216
+
1217
+ ---
1218
+
1219
+ ## 17.7 Spring Data JPA Specification による動的クエリ
1220
+
1221
+ JPA 版では、複雑な検索条件を組み合わせる際に Specification パターンを使用します。MyBatis 版では XML の `<if>` タグで動的 SQL を構築していましたが、JPA 版では型安全な Specification で実現します。
1222
+
1223
+ <details>
1224
+ <summary>AutoJournalEntrySpecifications.java</summary>
1225
+
1226
+ ```java
1227
+ package com.example.accounting.infrastructure.persistence.specification;
1228
+
1229
+ import com.example.accounting.domain.model.autojournal.AutoJournalEntry;
1230
+ import com.example.accounting.domain.model.autojournal.AutoJournalStatus;
1231
+ import org.springframework.data.jpa.domain.Specification;
1232
+
1233
+ import java.time.LocalDate;
1234
+
1235
+ /**
1236
+ * 自動仕訳エントリ検索用 Specification
1237
+ *
1238
+ * JPA 版では Specification パターンで動的クエリを構築
1239
+ * MyBatis 版の XML 動的 SQL を Java コードで表現
1240
+ *
1241
+ * 使用例:
1242
+ * var spec = AutoJournalEntrySpecifications.unposted()
1243
+ * .and(AutoJournalEntrySpecifications.postingDateBetween(from, to))
1244
+ * .and(AutoJournalEntrySpecifications.hasStatus(AutoJournalStatus.COMPLETED));
1245
+ * var entries = repository.findAll(spec);
1246
+ */
1247
+ public class AutoJournalEntrySpecifications {
1248
+
1249
+ /**
1250
+ * 未転記条件
1251
+ */
1252
+ public static Specification<AutoJournalEntry> unposted() {
1253
+ return (root, query, cb) -> cb.equal(root.get("postedFlag"), false);
1254
+ }
1255
+
1256
+ /**
1257
+ * ステータス条件
1258
+ */
1259
+ public static Specification<AutoJournalEntry> hasStatus(AutoJournalStatus status) {
1260
+ return (root, query, cb) -> cb.equal(root.get("status"), status);
1261
+ }
1262
+
1263
+ /**
1264
+ * 起票日範囲条件
1265
+ */
1266
+ public static Specification<AutoJournalEntry> postingDateBetween(
1267
+ LocalDate from, LocalDate to) {
1268
+ return (root, query, cb) -> cb.between(root.get("postingDate"), from, to);
1269
+ }
1270
+
1271
+ /**
1272
+ * 起票日条件
1273
+ */
1274
+ public static Specification<AutoJournalEntry> postingDateEquals(LocalDate date) {
1275
+ return (root, query, cb) -> cb.equal(root.get("postingDate"), date);
1276
+ }
1277
+
1278
+ /**
1279
+ * 売上番号条件
1280
+ */
1281
+ public static Specification<AutoJournalEntry> salesNumberEquals(String salesNumber) {
1282
+ return (root, query, cb) -> cb.equal(root.get("salesNumber"), salesNumber);
1283
+ }
1284
+
1285
+ /**
1286
+ * パターンコード条件
1287
+ */
1288
+ public static Specification<AutoJournalEntry> patternCodeEquals(String patternCode) {
1289
+ return (root, query, cb) -> cb.equal(root.get("patternCode"), patternCode);
1290
+ }
1291
+
1292
+ /**
1293
+ * 勘定科目コード条件
1294
+ */
1295
+ public static Specification<AutoJournalEntry> accountCodeEquals(String accountCode) {
1296
+ return (root, query, cb) -> cb.equal(root.get("accountCode"), accountCode);
1297
+ }
1298
+
1299
+ /**
1300
+ * 部門コード条件
1301
+ */
1302
+ public static Specification<AutoJournalEntry> departmentCodeEquals(String departmentCode) {
1303
+ return (root, query, cb) -> cb.equal(root.get("departmentCode"), departmentCode);
1304
+ }
1305
+
1306
+ /**
1307
+ * エラー有無条件
1308
+ */
1309
+ public static Specification<AutoJournalEntry> hasError() {
1310
+ return (root, query, cb) -> cb.isNotNull(root.get("errorCode"));
1311
+ }
1312
+
1313
+ /**
1314
+ * エラー無し条件
1315
+ */
1316
+ public static Specification<AutoJournalEntry> noError() {
1317
+ return (root, query, cb) -> cb.isNull(root.get("errorCode"));
1318
+ }
1319
+ }
1320
+ ```
1321
+
1322
+ </details>
1323
+
1324
+ ### Specification の使用例
1325
+
1326
+ ```java
1327
+ // 動的クエリの組み立て
1328
+ var spec = AutoJournalEntrySpecifications.unposted()
1329
+ .and(AutoJournalEntrySpecifications.postingDateBetween(
1330
+ LocalDate.of(2024, 4, 1),
1331
+ LocalDate.of(2024, 4, 30)))
1332
+ .and(AutoJournalEntrySpecifications.hasStatus(AutoJournalStatus.COMPLETED))
1333
+ .and(AutoJournalEntrySpecifications.noError());
1334
+
1335
+ // クエリ実行
1336
+ var entries = entryJpaRepository.findAll(spec);
1337
+ ```
1338
+
1339
+ ---
1340
+
1341
+ ## 17.8 自動仕訳サービスの実装
1342
+
1343
+ <details>
1344
+ <summary>AutoJournalService.java(一部抜粋)</summary>
1345
+
1346
+ ```java
1347
+ package com.example.accounting.application.service;
1348
+
1349
+ import com.example.accounting.application.dto.AutoJournalResult;
1350
+ import com.example.accounting.application.dto.SalesData;
1351
+ import com.example.accounting.application.port.out.AutoJournalRepository;
1352
+ import com.example.accounting.domain.model.autojournal.*;
1353
+ import com.example.accounting.domain.model.journal.*;
1354
+ import lombok.RequiredArgsConstructor;
1355
+ import lombok.extern.slf4j.Slf4j;
1356
+ import org.springframework.stereotype.Service;
1357
+ import org.springframework.transaction.annotation.Transactional;
1358
+
1359
+ import java.math.BigDecimal;
1360
+ import java.time.LocalDate;
1361
+ import java.time.LocalDateTime;
1362
+ import java.util.*;
1363
+
1364
+ /**
1365
+ * 自動仕訳サービス
1366
+ *
1367
+ * JPA 版でもビジネスロジックは MyBatis 版と同じ
1368
+ * リポジトリ経由でデータアクセスを行う
1369
+ */
1370
+ @Service
1371
+ @RequiredArgsConstructor
1372
+ @Slf4j
1373
+ public class AutoJournalService {
1374
+
1375
+ private final AutoJournalRepository autoJournalRepository;
1376
+ private final JournalRepository journalRepository;
1377
+
1378
+ /**
1379
+ * 売上データから自動仕訳を生成する
1380
+ */
1381
+ @Transactional
1382
+ public AutoJournalResult generateAutoJournals(List<SalesData> salesDataList) {
1383
+ var processNumber = generateProcessNumber();
1384
+ var startTime = LocalDateTime.now();
1385
+
1386
+ int successCount = 0;
1387
+ int errorCount = 0;
1388
+ BigDecimal totalAmount = BigDecimal.ZERO;
1389
+
1390
+ // 有効なパターンを取得
1391
+ var patterns = autoJournalRepository.findValidPatterns(LocalDate.now());
1392
+
1393
+ List<AutoJournalEntry> allEntries = new ArrayList<>();
1394
+
1395
+ for (SalesData sale : salesDataList) {
1396
+ try {
1397
+ // マッチするパターンを検索
1398
+ var matchedPattern = findMatchingPattern(patterns, sale);
1399
+
1400
+ if (matchedPattern.isPresent()) {
1401
+ // 自動仕訳データを生成
1402
+ var entries = createAutoJournalEntries(sale, matchedPattern.get());
1403
+ allEntries.addAll(entries);
1404
+
1405
+ successCount++;
1406
+ totalAmount = totalAmount.add(sale.getAmount());
1407
+ } else {
1408
+ // パターン不一致エラー
1409
+ allEntries.add(createErrorEntry(sale, "E001", "マッチするパターンが見つかりません"));
1410
+ errorCount++;
1411
+ }
1412
+ } catch (Exception e) {
1413
+ log.error("自動仕訳処理エラー: {}", sale.getSalesNumber(), e);
1414
+ allEntries.add(createErrorEntry(sale, "E999", e.getMessage()));
1415
+ errorCount++;
1416
+ }
1417
+ }
1418
+
1419
+ // JPA 版では saveAll でバッチ保存
1420
+ autoJournalRepository.saveAllEntries(allEntries);
1421
+
1422
+ // 処理履歴を保存
1423
+ var history = AutoJournalHistory.builder()
1424
+ .processNumber(processNumber)
1425
+ .processDateTime(startTime)
1426
+ .targetFromDate(salesDataList.stream()
1427
+ .map(SalesData::getSalesDate)
1428
+ .min(LocalDate::compareTo).orElse(LocalDate.now()))
1429
+ .targetToDate(salesDataList.stream()
1430
+ .map(SalesData::getSalesDate)
1431
+ .max(LocalDate::compareTo).orElse(LocalDate.now()))
1432
+ .totalCount(salesDataList.size())
1433
+ .successCount(successCount)
1434
+ .errorCount(errorCount)
1435
+ .totalAmount(totalAmount)
1436
+ .build();
1437
+
1438
+ autoJournalRepository.saveHistory(history);
1439
+
1440
+ return new AutoJournalResult(processNumber, successCount, errorCount, totalAmount);
1441
+ }
1442
+
1443
+ /**
1444
+ * マッチするパターンを検索
1445
+ */
1446
+ private Optional<AutoJournalPattern> findMatchingPattern(
1447
+ List<AutoJournalPattern> patterns, SalesData sale) {
1448
+ return patterns.stream()
1449
+ .filter(p -> p.matches(sale.getProductGroup(), sale.getCustomerGroup()))
1450
+ .filter(p -> p.isValidAt(sale.getSalesDate()))
1451
+ .findFirst();
1452
+ }
1453
+ }
1454
+ ```
1455
+
1456
+ </details>
1457
+
1458
+ ---
1459
+
1460
+ ## 17.9 ヘキサゴナルアーキテクチャでのポート設計
1461
+
1462
+ ### Output Port(リポジトリインターフェース)
1463
+
1464
+ <details>
1465
+ <summary>AutoJournalRepository.java</summary>
1466
+
1467
+ ```java
1468
+ package com.example.accounting.application.port.out;
1469
+
1470
+ import com.example.accounting.domain.model.autojournal.*;
1471
+
1472
+ import java.time.LocalDate;
1473
+ import java.util.List;
1474
+ import java.util.Optional;
1475
+
1476
+ /**
1477
+ * 自動仕訳リポジトリ(Output Port)
1478
+ *
1479
+ * ヘキサゴナルアーキテクチャにおける出力ポート
1480
+ * アプリケーション層はこのインターフェースに依存し、
1481
+ * インフラストラクチャ層で実装する
1482
+ */
1483
+ public interface AutoJournalRepository {
1484
+
1485
+ // パターンマスタ操作
1486
+ void savePattern(AutoJournalPattern pattern);
1487
+
1488
+ Optional<AutoJournalPattern> findPatternByCode(String patternCode);
1489
+
1490
+ Optional<AutoJournalPattern> findPatternWithAccounts(String patternCode);
1491
+
1492
+ List<AutoJournalPattern> findAllPatterns();
1493
+
1494
+ List<AutoJournalPattern> findValidPatterns(LocalDate date);
1495
+
1496
+ void updatePattern(AutoJournalPattern pattern);
1497
+
1498
+ void deletePattern(String patternCode);
1499
+
1500
+ // 自動仕訳エントリ操作
1501
+ void saveEntry(AutoJournalEntry entry);
1502
+
1503
+ void saveAllEntries(List<AutoJournalEntry> entries);
1504
+
1505
+ Optional<AutoJournalEntry> findEntryByNumber(String autoJournalNumber);
1506
+
1507
+ Optional<AutoJournalEntry> findEntryWithRelations(String autoJournalNumber);
1508
+
1509
+ List<AutoJournalEntry> findEntriesBySalesNumber(String salesNumber);
1510
+
1511
+ List<AutoJournalEntry> findUnpostedEntries();
1512
+
1513
+ List<AutoJournalEntry> findUnpostedEntriesByDate(LocalDate date);
1514
+
1515
+ List<AutoJournalEntry> findEntriesByStatus(AutoJournalStatus status);
1516
+
1517
+ void updateEntry(AutoJournalEntry entry);
1518
+
1519
+ // 処理履歴操作
1520
+ void saveHistory(AutoJournalHistory history);
1521
+
1522
+ Optional<AutoJournalHistory> findHistoryByNumber(String processNumber);
1523
+
1524
+ List<AutoJournalHistory> findHistoriesByDateRange(LocalDate fromDate, LocalDate toDate);
1525
+
1526
+ void deleteAll();
1527
+ }
1528
+ ```
1529
+
1530
+ </details>
1531
+
1532
+ ---
1533
+
1534
+ ## 17.10 まとめ:MyBatis 版と JPA 版の比較
1535
+
1536
+ ### データアクセス層のアーキテクチャ比較
1537
+
1538
+ ```plantuml
1539
+ @startuml
1540
+
1541
+ title MyBatis 版と JPA 版のアーキテクチャ比較
1542
+
1543
+ package "MyBatis 版" {
1544
+ [AutoJournalService] as MService
1545
+ [AutoJournalMapper] as MMapper
1546
+ [AutoJournalMapper.xml] as MXml
1547
+ database "PostgreSQL" as MDB
1548
+
1549
+ MService --> MMapper
1550
+ MMapper --> MXml
1551
+ MXml --> MDB
1552
+ }
1553
+
1554
+ package "JPA 版" {
1555
+ [AutoJournalService] as JService
1556
+ [AutoJournalRepository\n(Output Port)] as JPort
1557
+ [AutoJournalRepositoryImpl] as JImpl
1558
+ [AutoJournalPatternJpaRepository] as JJpa
1559
+ database "PostgreSQL" as JDB
1560
+
1561
+ JService --> JPort
1562
+ JPort <|.. JImpl
1563
+ JImpl --> JJpa
1564
+ JJpa --> JDB
1565
+ }
1566
+
1567
+ @enduml
1568
+ ```
1569
+
1570
+ ### JPA 版の特徴
1571
+
1572
+ | 観点 | JPA 版の特徴 |
1573
+ |-----|-------------|
1574
+ | 型安全性 | Specification パターンによる型安全な動的クエリ |
1575
+ | N+1 問題対策 | @EntityGraph による宣言的な解決 |
1576
+ | バッチ処理 | saveAll() による効率的な一括保存 |
1577
+ | ポート分離 | Output Port インターフェースによる抽象化 |
1578
+ | テスト容易性 | @DataJpaTest + TestContainers |
1579
+
1580
+ ---
1581
+
1582
+ ## 次章予告
1583
+
1584
+ [第18章](chapter18-orm.md)では、勘定科目残高テーブルの設計について、JPA 版での実装を解説します。