@k2works/claude-code-booster 0.1.3 → 0.2.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.
- package/README.md +14 -0
- package/bin/claude-code-booster +24 -4
- package/lib/assets/.claude/README.md +44 -40
- package/lib/assets/.claude/commands/analysis.md +230 -0
- package/lib/assets/.claude/commands/kill.md +109 -0
- package/lib/assets/.claude/commands/next.md +136 -0
- package/lib/assets/.claude/commands/plan.md +141 -91
- package/lib/assets/.claude/commands/progress.md +172 -0
- package/lib/assets/docs/reference/UI/350/250/255/350/250/210/343/202/254/343/202/244/343/203/211.md +446 -0
- package/lib/assets/docs/reference//343/202/242/343/203/274/343/202/255/343/203/206/343/202/257/343/203/201/343/203/243/350/250/255/350/250/210/343/202/254/343/202/244/343/203/211.md +1428 -0
- package/lib/assets/docs/reference//343/202/244/343/203/263/343/203/225/343/203/251/350/250/255/350/250/210/343/202/254/343/202/244/343/203/211.md +1879 -0
- package/lib/assets/docs/reference//343/203/206/343/202/271/343/203/210/346/210/246/347/225/245/343/202/254/343/202/244/343/203/211.md +1310 -0
- package/lib/assets/docs/reference//343/203/207/343/203/274/343/202/277/343/203/242/343/203/207/343/203/253/350/250/255/350/250/210/343/202/254/343/202/244/343/203/211.md +312 -0
- package/lib/assets/docs/reference//343/203/211/343/203/241/343/202/244/343/203/263/343/203/242/343/203/207/343/203/253/350/250/255/350/250/210/343/202/254/343/202/244/343/203/211.md +600 -0
- package/lib/assets/docs/reference//343/203/246/343/203/274/343/202/271/343/202/261/343/203/274/343/202/271/344/275/234/346/210/220/343/202/254/343/202/244/343/203/211.md +672 -0
- package/lib/assets/docs/reference//343/203/252/343/203/252/343/203/274/343/202/271/343/203/273/343/202/244/343/203/206/343/203/254/343/203/274/343/202/267/343/203/247/343/203/263/350/250/210/347/224/273/343/202/254/343/202/244/343/203/211.md +524 -0
- package/lib/assets/docs/reference//351/201/213/347/224/250/350/246/201/344/273/266/345/256/232/347/276/251/343/202/254/343/202/244/343/203/211.md +393 -0
- package/lib/assets/docs/reference//351/226/213/347/231/272/343/202/254/343/202/244/343/203/211.md +18 -173
- package/lib/assets/docs/reference//351/235/236/346/251/237/350/203/275/350/246/201/344/273/266/345/256/232/347/276/251/343/202/254/343/202/244/343/203/211.md +1231 -0
- package/lib/assets/docs/template//345/256/214/345/205/250/345/275/242/345/274/217/343/201/256/343/203/246/343/203/274/343/202/271/343/202/261/343/203/274/343/202/271.md +64 -0
- package/lib/assets/docs/template//350/246/201/344/273/266/345/256/232/347/276/251.md +467 -443
- package/package.json +1 -1
|
@@ -0,0 +1,1231 @@
|
|
|
1
|
+
# 非機能要件定義ガイド
|
|
2
|
+
|
|
3
|
+
## 概要
|
|
4
|
+
|
|
5
|
+
よいソフトウェアを作るための非機能要件定義について説明する。変更を楽に安全にできて役に立つソフトウェアを実現するため、品質属性を明確に定義し、測定可能な指標を設定する。
|
|
6
|
+
|
|
7
|
+
## 非機能要件の基本原則
|
|
8
|
+
|
|
9
|
+
### よいソフトウェアと非機能要件
|
|
10
|
+
|
|
11
|
+
変更を楽に安全にできて役に立つソフトウェアを作るため、非機能要件は以下の価値を提供する:
|
|
12
|
+
|
|
13
|
+
1. **安全な変更**: システムが安定して動作し続ける信頼性
|
|
14
|
+
2. **楽な変更**: 保守性と拡張性により変更コストを最小化
|
|
15
|
+
3. **役に立つ**: ユーザビリティとパフォーマンスによる価値提供
|
|
16
|
+
4. **継続的価値**: 可用性と運用性による長期的な価値維持
|
|
17
|
+
|
|
18
|
+
### 非機能要件の重要性
|
|
19
|
+
|
|
20
|
+
```plantuml
|
|
21
|
+
@startuml
|
|
22
|
+
rectangle "機能要件" as Func {
|
|
23
|
+
- 何を実現するか
|
|
24
|
+
- ビジネス要求
|
|
25
|
+
- ユースケース
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
rectangle "非機能要件" as NonFunc {
|
|
29
|
+
- どの程度うまく実現するか
|
|
30
|
+
- 品質属性
|
|
31
|
+
- システム制約
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
@enduml
|
|
35
|
+
```
|
|
36
|
+
|
|
37
|
+
## ISO/IEC 25010 品質モデル
|
|
38
|
+
|
|
39
|
+
### システム・ソフトウェア品質モデル
|
|
40
|
+
|
|
41
|
+
```plantuml
|
|
42
|
+
@startuml
|
|
43
|
+
rectangle "ISO/IEC 25010 品質特性" {
|
|
44
|
+
rectangle "機能適合性" {
|
|
45
|
+
- 機能完全性
|
|
46
|
+
- 機能正確性
|
|
47
|
+
- 機能適切性
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
rectangle "性能効率性" {
|
|
51
|
+
- 時間効率性
|
|
52
|
+
- 資源効率性
|
|
53
|
+
- 容量満足性
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
rectangle "互換性" {
|
|
57
|
+
- 共存性
|
|
58
|
+
- 相互運用性
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
rectangle "使用性" {
|
|
62
|
+
- 適切度認識性
|
|
63
|
+
- 習得性
|
|
64
|
+
- 運用性
|
|
65
|
+
- ユーザーエラー防止性
|
|
66
|
+
- UI快美性
|
|
67
|
+
- アクセシビリティ
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
rectangle "信頼性" {
|
|
71
|
+
- 成熟性
|
|
72
|
+
- 可用性
|
|
73
|
+
- 障害許容性
|
|
74
|
+
- 回復性
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
rectangle "セキュリティ" {
|
|
78
|
+
- 機密性
|
|
79
|
+
- 完全性
|
|
80
|
+
- 否認防止性
|
|
81
|
+
- 責任追跡性
|
|
82
|
+
- 真正性
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
rectangle "保守性" {
|
|
86
|
+
- モジュール性
|
|
87
|
+
- 再利用性
|
|
88
|
+
- 解析性
|
|
89
|
+
- 修正性
|
|
90
|
+
- 試験性
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
rectangle "移植性" {
|
|
94
|
+
- 適応性
|
|
95
|
+
- 設置性
|
|
96
|
+
- 置換性
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
@enduml
|
|
100
|
+
```
|
|
101
|
+
|
|
102
|
+
### 品質特性の詳細
|
|
103
|
+
|
|
104
|
+
#### 機能適合性(Functional Suitability)
|
|
105
|
+
- **機能完全性**: 指定されたタスクと目標をカバーする機能の度合い
|
|
106
|
+
- **機能正確性**: 正確な結果を提供する機能の度合い
|
|
107
|
+
- **機能適切性**: 指定されたタスクと目標を促進する機能の度合い
|
|
108
|
+
|
|
109
|
+
#### 性能効率性(Performance Efficiency)
|
|
110
|
+
- **時間効率性**: 応答時間、処理時間、スループット
|
|
111
|
+
- **資源効率性**: CPU、メモリ、ネットワーク、ストレージの使用量
|
|
112
|
+
- **容量満足性**: 最大限界値(ユーザー数、データ量など)
|
|
113
|
+
|
|
114
|
+
#### 互換性(Compatibility)
|
|
115
|
+
- **共存性**: 他のソフトウェアと共通環境でリソースを共有する度合い
|
|
116
|
+
- **相互運用性**: 他のシステムと情報交換し機能を利用する度合い
|
|
117
|
+
|
|
118
|
+
#### 使用性(Usability)
|
|
119
|
+
- **適切度認識性**: ユーザーが適切性を認識する度合い
|
|
120
|
+
- **習得性**: 学習のしやすさ
|
|
121
|
+
- **運用性**: 操作・制御のしやすさ
|
|
122
|
+
- **ユーザーエラー防止性**: エラーを防護する度合い
|
|
123
|
+
- **UI快美性**: 満足感を与える度合い
|
|
124
|
+
- **アクセシビリティ**: 幅広いユーザーが使用できる度合い
|
|
125
|
+
|
|
126
|
+
#### 信頼性(Reliability)
|
|
127
|
+
- **成熟性**: 通常運用下で信頼性要求を満足する度合い
|
|
128
|
+
- **可用性**: 運用可能で利用できる度合い
|
|
129
|
+
- **障害許容性**: 障害にもかかわらず動作する度合い
|
|
130
|
+
- **回復性**: 障害後に回復し影響データを復旧する度合い
|
|
131
|
+
|
|
132
|
+
#### セキュリティ(Security)
|
|
133
|
+
- **機密性**: 認可されたもののみがアクセスできる度合い
|
|
134
|
+
- **完全性**: データや計算方法への不正アクセスを防ぐ度合い
|
|
135
|
+
- **否認防止性**: アクションや事象が起きたことを証明する度合い
|
|
136
|
+
- **責任追跡性**: エンティティのアクションを一意に追跡する度合い
|
|
137
|
+
- **真正性**: 主張されたアイデンティティを証明する度合い
|
|
138
|
+
|
|
139
|
+
#### 保守性(Maintainability)
|
|
140
|
+
- **モジュール性**: 構成要素への変更が他に与える影響が最小限の度合い
|
|
141
|
+
- **再利用性**: 他のシステムで利用できる度合い
|
|
142
|
+
- **解析性**: 変更の影響を評価する度合い
|
|
143
|
+
- **修正性**: 欠陥除去や改善を効果的かつ効率的に行える度合い
|
|
144
|
+
- **試験性**: テスト基準を確立しテストを実行する度合い
|
|
145
|
+
|
|
146
|
+
#### 移植性(Portability)
|
|
147
|
+
- **適応性**: 異なるハードウェア・ソフトウェア環境に適応する度合い
|
|
148
|
+
- **設置性**: 指定された環境に設置する度合い
|
|
149
|
+
- **置換性**: 同じ目的の他のソフトウェアと置き換える度合い
|
|
150
|
+
|
|
151
|
+
## 非機能要件の分類
|
|
152
|
+
|
|
153
|
+
### FURPS+ モデル
|
|
154
|
+
|
|
155
|
+
```plantuml
|
|
156
|
+
@startuml
|
|
157
|
+
rectangle "FURPS+ モデル" {
|
|
158
|
+
rectangle "Functionality" {
|
|
159
|
+
- 機能性
|
|
160
|
+
- セキュリティ
|
|
161
|
+
- API
|
|
162
|
+
- インターフェース
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
rectangle "Usability" {
|
|
166
|
+
- 使いやすさ
|
|
167
|
+
- ユーザビリティ
|
|
168
|
+
- アクセシビリティ
|
|
169
|
+
- ドキュメント
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
rectangle "Reliability" {
|
|
173
|
+
- 信頼性
|
|
174
|
+
- 可用性
|
|
175
|
+
- 故障率
|
|
176
|
+
- 復旧時間
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
rectangle "Performance" {
|
|
180
|
+
- パフォーマンス
|
|
181
|
+
- 応答時間
|
|
182
|
+
- スループット
|
|
183
|
+
- 容量
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
rectangle "Supportability" {
|
|
187
|
+
- サポート性
|
|
188
|
+
- 保守性
|
|
189
|
+
- 拡張性
|
|
190
|
+
- 設定性
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
rectangle "Plus" {
|
|
194
|
+
- 実装制約
|
|
195
|
+
- インターフェース制約
|
|
196
|
+
- 運用制約
|
|
197
|
+
- パッケージ制約
|
|
198
|
+
- 法的制約
|
|
199
|
+
}
|
|
200
|
+
}
|
|
201
|
+
@enduml
|
|
202
|
+
```
|
|
203
|
+
|
|
204
|
+
### 会議室予約システムの非機能要件分類
|
|
205
|
+
|
|
206
|
+
#### 性能要件(Performance Requirements)
|
|
207
|
+
- **応答時間**: ユーザーアクションに対するレスポンス時間
|
|
208
|
+
- **スループット**: 同時処理可能な予約数・ユーザー数
|
|
209
|
+
- **容量**: 保存可能なデータ量、サポートするユーザー数
|
|
210
|
+
|
|
211
|
+
#### 可用性・信頼性要件(Availability & Reliability Requirements)
|
|
212
|
+
- **稼働率**: システムが利用可能な時間の割合
|
|
213
|
+
- **障害回復**: システム障害からの回復時間
|
|
214
|
+
- **データ保護**: データの損失防止とバックアップ
|
|
215
|
+
|
|
216
|
+
#### セキュリティ要件(Security Requirements)
|
|
217
|
+
- **認証・認可**: ユーザー識別とアクセス制御
|
|
218
|
+
- **データ保護**: 個人情報・機密情報の保護
|
|
219
|
+
- **監査**: アクセスログと操作履歴の記録
|
|
220
|
+
|
|
221
|
+
#### 使用性要件(Usability Requirements)
|
|
222
|
+
- **操作性**: 直感的で使いやすいインターフェース
|
|
223
|
+
- **アクセシビリティ**: 多様なユーザーへの対応
|
|
224
|
+
- **応答性**: レスポンシブデザインと快適な操作感
|
|
225
|
+
|
|
226
|
+
#### 保守性・拡張性要件(Maintainability & Scalability Requirements)
|
|
227
|
+
- **保守性**: コードの理解しやすさと変更容易性
|
|
228
|
+
- **拡張性**: 機能追加とユーザー増加への対応
|
|
229
|
+
- **監視性**: システム状態の把握と問題の早期発見
|
|
230
|
+
|
|
231
|
+
#### 互換性・移植性要件(Compatibility & Portability Requirements)
|
|
232
|
+
- **ブラウザ互換性**: 複数ブラウザでの動作保証
|
|
233
|
+
- **デバイス対応**: PC、タブレット、スマートフォン対応
|
|
234
|
+
- **システム連携**: 既存システムとの連携能力
|
|
235
|
+
|
|
236
|
+
## 会議室予約システムの非機能要件詳細
|
|
237
|
+
|
|
238
|
+
### 1. 性能要件(Performance Requirements)
|
|
239
|
+
|
|
240
|
+
#### 1.1 応答時間要件
|
|
241
|
+
|
|
242
|
+
```plantuml
|
|
243
|
+
@startuml
|
|
244
|
+
rectangle "応答時間要件" {
|
|
245
|
+
rectangle "ページ表示" {
|
|
246
|
+
- 初回ページ読み込み: < 3秒
|
|
247
|
+
- ページ遷移: < 1秒
|
|
248
|
+
- 部分更新: < 500ms
|
|
249
|
+
}
|
|
250
|
+
|
|
251
|
+
rectangle "API応答" {
|
|
252
|
+
- 会議室検索: < 1秒
|
|
253
|
+
- 予約作成: < 2秒
|
|
254
|
+
- 予約変更・キャンセル: < 1秒
|
|
255
|
+
- ユーザー認証: < 1秒
|
|
256
|
+
}
|
|
257
|
+
|
|
258
|
+
rectangle "バッチ処理" {
|
|
259
|
+
- データ集計: < 30秒
|
|
260
|
+
- レポート生成: < 60秒
|
|
261
|
+
- データアーカイブ: < 10分
|
|
262
|
+
}
|
|
263
|
+
}
|
|
264
|
+
@enduml
|
|
265
|
+
```
|
|
266
|
+
|
|
267
|
+
**測定方法**:
|
|
268
|
+
```javascript
|
|
269
|
+
// フロントエンド性能測定
|
|
270
|
+
const observer = new PerformanceObserver((list) => {
|
|
271
|
+
list.getEntries().forEach((entry) => {
|
|
272
|
+
console.log(`${entry.name}: ${entry.duration}ms`);
|
|
273
|
+
});
|
|
274
|
+
});
|
|
275
|
+
observer.observe({ entryTypes: ['navigation', 'measure'] });
|
|
276
|
+
|
|
277
|
+
// API応答時間測定
|
|
278
|
+
const startTime = performance.now();
|
|
279
|
+
await fetch('/api/reservations');
|
|
280
|
+
const endTime = performance.now();
|
|
281
|
+
console.log(`API応答時間: ${endTime - startTime}ms`);
|
|
282
|
+
```
|
|
283
|
+
|
|
284
|
+
**実装例**:
|
|
285
|
+
```java
|
|
286
|
+
// バックエンド性能監視
|
|
287
|
+
@RestController
|
|
288
|
+
@Timed // Micrometer アノテーション
|
|
289
|
+
public class ReservationController {
|
|
290
|
+
|
|
291
|
+
@GetMapping("/reservations")
|
|
292
|
+
@Timed(name = "reservation.search", description = "予約検索処理時間")
|
|
293
|
+
public ResponseEntity<List<ReservationResponse>> searchReservations(
|
|
294
|
+
@RequestParam ReservationSearchCriteria criteria) {
|
|
295
|
+
// 処理実装
|
|
296
|
+
}
|
|
297
|
+
}
|
|
298
|
+
|
|
299
|
+
// データベースクエリ最適化
|
|
300
|
+
@Repository
|
|
301
|
+
public class ReservationRepository {
|
|
302
|
+
|
|
303
|
+
@Query(value = """
|
|
304
|
+
SELECT r FROM Reservation r
|
|
305
|
+
WHERE r.roomId = :roomId
|
|
306
|
+
AND r.startTime >= :startDate
|
|
307
|
+
AND r.endTime <= :endDate
|
|
308
|
+
AND r.status IN ('CONFIRMED', 'PENDING')
|
|
309
|
+
""")
|
|
310
|
+
List<Reservation> findActiveByRoomAndDateRange(
|
|
311
|
+
@Param("roomId") UUID roomId,
|
|
312
|
+
@Param("startDate") LocalDateTime startDate,
|
|
313
|
+
@Param("endDate") LocalDateTime endDate);
|
|
314
|
+
}
|
|
315
|
+
```
|
|
316
|
+
|
|
317
|
+
#### 1.2 スループット要件
|
|
318
|
+
|
|
319
|
+
- **同時ユーザー数**: 100名(ピーク時200名まで対応)
|
|
320
|
+
- **予約処理能力**: 毎秒10件の予約処理
|
|
321
|
+
- **データ容量**: 年間10,000件の予約データ、5年間保持
|
|
322
|
+
|
|
323
|
+
```java
|
|
324
|
+
// 負荷テスト設定例
|
|
325
|
+
@Test
|
|
326
|
+
@LoadTest(users = 100, duration = 300) // 100ユーザー、5分間
|
|
327
|
+
void 予約作成の負荷テスト() {
|
|
328
|
+
CreateReservationRequest request = createTestRequest();
|
|
329
|
+
|
|
330
|
+
ResponseEntity<ReservationResponse> response = restTemplate.postForEntity(
|
|
331
|
+
"/api/reservations", request, ReservationResponse.class);
|
|
332
|
+
|
|
333
|
+
assertThat(response.getStatusCode()).isEqualTo(HttpStatus.CREATED);
|
|
334
|
+
assertThat(response.getBody().getReservationId()).isNotNull();
|
|
335
|
+
}
|
|
336
|
+
```
|
|
337
|
+
|
|
338
|
+
### 2. 可用性・信頼性要件(Availability & Reliability Requirements)
|
|
339
|
+
|
|
340
|
+
#### 2.1 稼働率要件
|
|
341
|
+
|
|
342
|
+
```plantuml
|
|
343
|
+
@startuml
|
|
344
|
+
rectangle "可用性・信頼性要件" {
|
|
345
|
+
rectangle "可用性レベル" {
|
|
346
|
+
- 稼働率 99.9%: 年間ダウンタイム < 8.77時間
|
|
347
|
+
- 営業時間内 99.95%: 月間ダウンタイム < 22分
|
|
348
|
+
- 計画停止除く: 営業時間外のメンテナンス許可
|
|
349
|
+
}
|
|
350
|
+
|
|
351
|
+
rectangle "障害対応" {
|
|
352
|
+
- 検知時間: < 5分
|
|
353
|
+
- 初期対応: < 15分
|
|
354
|
+
- 復旧時間: < 2時間(重大障害)
|
|
355
|
+
- 回復時間: < 30分(軽微な障害)
|
|
356
|
+
}
|
|
357
|
+
|
|
358
|
+
rectangle "データ保護" {
|
|
359
|
+
- バックアップ: 日次・週次・月次
|
|
360
|
+
- 復旧時間: < 4時間(RPO: 1時間以内)
|
|
361
|
+
- データ整合性: トランザクション保証
|
|
362
|
+
}
|
|
363
|
+
}
|
|
364
|
+
@enduml
|
|
365
|
+
```
|
|
366
|
+
|
|
367
|
+
**実装例**:
|
|
368
|
+
```yaml
|
|
369
|
+
# Docker Compose でのヘルスチェック
|
|
370
|
+
services:
|
|
371
|
+
app:
|
|
372
|
+
image: meeting-room-app:latest
|
|
373
|
+
healthcheck:
|
|
374
|
+
test: ["CMD", "curl", "-f", "http://localhost:8080/actuator/health"]
|
|
375
|
+
interval: 30s
|
|
376
|
+
timeout: 10s
|
|
377
|
+
retries: 3
|
|
378
|
+
start_period: 60s
|
|
379
|
+
restart: unless-stopped
|
|
380
|
+
|
|
381
|
+
database:
|
|
382
|
+
image: postgres:15
|
|
383
|
+
healthcheck:
|
|
384
|
+
test: ["CMD-SHELL", "pg_isready -U ${POSTGRES_USER}"]
|
|
385
|
+
interval: 10s
|
|
386
|
+
timeout: 5s
|
|
387
|
+
retries: 5
|
|
388
|
+
```
|
|
389
|
+
|
|
390
|
+
```java
|
|
391
|
+
// Spring Boot Actuator によるヘルスチェック
|
|
392
|
+
@Component
|
|
393
|
+
public class DatabaseHealthIndicator implements HealthIndicator {
|
|
394
|
+
|
|
395
|
+
private final DataSource dataSource;
|
|
396
|
+
|
|
397
|
+
@Override
|
|
398
|
+
public Health health() {
|
|
399
|
+
try (Connection connection = dataSource.getConnection()) {
|
|
400
|
+
if (connection.isValid(1)) {
|
|
401
|
+
return Health.up()
|
|
402
|
+
.withDetail("database", "Available")
|
|
403
|
+
.withDetail("validationQuery", "SELECT 1")
|
|
404
|
+
.build();
|
|
405
|
+
}
|
|
406
|
+
} catch (SQLException e) {
|
|
407
|
+
return Health.down(e)
|
|
408
|
+
.withDetail("database", "Unavailable")
|
|
409
|
+
.build();
|
|
410
|
+
}
|
|
411
|
+
return Health.down()
|
|
412
|
+
.withDetail("database", "Unknown")
|
|
413
|
+
.build();
|
|
414
|
+
}
|
|
415
|
+
}
|
|
416
|
+
```
|
|
417
|
+
|
|
418
|
+
#### 2.2 障害許容性
|
|
419
|
+
|
|
420
|
+
```java
|
|
421
|
+
// Circuit Breaker パターン実装
|
|
422
|
+
@Component
|
|
423
|
+
public class ExternalServiceClient {
|
|
424
|
+
|
|
425
|
+
private final CircuitBreaker circuitBreaker;
|
|
426
|
+
|
|
427
|
+
public ExternalServiceClient() {
|
|
428
|
+
this.circuitBreaker = CircuitBreaker.ofDefaults("externalService");
|
|
429
|
+
circuitBreaker.getEventPublisher()
|
|
430
|
+
.onStateTransition(event ->
|
|
431
|
+
log.info("Circuit breaker state transition: {}", event));
|
|
432
|
+
}
|
|
433
|
+
|
|
434
|
+
public Optional<String> callExternalService(String request) {
|
|
435
|
+
return circuitBreaker.executeSupplier(() -> {
|
|
436
|
+
// 外部サービス呼び出し
|
|
437
|
+
return externalServiceCall(request);
|
|
438
|
+
});
|
|
439
|
+
}
|
|
440
|
+
}
|
|
441
|
+
|
|
442
|
+
// 再試行機能
|
|
443
|
+
@Retryable(
|
|
444
|
+
value = {DataAccessException.class},
|
|
445
|
+
maxAttempts = 3,
|
|
446
|
+
backoff = @Backoff(delay = 1000, multiplier = 2)
|
|
447
|
+
)
|
|
448
|
+
public Reservation saveReservation(Reservation reservation) {
|
|
449
|
+
return reservationRepository.save(reservation);
|
|
450
|
+
}
|
|
451
|
+
```
|
|
452
|
+
|
|
453
|
+
### 3. セキュリティ要件(Security Requirements)
|
|
454
|
+
|
|
455
|
+
#### 3.1 認証・認可要件
|
|
456
|
+
|
|
457
|
+
```plantuml
|
|
458
|
+
@startuml
|
|
459
|
+
rectangle "セキュリティ要件" {
|
|
460
|
+
rectangle "認証" {
|
|
461
|
+
- JWT Bearer Token: 有効期限 24時間
|
|
462
|
+
- リフレッシュトークン: 有効期限 30日
|
|
463
|
+
- 多要素認証: 管理者必須
|
|
464
|
+
- パスワード強度: 8文字以上、複雑性要求
|
|
465
|
+
}
|
|
466
|
+
|
|
467
|
+
rectangle "認可" {
|
|
468
|
+
- RBAC: ロールベースアクセス制御
|
|
469
|
+
- リソースレベル制御: 予約・会議室・ユーザー
|
|
470
|
+
- API エンドポイント保護: 全API認証必須
|
|
471
|
+
}
|
|
472
|
+
|
|
473
|
+
rectangle "データ保護" {
|
|
474
|
+
- 暗号化: 保存時・転送時
|
|
475
|
+
- 個人情報保護: GDPR準拠
|
|
476
|
+
- 監査ログ: 全操作記録
|
|
477
|
+
}
|
|
478
|
+
}
|
|
479
|
+
@enduml
|
|
480
|
+
```
|
|
481
|
+
|
|
482
|
+
**実装例**:
|
|
483
|
+
```java
|
|
484
|
+
// JWT セキュリティ設定
|
|
485
|
+
@Configuration
|
|
486
|
+
@EnableWebSecurity
|
|
487
|
+
@EnableMethodSecurity
|
|
488
|
+
public class SecurityConfig {
|
|
489
|
+
|
|
490
|
+
@Bean
|
|
491
|
+
public SecurityFilterChain filterChain(HttpSecurity http) throws Exception {
|
|
492
|
+
return http
|
|
493
|
+
.csrf(csrf -> csrf.disable())
|
|
494
|
+
.sessionManagement(session ->
|
|
495
|
+
session.sessionCreationPolicy(SessionCreationPolicy.STATELESS))
|
|
496
|
+
.authorizeHttpRequests(auth -> auth
|
|
497
|
+
.requestMatchers("/api/auth/**").permitAll()
|
|
498
|
+
.requestMatchers(HttpMethod.GET, "/api/rooms").hasAnyRole("USER", "ADMIN")
|
|
499
|
+
.requestMatchers("/api/reservations/**").hasAnyRole("USER", "ADMIN")
|
|
500
|
+
.requestMatchers("/api/admin/**").hasRole("ADMIN")
|
|
501
|
+
.anyRequest().authenticated())
|
|
502
|
+
.oauth2ResourceServer(oauth2 -> oauth2
|
|
503
|
+
.jwt(jwt -> jwt.jwtAuthenticationConverter(jwtAuthConverter())))
|
|
504
|
+
.build();
|
|
505
|
+
}
|
|
506
|
+
|
|
507
|
+
@Bean
|
|
508
|
+
public PasswordEncoder passwordEncoder() {
|
|
509
|
+
return new BCryptPasswordEncoder(12); // 強度12
|
|
510
|
+
}
|
|
511
|
+
}
|
|
512
|
+
|
|
513
|
+
// 認可制御
|
|
514
|
+
@PreAuthorize("hasRole('ADMIN') or @reservationSecurityService.canAccessReservation(#reservationId, authentication.name)")
|
|
515
|
+
public ReservationResponse getReservation(@PathVariable String reservationId) {
|
|
516
|
+
// 実装
|
|
517
|
+
}
|
|
518
|
+
|
|
519
|
+
// 監査ログ
|
|
520
|
+
@EventListener
|
|
521
|
+
public class AuditEventListener {
|
|
522
|
+
|
|
523
|
+
@Async
|
|
524
|
+
@EventListener
|
|
525
|
+
public void handleAuditEvent(AuditEvent event) {
|
|
526
|
+
AuditLog auditLog = AuditLog.builder()
|
|
527
|
+
.userId(event.getPrincipal())
|
|
528
|
+
.action(event.getType())
|
|
529
|
+
.resource(event.getData().toString())
|
|
530
|
+
.timestamp(LocalDateTime.now())
|
|
531
|
+
.ipAddress(getClientIpAddress())
|
|
532
|
+
.userAgent(getUserAgent())
|
|
533
|
+
.build();
|
|
534
|
+
|
|
535
|
+
auditLogRepository.save(auditLog);
|
|
536
|
+
}
|
|
537
|
+
}
|
|
538
|
+
```
|
|
539
|
+
|
|
540
|
+
#### 3.2 データ暗号化
|
|
541
|
+
|
|
542
|
+
```java
|
|
543
|
+
// 個人情報暗号化
|
|
544
|
+
@Entity
|
|
545
|
+
@Table(name = "users")
|
|
546
|
+
public class User {
|
|
547
|
+
|
|
548
|
+
@Id
|
|
549
|
+
private UUID id;
|
|
550
|
+
|
|
551
|
+
@Column(name = "username")
|
|
552
|
+
private String username;
|
|
553
|
+
|
|
554
|
+
@Convert(converter = EncryptedStringConverter.class)
|
|
555
|
+
@Column(name = "email")
|
|
556
|
+
private String email; // 暗号化保存
|
|
557
|
+
|
|
558
|
+
@Convert(converter = EncryptedStringConverter.class)
|
|
559
|
+
@Column(name = "full_name")
|
|
560
|
+
private String fullName; // 暗号化保存
|
|
561
|
+
|
|
562
|
+
@Column(name = "password_hash")
|
|
563
|
+
private String passwordHash; // BCrypt ハッシュ
|
|
564
|
+
}
|
|
565
|
+
|
|
566
|
+
// 暗号化コンバーター
|
|
567
|
+
@Component
|
|
568
|
+
public class EncryptedStringConverter implements AttributeConverter<String, String> {
|
|
569
|
+
|
|
570
|
+
@Autowired
|
|
571
|
+
private AESEncryptionService encryptionService;
|
|
572
|
+
|
|
573
|
+
@Override
|
|
574
|
+
public String convertToDatabaseColumn(String attribute) {
|
|
575
|
+
return attribute == null ? null : encryptionService.encrypt(attribute);
|
|
576
|
+
}
|
|
577
|
+
|
|
578
|
+
@Override
|
|
579
|
+
public String convertToEntityAttribute(String dbData) {
|
|
580
|
+
return dbData == null ? null : encryptionService.decrypt(dbData);
|
|
581
|
+
}
|
|
582
|
+
}
|
|
583
|
+
```
|
|
584
|
+
|
|
585
|
+
### 4. 使用性要件(Usability Requirements)
|
|
586
|
+
|
|
587
|
+
#### 4.1 ユーザビリティ要件
|
|
588
|
+
|
|
589
|
+
```plantuml
|
|
590
|
+
@startuml
|
|
591
|
+
rectangle "ユーザビリティ要件" {
|
|
592
|
+
rectangle "操作性" {
|
|
593
|
+
- 3クリック原則: 主要機能は3クリック以内
|
|
594
|
+
- 予約完了時間: < 2分(初回ユーザー)
|
|
595
|
+
- 学習時間: < 30分(新規ユーザー)
|
|
596
|
+
- エラー回復: 1クリックで前の状態に復帰
|
|
597
|
+
}
|
|
598
|
+
|
|
599
|
+
rectangle "アクセシビリティ" {
|
|
600
|
+
- WCAG 2.1 AA: Web Content Accessibility Guidelines 準拠
|
|
601
|
+
- キーボード操作: 全機能をキーボードで操作可能
|
|
602
|
+
- スクリーンリーダー: 対応必須
|
|
603
|
+
- 色覚対応: カラーユニバーサルデザイン
|
|
604
|
+
}
|
|
605
|
+
|
|
606
|
+
rectangle "レスポンシブ" {
|
|
607
|
+
- デスクトップ: 1920x1080以上推奨
|
|
608
|
+
- タブレット: 768px以上対応
|
|
609
|
+
- スマートフォン: 375px以上対応
|
|
610
|
+
- ブラウザ: Chrome, Firefox, Safari, Edge 最新版
|
|
611
|
+
}
|
|
612
|
+
}
|
|
613
|
+
@enduml
|
|
614
|
+
```
|
|
615
|
+
|
|
616
|
+
**実装例**:
|
|
617
|
+
```tsx
|
|
618
|
+
// アクセシビリティ対応コンポーネント
|
|
619
|
+
interface ReservationFormProps {
|
|
620
|
+
onSubmit: (data: ReservationFormData) => void;
|
|
621
|
+
}
|
|
622
|
+
|
|
623
|
+
export const ReservationForm: React.FC<ReservationFormProps> = ({ onSubmit }) => {
|
|
624
|
+
return (
|
|
625
|
+
<form onSubmit={handleSubmit} role="form" aria-label="会議室予約フォーム">
|
|
626
|
+
<fieldset>
|
|
627
|
+
<legend>予約情報</legend>
|
|
628
|
+
|
|
629
|
+
<div className="form-group">
|
|
630
|
+
<label htmlFor="room-select" className="required">
|
|
631
|
+
会議室選択
|
|
632
|
+
</label>
|
|
633
|
+
<select
|
|
634
|
+
id="room-select"
|
|
635
|
+
aria-required="true"
|
|
636
|
+
aria-describedby="room-help"
|
|
637
|
+
{...register('roomId', { required: '会議室を選択してください' })}
|
|
638
|
+
>
|
|
639
|
+
<option value="">選択してください</option>
|
|
640
|
+
{rooms.map(room => (
|
|
641
|
+
<option key={room.id} value={room.id}>
|
|
642
|
+
{room.name} (収容人数: {room.capacity}名)
|
|
643
|
+
</option>
|
|
644
|
+
))}
|
|
645
|
+
</select>
|
|
646
|
+
<div id="room-help" className="help-text">
|
|
647
|
+
希望する会議室を選択してください
|
|
648
|
+
</div>
|
|
649
|
+
{errors.roomId && (
|
|
650
|
+
<div role="alert" className="error-message">
|
|
651
|
+
{errors.roomId.message}
|
|
652
|
+
</div>
|
|
653
|
+
)}
|
|
654
|
+
</div>
|
|
655
|
+
|
|
656
|
+
<div className="form-group">
|
|
657
|
+
<label htmlFor="date-input" className="required">
|
|
658
|
+
利用日
|
|
659
|
+
</label>
|
|
660
|
+
<input
|
|
661
|
+
type="date"
|
|
662
|
+
id="date-input"
|
|
663
|
+
aria-required="true"
|
|
664
|
+
min={new Date().toISOString().split('T')[0]}
|
|
665
|
+
{...register('date', { required: '利用日を選択してください' })}
|
|
666
|
+
/>
|
|
667
|
+
</div>
|
|
668
|
+
|
|
669
|
+
<button
|
|
670
|
+
type="submit"
|
|
671
|
+
disabled={isSubmitting}
|
|
672
|
+
aria-describedby="submit-help"
|
|
673
|
+
>
|
|
674
|
+
{isSubmitting ? '予約中...' : '予約する'}
|
|
675
|
+
</button>
|
|
676
|
+
<div id="submit-help" className="help-text">
|
|
677
|
+
入力内容を確認して予約ボタンをクリックしてください
|
|
678
|
+
</div>
|
|
679
|
+
</fieldset>
|
|
680
|
+
</form>
|
|
681
|
+
);
|
|
682
|
+
};
|
|
683
|
+
|
|
684
|
+
// レスポンシブデザイン
|
|
685
|
+
const ReservationCard = styled.div`
|
|
686
|
+
padding: 1rem;
|
|
687
|
+
border: 1px solid #e2e8f0;
|
|
688
|
+
border-radius: 0.5rem;
|
|
689
|
+
|
|
690
|
+
@media (min-width: 768px) {
|
|
691
|
+
padding: 1.5rem;
|
|
692
|
+
display: grid;
|
|
693
|
+
grid-template-columns: 1fr auto;
|
|
694
|
+
gap: 1rem;
|
|
695
|
+
}
|
|
696
|
+
|
|
697
|
+
@media (min-width: 1024px) {
|
|
698
|
+
padding: 2rem;
|
|
699
|
+
}
|
|
700
|
+
`;
|
|
701
|
+
```
|
|
702
|
+
|
|
703
|
+
#### 4.2 ユーザーエクスペリエンス指標
|
|
704
|
+
|
|
705
|
+
```javascript
|
|
706
|
+
// Core Web Vitals 測定
|
|
707
|
+
import { getCLS, getFID, getFCP, getLCP, getTTFB } from 'web-vitals';
|
|
708
|
+
|
|
709
|
+
function sendToAnalytics(metric) {
|
|
710
|
+
// 分析ツールへの送信
|
|
711
|
+
analytics.track('Performance Metric', {
|
|
712
|
+
name: metric.name,
|
|
713
|
+
value: metric.value,
|
|
714
|
+
rating: metric.rating
|
|
715
|
+
});
|
|
716
|
+
}
|
|
717
|
+
|
|
718
|
+
// 各指標の測定
|
|
719
|
+
getCLS(sendToAnalytics); // Cumulative Layout Shift < 0.1
|
|
720
|
+
getFID(sendToAnalytics); // First Input Delay < 100ms
|
|
721
|
+
getFCP(sendToAnalytics); // First Contentful Paint < 1.8s
|
|
722
|
+
getLCP(sendToAnalytics); // Largest Contentful Paint < 2.5s
|
|
723
|
+
getTTFB(sendToAnalytics); // Time to First Byte < 0.8s
|
|
724
|
+
|
|
725
|
+
// ユーザー行動分析
|
|
726
|
+
const trackUserInteraction = (action, element, duration) => {
|
|
727
|
+
analytics.track('User Interaction', {
|
|
728
|
+
action,
|
|
729
|
+
element,
|
|
730
|
+
duration,
|
|
731
|
+
timestamp: new Date().toISOString(),
|
|
732
|
+
userId: getCurrentUserId()
|
|
733
|
+
});
|
|
734
|
+
};
|
|
735
|
+
|
|
736
|
+
// 予約プロセス分析
|
|
737
|
+
const reservationFlowTracking = {
|
|
738
|
+
searchStart: () => trackUserInteraction('search_start', 'room_search', 0),
|
|
739
|
+
searchComplete: (duration) => trackUserInteraction('search_complete', 'room_search', duration),
|
|
740
|
+
reservationStart: () => trackUserInteraction('reservation_start', 'reservation_form', 0),
|
|
741
|
+
reservationComplete: (duration) => trackUserInteraction('reservation_complete', 'reservation_form', duration),
|
|
742
|
+
reservationAbandoned: (step, duration) => trackUserInteraction('reservation_abandoned', step, duration)
|
|
743
|
+
};
|
|
744
|
+
```
|
|
745
|
+
|
|
746
|
+
### 5. 保守性・拡張性要件(Maintainability & Scalability Requirements)
|
|
747
|
+
|
|
748
|
+
#### 5.1 保守性要件
|
|
749
|
+
|
|
750
|
+
```plantuml
|
|
751
|
+
@startuml
|
|
752
|
+
rectangle "保守性要件" {
|
|
753
|
+
rectangle "コード品質" {
|
|
754
|
+
- テストカバレッジ: > 80%(ドメイン層 > 90%)
|
|
755
|
+
- 複雑度: Cyclomatic Complexity < 10
|
|
756
|
+
- 重複コード: < 3%
|
|
757
|
+
- 技術的負債: SonarQube Rating A
|
|
758
|
+
}
|
|
759
|
+
|
|
760
|
+
rectangle "ドキュメント" {
|
|
761
|
+
- API ドキュメント: OpenAPI 3.0 準拠
|
|
762
|
+
- アーキテクチャ図: C4 モデル
|
|
763
|
+
- 運用手順書: 障害対応・デプロイ手順
|
|
764
|
+
- 更新頻度: コード変更と同時更新
|
|
765
|
+
}
|
|
766
|
+
|
|
767
|
+
rectangle "監視・ログ" {
|
|
768
|
+
- 構造化ログ: JSON形式、分析可能
|
|
769
|
+
- メトリクス: Micrometer + Prometheus
|
|
770
|
+
- 分散トレーシング: Jaeger 対応
|
|
771
|
+
- アラート: SLA 違反時の自動通知
|
|
772
|
+
}
|
|
773
|
+
}
|
|
774
|
+
@enduml
|
|
775
|
+
```
|
|
776
|
+
|
|
777
|
+
**実装例**:
|
|
778
|
+
```java
|
|
779
|
+
// 構造化ログ
|
|
780
|
+
@RestController
|
|
781
|
+
@Slf4j
|
|
782
|
+
public class ReservationController {
|
|
783
|
+
|
|
784
|
+
@PostMapping("/reservations")
|
|
785
|
+
public ResponseEntity<ReservationResponse> createReservation(
|
|
786
|
+
@RequestBody CreateReservationRequest request,
|
|
787
|
+
HttpServletRequest httpRequest) {
|
|
788
|
+
|
|
789
|
+
MDC.put("userId", getCurrentUserId());
|
|
790
|
+
MDC.put("correlationId", UUID.randomUUID().toString());
|
|
791
|
+
MDC.put("ipAddress", getClientIpAddress(httpRequest));
|
|
792
|
+
|
|
793
|
+
try {
|
|
794
|
+
log.info("Creating reservation: roomId={}, startTime={}",
|
|
795
|
+
request.getRoomId(), request.getStartTime());
|
|
796
|
+
|
|
797
|
+
ReservationId reservationId = createReservationUseCase.execute(
|
|
798
|
+
mapToCommand(request));
|
|
799
|
+
|
|
800
|
+
log.info("Reservation created successfully: reservationId={}", reservationId);
|
|
801
|
+
|
|
802
|
+
return ResponseEntity.status(HttpStatus.CREATED)
|
|
803
|
+
.body(ReservationResponse.from(reservationId));
|
|
804
|
+
|
|
805
|
+
} catch (Exception e) {
|
|
806
|
+
log.error("Failed to create reservation", e);
|
|
807
|
+
throw e;
|
|
808
|
+
} finally {
|
|
809
|
+
MDC.clear();
|
|
810
|
+
}
|
|
811
|
+
}
|
|
812
|
+
}
|
|
813
|
+
|
|
814
|
+
// メトリクス収集
|
|
815
|
+
@Component
|
|
816
|
+
@Service
|
|
817
|
+
public class ReservationMetricsService {
|
|
818
|
+
|
|
819
|
+
private final MeterRegistry meterRegistry;
|
|
820
|
+
private final Counter reservationCreatedCounter;
|
|
821
|
+
private final Counter reservationFailedCounter;
|
|
822
|
+
private final Timer reservationProcessingTimer;
|
|
823
|
+
|
|
824
|
+
public ReservationMetricsService(MeterRegistry meterRegistry) {
|
|
825
|
+
this.meterRegistry = meterRegistry;
|
|
826
|
+
this.reservationCreatedCounter = Counter.builder("reservation.created")
|
|
827
|
+
.description("Created reservations count")
|
|
828
|
+
.register(meterRegistry);
|
|
829
|
+
this.reservationFailedCounter = Counter.builder("reservation.failed")
|
|
830
|
+
.description("Failed reservations count")
|
|
831
|
+
.register(meterRegistry);
|
|
832
|
+
this.reservationProcessingTimer = Timer.builder("reservation.processing")
|
|
833
|
+
.description("Reservation processing time")
|
|
834
|
+
.register(meterRegistry);
|
|
835
|
+
}
|
|
836
|
+
|
|
837
|
+
public void recordReservationCreated() {
|
|
838
|
+
reservationCreatedCounter.increment();
|
|
839
|
+
}
|
|
840
|
+
|
|
841
|
+
public void recordReservationFailed(String reason) {
|
|
842
|
+
reservationFailedCounter.increment(Tags.of("reason", reason));
|
|
843
|
+
}
|
|
844
|
+
|
|
845
|
+
public Timer.Sample startProcessingTimer() {
|
|
846
|
+
return Timer.start(meterRegistry);
|
|
847
|
+
}
|
|
848
|
+
}
|
|
849
|
+
```
|
|
850
|
+
|
|
851
|
+
#### 5.2 拡張性要件
|
|
852
|
+
|
|
853
|
+
```java
|
|
854
|
+
// 水平スケーリング対応
|
|
855
|
+
@Configuration
|
|
856
|
+
@EnableCaching
|
|
857
|
+
public class CacheConfig {
|
|
858
|
+
|
|
859
|
+
@Bean
|
|
860
|
+
public CacheManager cacheManager() {
|
|
861
|
+
RedisCacheManager.Builder builder = RedisCacheManager
|
|
862
|
+
.RedisCacheManagerBuilder
|
|
863
|
+
.fromConnectionFactory(jedisConnectionFactory())
|
|
864
|
+
.cacheDefaults(cacheConfiguration());
|
|
865
|
+
|
|
866
|
+
return builder.build();
|
|
867
|
+
}
|
|
868
|
+
|
|
869
|
+
private RedisCacheConfiguration cacheConfiguration() {
|
|
870
|
+
return RedisCacheConfiguration.defaultCacheConfig()
|
|
871
|
+
.entryTtl(Duration.ofMinutes(10))
|
|
872
|
+
.serializeKeysWith(RedisSerializationContext.SerializationPair
|
|
873
|
+
.fromSerializer(new StringRedisSerializer()))
|
|
874
|
+
.serializeValuesWith(RedisSerializationContext.SerializationPair
|
|
875
|
+
.fromSerializer(new GenericJackson2JsonRedisSerializer()));
|
|
876
|
+
}
|
|
877
|
+
}
|
|
878
|
+
|
|
879
|
+
// データベースパーティショニング
|
|
880
|
+
@Entity
|
|
881
|
+
@Table(name = "reservations")
|
|
882
|
+
@PartitionKey("DATE(start_time)") // 日付によるパーティショニング
|
|
883
|
+
public class ReservationEntity {
|
|
884
|
+
// エンティティ定義
|
|
885
|
+
}
|
|
886
|
+
|
|
887
|
+
// マイクロサービス化の準備
|
|
888
|
+
@FeignClient(name = "notification-service", url = "${services.notification.url}")
|
|
889
|
+
public interface NotificationServiceClient {
|
|
890
|
+
|
|
891
|
+
@PostMapping("/notifications")
|
|
892
|
+
ResponseEntity<Void> sendNotification(@RequestBody NotificationRequest request);
|
|
893
|
+
}
|
|
894
|
+
```
|
|
895
|
+
|
|
896
|
+
### 6. 運用要件(Operational Requirements)
|
|
897
|
+
|
|
898
|
+
#### 6.1 監視・運用要件
|
|
899
|
+
|
|
900
|
+
```yaml
|
|
901
|
+
# Kubernetes デプロイメント設定
|
|
902
|
+
apiVersion: apps/v1
|
|
903
|
+
kind: Deployment
|
|
904
|
+
metadata:
|
|
905
|
+
name: meeting-room-app
|
|
906
|
+
spec:
|
|
907
|
+
replicas: 3
|
|
908
|
+
selector:
|
|
909
|
+
matchLabels:
|
|
910
|
+
app: meeting-room-app
|
|
911
|
+
template:
|
|
912
|
+
spec:
|
|
913
|
+
containers:
|
|
914
|
+
- name: app
|
|
915
|
+
image: meeting-room-app:latest
|
|
916
|
+
ports:
|
|
917
|
+
- containerPort: 8080
|
|
918
|
+
resources:
|
|
919
|
+
requests:
|
|
920
|
+
memory: "512Mi"
|
|
921
|
+
cpu: "500m"
|
|
922
|
+
limits:
|
|
923
|
+
memory: "1Gi"
|
|
924
|
+
cpu: "1000m"
|
|
925
|
+
livenessProbe:
|
|
926
|
+
httpGet:
|
|
927
|
+
path: /actuator/health/liveness
|
|
928
|
+
port: 8080
|
|
929
|
+
initialDelaySeconds: 60
|
|
930
|
+
periodSeconds: 30
|
|
931
|
+
readinessProbe:
|
|
932
|
+
httpGet:
|
|
933
|
+
path: /actuator/health/readiness
|
|
934
|
+
port: 8080
|
|
935
|
+
initialDelaySeconds: 30
|
|
936
|
+
periodSeconds: 10
|
|
937
|
+
env:
|
|
938
|
+
- name: JAVA_OPTS
|
|
939
|
+
value: "-Xms512m -Xmx1g -XX:+UseG1GC"
|
|
940
|
+
- name: SPRING_PROFILES_ACTIVE
|
|
941
|
+
value: "production"
|
|
942
|
+
```
|
|
943
|
+
|
|
944
|
+
```java
|
|
945
|
+
// カスタムヘルスチェック
|
|
946
|
+
@Component
|
|
947
|
+
public class ReservationSystemHealthIndicator implements HealthIndicator {
|
|
948
|
+
|
|
949
|
+
private final ReservationRepository reservationRepository;
|
|
950
|
+
private final NotificationServiceClient notificationService;
|
|
951
|
+
|
|
952
|
+
@Override
|
|
953
|
+
public Health health() {
|
|
954
|
+
Health.Builder status = Health.up();
|
|
955
|
+
|
|
956
|
+
try {
|
|
957
|
+
// データベース接続確認
|
|
958
|
+
long activeReservationsCount = reservationRepository.countActiveReservations();
|
|
959
|
+
status.withDetail("database", Map.of(
|
|
960
|
+
"status", "UP",
|
|
961
|
+
"activeReservations", activeReservationsCount
|
|
962
|
+
));
|
|
963
|
+
|
|
964
|
+
// 外部サービス接続確認
|
|
965
|
+
boolean notificationServiceUp = checkNotificationService();
|
|
966
|
+
status.withDetail("notificationService", Map.of(
|
|
967
|
+
"status", notificationServiceUp ? "UP" : "DOWN"
|
|
968
|
+
));
|
|
969
|
+
|
|
970
|
+
// システムリソース確認
|
|
971
|
+
status.withDetail("system", getSystemMetrics());
|
|
972
|
+
|
|
973
|
+
} catch (Exception e) {
|
|
974
|
+
return Health.down(e).build();
|
|
975
|
+
}
|
|
976
|
+
|
|
977
|
+
return status.build();
|
|
978
|
+
}
|
|
979
|
+
|
|
980
|
+
private Map<String, Object> getSystemMetrics() {
|
|
981
|
+
Runtime runtime = Runtime.getRuntime();
|
|
982
|
+
long maxMemory = runtime.maxMemory();
|
|
983
|
+
long totalMemory = runtime.totalMemory();
|
|
984
|
+
long freeMemory = runtime.freeMemory();
|
|
985
|
+
|
|
986
|
+
return Map.of(
|
|
987
|
+
"memory.max", maxMemory,
|
|
988
|
+
"memory.total", totalMemory,
|
|
989
|
+
"memory.free", freeMemory,
|
|
990
|
+
"memory.used", totalMemory - freeMemory,
|
|
991
|
+
"processors", runtime.availableProcessors()
|
|
992
|
+
);
|
|
993
|
+
}
|
|
994
|
+
}
|
|
995
|
+
```
|
|
996
|
+
|
|
997
|
+
## 非機能要件の測定と検証
|
|
998
|
+
|
|
999
|
+
### 1. パフォーマンステスト
|
|
1000
|
+
|
|
1001
|
+
```java
|
|
1002
|
+
// JMeter 負荷テストシナリオ
|
|
1003
|
+
@Test
|
|
1004
|
+
public class PerformanceTest {
|
|
1005
|
+
|
|
1006
|
+
@LoadTest(
|
|
1007
|
+
users = 100,
|
|
1008
|
+
rampUp = 60, // 60秒で100ユーザーまで増加
|
|
1009
|
+
duration = 300 // 5分間実行
|
|
1010
|
+
)
|
|
1011
|
+
public void 予約作成負荷テスト() {
|
|
1012
|
+
// 予約作成APIの負荷テスト
|
|
1013
|
+
given()
|
|
1014
|
+
.contentType(ContentType.JSON)
|
|
1015
|
+
.body(createReservationRequest())
|
|
1016
|
+
.when()
|
|
1017
|
+
.post("/api/reservations")
|
|
1018
|
+
.then()
|
|
1019
|
+
.statusCode(201)
|
|
1020
|
+
.time(lessThan(2000L)); // 2秒以内
|
|
1021
|
+
}
|
|
1022
|
+
|
|
1023
|
+
@PerformanceTest
|
|
1024
|
+
@JvmOptions({"-Xms1g", "-Xmx2g", "-XX:+UseG1GC"})
|
|
1025
|
+
public void メモリ使用量テスト() {
|
|
1026
|
+
// メモリ使用量とGCの影響を測定
|
|
1027
|
+
for (int i = 0; i < 10000; i++) {
|
|
1028
|
+
createAndProcessReservation();
|
|
1029
|
+
}
|
|
1030
|
+
|
|
1031
|
+
// メモリ使用量を確認
|
|
1032
|
+
assertThat(getUsedMemoryMB()).isLessThan(1500); // 1.5GB以下
|
|
1033
|
+
assertThat(getGcPauseTimeMs()).isLessThan(100); // GC停止時間100ms以下
|
|
1034
|
+
}
|
|
1035
|
+
}
|
|
1036
|
+
```
|
|
1037
|
+
|
|
1038
|
+
### 2. セキュリティテスト
|
|
1039
|
+
|
|
1040
|
+
```java
|
|
1041
|
+
// セキュリティテスト
|
|
1042
|
+
@SpringBootTest
|
|
1043
|
+
@AutoConfigureMockMvc
|
|
1044
|
+
public class SecurityTest {
|
|
1045
|
+
|
|
1046
|
+
@Test
|
|
1047
|
+
public void 認証なしでアクセスすると401エラー() throws Exception {
|
|
1048
|
+
mockMvc.perform(get("/api/reservations"))
|
|
1049
|
+
.andExpect(status().isUnauthorized());
|
|
1050
|
+
}
|
|
1051
|
+
|
|
1052
|
+
@Test
|
|
1053
|
+
public void 無効なJWTトークンで401エラー() throws Exception {
|
|
1054
|
+
mockMvc.perform(get("/api/reservations")
|
|
1055
|
+
.header("Authorization", "Bearer invalid-token"))
|
|
1056
|
+
.andExpected(status().isUnauthorized());
|
|
1057
|
+
}
|
|
1058
|
+
|
|
1059
|
+
@Test
|
|
1060
|
+
@WithMockUser(roles = "USER")
|
|
1061
|
+
public void 管理者権限なしで管理APIにアクセスすると403エラー() throws Exception {
|
|
1062
|
+
mockMvc.perform(get("/api/admin/users"))
|
|
1063
|
+
.andExpect(status().isForbidden());
|
|
1064
|
+
}
|
|
1065
|
+
|
|
1066
|
+
@Test
|
|
1067
|
+
public void SQLインジェクション攻撃をブロック() throws Exception {
|
|
1068
|
+
String maliciousInput = "'; DROP TABLE users; --";
|
|
1069
|
+
|
|
1070
|
+
mockMvc.perform(get("/api/reservations")
|
|
1071
|
+
.param("search", maliciousInput))
|
|
1072
|
+
.andExpect(status().isBadRequest())
|
|
1073
|
+
.andExpect(jsonPath("$.error").value("INVALID_INPUT"));
|
|
1074
|
+
}
|
|
1075
|
+
}
|
|
1076
|
+
```
|
|
1077
|
+
|
|
1078
|
+
### 3. ユーザビリティテスト
|
|
1079
|
+
|
|
1080
|
+
```javascript
|
|
1081
|
+
// Cypress E2Eテスト
|
|
1082
|
+
describe('予約作成フロー', () => {
|
|
1083
|
+
it('新規ユーザーでも2分以内に予約完了できる', () => {
|
|
1084
|
+
const startTime = Date.now();
|
|
1085
|
+
|
|
1086
|
+
cy.visit('/login');
|
|
1087
|
+
cy.get('[data-cy=username]').type('newuser@example.com');
|
|
1088
|
+
cy.get('[data-cy=password]').type('password123');
|
|
1089
|
+
cy.get('[data-cy=login-button]').click();
|
|
1090
|
+
|
|
1091
|
+
// 会議室検索
|
|
1092
|
+
cy.get('[data-cy=room-search]').click();
|
|
1093
|
+
cy.get('[data-cy=capacity-filter]').select('10');
|
|
1094
|
+
cy.get('[data-cy=date-picker]').type('2024-12-01');
|
|
1095
|
+
cy.get('[data-cy=search-button]').click();
|
|
1096
|
+
|
|
1097
|
+
// 予約作成
|
|
1098
|
+
cy.get('[data-cy=room-card]').first().click();
|
|
1099
|
+
cy.get('[data-cy=time-slot]').contains('10:00-12:00').click();
|
|
1100
|
+
cy.get('[data-cy=purpose]').type('定例会議');
|
|
1101
|
+
cy.get('[data-cy=reserve-button]').click();
|
|
1102
|
+
|
|
1103
|
+
// 完了確認
|
|
1104
|
+
cy.get('[data-cy=success-message]').should('be.visible');
|
|
1105
|
+
|
|
1106
|
+
const endTime = Date.now();
|
|
1107
|
+
const duration = (endTime - startTime) / 1000; // 秒換算
|
|
1108
|
+
|
|
1109
|
+
expect(duration).to.be.lessThan(120); // 2分以内
|
|
1110
|
+
});
|
|
1111
|
+
|
|
1112
|
+
it('アクセシビリティ基準を満たす', () => {
|
|
1113
|
+
cy.visit('/reservations');
|
|
1114
|
+
cy.injectAxe(); // axe-core ライブラリ
|
|
1115
|
+
|
|
1116
|
+
cy.checkA11y(null, {
|
|
1117
|
+
rules: {
|
|
1118
|
+
'color-contrast': { enabled: true },
|
|
1119
|
+
'keyboard-navigation': { enabled: true },
|
|
1120
|
+
'focus-management': { enabled: true }
|
|
1121
|
+
}
|
|
1122
|
+
});
|
|
1123
|
+
});
|
|
1124
|
+
});
|
|
1125
|
+
```
|
|
1126
|
+
|
|
1127
|
+
### 4. 継続的監視
|
|
1128
|
+
|
|
1129
|
+
```yaml
|
|
1130
|
+
# Prometheus 監視ルール
|
|
1131
|
+
groups:
|
|
1132
|
+
- name: meeting-room-system
|
|
1133
|
+
rules:
|
|
1134
|
+
- alert: HighResponseTime
|
|
1135
|
+
expr: histogram_quantile(0.95, rate(http_request_duration_seconds_bucket[5m])) > 2
|
|
1136
|
+
for: 2m
|
|
1137
|
+
labels:
|
|
1138
|
+
severity: warning
|
|
1139
|
+
annotations:
|
|
1140
|
+
summary: "API応答時間が遅延"
|
|
1141
|
+
description: "95パーセンタイルの応答時間が2秒を超えています"
|
|
1142
|
+
|
|
1143
|
+
- alert: HighErrorRate
|
|
1144
|
+
expr: rate(http_requests_total{status=~"5.."}[5m]) > 0.05
|
|
1145
|
+
for: 1m
|
|
1146
|
+
labels:
|
|
1147
|
+
severity: critical
|
|
1148
|
+
annotations:
|
|
1149
|
+
summary: "エラー率が高い"
|
|
1150
|
+
description: "5xx エラーの発生率が5%を超えています"
|
|
1151
|
+
|
|
1152
|
+
- alert: DatabaseConnectionFailure
|
|
1153
|
+
expr: health_database_status != 1
|
|
1154
|
+
for: 30s
|
|
1155
|
+
labels:
|
|
1156
|
+
severity: critical
|
|
1157
|
+
annotations:
|
|
1158
|
+
summary: "データベース接続失敗"
|
|
1159
|
+
description: "データベースへの接続が失敗しています"
|
|
1160
|
+
```
|
|
1161
|
+
|
|
1162
|
+
## 非機能要件のライフサイクル管理
|
|
1163
|
+
|
|
1164
|
+
### 1. 要件定義フェーズ
|
|
1165
|
+
|
|
1166
|
+
- ステークホルダー特定
|
|
1167
|
+
- 品質属性の優先順位付け
|
|
1168
|
+
- 測定可能な目標値設定
|
|
1169
|
+
- 検証方法の決定
|
|
1170
|
+
- 受け入れ基準の合意
|
|
1171
|
+
|
|
1172
|
+
### 2. 設計・実装フェーズ
|
|
1173
|
+
|
|
1174
|
+
- アーキテクチャ設計への反映
|
|
1175
|
+
- 非機能要件を満たす技術選択
|
|
1176
|
+
- 測定・監視機能の実装
|
|
1177
|
+
- テスト戦略への組み込み
|
|
1178
|
+
|
|
1179
|
+
### 3. テスト・検証フェーズ
|
|
1180
|
+
|
|
1181
|
+
- パフォーマンステスト実行
|
|
1182
|
+
- セキュリティテスト実行
|
|
1183
|
+
- ユーザビリティテスト実行
|
|
1184
|
+
- 受け入れ基準の検証
|
|
1185
|
+
- 改善点の特定と対策
|
|
1186
|
+
|
|
1187
|
+
### 4. 運用フェーズ
|
|
1188
|
+
|
|
1189
|
+
- 監視システム構築
|
|
1190
|
+
- SLAの設定
|
|
1191
|
+
- 定期的なパフォーマンス測定
|
|
1192
|
+
- ユーザーフィードバック収集
|
|
1193
|
+
- 継続的改善
|
|
1194
|
+
|
|
1195
|
+
## まとめ
|
|
1196
|
+
|
|
1197
|
+
### 非機能要件定義の要点
|
|
1198
|
+
|
|
1199
|
+
1. **品質属性の明確化**
|
|
1200
|
+
- ISO/IEC 25010 に基づく体系的な分類
|
|
1201
|
+
- 測定可能な目標値の設定
|
|
1202
|
+
- ステークホルダーとの合意形成
|
|
1203
|
+
|
|
1204
|
+
2. **実装と検証の統合**
|
|
1205
|
+
- アーキテクチャ設計への反映
|
|
1206
|
+
- 自動テストによる継続的検証
|
|
1207
|
+
- 運用監視との連携
|
|
1208
|
+
|
|
1209
|
+
3. **継続的改善**
|
|
1210
|
+
- 実運用データに基づく評価
|
|
1211
|
+
- ユーザーフィードバックの活用
|
|
1212
|
+
- 技術進歩に応じた要件見直し
|
|
1213
|
+
|
|
1214
|
+
4. **リスク管理**
|
|
1215
|
+
- 品質属性間のトレードオフの認識
|
|
1216
|
+
- 優先順位に基づく段階的実装
|
|
1217
|
+
- 早期の問題発見と対策
|
|
1218
|
+
|
|
1219
|
+
### 非機能要件がもたらす価値
|
|
1220
|
+
|
|
1221
|
+
**短期的価値**:
|
|
1222
|
+
- システムの安定性と信頼性の確保
|
|
1223
|
+
- ユーザー満足度の向上
|
|
1224
|
+
- 運用コストの削減
|
|
1225
|
+
|
|
1226
|
+
**長期的価値**:
|
|
1227
|
+
- 保守性による開発効率の維持
|
|
1228
|
+
- 拡張性による事業成長への対応
|
|
1229
|
+
- セキュリティによるリスク回避
|
|
1230
|
+
|
|
1231
|
+
非機能要件は、変更を楽に安全にできて役に立つソフトウェアを実現するための基盤である。適切な定義と継続的な管理により、持続的な価値提供を可能にする。
|