@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.
@@ -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 가능.