@walwal-harness/cli 4.0.0-alpha.2 → 4.0.0-alpha.20
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 +226 -273
- package/assets/templates/config.json +1 -48
- package/assets/templates/gitignore-append.txt +1 -0
- package/bin/init.js +1 -0
- package/package.json +1 -1
- package/scripts/harness-dashboard-v4.sh +58 -81
- package/scripts/harness-next.sh +4 -15
- package/scripts/harness-prompts-v4.sh +106 -0
- package/scripts/harness-queue-manager.sh +59 -5
- package/scripts/harness-session-start.sh +18 -0
- package/scripts/harness-studio-v4.sh +69 -69
- package/scripts/harness-team-worker.sh +136 -123
- package/scripts/harness-user-prompt-submit.sh +31 -1
- package/skills/dispatcher/SKILL.md +7 -2
- package/skills/evaluator-functional-flutter/SKILL.md +0 -206
- package/skills/evaluator-functional-flutter/references/ia-compliance.md +0 -77
- package/skills/evaluator-functional-flutter/references/scoring-rubric.md +0 -132
- package/skills/evaluator-functional-flutter/references/static-check-rules.md +0 -99
- package/skills/generator-frontend-flutter/SKILL.md +0 -173
- package/skills/generator-frontend-flutter/references/anti-patterns.md +0 -320
- package/skills/generator-frontend-flutter/references/api-layer-pattern.md +0 -233
- package/skills/generator-frontend-flutter/references/flutter-web-pattern.md +0 -273
- package/skills/generator-frontend-flutter/references/i18n-pattern.md +0 -102
- package/skills/generator-frontend-flutter/references/riverpod-pattern.md +0 -199
|
@@ -1,173 +0,0 @@
|
|
|
1
|
-
---
|
|
2
|
-
name: harness-generator-frontend-flutter
|
|
3
|
-
description: "하네스 Flutter Frontend Generator. Flutter(Dart) + Riverpod + integrated_data_layer(Retrofit) 기반 모바일 앱을 구현한다. ARB 다국어, JsonSerializable, NotifierProvider 패턴 준수. api-contract.json이 Source of Truth — UI 추론 금지."
|
|
4
|
-
disable-model-invocation: true
|
|
5
|
-
---
|
|
6
|
-
|
|
7
|
-
# Generator-Frontend-Flutter — Dart + Riverpod + integrated_data_layer
|
|
8
|
-
|
|
9
|
-
## Session Boundary Protocol
|
|
10
|
-
|
|
11
|
-
### On Start
|
|
12
|
-
1. `.harness/progress.json` 읽기 — `next_agent`가 `"generator-frontend-flutter"`인지 확인
|
|
13
|
-
2. `.harness/actions/pipeline.json`의 `fe_stack == "flutter"` 재확인 — 아니면 즉시 STOP + 사용자에게 불일치 보고
|
|
14
|
-
3. progress.json 업데이트: `current_agent` → `"generator-frontend-flutter"`, `agent_status` → `"running"`, `updated_at` 갱신
|
|
15
|
-
4. `failure` 필드 확인 — retry인 경우 평가 문서의 실패 사유 우선 읽기
|
|
16
|
-
|
|
17
|
-
### On Complete
|
|
18
|
-
1. progress.json 업데이트:
|
|
19
|
-
- `agent_status` → `"completed"`
|
|
20
|
-
- `completed_agents`에 `"generator-frontend-flutter"` 추가
|
|
21
|
-
- `next_agent` → `"evaluator-functional-flutter"`
|
|
22
|
-
- `failure` 필드 초기화
|
|
23
|
-
2. `feature-list.json`의 해당 feature `passes`에 `"generator-frontend-flutter"` 추가
|
|
24
|
-
3. `.harness/progress.log`에 요약 추가
|
|
25
|
-
4. **STOP. 다음 에이전트를 직접 호출하지 않는다.**
|
|
26
|
-
5. 출력: `"✓ Generator-Frontend-Flutter 완료. bash scripts/harness-next.sh 실행하여 다음 단계 확인."`
|
|
27
|
-
|
|
28
|
-
## Startup
|
|
29
|
-
|
|
30
|
-
1. `AGENTS.md` 읽기 — IA-MAP, 권한 확인 (Flutter 소유 경로)
|
|
31
|
-
2. `.harness/gotchas/generator-frontend-flutter.md` 읽기 (없으면 skip) — **과거 실수 반복 금지**
|
|
32
|
-
3. `.harness/memory.md` 읽기 — **프로젝트 공유 학습 규칙 적용**
|
|
33
|
-
4. `pwd` + `.harness/progress.json` + `git log --oneline -20`
|
|
34
|
-
5. `.harness/actions/api-contract.json` 읽기 — **서버 API 계약이 Source of Truth**
|
|
35
|
-
6. `.harness/actions/feature-list.json` — `layer: "frontend"` 필터
|
|
36
|
-
7. `.harness/actions/pipeline.json` 의 **`fe_target`** 확인 (`web` | `mobile` | `desktop`)
|
|
37
|
-
- 이 값에 따라 허용되는 API, 빌드 명령, 안티패턴이 달라진다 (아래 "fe_target 분기" 섹션 참조)
|
|
38
|
-
8. `pubspec.yaml` 확인 — Flutter 버전, Riverpod/Retrofit/json_serializable 의존성 존재 확인
|
|
39
|
-
- `fe_target = web` 인 경우 `flutter config --enable-web` 활성화 + `web/index.html` 존재 확인
|
|
40
|
-
|
|
41
|
-
## AGENTS.md — 읽기 전용
|
|
42
|
-
|
|
43
|
-
`[FE]` + `→ Generator-Frontend-Flutter` 소유 경로만 쓰기 가능. 일반적으로:
|
|
44
|
-
- `lib/ui/pages/`, `lib/ui/component/`, `lib/l10n/`
|
|
45
|
-
- `integrated_data_layer/lib/`, `integrated_data_layer/test/`
|
|
46
|
-
- `assets/strings/`
|
|
47
|
-
|
|
48
|
-
Backend 코드, `.harness/`, `AGENTS.md` 수정 금지.
|
|
49
|
-
|
|
50
|
-
## Sprint Workflow
|
|
51
|
-
|
|
52
|
-
1. **Sprint Contract FE 섹션 추가** — 페이지, VM, API 연동, 성공 기준
|
|
53
|
-
2. **api-contract.json → Retrofit/JsonSerializable 변환**
|
|
54
|
-
3. **구현** — 아래 4개 레퍼런스를 반드시 참조
|
|
55
|
-
4. **코드 생성**: `flutter pub run build_runner build --delete-conflicting-outputs`
|
|
56
|
-
5. **Self-Verification** — `flutter analyze` + `flutter test` 통과
|
|
57
|
-
6. **Handoff** → Evaluator-Functional-Flutter
|
|
58
|
-
|
|
59
|
-
## 개발론 레퍼런스 (점진적 로딩)
|
|
60
|
-
|
|
61
|
-
| 문서 | 내용 | 언제 로드 |
|
|
62
|
-
|------|------|----------|
|
|
63
|
-
| [API Layer Pattern](references/api-layer-pattern.md) | integrated_data_layer 구조, Request/Response, Retrofit | API 연동 시 |
|
|
64
|
-
| [Riverpod Pattern](references/riverpod-pattern.md) | Page+VM 쌍, NotifierProvider, family 패턴 | 페이지/위젯 구현 시 |
|
|
65
|
-
| [i18n Pattern](references/i18n-pattern.md) | ARB 파일, LocaleAssist, 키 네이밍 | 문자열 추가 시 |
|
|
66
|
-
| [Anti-Patterns](references/anti-patterns.md) | 금지 API, 하드코딩, bridges/, StatefulWidget 직접 호출 (fe_target 별 차이 포함) | 구현 완료 후 셀프 체크 |
|
|
67
|
-
| [Flutter Web Pattern](references/flutter-web-pattern.md) | Web 전용 — `dart:html`/`package:web` 허용, 라우팅, 빌드, hosting | `fe_target = web` 일 때만 |
|
|
68
|
-
|
|
69
|
-
## fe_target 분기
|
|
70
|
-
|
|
71
|
-
`pipeline.json.fe_target` 에 따라 적용되는 규칙이 다르다:
|
|
72
|
-
|
|
73
|
-
### `fe_target = web` (Flutter Web)
|
|
74
|
-
|
|
75
|
-
- **빌드**: `flutter build web --release` / 개발 서버: `flutter run -d chrome`
|
|
76
|
-
- **HTML/JS 인터롭 허용**: `dart:html`, `package:web`, `dart:js_interop` 사용 가능
|
|
77
|
-
- 단, 가능하면 cross-platform 코드를 우선하고 web 전용 코드는 `if (kIsWeb)` 가드 또는 conditional import
|
|
78
|
-
- **라우팅**: `go_router` 권장 (URL 동기화). `Navigator 1.0` 의 `MaterialApp.routes` 도 OK 지만 hash routing 주의
|
|
79
|
-
- **자산**: `web/index.html`, `web/manifest.json`, `web/icons/` 관리. SEO 가 필요하면 `<meta>` 태그 명시
|
|
80
|
-
- **CORS**: 백엔드 API 가 `localhost:포트` 에서 다른 origin 일 수 있으므로 CORS 설정 확인
|
|
81
|
-
- **Eval 흐름**: Playwright 기반 `evaluator-functional` + `evaluator-visual` 이 정상 작동 (브라우저 E2E + 시각 검증)
|
|
82
|
-
- 상세 → [Flutter Web Pattern](references/flutter-web-pattern.md)
|
|
83
|
-
|
|
84
|
-
### `fe_target = mobile` (Android / iOS)
|
|
85
|
-
|
|
86
|
-
- **빌드**: `flutter build apk` / `flutter build ios` / 개발: `flutter run -d <device>`
|
|
87
|
-
- **HTML/JS 인터롭 금지**: `dart:html`, `package:web`, `universal_html` 직접 참조 → 빌드 실패
|
|
88
|
-
- **플랫폼 채널**: 네이티브 기능 사용 시 `MethodChannel` 또는 검증된 plugin 사용
|
|
89
|
-
- **권한**: `AndroidManifest.xml` / `Info.plist` 에 권한 명시
|
|
90
|
-
- **Eval 흐름**: 정적 분석 기반 `evaluator-functional-flutter` 사용 (Playwright 불가)
|
|
91
|
-
|
|
92
|
-
### `fe_target = desktop` (macOS / Windows / Linux)
|
|
93
|
-
|
|
94
|
-
- **빌드**: `flutter build macos` / `flutter build windows` / `flutter build linux`
|
|
95
|
-
- **HTML/JS 인터롭 금지**: 모바일과 동일
|
|
96
|
-
- **윈도우 관리**: `window_manager` 등 플러그인으로 윈도우 크기/위치 제어
|
|
97
|
-
- **Eval 흐름**: 정적 분석 기반 `evaluator-functional-flutter` 사용
|
|
98
|
-
|
|
99
|
-
## 핵심 규칙
|
|
100
|
-
|
|
101
|
-
### api-contract.json → Dart 변환 규칙
|
|
102
|
-
|
|
103
|
-
- 각 엔드포인트 → `rest_api.dart`에 Retrofit 어노테이션 (`@GET`, `@POST`, `@PUT`, `@DELETE`)
|
|
104
|
-
- Request body → `2_data_sources/remote/request/body/xxx_body.dart` (`@JsonSerializable(includeIfNull: false)`)
|
|
105
|
-
- Response → `2_data_sources/remote/response/xxx_response.dart` (`ClueResponseImpl<T>` 상속 or 재사용)
|
|
106
|
-
- **필수 지시가 없으면 모든 필드는 Nullable** — 서버가 언제든 필드를 누락할 수 있음
|
|
107
|
-
- **기존 응답 타입 재사용 우선** — 동일 구조면 새 클래스 생성 금지
|
|
108
|
-
- Repository 래퍼 메서드 추가 → `1_repositories/xxx_repository.dart`
|
|
109
|
-
|
|
110
|
-
### UI / 상태관리
|
|
111
|
-
|
|
112
|
-
- 모든 페이지 = `xxx_page.dart` + `xxx_page_vm.dart` 쌍
|
|
113
|
-
- Page는 `ConsumerStatefulWidget` 또는 `ConsumerWidget`
|
|
114
|
-
- VM은 `NotifierProvider<Notifier, State>` (다중 인스턴스는 `.family`)
|
|
115
|
-
- State는 `Equatable` + `copyWith` 불변성
|
|
116
|
-
- `ref.watch` (build 내) / `ref.read` (이벤트 핸들러, initState)
|
|
117
|
-
- API 호출은 반드시 VM 안에서 `ref.read(dataLayer).xxx.method()`
|
|
118
|
-
- Provider 네이밍: `p` + PascalCase + `Provider` (예: `pHomePageProvider`)
|
|
119
|
-
|
|
120
|
-
### 다국어 (i18n)
|
|
121
|
-
|
|
122
|
-
- 영문 전용 지시가 없는 한 **모든 사용자 노출 문자열은 ARB 경유 필수**
|
|
123
|
-
- ARB: `lib/l10n/app_en.arb`, `app_ko.arb`, `app_ja.arb`
|
|
124
|
-
- 코드 접근: `LocaleAssist().of.키이름`
|
|
125
|
-
- 키 네이밍: camelCase (예: `doorOpen`, `networkError`)
|
|
126
|
-
- 모든 언어 파일에 동일 키 동시 추가
|
|
127
|
-
|
|
128
|
-
### 코드 생성
|
|
129
|
-
|
|
130
|
-
Request Body / Response / rest_api.dart 수정 후 **반드시** 실행:
|
|
131
|
-
|
|
132
|
-
```bash
|
|
133
|
-
cd <integrated_data_layer 경로>
|
|
134
|
-
flutter pub run build_runner build --delete-conflicting-outputs
|
|
135
|
-
```
|
|
136
|
-
|
|
137
|
-
## Self-Verification (Handoff 전)
|
|
138
|
-
|
|
139
|
-
1. `flutter analyze` → 경고 0개 (info 수준은 허용, warning/error 금지)
|
|
140
|
-
2. `flutter test` → 100% 통과
|
|
141
|
-
3. integrated_data_layer의 새 Request Body / Response에 `fromJson`/`toJson` 왕복 테스트 존재
|
|
142
|
-
4. `grep -rn 'dart:html\|universal_html\|print(\|console\.log' lib/ integrated_data_layer/lib/` → 0개
|
|
143
|
-
5. `grep -rn "bridges/" lib/` → 신규 코드 내 참조 0개
|
|
144
|
-
6. 하드코딩 색상(`Color(0xFF...`) → `ColorManager` 경유 확인
|
|
145
|
-
7. 새 페이지/위젯은 `StatefulWidget`에서 직접 API 호출하지 않는지 확인 (VM/Provider 경유)
|
|
146
|
-
|
|
147
|
-
## 금지 사항
|
|
148
|
-
|
|
149
|
-
- Backend 코드 수정, 서버 API 경로 임의 변경
|
|
150
|
-
- api-contract.json에 없는 엔드포인트 호출/추가
|
|
151
|
-
- `dart:html`, `universal_html` 직접 참조
|
|
152
|
-
- `print()` / `debugPrint` 남발 — `logger` 사용
|
|
153
|
-
- 하드코딩 색상 → `ColorManager` 강제
|
|
154
|
-
- `StatefulWidget` 내 직접 API 호출 → VM/Provider 경유
|
|
155
|
-
- `bridges/` 신규 개발 — BLOC 기반 레거시, 유지보수만 허용
|
|
156
|
-
- AI 추측으로 서버 응답 구조를 만들지 말 것 (계약이 Source of Truth)
|
|
157
|
-
- API 키/시크릿 하드코딩
|
|
158
|
-
|
|
159
|
-
## 명령어
|
|
160
|
-
|
|
161
|
-
```bash
|
|
162
|
-
flutter pub get # 의존성
|
|
163
|
-
flutter run # 실행
|
|
164
|
-
flutter test # 테스트
|
|
165
|
-
flutter analyze # 정적 분석
|
|
166
|
-
flutter pub run build_runner build --delete-conflicting-outputs # 코드 생성
|
|
167
|
-
```
|
|
168
|
-
|
|
169
|
-
## After Completion
|
|
170
|
-
|
|
171
|
-
1. sprint-contract.md의 FE 섹션에 완료 항목 체크
|
|
172
|
-
2. Self-Verification 체크리스트 결과 요약
|
|
173
|
-
3. Session Boundary Protocol On Complete 실행
|
|
@@ -1,320 +0,0 @@
|
|
|
1
|
-
---
|
|
2
|
-
docmeta:
|
|
3
|
-
id: anti-patterns
|
|
4
|
-
title: Flutter Anti-Patterns & Forbidden APIs
|
|
5
|
-
type: output
|
|
6
|
-
createdAt: 2026-04-09T00:00:00Z
|
|
7
|
-
updatedAt: 2026-04-09T00:00:00Z
|
|
8
|
-
source:
|
|
9
|
-
producer: agent
|
|
10
|
-
skillId: harness-generator-frontend-flutter
|
|
11
|
-
inputs:
|
|
12
|
-
- documentId: clue-fe-flutter-skill
|
|
13
|
-
uri: ../../../../../moon_web/clue-fe-flutter.skill
|
|
14
|
-
relation: output-from
|
|
15
|
-
sections:
|
|
16
|
-
- sourceRange:
|
|
17
|
-
startLine: 62
|
|
18
|
-
endLine: 103
|
|
19
|
-
targetRange:
|
|
20
|
-
startLine: 32
|
|
21
|
-
endLine: 180
|
|
22
|
-
tags:
|
|
23
|
-
- flutter
|
|
24
|
-
- anti-patterns
|
|
25
|
-
- forbidden
|
|
26
|
-
- lint
|
|
27
|
-
---
|
|
28
|
-
|
|
29
|
-
# Flutter Anti-Patterns & Forbidden APIs
|
|
30
|
-
|
|
31
|
-
**Generator는 구현 완료 전 이 문서를 보고 셀프 체크한다.**
|
|
32
|
-
**Evaluator-Functional-Flutter 는 이 문서의 패턴을 `grep` 기반 정적 검증에 사용한다.**
|
|
33
|
-
|
|
34
|
-
각 항목은 **패턴 → 이유 → 대체 방법** 형태.
|
|
35
|
-
|
|
36
|
-
## 1. 웹 전용 API 직접 참조 (fe_target 별 차이)
|
|
37
|
-
|
|
38
|
-
### `fe_target = mobile` 또는 `desktop` — 금지
|
|
39
|
-
```dart
|
|
40
|
-
import 'dart:html'; // ✗
|
|
41
|
-
import 'package:universal_html/html.dart'; // ✗
|
|
42
|
-
import 'package:web/web.dart'; // ✗ (가드 없으면)
|
|
43
|
-
```
|
|
44
|
-
|
|
45
|
-
### 이유
|
|
46
|
-
Flutter 모바일/데스크톱 빌드에서 `dart:html` 참조는 즉시 빌드 실패한다.
|
|
47
|
-
가드 없이 import 하면 cross-platform 코드가 깨진다.
|
|
48
|
-
|
|
49
|
-
### 대체
|
|
50
|
-
- `kIsWeb` 분기로 플랫폼 체크 후 조건부 import
|
|
51
|
-
- `if (kIsWeb) { ... }` 가드 안에서만 web API 사용
|
|
52
|
-
- 또는 conditional import (`stub.dart` / `web.dart` / `io.dart`)
|
|
53
|
-
|
|
54
|
-
### `fe_target = web` — 허용 (단, 권장 패턴 준수)
|
|
55
|
-
```dart
|
|
56
|
-
import 'package:web/web.dart' as web; // ✓ 권장 (modern)
|
|
57
|
-
import 'dart:js_interop'; // ✓ JS interop
|
|
58
|
-
import 'dart:html'; // ✓ (legacy, 신규 코드는 package:web 권장)
|
|
59
|
-
```
|
|
60
|
-
|
|
61
|
-
### Web 권장 패턴
|
|
62
|
-
- 새 코드는 `package:web` + `dart:js_interop` 사용 (Flutter 3.7+ 에서 stable)
|
|
63
|
-
- `dart:html` 은 legacy — 마이그레이션 대상이지만 즉시 금지는 아님
|
|
64
|
-
- 가능하면 web 전용 코드를 별도 파일로 분리하고 conditional import:
|
|
65
|
-
```dart
|
|
66
|
-
// _web_helper.dart 또는 conditional import
|
|
67
|
-
import 'platform_helper.dart'
|
|
68
|
-
if (dart.library.html) 'platform_helper_web.dart'
|
|
69
|
-
if (dart.library.io) 'platform_helper_io.dart';
|
|
70
|
-
```
|
|
71
|
-
|
|
72
|
-
### 검증
|
|
73
|
-
```bash
|
|
74
|
-
# fe_target = mobile/desktop: 0개여야 함
|
|
75
|
-
grep -rn "dart:html\|universal_html\|package:web" lib/ integrated_data_layer/lib/
|
|
76
|
-
|
|
77
|
-
# fe_target = web: 매치 OK, 단 'kIsWeb' 가드 또는 conditional import 와 함께 쓰는지 인접 라인 확인
|
|
78
|
-
```
|
|
79
|
-
|
|
80
|
-
---
|
|
81
|
-
|
|
82
|
-
## 2. print / console 남발
|
|
83
|
-
|
|
84
|
-
### 금지
|
|
85
|
-
```dart
|
|
86
|
-
print("user clicked");
|
|
87
|
-
debugPrint("response: $data");
|
|
88
|
-
```
|
|
89
|
-
|
|
90
|
-
### 이유
|
|
91
|
-
프로덕션 빌드에 로그가 섞이면 성능/보안 문제. 통일된 로거가 없으면 분석 불가.
|
|
92
|
-
|
|
93
|
-
### 대체
|
|
94
|
-
프로젝트의 `logger` 모듈 사용 (`Logger().d()`, `.i()`, `.w()`, `.e()`).
|
|
95
|
-
디버그 전용 출력도 `kDebugMode` 가드 필수.
|
|
96
|
-
|
|
97
|
-
### 검증
|
|
98
|
-
```bash
|
|
99
|
-
grep -rn "^\s*print(" lib/ integrated_data_layer/lib/
|
|
100
|
-
grep -rn "console\.log" lib/
|
|
101
|
-
```
|
|
102
|
-
|
|
103
|
-
---
|
|
104
|
-
|
|
105
|
-
## 3. 하드코딩 색상
|
|
106
|
-
|
|
107
|
-
### 금지
|
|
108
|
-
```dart
|
|
109
|
-
Container(color: Color(0xFF6682FF))
|
|
110
|
-
Text('Hello', style: TextStyle(color: Colors.blue))
|
|
111
|
-
```
|
|
112
|
-
|
|
113
|
-
### 이유
|
|
114
|
-
디자인 시스템 붕괴 + 다크모드/브랜딩 변경 시 전역 수정 불가.
|
|
115
|
-
|
|
116
|
-
### 대체
|
|
117
|
-
```dart
|
|
118
|
-
Container(color: ColorManager.primary)
|
|
119
|
-
Text('Hello', style: TextStyle(color: ColorManager.textPrimary))
|
|
120
|
-
```
|
|
121
|
-
|
|
122
|
-
`ColorManager` (또는 프로젝트의 Design Token) 경유 필수.
|
|
123
|
-
|
|
124
|
-
### 검증
|
|
125
|
-
```bash
|
|
126
|
-
grep -rn "Color(0x" lib/ui/ | grep -v "ColorManager\|color_manager"
|
|
127
|
-
```
|
|
128
|
-
신규 파일에서 결과 0개여야 함 (레거시 파일은 예외).
|
|
129
|
-
|
|
130
|
-
---
|
|
131
|
-
|
|
132
|
-
## 4. StatefulWidget 내 직접 API 호출
|
|
133
|
-
|
|
134
|
-
### 금지
|
|
135
|
-
```dart
|
|
136
|
-
class _MyPageState extends State<MyPage> {
|
|
137
|
-
@override
|
|
138
|
-
void initState() {
|
|
139
|
-
super.initState();
|
|
140
|
-
DataLayer.instance.ac.getExample(exampleId: 1).then((res) { // ✗
|
|
141
|
-
setState(() { _data = res; });
|
|
142
|
-
});
|
|
143
|
-
}
|
|
144
|
-
}
|
|
145
|
-
```
|
|
146
|
-
|
|
147
|
-
### 이유
|
|
148
|
-
테스트 불가능, 상태 공유 불가, UI와 비즈니스 로직 결합.
|
|
149
|
-
|
|
150
|
-
### 대체
|
|
151
|
-
VM (`NotifierProvider`) 에 API 호출을 옮기고 Page는 `ConsumerWidget` 으로.
|
|
152
|
-
상세 → [riverpod-pattern.md](./riverpod-pattern.md)
|
|
153
|
-
|
|
154
|
-
### 검증
|
|
155
|
-
`grep` 으로 자동 감지 어려움 — Evaluator가 신규 `*_page.dart` 파일에서
|
|
156
|
-
`dataLayer\|DataLayer` 호출 패턴을 확인하고, 파트너 `*_page_vm.dart` 파일에
|
|
157
|
-
동일 호출이 있는지 대조한다.
|
|
158
|
-
|
|
159
|
-
---
|
|
160
|
-
|
|
161
|
-
## 5. bridges/ 신규 사용
|
|
162
|
-
|
|
163
|
-
### 금지
|
|
164
|
-
```dart
|
|
165
|
-
import 'package:clue_mobile_app/bridges/xxx.dart';
|
|
166
|
-
```
|
|
167
|
-
신규 코드에서 참조 금지.
|
|
168
|
-
|
|
169
|
-
### 이유
|
|
170
|
-
BLOC 기반 레거시 — Riverpod 마이그레이션 방침에 따라 유지보수만 허용.
|
|
171
|
-
|
|
172
|
-
### 대체
|
|
173
|
-
신규 상태관리는 `NotifierProvider` + `Notifier<State>`.
|
|
174
|
-
|
|
175
|
-
### 검증
|
|
176
|
-
신규/수정 파일의 git diff에서 `bridges/` import 추가 여부 확인.
|
|
177
|
-
```bash
|
|
178
|
-
git diff --name-only HEAD~1 | xargs grep -l "bridges/" 2>/dev/null
|
|
179
|
-
```
|
|
180
|
-
새로 추가된 줄이 있으면 FAIL.
|
|
181
|
-
|
|
182
|
-
---
|
|
183
|
-
|
|
184
|
-
## 6. 하드코딩 문자열 (다국어 미처리)
|
|
185
|
-
|
|
186
|
-
### 금지
|
|
187
|
-
```dart
|
|
188
|
-
Text('취소')
|
|
189
|
-
Text('로그인 실패')
|
|
190
|
-
```
|
|
191
|
-
|
|
192
|
-
### 이유
|
|
193
|
-
i18n 원칙 위반 — 다국어 전환 시 즉시 깨짐.
|
|
194
|
-
|
|
195
|
-
### 대체
|
|
196
|
-
```dart
|
|
197
|
-
Text(LocaleAssist().of.cancel)
|
|
198
|
-
Text(LocaleAssist().of.loginFailed)
|
|
199
|
-
```
|
|
200
|
-
|
|
201
|
-
상세 → [i18n-pattern.md](./i18n-pattern.md)
|
|
202
|
-
|
|
203
|
-
### 검증
|
|
204
|
-
```bash
|
|
205
|
-
grep -rn "Text('[가-힣ぁ-んァ-ヶ一-龯]" lib/ui/
|
|
206
|
-
```
|
|
207
|
-
신규 파일에서 결과 0개여야 함.
|
|
208
|
-
|
|
209
|
-
---
|
|
210
|
-
|
|
211
|
-
## 7. JsonSerializable 누락 / includeIfNull 잘못 설정
|
|
212
|
-
|
|
213
|
-
### 금지
|
|
214
|
-
```dart
|
|
215
|
-
// @JsonSerializable 없이 수동 fromJson/toJson
|
|
216
|
-
@JsonSerializable() // includeIfNull 누락 → 기본값 true
|
|
217
|
-
class UserBody { ... }
|
|
218
|
-
```
|
|
219
|
-
|
|
220
|
-
### 이유
|
|
221
|
-
서버가 null 필드를 bad request 처리하는 API 계약 위배. 코드 생성 불일치.
|
|
222
|
-
|
|
223
|
-
### 대체
|
|
224
|
-
```dart
|
|
225
|
-
@JsonSerializable(includeIfNull: false)
|
|
226
|
-
class UserBody { ... }
|
|
227
|
-
```
|
|
228
|
-
|
|
229
|
-
Request Body는 **항상** `includeIfNull: false`.
|
|
230
|
-
|
|
231
|
-
### 검증
|
|
232
|
-
```bash
|
|
233
|
-
grep -rn "@JsonSerializable" integrated_data_layer/lib/2_data_sources/remote/request/body/ | grep -v "includeIfNull: false"
|
|
234
|
-
```
|
|
235
|
-
결과 0개여야 함.
|
|
236
|
-
|
|
237
|
-
---
|
|
238
|
-
|
|
239
|
-
## 8. Non-null 남용
|
|
240
|
-
|
|
241
|
-
### 금지
|
|
242
|
-
```dart
|
|
243
|
-
// 서버 응답에 대해 non-null 단정
|
|
244
|
-
class ExampleData {
|
|
245
|
-
final String name; // ✗ 필수 지시 없으면 Nullable
|
|
246
|
-
final int id;
|
|
247
|
-
}
|
|
248
|
-
```
|
|
249
|
-
|
|
250
|
-
### 이유
|
|
251
|
-
서버는 언제든 필드를 누락할 수 있음. non-null은 런타임 crash 유발.
|
|
252
|
-
|
|
253
|
-
### 대체
|
|
254
|
-
```dart
|
|
255
|
-
class ExampleData {
|
|
256
|
-
final String? name;
|
|
257
|
-
final int? id;
|
|
258
|
-
}
|
|
259
|
-
```
|
|
260
|
-
|
|
261
|
-
**필수 지시가 api-contract.json 에 명시된 필드만 non-null 허용.**
|
|
262
|
-
|
|
263
|
-
---
|
|
264
|
-
|
|
265
|
-
## 9. API 키 / 시크릿 하드코딩
|
|
266
|
-
|
|
267
|
-
### 금지
|
|
268
|
-
```dart
|
|
269
|
-
const apiKey = "sk-1234567890abcdef";
|
|
270
|
-
const Dio().options.headers['Authorization'] = 'Bearer xxx';
|
|
271
|
-
```
|
|
272
|
-
|
|
273
|
-
### 이유
|
|
274
|
-
APK 디컴파일로 즉시 노출 — 보안 사고.
|
|
275
|
-
|
|
276
|
-
### 대체
|
|
277
|
-
`--dart-define` 또는 `flutter_dotenv` + `.env` (gitignore).
|
|
278
|
-
런타임에 서버에서 발급받는 토큰 사용.
|
|
279
|
-
|
|
280
|
-
### 검증
|
|
281
|
-
```bash
|
|
282
|
-
grep -rEn "(api[_-]?key|secret|token)\s*=\s*['\"][A-Za-z0-9]{16,}['\"]" lib/ integrated_data_layer/lib/
|
|
283
|
-
```
|
|
284
|
-
|
|
285
|
-
---
|
|
286
|
-
|
|
287
|
-
## 셀프 체크 스크립트
|
|
288
|
-
|
|
289
|
-
Generator는 handoff 전 다음 명령을 모두 실행하고 결과를 sprint-contract.md에 기록한다.
|
|
290
|
-
**`fe_target` 에 따라 1번 룰의 적용 여부가 달라진다.**
|
|
291
|
-
|
|
292
|
-
```bash
|
|
293
|
-
# fe_target 읽기
|
|
294
|
-
FE_TARGET=$(jq -r '.fe_target // "web"' .harness/actions/pipeline.json 2>/dev/null || echo "web")
|
|
295
|
-
|
|
296
|
-
# 1. 웹 API 직접 참조 — fe_target=web 이 아닐 때만 검사
|
|
297
|
-
if [ "$FE_TARGET" != "web" ]; then
|
|
298
|
-
grep -rn "dart:html\|universal_html\|package:web" lib/ integrated_data_layer/lib/ || echo "OK (FL-01 mobile/desktop)"
|
|
299
|
-
else
|
|
300
|
-
# web 타겟: 가드 없는 사용만 경고 (수동 검토)
|
|
301
|
-
grep -rn "dart:html\|package:web" lib/ | grep -v "kIsWeb\|conditional import" || echo "OK (FL-01 web — manual review)"
|
|
302
|
-
fi
|
|
303
|
-
|
|
304
|
-
# 2. print 남발
|
|
305
|
-
grep -rn "^\s*print(" lib/ integrated_data_layer/lib/ || echo "OK"
|
|
306
|
-
|
|
307
|
-
# 3. 하드코딩 색상 (신규 파일만 — git diff 기반)
|
|
308
|
-
git diff --name-only --diff-filter=A HEAD | grep '\.dart$' | xargs grep -n "Color(0x" 2>/dev/null || echo "OK"
|
|
309
|
-
|
|
310
|
-
# 4. bridges/ 신규 참조
|
|
311
|
-
git diff HEAD | grep "^+" | grep "bridges/" || echo "OK"
|
|
312
|
-
|
|
313
|
-
# 5. JsonSerializable + includeIfNull 확인
|
|
314
|
-
grep -rn "@JsonSerializable" integrated_data_layer/lib/2_data_sources/remote/request/body/ | grep -v "includeIfNull: false" || echo "OK"
|
|
315
|
-
|
|
316
|
-
# 6. 하드코딩 한글 (신규 UI)
|
|
317
|
-
git diff --name-only --diff-filter=AM HEAD | grep 'lib/ui/.*\.dart$' | xargs grep -n "Text('[가-힣]" 2>/dev/null || echo "OK"
|
|
318
|
-
```
|
|
319
|
-
|
|
320
|
-
모두 `OK` 여야 handoff 가능.
|