@edupia-tutor/spec-driven-docs 0.14.0 → 0.14.2
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/bin/index.js +12 -1
- package/commands/debug.md +436 -436
- package/commands/debug.tmpl +111 -111
- package/commands/define-product.md +350 -345
- package/commands/define-product.tmpl +69 -64
- package/commands/dev-gen-test.md +365 -365
- package/commands/dev-gen-test.tmpl +63 -63
- package/commands/dev-run-test.md +376 -376
- package/commands/dev-run-test.tmpl +74 -74
- package/commands/dev-smoke-test.md +341 -341
- package/commands/dev-smoke-test.tmpl +60 -60
- package/commands/fix-bug.md +403 -403
- package/commands/fix-bug.tmpl +78 -78
- package/commands/generate-bdd.md +513 -513
- package/commands/generate-bdd.tmpl +211 -211
- package/commands/generate-code.md +481 -483
- package/commands/generate-code.tmpl +179 -181
- package/commands/generate-design-spec.md +497 -497
- package/commands/generate-design-spec.tmpl +220 -220
- package/commands/generate-prd.md +452 -400
- package/commands/generate-prd.tmpl +50 -200
- package/commands/generate-spec-manifest.md +340 -340
- package/commands/generate-spec-manifest.tmpl +59 -59
- package/commands/generate-tech-docs.md +365 -365
- package/commands/generate-tech-docs.tmpl +84 -84
- package/commands/learn.md +347 -347
- package/commands/learn.tmpl +22 -22
- package/commands/map-testids.md +322 -322
- package/commands/map-testids.tmpl +41 -41
- package/commands/propose-scenario.md +335 -335
- package/commands/propose-scenario.tmpl +54 -54
- package/commands/qc-analyze.md +323 -324
- package/commands/qc-analyze.tmpl +42 -43
- package/commands/qc-design-test.md +304 -304
- package/commands/qc-design-test.tmpl +23 -23
- package/commands/qc-plan.md +297 -297
- package/commands/qc-plan.tmpl +16 -16
- package/commands/qc-report.md +302 -302
- package/commands/qc-report.tmpl +21 -21
- package/commands/qc-review.md +298 -298
- package/commands/qc-review.tmpl +17 -17
- package/commands/qc-run-test.md +337 -337
- package/commands/qc-run-test.tmpl +35 -35
- package/commands/refine-prd.md +428 -430
- package/commands/refine-prd.tmpl +62 -62
- package/commands/report-bug.md +351 -351
- package/commands/report-bug.tmpl +70 -70
- package/commands/review-code.md +364 -364
- package/commands/review-code.tmpl +39 -39
- package/commands/review-context.md +578 -580
- package/commands/review-context.tmpl +212 -212
- package/commands/review-tech-docs.md +427 -427
- package/commands/review-tech-docs.tmpl +146 -146
- package/commands/setup-ai-first.md +239 -239
- package/commands/setup-ai-first.tmpl +133 -133
- package/commands/sync.md +145 -145
- package/commands/sync.tmpl +93 -93
- package/commands/update-framework.md +88 -88
- package/commands/update-framework.tmpl +36 -36
- package/commands/validate-traces.md +381 -381
- package/commands/validate-traces.tmpl +100 -100
- package/core/FRAMEWORK_VERSION +1 -1
- package/core/commands/debug.md +436 -436
- package/core/commands/define-product.md +350 -345
- package/core/commands/dev-gen-test.md +365 -365
- package/core/commands/dev-run-test.md +376 -376
- package/core/commands/dev-smoke-test.md +341 -341
- package/core/commands/fix-bug.md +403 -403
- package/core/commands/generate-bdd.md +513 -513
- package/core/commands/generate-code.md +481 -483
- package/core/commands/generate-design-spec.md +497 -497
- package/core/commands/generate-prd.md +452 -400
- package/core/commands/generate-spec-manifest.md +340 -340
- package/core/commands/generate-tech-docs.md +365 -365
- package/core/commands/learn.md +347 -347
- package/core/commands/map-testids.md +322 -322
- package/core/commands/propose-scenario.md +335 -335
- package/core/commands/qc-analyze.md +323 -324
- package/core/commands/qc-design-test.md +304 -304
- package/core/commands/qc-plan.md +297 -297
- package/core/commands/qc-report.md +302 -302
- package/core/commands/qc-review.md +298 -298
- package/core/commands/qc-run-test.md +337 -337
- package/core/commands/refine-prd.md +428 -430
- package/core/commands/report-bug.md +351 -351
- package/core/commands/review-code.md +364 -364
- package/core/commands/review-context.md +578 -580
- package/core/commands/review-tech-docs.md +427 -427
- package/core/commands/setup-ai-first.md +239 -239
- package/core/commands/sync.md +145 -145
- package/core/commands/update-framework.md +88 -88
- package/core/commands/validate-traces.md +381 -381
- package/core/skills/code/SKILL.md +389 -389
- package/core/skills/debug/SKILL.md +391 -391
- package/core/skills/design-spec/SKILL.md +318 -318
- package/core/skills/discovery/SKILL.md +7 -547
- package/core/skills/prd/SKILL.md +298 -394
- package/core/skills/setup-ai-first/SKILL.md +80 -80
- package/core/skills/spec/SKILL.md +178 -178
- package/core/skills/test/SKILL.md +604 -604
- package/core/steps/capture-lesson.md +44 -44
- package/core/steps/context-loader.md +175 -175
- package/core/steps/gate.md +54 -54
- package/core/steps/report-footer.md +52 -52
- package/core/steps/review-fanout.md +85 -87
- package/core/steps/spawn-agent.md +45 -45
- package/core/steps/trace-mirror.md +21 -21
- package/core/templates/architecture.template.md +37 -37
- package/core/templates/design-spec.template.md +77 -77
- package/core/templates/platform-guide.template.md +47 -47
- package/core/templates/prd.template.md +107 -232
- package/core/templates/product-definition.template.md +101 -88
- package/core/templates/project-context.yaml +2 -2
- package/docs/01-getting-started/core-concepts.md +1 -1
- package/docs/01-getting-started/quickstart.md +7 -7
- package/docs/02-guides/developer/bdd-and-trace.md +1 -1
- package/docs/02-guides/developer/scenarios.md +5 -5
- package/docs/02-guides/product-owner/handoff-checklist.md +1 -1
- package/docs/02-guides/product-owner/scenarios.md +23 -23
- package/docs/02-guides/tester/bug-reporting.md +2 -2
- package/docs/02-guides/tester/reading-specs.md +2 -2
- package/docs/02-guides/tester/scenarios.md +1 -1
- package/docs/02-guides/tester/spec-manifest.md +3 -3
- package/docs/02-guides/tester/workflow.md +1 -1
- package/docs/03-concepts/architecture.md +3 -3
- package/docs/03-concepts/pipeline.md +3 -3
- package/docs/04-operations/publishing.md +20 -3
- package/docs/04-operations/sync-and-update.md +5 -5
- package/docs/05-reference/command-cheatsheet.md +2 -2
- package/docs/05-reference/commands.md +8 -8
- package/package.json +1 -1
- package/scripts/migrate-specs.js +5 -3
- package/scripts/rename-prd-files.js +174 -0
- package/skills/code/SKILL.md +389 -389
- package/skills/code/SKILL.tmpl +56 -56
- package/skills/debug/SKILL.md +391 -391
- package/skills/debug/SKILL.tmpl +60 -60
- package/skills/design-spec/SKILL.md +318 -318
- package/skills/design-spec/SKILL.tmpl +37 -37
- package/skills/discovery/SKILL.md +7 -547
- package/skills/discovery/SKILL.tmpl +7 -140
- package/skills/prd/SKILL.md +298 -394
- package/skills/prd/SKILL.tmpl +40 -151
- package/skills/setup-ai-first/SKILL.md +80 -80
- package/skills/setup-ai-first/SKILL.tmpl +28 -28
- package/skills/spec/SKILL.md +178 -178
- package/skills/spec/SKILL.tmpl +20 -20
- package/skills/test/SKILL.md +604 -604
- package/skills/test/SKILL.tmpl +44 -44
- package/steps/capture-lesson.md +44 -44
- package/steps/context-loader.md +175 -175
- package/steps/gate.md +54 -54
- package/steps/report-footer.md +52 -52
- package/steps/review-fanout.md +85 -87
- package/steps/spawn-agent.md +45 -45
- package/steps/trace-mirror.md +21 -21
- package/templates/architecture.template.md +37 -37
- package/templates/design-spec.template.md +77 -77
- package/templates/platform-guide.template.md +47 -47
- package/templates/prd.template.md +107 -232
- package/templates/product-definition.template.md +101 -88
- package/templates/project-context.yaml +2 -2
|
@@ -1,152 +1,152 @@
|
|
|
1
|
-
# /review-context — Review PRD
|
|
1
|
+
# /review-context — Review PRD hoặc BDD về Chất lượng & Tính nhất quán
|
|
2
2
|
|
|
3
|
-
**READ-ONLY
|
|
4
|
-
**
|
|
3
|
+
**Chế độ phân tích READ-ONLY — ghi file findings, KHÔNG sửa target.**
|
|
4
|
+
**Dùng `--resume` để áp dụng các finding được chấp nhận.**
|
|
5
5
|
|
|
6
6
|
## Gate
|
|
7
|
-
# Gate —
|
|
7
|
+
# Gate — Quy trình vào chuẩn cho mọi lệnh
|
|
8
8
|
|
|
9
|
-
|
|
9
|
+
Mọi lệnh PHẢI chạy gate này trước khi thực thi phần logic riêng của nó.
|
|
10
10
|
|
|
11
|
-
##
|
|
11
|
+
## Bước 0 — Kiểm tra chế độ Sub-Agent
|
|
12
12
|
|
|
13
|
-
|
|
13
|
+
Trước tiên, kiểm tra xem `$ARGUMENTS` có phải là payload JSON từ một orchestrator hay không:
|
|
14
14
|
|
|
15
|
-
1.
|
|
16
|
-
2.
|
|
17
|
-
- **
|
|
18
|
-
-
|
|
19
|
-
-
|
|
20
|
-
-
|
|
21
|
-
-
|
|
22
|
-
-
|
|
23
|
-
3.
|
|
15
|
+
1. Thử parse `$ARGUMENTS` dưới dạng JSON.
|
|
16
|
+
2. Nếu parse thành công **và** chứa `"_agent_mode": true`:
|
|
17
|
+
- **Bỏ qua hoàn toàn Bước 1, 2 và 3 của Gate này.**
|
|
18
|
+
- Đặt target file = `payload.target_file`
|
|
19
|
+
- Đặt loaded context = `payload.context` (KHÔNG chạy context-loader.md)
|
|
20
|
+
- Đặt phạm vi UC = `payload.uc_id` (chỉ xử lý UC này)
|
|
21
|
+
- Đặt line range = `payload.uc_section` (chỉ đọc đúng section đó của PRD)
|
|
22
|
+
- Đi thẳng tới phần logic riêng của lệnh.
|
|
23
|
+
3. Nếu `$ARGUMENTS` không phải JSON hoặc không có `_agent_mode` → tiếp tục sang Bước 1 (chế độ thường).
|
|
24
24
|
|
|
25
|
-
##
|
|
25
|
+
## Bước 0-B — Kiểm tra Model
|
|
26
26
|
|
|
27
|
-
*
|
|
27
|
+
*Bỏ qua bước này nếu `_agent_mode: true` (sub-agent — orchestrator đã kiểm tra rồi).*
|
|
28
28
|
|
|
29
|
-
|
|
30
|
-
|
|
29
|
+
Các lệnh sinh nội dung và review phức tạp đòi hỏi khả năng suy luận mạnh.
|
|
30
|
+
Dùng model nhỏ hơn sẽ rủi ro: bỏ sót edge case, phân tích spec thiếu sót, vi phạm kiến trúc.
|
|
31
31
|
|
|
32
|
-
|
|
32
|
+
Hiển thị và chờ phản hồi:
|
|
33
33
|
|
|
34
34
|
```
|
|
35
35
|
⚙️ MODEL CHECK
|
|
36
36
|
──────────────────────────────────────────────────────────────────
|
|
37
|
-
Recommended : claude-opus-4 (
|
|
38
|
-
Why needed :
|
|
39
|
-
|
|
37
|
+
Recommended : claude-opus-4 (hoặc model Opus mới nhất)
|
|
38
|
+
Why needed : Phân tích spec, review kiến trúc, sinh code đòi hỏi
|
|
39
|
+
suy luận sâu. Model nhỏ hơn dễ bỏ sót edge case.
|
|
40
40
|
|
|
41
|
-
|
|
42
|
-
• Settings → Model →
|
|
43
|
-
•
|
|
41
|
+
Cách đổi trong Claude Code:
|
|
42
|
+
• Settings → Model → chọn "claude-opus"
|
|
43
|
+
• hoặc: /model → chọn claude-opus
|
|
44
44
|
|
|
45
|
-
|
|
46
|
-
Y —
|
|
47
|
-
S —
|
|
45
|
+
Đang chạy claude-opus?
|
|
46
|
+
Y — đúng, đang dùng claude-opus → tiếp tục
|
|
47
|
+
S — bỏ qua kiểm tra (tôi chấp nhận rủi ro chất lượng thấp hơn với model hiện tại)
|
|
48
48
|
──────────────────────────────────────────────────────────────────
|
|
49
49
|
```
|
|
50
50
|
|
|
51
|
-
- "Y" →
|
|
52
|
-
- "S" →
|
|
53
|
-
- "N"
|
|
51
|
+
- "Y" → tiếp tục sang Bước 1.
|
|
52
|
+
- "S" → tiếp tục sang Bước 1 (người dùng chấp nhận rủi ro, thêm ⚠️ vào report cuối).
|
|
53
|
+
- "N" hoặc bất kỳ giá trị nào khác → **DỪNG.** Xuất: "Vui lòng chuyển sang claude-opus rồi chạy lại lệnh này."
|
|
54
54
|
|
|
55
|
-
##
|
|
55
|
+
## Bước 1 — Xác định Target File
|
|
56
56
|
|
|
57
|
-
1.
|
|
58
|
-
2.
|
|
59
|
-
- **BDD
|
|
60
|
-
- **PRD
|
|
61
|
-
- **tech-docs
|
|
62
|
-
- **design-spec
|
|
57
|
+
1. Nếu `$ARGUMENTS` được cung cấp và trỏ tới một file tồn tại → dùng trực tiếp làm target.
|
|
58
|
+
2. Nếu `$ARGUMENTS` là một **UC-ID / ticket ID / tên rút gọn** (không có path) → phân giải thành file bằng cách glob theo bố cục feature-package. `{prd-slug}` lúc này **chưa biết**, nên dùng wildcard `*` cho segment đó, và `**` đệ quy dưới `bdd/` để phủ hết các thư mục con theo platform (`bdd/web/`, `bdd/app/`, `bdd/system/`):
|
|
59
|
+
- **Lệnh BDD** (target là `.feature`): `{specs_dir}/{domain}/*/bdd/**/{UC-ID}*.feature` — hoặc `{specs_dir}/*/*/bdd/**/{UC-ID}*.feature` nếu domain cũng chưa biết. Nếu lệnh ngụ ý một platform/scope cụ thể (vd: system tech-doc cần BDD `system/`), ưu tiên kết quả trong thư mục con platform đó.
|
|
60
|
+
- **Lệnh PRD** (target là file PRD `{TICKET-ID}-{prd-slug}.md` — file `.md` duy nhất ở gốc feature folder, cạnh `bdd/`): `{specs_dir}/{domain}/*/{TICKET-ID}*.md` nếu biết TICKET-ID; nếu không, `{specs_dir}/{domain}/*/*.md` (khớp feature folder có id tương ứng), hoặc `{specs_dir}/*/*/*.md` nếu domain cũng chưa biết. *(Glob `*/*.md` ở cấp gốc folder chỉ khớp PRD — tech-docs/design-spec `.md` nằm sâu hơn trong thư mục con.)*
|
|
61
|
+
- **Lệnh tech-docs**: `{specs_dir}/{domain}/*/tech-docs/{UC-ID}*-tech-design*.md`.
|
|
62
|
+
- **Lệnh design-spec**: `{specs_dir}/{domain}/*/design-spec/{TICKET-ID}*.md`.
|
|
63
63
|
|
|
64
|
-
|
|
65
|
-
3.
|
|
66
|
-
-
|
|
67
|
-
-
|
|
68
|
-
-
|
|
64
|
+
Khi một file khớp: đặt nó làm target **và** ghi lại `domain` + `prd_slug` từ path của nó (theo quy tắc trích xuất trong `context-loader.md` Bước 1 — `prd_slug` = segment đầu tiên sau `{specs_dir}/{domain}/`). Mọi path mà lệnh đọc/ghi về sau (BDD/tech-docs/design-spec/trace cùng cấp) đều dùng **`prd_slug` đã phân giải đó**, nên tất cả artifact nằm chung một feature package. Nếu nhiều file khớp (vd: nhiều platform), chọn theo platform/scope của lệnh hoặc liệt kê ra và hỏi.
|
|
65
|
+
3. Nếu `$ARGUMENTS` rỗng hoặc không tìm thấy file khớp:
|
|
66
|
+
- Liệt kê các file trong thư mục liên quan của lệnh này (vd: `specs/*/*/*.md` — file PRD ở gốc mỗi feature folder — cho lệnh PRD, `specs/*/*/bdd/**/*.feature` cho lệnh BDD).
|
|
67
|
+
- Hiển thị danh sách cho người dùng và hỏi: "Bạn muốn làm việc với file nào? (Nhập số thứ tự hoặc tên file)"
|
|
68
|
+
- Chờ người dùng chọn rồi mới tiếp tục.
|
|
69
69
|
|
|
70
|
-
##
|
|
70
|
+
## Bước 2 — Chạy Context Loader
|
|
71
71
|
|
|
72
|
-
|
|
73
|
-
|
|
72
|
+
Nạp toàn bộ context của dự án bằng cách làm theo quy trình trong `steps/context-loader.md`.
|
|
73
|
+
Lưu toàn bộ context đã nạp vào bộ nhớ để dùng xuyên suốt phiên làm việc của lệnh.
|
|
74
74
|
|
|
75
|
-
##
|
|
75
|
+
## Bước 3 — CHECKPOINT
|
|
76
76
|
|
|
77
|
-
|
|
77
|
+
Sau khi hoàn thành Bước 1 và 2, hiển thị bản tóm tắt và chờ xác nhận:
|
|
78
78
|
|
|
79
79
|
```
|
|
80
80
|
CHECKPOINT
|
|
81
81
|
-----------
|
|
82
82
|
Target : {resolved file path}
|
|
83
|
-
Project : {project.name
|
|
83
|
+
Project : {project.name từ project-context.yaml}
|
|
84
84
|
Tech stack : {language} / {framework}
|
|
85
|
-
Module : {module
|
|
86
|
-
Domains : {
|
|
85
|
+
Module : {module nếu có, else "not configured"}
|
|
86
|
+
Domains : {danh sách domain, ngăn cách bởi dấu phẩy}
|
|
87
87
|
|
|
88
|
-
|
|
88
|
+
Tiếp tục? (Y/N)
|
|
89
89
|
```
|
|
90
90
|
|
|
91
|
-
|
|
92
|
-
- "Y" →
|
|
93
|
-
- "N" →
|
|
91
|
+
Chờ người dùng trả lời rõ ràng "Y" hoặc "N" rồi mới tiếp tục.
|
|
92
|
+
- "Y" → tiếp tục sang các bước riêng của lệnh bên dưới.
|
|
93
|
+
- "N" → dừng lại và hỏi người dùng muốn thay đổi gì.
|
|
94
94
|
|
|
95
95
|
|
|
96
|
-
*
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
96
|
+
*Lưu ý: Với lệnh này, target ở Bước 1 là một file PRD `.md` hoặc file BDD `.feature`.
|
|
97
|
+
Nếu path là file PRD (`{TICKET-ID}-{prd-slug}.md` — file `.md` ở gốc feature folder `{paths.specs_dir}/{domain}/{prd-slug}/`) → PRD Review Mode.
|
|
98
|
+
Nếu path kết thúc bằng `.feature` → BDD Review Mode.
|
|
99
|
+
Nếu `$ARGUMENTS` chứa `--resume` → bỏ qua sang Resume Mode bên dưới.
|
|
100
|
+
Nếu `$ARGUMENTS` chứa `--fix` → bỏ qua sang Fix Mode bên dưới (áp dụng ngay mọi finding auto-fixable).*
|
|
101
101
|
|
|
102
102
|
## Context
|
|
103
|
-
# Context Loader —
|
|
103
|
+
# Context Loader — Nạp toàn bộ context dự án
|
|
104
104
|
|
|
105
|
-
|
|
105
|
+
Thực hiện các bước theo đúng thứ tự. Lưu mọi thứ vào bộ nhớ trong suốt phiên làm việc của lệnh.
|
|
106
106
|
|
|
107
|
-
**
|
|
108
|
-
-
|
|
109
|
-
-
|
|
110
|
-
-
|
|
111
|
-
-
|
|
112
|
-
-
|
|
107
|
+
**Hướng dẫn ưu tiên (chống lost-in-middle):**
|
|
108
|
+
- Bước 1–2 là PROJECT-CONFIG — nạp trước, phân giải mọi path và metadata.
|
|
109
|
+
- Bước 3 là CRITICAL — kiến trúc + coding standards, là các sự thật ưu tiên cao nhất khi sinh nội dung.
|
|
110
|
+
- Bước 4 là SAFETY — quy tắc bảo vệ dữ liệu, thực thi ngầm suốt cả phiên.
|
|
111
|
+
- Bước 5–6 là DOMAIN KNOWLEDGE — thuật ngữ và định nghĩa entity.
|
|
112
|
+
- Bước 7 là WORKING MEMORY RECAP — chốt các sự thật quan trọng lên đầu bộ nhớ làm việc.
|
|
113
113
|
|
|
114
114
|
---
|
|
115
115
|
|
|
116
|
-
##
|
|
116
|
+
## Bước 1 — [PROJECT-CONFIG] Nạp project-context.yaml
|
|
117
117
|
|
|
118
|
-
|
|
118
|
+
Đọc `.agent/project-context.yaml`. Trích xuất và lưu:
|
|
119
119
|
|
|
120
120
|
**Tech Stack:**
|
|
121
|
-
- `tech_stack.language` →
|
|
122
|
-
- `tech_stack.framework` →
|
|
123
|
-
- `tech_stack.build_tool` → build tool (
|
|
124
|
-
- `tech_stack.test_framework` → test framework (
|
|
125
|
-
- `tech_stack.database` → database (
|
|
126
|
-
- `tech_stack.module` →
|
|
121
|
+
- `tech_stack.language` → ngôn ngữ đang dùng (vd: Java 17, TypeScript, C#, Go)
|
|
122
|
+
- `tech_stack.framework` → framework đang dùng (vd: Spring Boot 3.2, Angular 17, .NET 8)
|
|
123
|
+
- `tech_stack.build_tool` → build tool (vd: Maven, npm, dotnet, go)
|
|
124
|
+
- `tech_stack.test_framework` → test framework (vd: JUnit 5 + Mockito, Jest, xUnit)
|
|
125
|
+
- `tech_stack.database` → database (vd: PostgreSQL, MySQL, MongoDB)
|
|
126
|
+
- `tech_stack.module` → module profile đang dùng (vd: java-spring, angular, dotnet, golang, context-engineering)
|
|
127
127
|
|
|
128
128
|
**Conventions:**
|
|
129
|
-
- `conventions.build_command` →
|
|
130
|
-
- `conventions.test_command` →
|
|
131
|
-
- `conventions.service_run` →
|
|
132
|
-
- `conventions.ticket_prefix` → ticket ID
|
|
129
|
+
- `conventions.build_command` → cách compile/build
|
|
130
|
+
- `conventions.test_command` → cách chạy test
|
|
131
|
+
- `conventions.service_run` → cách khởi động service
|
|
132
|
+
- `conventions.ticket_prefix` → tiền tố ticket ID (vd: PROJ, FEAT, UC)
|
|
133
133
|
|
|
134
134
|
**Domains:**
|
|
135
|
-
- `domains` →
|
|
136
|
-
|
|
137
|
-
**Paths (
|
|
138
|
-
- `paths.specs_dir` → spec
|
|
139
|
-
- `paths.refinement_dir` → findings/review
|
|
140
|
-
- `paths.qc_dir` → QC automation
|
|
141
|
-
- `paths.qc_skills_dir` →
|
|
142
|
-
- `paths.product_definitions_dir` → product
|
|
143
|
-
- `paths.domain_knowledge_dir` → domain knowledge
|
|
144
|
-
- `paths.business_dictionary` → path
|
|
145
|
-
- `paths.core_entities` → path
|
|
146
|
-
- `paths.tech_docs_dir` →
|
|
147
|
-
- `paths.trace_dir` →
|
|
148
|
-
|
|
149
|
-
|
|
135
|
+
- `domains` → danh sách các business domain đang hoạt động
|
|
136
|
+
|
|
137
|
+
**Paths (nếu có):**
|
|
138
|
+
- `paths.specs_dir` → gốc của spec artifact — PRD, BDD, tech-docs, design-spec. Cấu trúc: `{specs_dir}/{domain}/{prd-slug}/{ {TICKET-ID}-{prd-slug}.md | bdd/ | tech-docs/ | design-spec/}` (file PRD đặt tên `{TICKET-ID}-{prd-slug}.md`, là file `.md` duy nhất ở gốc feature folder)
|
|
139
|
+
- `paths.refinement_dir` → thư mục output cho findings/review
|
|
140
|
+
- `paths.qc_dir` → gốc artifact QC automation (hiện ở top-level, mỗi UC một thư mục con: `{qc_dir}/{UC-ID}/`)
|
|
141
|
+
- `paths.qc_skills_dir` → nơi các lệnh qc-* nạp QC skill (mặc định bundled `.agent/skills/qc`; override sang repo/submodule riêng của team QC để bản nâng cấp framework không ghi đè)
|
|
142
|
+
- `paths.product_definitions_dir` → gốc product definition
|
|
143
|
+
- `paths.domain_knowledge_dir` → gốc domain knowledge
|
|
144
|
+
- `paths.business_dictionary` → path tới business-dictionary.md
|
|
145
|
+
- `paths.core_entities` → path tới core-entities.md
|
|
146
|
+
- `paths.tech_docs_dir` → gốc tài liệu kỹ thuật (gộp với specs_dir trong bố cục feature-package — tech-docs nằm dưới `{specs_dir}/{domain}/{prd-slug}/tech-docs/`)
|
|
147
|
+
- `paths.trace_dir` → thư mục trạng thái trace; cấu trúc: `.trace/{domain}/{prd-slug}/{UC-ID}.tsv`
|
|
148
|
+
|
|
149
|
+
Nếu không có section `paths`, dùng các giá trị mặc định:
|
|
150
150
|
- `specs_dir` = `specs`
|
|
151
151
|
- `refinement_dir` = `.agent/review`
|
|
152
152
|
- `qc_dir` = `docs`
|
|
@@ -158,184 +158,184 @@ If `paths` section is absent, use these defaults:
|
|
|
158
158
|
- `tech_docs_dir` = `specs`
|
|
159
159
|
- `trace_dir` = `.trace`
|
|
160
160
|
|
|
161
|
-
|
|
161
|
+
Lưu ý: Trong bố cục feature-package, `specs_dir` là gốc thống nhất. Mọi loại spec artifact (PRD, BDD, tech-docs, design-spec) đều nằm dưới `{specs_dir}/{domain}/{prd-slug}/`. `prd-slug` là tên folder feature-package, không phải một biến config riêng.
|
|
162
162
|
|
|
163
|
-
**
|
|
164
|
-
- `specs/payment/create-invoice/
|
|
165
|
-
- `specs/payment/create-invoice/bdd/system/PAY-UC1.feature` → `prd_slug = create-invoice` *(
|
|
166
|
-
- `specs/payment/create-invoice/bdd/web/PAY-UC1.feature` → `prd_slug = create-invoice` *(
|
|
167
|
-
- `specs/payment/create-invoice/tech-docs/PAY-UC1-tech-design.md` → `prd_slug = create-invoice` *(
|
|
163
|
+
**Cách trích xuất `prd_slug` (đúng cho MỌI target file, bất kể độ sâu lồng nhau):** với một path target dạng `{specs_dir}/{domain}/{prd-slug}/...`, lấy **segment path đầu tiên sau `{specs_dir}/{domain}/`** — tức vị trí `{prd-slug}`. KHÔNG dùng folder cha trực tiếp của file, vì artifact BDD/tech-docs/design-spec lồng sâu hơn một hoặc hai cấp bên trong package. Ví dụ:
|
|
164
|
+
- `specs/payment/create-invoice/PAY01-create-invoice.md` → `prd_slug = create-invoice`
|
|
165
|
+
- `specs/payment/create-invoice/bdd/system/PAY-UC1.feature` → `prd_slug = create-invoice` *(KHÔNG phải `system`)*
|
|
166
|
+
- `specs/payment/create-invoice/bdd/web/PAY-UC1.feature` → `prd_slug = create-invoice` *(KHÔNG phải `web`)*
|
|
167
|
+
- `specs/payment/create-invoice/tech-docs/PAY-UC1-tech-design.md` → `prd_slug = create-invoice` *(KHÔNG phải `tech-docs`)*
|
|
168
168
|
- `specs/payment/create-invoice/design-spec/PAY-design-spec-web.md` → `prd_slug = create-invoice`
|
|
169
169
|
|
|
170
|
-
|
|
170
|
+
Mọi artifact cùng cấp của một feature (PRD, BDD của từng platform, tech-docs BE + FE, design-spec, và trace TSV) đều phân giải về **cùng một `prd_slug`** — nên một BDD **system** hay tech-doc **system/BE** được tổng hợp sẽ nằm chung package `{specs_dir}/{domain}/{prd-slug}/` với các artifact web/app mà nó được suy ra từ đó.
|
|
171
171
|
|
|
172
|
-
|
|
172
|
+
Nếu `tech_stack.module` được đặt, đồng thời nạp `.agent/modules/{module}/stack-profile.yaml` nếu file tồn tại.
|
|
173
173
|
|
|
174
174
|
---
|
|
175
175
|
|
|
176
|
-
##
|
|
176
|
+
## Bước 1.5 — [SERVICE ROUTING] Phân giải path service (chế độ umbrella)
|
|
177
177
|
|
|
178
|
-
*
|
|
178
|
+
*Bỏ qua hoàn toàn bước này nếu `setup.mode` không phải `"umbrella"` và không có section `services` trong project-context.yaml.*
|
|
179
179
|
|
|
180
|
-
|
|
180
|
+
Nếu có section `services`:
|
|
181
181
|
|
|
182
|
-
**1.
|
|
183
|
-
-
|
|
184
|
-
-
|
|
185
|
-
*(
|
|
186
|
-
-
|
|
182
|
+
**1. Phát hiện active domain** (theo thứ tự ưu tiên):
|
|
183
|
+
- Đọc `@trace.domain` từ frontmatter của target file (nếu Gate đã nạp một target file)
|
|
184
|
+
- Trích xuất từ path target file: `domain` = segment đầu tiên sau base path `specs_dir`; `prd_slug` = segment kế tiếp (folder feature-package). Điều này đúng ở mọi độ sâu target — xem quy tắc trích xuất `prd_slug` ở Bước 1.
|
|
185
|
+
*(vd: `specs/user/create-account/USR01-create-account.md` **và** `specs/user/create-account/bdd/system/UC1.feature` đều → domain = `user`, prd_slug = `create-account`)*
|
|
186
|
+
- Nếu `$ARGUMENTS` chứa một path, trích xuất segment domain sau `specs_dir`
|
|
187
187
|
|
|
188
|
-
**2. Route
|
|
189
|
-
- Override `paths.specs_dir` → `services.{domain}.specs_dir` — **
|
|
190
|
-
- Override `paths.tech_docs_dir` → `services.{domain}.tech_docs_dir` — **
|
|
191
|
-
-
|
|
192
|
-
-
|
|
193
|
-
-
|
|
188
|
+
**2. Route tới service** — nếu active domain khớp với một key trong `services`:
|
|
189
|
+
- Override `paths.specs_dir` → `services.{domain}.specs_dir` — **chỉ khi `setup.spec_source` KHÔNG được đặt.** Khi `spec_source` ĐƯỢC đặt, MỌI BDD (web/app/**system**) là artifact dùng chung liên team → để bước 4 route sang spec repo; KHÔNG pin theo service ở đây.
|
|
190
|
+
- Override `paths.tech_docs_dir` → `services.{domain}.tech_docs_dir` — **chỉ khi `setup.spec_source` KHÔNG được đặt.** Khi `spec_source` ĐƯỢC đặt, tech-design (API contract) là artifact liên team và phải nằm trong spec repo dùng chung (xử lý ở bước 4), nên để bước 4 route `tech_docs_dir` — KHÔNG pin theo service ở đây.
|
|
191
|
+
- Lưu `active_service` = `services.{domain}.path`
|
|
192
|
+
- Lưu `active_service_module` = `services.{domain}.module`
|
|
193
|
+
- Nếu service có `module` riêng → dùng nó làm `active_module` (override `tech_stack.module`)
|
|
194
194
|
|
|
195
|
-
**3. Fallback** —
|
|
196
|
-
-
|
|
197
|
-
-
|
|
195
|
+
**3. Fallback** — nếu không phát hiện được domain hoặc không có service key khớp:
|
|
196
|
+
- Giữ path mặc định từ Bước 1
|
|
197
|
+
- Đặt `active_service = unresolved`
|
|
198
198
|
|
|
199
|
-
**4.
|
|
200
|
-
- Override `paths.specs_dir` → `{spec_source}/specs` — **
|
|
201
|
-
- Override `paths.tech_docs_dir` → `{spec_source}/specs` — **
|
|
199
|
+
**4. Tự động override theo spec source** — nếu `setup.spec_source` được đặt VÀ path tương ứng chưa được set tường minh trong `paths:`:
|
|
200
|
+
- Override `paths.specs_dir` → `{spec_source}/specs` — **luôn khi `spec_source` được đặt.** Mọi spec artifact (PRD, BDD, tech-docs, design-spec) nằm dưới gốc spec thống nhất trong spec repo dùng chung theo bố cục feature-package: `{spec_source}/specs/{domain}/{prd-slug}/`. Mọi umbrella (FE/App/BE) đều đọc từ đây. *(`specs/` theo service chỉ khi không có `spec_source`.)*
|
|
201
|
+
- Override `paths.tech_docs_dir` → `{spec_source}/specs` — **luôn khi `spec_source` được đặt** (bước 2 không còn pin tech-docs theo service trong trường hợp này). Tech-docs nằm tại `{spec_source}/specs/{domain}/{prd-slug}/tech-docs/`. Tech-design CHÍNH LÀ API contract liên team: BE viết ở đây, FE/App đọc nó từ cùng spec submodule tại `/generate-code --phase=integration`. *(tech-docs theo service chỉ xảy ra khi không có `spec_source` — repo BE thuần đa-service không có spec module dùng chung.)*
|
|
202
202
|
- Override `paths.domain_knowledge_dir` → `{spec_source}/specs/domain-knowledge`
|
|
203
203
|
- Override `paths.business_dictionary` → `{spec_source}/specs/domain-knowledge/business-dictionary.md`
|
|
204
204
|
- Override `paths.core_entities` → `{spec_source}/specs/domain-knowledge/core-entities.md`
|
|
205
205
|
- Override `paths.bug_reports_dir` → `{spec_source}/feedback/bug-reports`
|
|
206
206
|
- Override `paths.bdd_proposals_dir` → `{spec_source}/feedback/bdd-proposals`
|
|
207
207
|
- Override `paths.prd_change_requests_dir` → `{spec_source}/feedback/prd-change-requests`
|
|
208
|
-
- Override `paths.trace_dir` → `{spec_source}/.trace` — **
|
|
208
|
+
- Override `paths.trace_dir` → `{spec_source}/.trace` — **luôn khi `spec_source` được đặt.** Trace TSV được gộp vào spec repo (một nơi authoritative duy nhất, không tách theo service) để PM/PO có một chỗ duy nhất quản lý trạng thái. Cấu trúc bên trong: `.trace/{domain}/{prd-slug}/{UC-ID}.tsv`. Các lệnh phía code (`/generate-code`, `/dev-run-test`, `/qc-run-test`) chạy từ `service_root` nhưng **ghi trace row của chúng vào `{spec_source}/.trace/{domain}/{prd-slug}/`** — giống như chúng đã push `feedback/` vào đó. *(`.trace` theo service chỉ khi không có `spec_source`.)*
|
|
209
209
|
|
|
210
|
-
> **
|
|
210
|
+
> **Vì sao đặt dưới `spec_source`:** PRD, BDD, tech-docs, design-spec, domain knowledge, feedback của tester, **và trạng thái coverage `.trace/`** đều là **artifact liên team** — chúng nằm trong **spec repo dùng chung** theo bố cục feature-package để mọi umbrella (FE/App/BE) và PM đọc từ một nguồn qua `/sync`. Trong bố cục feature-package, một folder `specs/{domain}/{prd-slug}/` gom tất cả loại artifact của một PRD, giúp spec repo tự đủ và dễ điều hướng theo feature. Service submodule chỉ chứa **code** (+ tooling build/test). `.trace/` và `feedback/` là khu vực **ghi** của dev/QC trong spec repo. Ở chế độ single-service (không có `spec_source`), mọi thứ mặc định dưới gốc repo — vẫn là một repo.
|
|
211
211
|
|
|
212
212
|
---
|
|
213
213
|
|
|
214
|
-
##
|
|
214
|
+
## Bước 1.6 — [SERVICE CONVENTIONS] Nạp convention riêng của service (chế độ umbrella)
|
|
215
215
|
|
|
216
|
-
*
|
|
216
|
+
*Bỏ qua hoàn toàn bước này nếu `active_service` là `"unresolved"` hoặc context ở chế độ single-service.*
|
|
217
217
|
|
|
218
|
-
|
|
218
|
+
Khi `active_service` đã được phân giải thành một path thật ở Bước 1.5 (vd: `user-service/`):
|
|
219
219
|
|
|
220
|
-
**1.
|
|
220
|
+
**1. Định vị config của service** — thử theo thứ tự ưu tiên:
|
|
221
221
|
- `{active_service}/.agent/project-context.yaml`
|
|
222
222
|
- `{active_service}/project-context.yaml`
|
|
223
223
|
|
|
224
|
-
**2.
|
|
224
|
+
**2. Nếu tìm thấy, override bằng giá trị riêng của service:**
|
|
225
225
|
|
|
226
|
-
|
|
|
226
|
+
| Biến | Nguồn |
|
|
227
227
|
|----------|--------|
|
|
228
|
-
| `conventions.test_command` |
|
|
229
|
-
| `conventions.build_command` |
|
|
230
|
-
| `paths.trace_dir` | **
|
|
231
|
-
| `paths.specs_dir` | **
|
|
228
|
+
| `conventions.test_command` | `conventions.test_command` của service |
|
|
229
|
+
| `conventions.build_command` | `conventions.build_command` của service |
|
|
230
|
+
| `paths.trace_dir` | **Nếu `spec_source` được đặt → giữ route spec-repo của bước 4 (`{spec_source}/.trace`); bỏ qua mọi `trace_dir` cấp service.** Chỉ khi không có `spec_source`: `{active_service}/{service paths.trace_dir}` (mặc định `{active_service}/.trace`). |
|
|
231
|
+
| `paths.specs_dir` | **Nếu `spec_source` được đặt → giữ route spec-repo của bước 4 (`{spec_source}/specs`); bỏ qua mọi `specs_dir` cấp service** (mọi spec artifact đều liên team, không bao giờ theo service ở chế độ này). Chỉ khi không có `spec_source`: `{active_service}/{service paths.specs_dir}` nếu được set, else dùng override ở Bước 1.5. |
|
|
232
232
|
|
|
233
|
-
**3.
|
|
234
|
-
-
|
|
235
|
-
- **
|
|
233
|
+
**3. Lưu** `service_root = {active_service}` làm mốc thư mục làm việc cho mọi lệnh phía sau:
|
|
234
|
+
- Các lệnh shell (`/dev-run-test`, `/dev-gen-test`) chạy **bên trong** `service_root`
|
|
235
|
+
- **File source/test** được ghi tương đối với `service_root`; **trace TSV** được ghi vào `{paths.trace_dir}` (là spec repo khi `spec_source` được đặt — một thao tác ghi liên-repo, commit/push vào spec submodule giống như `feedback/`).
|
|
236
236
|
|
|
237
|
-
**4.
|
|
237
|
+
**4. Nếu không tìm thấy config của service** — giữ mặc định umbrella, vẫn set `service_root = {active_service}` (luôn cần mốc path kể cả khi không có config override).
|
|
238
238
|
|
|
239
239
|
---
|
|
240
240
|
|
|
241
|
-
##
|
|
241
|
+
## Bước 2 — [PROJECT-CONFIG] Nạp module stack profile (có điều kiện)
|
|
242
242
|
|
|
243
|
-
|
|
244
|
-
Merge framework
|
|
245
|
-
|
|
243
|
+
Nếu `tech_stack.module` được đặt, đọc `.agent/modules/{module}/stack-profile.yaml`.
|
|
244
|
+
Merge các convention riêng của framework (layer pattern, test pattern, quy tắc đặt tên) vào context đã nạp.
|
|
245
|
+
Nếu file không tồn tại → bỏ qua âm thầm.
|
|
246
246
|
|
|
247
247
|
---
|
|
248
248
|
|
|
249
|
-
##
|
|
249
|
+
## Bước 3 — [CRITICAL] Nạp CLAUDE.md (phân tầng: root + service overlay)
|
|
250
250
|
|
|
251
|
-
|
|
251
|
+
*Đây là context ưu tiên cao nhất — nó định nghĩa CÁCH viết code và tài liệu cho dự án này.*
|
|
252
252
|
|
|
253
|
-
CLAUDE.md
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
253
|
+
CLAUDE.md được nạp theo **hai tầng** để các quy tắc toàn-umbrella và kiến trúc/coding standards
|
|
254
|
+
riêng của service kết hợp đúng cách. Agent luôn đứng ở gốc umbrella, nhưng code triển khai nằm
|
|
255
|
+
trong một service submodule với stack, kiến trúc, và convention RIÊNG của nó — nên CLAUDE.md của
|
|
256
|
+
service phải thắng khi sinh code.
|
|
257
257
|
|
|
258
|
-
**
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
258
|
+
**Tầng 1 — [BASE] Root CLAUDE.md (toàn umbrella).**
|
|
259
|
+
Đọc `CLAUDE.md` ở gốc repo. Coi nội dung của nó là **nền tảng dùng chung** cho cả umbrella —
|
|
260
|
+
git convention, tư thế bảo vệ dữ liệu, quy tắc xuyên suốt, và (ở chế độ single-service) là
|
|
261
|
+
kiến trúc + coding standards duy nhất của dự án.
|
|
262
262
|
|
|
263
|
-
**
|
|
264
|
-
*
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
Overlay
|
|
268
|
-
coding standards,
|
|
269
|
-
(
|
|
263
|
+
**Tầng 2 — [OVERLAY] Service CLAUDE.md (chỉ chế độ umbrella).**
|
|
264
|
+
*Chỉ chạy nếu `service_root` đã được set ở Bước 1.6 (tức đã route tới một service thật).*
|
|
265
|
+
Đọc `{service_root}/CLAUDE.md`. File này định nghĩa kiến trúc + coding standards của **stack
|
|
266
|
+
thực sự đang được triển khai** (vd: `user-service` = java-spring, `web` = nextjs).
|
|
267
|
+
Overlay nó lên trên Tầng 1: **khi có xung đột, giá trị của service THẮNG** cho kiến trúc,
|
|
268
|
+
coding standards, và error handling. Các giá trị Tầng 1 mà service không định nghĩa lại
|
|
269
|
+
(vd: git convention, banned pattern dùng chung toàn tổ chức) vẫn có hiệu lực.
|
|
270
270
|
|
|
271
|
-
|
|
271
|
+
Từ kết quả **đã merge**, trích xuất và lưu:
|
|
272
272
|
|
|
273
|
-
- **§1 Project Overview** →
|
|
274
|
-
- **§2 Architecture** → layer
|
|
275
|
-
- **§3 Coding Standards** →
|
|
276
|
-
- **§5 Error Handling** → exception
|
|
277
|
-
- **§7 Git Conventions** →
|
|
273
|
+
- **§1 Project Overview** → tên dự án, ngôn ngữ, framework, lệnh build/test, domains
|
|
274
|
+
- **§2 Architecture** → thứ tự layer (vd: Controller → Facade → Service → Repository), quy tắc kiến trúc — *service overlay thắng*
|
|
275
|
+
- **§3 Coding Standards** → quy tắc đặt tên (class, method), kiểu response wrapper, pattern bị cấm — *service overlay thắng*
|
|
276
|
+
- **§5 Error Handling** → kiểu exception, mapping HTTP status code, tên class not-found exception — *service overlay thắng*
|
|
277
|
+
- **§7 Git Conventions** → pattern đặt tên branch, format commit message — *lấy theo root trừ khi service định nghĩa lại*
|
|
278
278
|
|
|
279
|
-
**
|
|
280
|
-
-
|
|
281
|
-
-
|
|
282
|
-
-
|
|
283
|
-
-
|
|
279
|
+
**Quy tắc phân giải:**
|
|
280
|
+
- Nếu cả hai tầng tồn tại → merge như trên; ghi `claude_md_source = root + {service_root}`.
|
|
281
|
+
- Nếu chỉ có service overlay (không có root CLAUDE.md) → dùng file service một mình; `claude_md_source = {service_root}`.
|
|
282
|
+
- Nếu `service_root` được set nhưng `{service_root}/CLAUDE.md` **thiếu** → fallback về root CLAUDE.md và gắn cờ ⚠️ trong recap Bước 7 (service không có định nghĩa kiến trúc/coding-standards — việc sinh code sẽ dùng mặc định umbrella, có thể sai stack).
|
|
283
|
+
- Nếu cả hai đều không tồn tại → ghi nhận CLAUDE.md thiếu và tiếp tục chỉ với dữ liệu từ project-context.yaml.
|
|
284
284
|
|
|
285
285
|
---
|
|
286
286
|
|
|
287
|
-
##
|
|
287
|
+
## Bước 4 — [SAFETY] Nạp quy tắc bảo vệ dữ liệu
|
|
288
288
|
|
|
289
|
-
|
|
289
|
+
Đọc `.agent/rules/data-protection.md` (hoặc `rules/data-protection.md` từ bản cài đặt framework).
|
|
290
290
|
|
|
291
|
-
|
|
291
|
+
Lưu các pattern file nhạy cảm — bạn **tuyệt đối không** đọc, ghi, hiển thị, hay tham chiếu nội dung từ các file khớp những pattern đó trong suốt cả phiên.
|
|
292
292
|
|
|
293
|
-
|
|
293
|
+
Nếu cả hai file đều không tồn tại → áp dụng mặc định built-in: không bao giờ truy cập `.env*`, `*.key`, `*.pem`, `*secret*`, `*password*`, `*credential*`.
|
|
294
294
|
|
|
295
295
|
---
|
|
296
296
|
|
|
297
|
-
##
|
|
297
|
+
## Bước 5 — [DOMAIN] Nạp Business Dictionary (có điều kiện)
|
|
298
298
|
|
|
299
|
-
|
|
299
|
+
Kiểm tra file business dictionary có tồn tại không (dùng `paths.business_dictionary` đã phân giải ở Bước 1).
|
|
300
300
|
|
|
301
|
-
|
|
302
|
-
- **Canonical Terms** →
|
|
303
|
-
- **Banned Terms** →
|
|
304
|
-
- **Status / Enum Registry** →
|
|
301
|
+
Nếu tồn tại, đọc và trích xuất:
|
|
302
|
+
- **Canonical Terms** → danh sách đầy đủ các thuật ngữ chuẩn và định nghĩa
|
|
303
|
+
- **Banned Terms** → danh sách đầy đủ các thuật ngữ bị cấm và bản thay thế chuẩn
|
|
304
|
+
- **Status / Enum Registry** → các giá trị enum được phép theo từng entity
|
|
305
305
|
|
|
306
|
-
|
|
307
|
-
-
|
|
308
|
-
-
|
|
306
|
+
Lưu danh sách banned term để **thực thi chủ động** suốt phiên làm việc của lệnh:
|
|
307
|
+
- Khi sinh bất kỳ văn bản nào (PRD, BDD, comment code, tech docs), kiểm tra không có banned term nào xuất hiện
|
|
308
|
+
- Tự động thay banned term bằng bản chuẩn tương đương
|
|
309
309
|
|
|
310
|
-
|
|
310
|
+
Nếu file không tồn tại → bỏ qua âm thầm. Không cảnh báo hay chặn.
|
|
311
311
|
|
|
312
312
|
---
|
|
313
313
|
|
|
314
|
-
##
|
|
314
|
+
## Bước 6 — [DOMAIN] Nạp Core Entities (có điều kiện)
|
|
315
315
|
|
|
316
|
-
|
|
317
|
-
|
|
316
|
+
Kiểm tra file core entities có tồn tại tại `paths.core_entities` không (đã phân giải ở Bước 1).
|
|
317
|
+
Path mặc định: `specs/domain-knowledge/core-entities.md`.
|
|
318
318
|
|
|
319
|
-
|
|
320
|
-
- **Entity catalog** →
|
|
321
|
-
- **Field name registry** →
|
|
322
|
-
- **Relationship map** →
|
|
319
|
+
Nếu tồn tại, đọc và lưu:
|
|
320
|
+
- **Entity catalog** → với mỗi entity: tên, mục đích, service sở hữu, các field chính (tên + kiểu), business invariant, và quan hệ
|
|
321
|
+
- **Field name registry** → tên field chuẩn dùng trong code và tài liệu được sinh ra
|
|
322
|
+
- **Relationship map** → cách các entity liên hệ với nhau (1:N, N:N, embedded, v.v.)
|
|
323
323
|
|
|
324
|
-
**
|
|
325
|
-
-
|
|
326
|
-
-
|
|
327
|
-
-
|
|
324
|
+
**Cách dùng catalog này:**
|
|
325
|
+
- Khi sinh code: dùng tên field, kiểu, và quan hệ định nghĩa ở đây — KHÔNG suy đoán từ code có sẵn
|
|
326
|
+
- Khi sinh PRD/BDD: tham chiếu tên entity từ catalog này để nhất quán
|
|
327
|
+
- Khi sinh tech-docs: dùng catalog này làm nguồn chân lý cho định nghĩa entity
|
|
328
328
|
|
|
329
|
-
|
|
329
|
+
Nếu file không tồn tại → bỏ qua âm thầm.
|
|
330
330
|
|
|
331
331
|
---
|
|
332
332
|
|
|
333
|
-
##
|
|
333
|
+
## Bước 6.5 — [PLATFORM] Suy ra active_module và platform_type
|
|
334
334
|
|
|
335
|
-
|
|
335
|
+
Dùng `tech_stack.module` đã nạp ở Bước 1, suy ra và lưu hai biến để mọi lệnh phía sau dùng:
|
|
336
336
|
|
|
337
337
|
```
|
|
338
|
-
active_module = tech_stack.module (
|
|
338
|
+
active_module = tech_stack.module (vd: "java-spring", "react", "flutter")
|
|
339
339
|
```
|
|
340
340
|
|
|
341
341
|
| `platform_type` | Modules |
|
|
@@ -344,155 +344,155 @@ active_module = tech_stack.module (e.g. "java-spring", "react", "flutter")
|
|
|
344
344
|
| `web-frontend` | `react`, `nextjs`, `vue`, `nuxt`, `angular` |
|
|
345
345
|
| `mobile` | `flutter`, `react-native`, `ios-swiftui`, `android-compose` |
|
|
346
346
|
|
|
347
|
-
|
|
347
|
+
Nếu `tech_stack.module` rỗng hoặc không nhận diện được → set `platform_type = "unknown"` và gắn cờ ⚠️ trong recap Bước 7.
|
|
348
348
|
|
|
349
|
-
|
|
349
|
+
Hai biến này (`active_module`, `platform_type`) là nguồn chuẩn cho mọi logic rẽ nhánh trong các lệnh cần hành vi riêng theo platform (dev-gen-test, debug, fix-bug, dev-smoke-test).
|
|
350
350
|
|
|
351
351
|
---
|
|
352
352
|
|
|
353
|
-
##
|
|
353
|
+
## Bước 6.7 — [GUARDRAILS] Nạp Project Lessons (có điều kiện)
|
|
354
354
|
|
|
355
|
-
*
|
|
356
|
-
|
|
355
|
+
*Các lỗi tích luỹ mà AI không được lặp lại trong dự án này. Chúng được bổ sung dần qua `/learn`
|
|
356
|
+
hoặc được chấp nhận trong `/review-code`, `/fix-bug`, `/debug`.*
|
|
357
357
|
|
|
358
|
-
|
|
359
|
-
-
|
|
360
|
-
- Else
|
|
361
|
-
-
|
|
358
|
+
Phân giải path file lessons:
|
|
359
|
+
- Dùng `paths.lessons_file` nếu được set (có thể bị service override ở chế độ umbrella, Bước 1.6)
|
|
360
|
+
- Else mặc định `specs/domain-knowledge/lessons-learned.md`
|
|
361
|
+
- Ở chế độ umbrella/service (khi `service_root` được set), nếu `paths.lessons_file` chưa set, mặc định `{service_root}/.agent/project-lessons.md`
|
|
362
362
|
|
|
363
|
-
|
|
364
|
-
-
|
|
365
|
-
-
|
|
366
|
-
-
|
|
363
|
+
Nếu file tồn tại, đọc và lưu TẤT CẢ lesson làm **GUARDRAIL ĐANG HOẠT ĐỘNG** cho phiên:
|
|
364
|
+
- Coi **Rule** của mỗi lesson là ràng buộc cứng — cùng mức ưu tiên với coding standards trong CLAUDE.md (Bước 3).
|
|
365
|
+
- Trước khi sinh hoặc sửa bất kỳ artifact nào (PRD, BDD, tech-doc, code, test), đối chiếu output với mọi lesson có `category` khớp lệnh hiện tại VÀ `scope` khớp target (domain / file).
|
|
366
|
+
- Nếu output sinh ra vi phạm một lesson → sửa **trước khi** trình bày, và ghi rõ lesson nào (`L-NNN`) đã được áp dụng.
|
|
367
367
|
|
|
368
|
-
|
|
368
|
+
Nếu file không tồn tại → bỏ qua âm thầm (chưa có lesson nào được ghi nhận).
|
|
369
369
|
|
|
370
370
|
---
|
|
371
371
|
|
|
372
|
-
##
|
|
372
|
+
## Bước 7 — [RECAP] Working Memory Recap (chống lost-in-middle)
|
|
373
373
|
|
|
374
|
-
|
|
375
|
-
|
|
376
|
-
(recency
|
|
374
|
+
Sau khi nạp toàn bộ context, tổng hợp và xuất một khối tóm tắt gọn.
|
|
375
|
+
Recap này đảm bảo các sự thật quan trọng nhất được nêu ở CUỐI quá trình nạp context
|
|
376
|
+
(hiệu ứng recency — tươi mới nhất trong bộ nhớ làm việc khi bắt đầu task).
|
|
377
377
|
|
|
378
|
-
|
|
378
|
+
Xuất đúng khối này:
|
|
379
379
|
```
|
|
380
380
|
[CTX LOADED]
|
|
381
381
|
Stack : {language} / {framework} / {database}
|
|
382
382
|
Platform : {active_module} ({platform_type})
|
|
383
|
-
Layers : {
|
|
384
|
-
CLAUDE.md : {root + {service_root} | {service_root}
|
|
383
|
+
Layers : {thứ tự layer từ CLAUDE.md §2 đã merge, vd: Controller → Facade → Service → Repository}
|
|
384
|
+
CLAUDE.md : {root + {service_root} | chỉ {service_root} | chỉ root | ⚠️ service overlay THIẾU — dùng root | missing}
|
|
385
385
|
Ticket : {ticket_prefix}-
|
|
386
386
|
Dict : {loaded — N canonical terms, M banned terms | missing}
|
|
387
387
|
Entities : {loaded — EntityA, EntityB, EntityC | missing}
|
|
388
|
-
Lessons : {loaded — N guardrails |
|
|
388
|
+
Lessons : {loaded — N guardrails | chưa có}
|
|
389
389
|
Service : {active_service} ({active_service_module}) | single-service
|
|
390
|
-
Svc Root : {service_root} — conventions + trace_dir
|
|
391
|
-
Status : {FULL | PARTIAL —
|
|
390
|
+
Svc Root : {service_root} — đã nạp conventions + trace_dir từ config service | —
|
|
391
|
+
Status : {FULL | PARTIAL — thiếu: CLAUDE.md / business-dict / core-entities | MINIMAL}
|
|
392
392
|
```
|
|
393
393
|
|
|
394
|
-
|
|
394
|
+
Nếu bất kỳ file CRITICAL nào thiếu (CLAUDE.md), gắn cờ rõ ràng để người dùng quyết định có tiếp tục hay không.
|
|
395
395
|
|
|
396
396
|
---
|
|
397
397
|
|
|
398
|
-
##
|
|
398
|
+
## Hoàn tất nạp Context
|
|
399
399
|
|
|
400
|
-
|
|
401
|
-
-
|
|
402
|
-
-
|
|
403
|
-
- Coding standards
|
|
404
|
-
-
|
|
405
|
-
-
|
|
406
|
-
- Entity catalog (field
|
|
407
|
-
-
|
|
400
|
+
Sau khi hoàn thành tất cả các bước, bạn đã nạp:
|
|
401
|
+
- Định danh dự án, tech stack, convention module
|
|
402
|
+
- Quy tắc kiến trúc và thứ tự layer ← **[CRITICAL — giữ trong bộ nhớ làm việc]**
|
|
403
|
+
- Coding standards và quy tắc đặt tên ← **[CRITICAL — giữ trong bộ nhớ làm việc]**
|
|
404
|
+
- Quy tắc bảo vệ dữ liệu (pattern file nhạy cảm không bao giờ truy cập)
|
|
405
|
+
- Quy tắc thuật ngữ kèm danh sách banned term ← **[DOMAIN — áp dụng cho mọi từ được sinh ra]**
|
|
406
|
+
- Entity catalog (tên field, kiểu, invariant) ← **[DOMAIN — dùng khi sinh code]**
|
|
407
|
+
- Toàn bộ path đã cấu hình
|
|
408
408
|
|
|
409
|
-
|
|
409
|
+
Tiếp tục sang bước kế tiếp của lệnh đang gọi.
|
|
410
410
|
|
|
411
411
|
|
|
412
412
|
---
|
|
413
413
|
|
|
414
|
-
##
|
|
414
|
+
## Phát hiện Review Mode
|
|
415
415
|
|
|
416
|
-
|
|
417
|
-
- `.feature`
|
|
418
|
-
-
|
|
419
|
-
-
|
|
416
|
+
Sau khi phân giải target file:
|
|
417
|
+
- File `.feature` → **BDD Review Mode** (nhảy tới section BDD)
|
|
418
|
+
- File `.md` ở gốc feature folder `{paths.specs_dir}/*/*/` (không phải dưới `bdd/`·`tech-docs/`·`design-spec/`) → **PRD Review Mode** (tiếp tục bên dưới)
|
|
419
|
+
- Không xác định → hỏi: "Đây là file PRD hay file BDD feature? (prd/bdd)"
|
|
420
420
|
|
|
421
|
-
|
|
422
|
-
- `--fix`
|
|
423
|
-
- `--resume`
|
|
421
|
+
Đồng thời kiểm tra flag:
|
|
422
|
+
- Có `--fix` → sau khi chạy hết các check, áp dụng ngay các finding `auto_fixable: true` (bỏ qua Review Board)
|
|
423
|
+
- Có `--resume` → bỏ qua phân tích hoàn toàn, sang Resume Mode
|
|
424
424
|
|
|
425
|
-
|
|
425
|
+
Suy ra tên file findings output:
|
|
426
426
|
- PRD: `{paths.refinement_dir}/{prd-slug}-review-context-findings.yaml`
|
|
427
427
|
- BDD: `{paths.refinement_dir}/{uc-id}-review-bdd-findings.yaml`
|
|
428
428
|
|
|
429
429
|
---
|
|
430
430
|
|
|
431
|
-
## Review
|
|
432
|
-
#
|
|
431
|
+
## Quy trình Review
|
|
432
|
+
# Review Fan-Out toàn diện + Hội tụ về độ đầy đủ
|
|
433
433
|
|
|
434
|
-
**
|
|
435
|
-
|
|
436
|
-
(
|
|
437
|
-
fan out
|
|
438
|
-
|
|
434
|
+
**Vì sao có cái này:** Một lượt review đơn không bao giờ liệt kê hết mọi vấn đề cùng lúc — model
|
|
435
|
+
dừng ở mức "đủ" findings, nên mỗi vòng review sau lại lòi ra vấn đề *mới*
|
|
436
|
+
(đập chuột chũi). Quy trình này ép review **hội tụ trong một lần chạy lệnh**:
|
|
437
|
+
fan out song song theo các chiều review, rồi lặp một critic độ-đầy-đủ cho tới khi một
|
|
438
|
+
vòng không sinh thêm gì mới, *trước khi* ghi file findings.
|
|
439
439
|
|
|
440
|
-
|
|
441
|
-
- **DIMENSIONS** —
|
|
442
|
-
(`/refine-prd` →
|
|
443
|
-
- **FINDINGS SCHEMA** —
|
|
440
|
+
Lệnh gọi cung cấp hai thứ:
|
|
441
|
+
- **DIMENSIONS** — danh sách các chiều review để fan out
|
|
442
|
+
(`/refine-prd` → 4 lăng kính; `/review-context` → các P-check hoặc B-check).
|
|
443
|
+
- **FINDINGS SCHEMA** — dạng YAML mà mỗi finding phải theo (định nghĩa trong lệnh).
|
|
444
444
|
|
|
445
|
-
> **
|
|
446
|
-
>
|
|
447
|
-
> sub-agent.
|
|
445
|
+
> **Bỏ qua ở chế độ sub-agent:** Nếu Gate Bước 0 đã set `_agent_mode: true`, toàn bộ
|
|
446
|
+
> quy trình này bị **bỏ qua** — orchestrator đã chạy sẵn một dimension/UC cho mỗi
|
|
447
|
+
> sub-agent. Chạy các check của lệnh trực tiếp trên section đã giới hạn và trả về findings.
|
|
448
448
|
|
|
449
449
|
---
|
|
450
450
|
|
|
451
|
-
## Phase 1 —
|
|
451
|
+
## Phase 1 — Quét dimension song song
|
|
452
452
|
|
|
453
|
-
**
|
|
454
|
-
|
|
455
|
-
|
|
456
|
-
|
|
453
|
+
**Bao nhiêu sub-agent:** *số lượng* agent không phải là đòn bẩy độ đầy đủ — bề rộng được
|
|
454
|
+
cố định bởi taxonomy DIMENSION (thêm agent vào cùng một dimension chỉ tìm lại cùng vấn đề),
|
|
455
|
+
còn *độ sâu* thuộc về vòng lặp critic ở Phase 2. Chọn **độ mịn fan-out**
|
|
456
|
+
theo kích thước target, tái dùng ngưỡng của `steps/spawn-agent.md`:
|
|
457
457
|
|
|
458
|
-
|
|
|
458
|
+
| Kích thước target | Độ mịn | Số agent |
|
|
459
459
|
|-------------|-------------|-------------|
|
|
460
|
-
| ≤ 3
|
|
461
|
-
| > 3
|
|
460
|
+
| ≤ 3 UC **và** ≤ 300 dòng | một agent cho mỗi DIMENSION trên cả file | = số dimension |
|
|
461
|
+
| > 3 UC **hoặc** > 300 dòng | một agent cho mỗi **DIMENSION × phạm vi UC** (các UC + một phạm vi PRD-global), gom batch để vừa giới hạn agent | `dimensions × (UCs + 1)`, có cap (xem dưới) |
|
|
462
462
|
|
|
463
|
-
|
|
464
|
-
|
|
463
|
+
Độ mịn lớn hơn giữ context của mỗi sub-agent nhỏ và quét nó vét cạn trên một
|
|
464
|
+
UC duy nhất — chính là điều ngăn bỏ sót trên các PRD lớn.
|
|
465
465
|
|
|
466
|
-
> **
|
|
467
|
-
>
|
|
468
|
-
> problem statement, terminology, glossary, changelog)
|
|
469
|
-
> fan out
|
|
470
|
-
> `uc_id: ""`)
|
|
471
|
-
> (
|
|
466
|
+
> **Các section global (không thuộc UC) — bắt buộc ở chế độ `DIMENSION × UC`.** Mỗi agent per-UC chỉ
|
|
467
|
+
> thấy một UC, nên các section toàn-PRD không thuộc UC nào (scope, success metric,
|
|
468
|
+
> problem statement, terminology, glossary, changelog) sẽ không được quét. Khi nào
|
|
469
|
+
> fan out theo UC, cũng phải thêm một phạm vi **"PRD-global"** (các section không thuộc UC, finding nhận
|
|
470
|
+
> `uc_id: ""`) bên cạnh danh sách UC. Nên số agent tự nhiên là `dimensions × (UCs + 1)`.
|
|
471
|
+
> (Không cần ở chế độ whole-file — ở đó mỗi agent đã thấy các section global rồi.)
|
|
472
472
|
|
|
473
|
-
### Agent cap — batch
|
|
473
|
+
### Agent cap — gom batch các UC khi fan-out quá rộng
|
|
474
474
|
|
|
475
|
-
`dimensions × (UCs + 1)`
|
|
476
|
-
|
|
475
|
+
`dimensions × (UCs + 1)` có thể bùng nổ trên PRD lớn (vd 6 check × (8 UC + 1) = 54
|
|
476
|
+
agent). Giới hạn mỗi wave ở **`AGENT_CAP = 12`** agent và gom batch các phạm vi UC cho vừa:
|
|
477
477
|
|
|
478
|
-
1.
|
|
479
|
-
2.
|
|
480
|
-
-
|
|
481
|
-
- Else
|
|
482
|
-
(
|
|
483
|
-
|
|
484
|
-
3.
|
|
478
|
+
1. Dựng danh sách phạm vi = `[UC1, UC2, …, UCn, PRD-global]` (độ dài `UCs + 1`).
|
|
479
|
+
2. Tính số-phạm-vi-mỗi-bucket: `groups = max(1, floor(AGENT_CAP / dimensions))`.
|
|
480
|
+
- Nếu `groups ≥ UCs + 1` → không cần batch, chạy một agent cho mỗi `DIMENSION × scope`.
|
|
481
|
+
- Else chia danh sách phạm vi thành `groups` bucket liền kề kích thước xấp xỉ bằng nhau
|
|
482
|
+
(giữ `PRD-global` ở bucket riêng nếu vừa; nếu không thì gắn vào bucket cuối).
|
|
483
|
+
Mỗi agent khi đó xử lý **một DIMENSION trên một bucket UC**.
|
|
484
|
+
3. Kích thước wave kết quả = `dimensions × groups ≤ AGENT_CAP`.
|
|
485
485
|
|
|
486
|
-
|
|
487
|
-
file,
|
|
488
|
-
|
|
486
|
+
Một agent đã batch review nhiều UC cùng lúc — vẫn giới hạn chặt hơn nhiều so với cả
|
|
487
|
+
file, nên độ phủ vẫn cao. `AGENT_CAP` là núm chỉnh duy nhất; tăng nếu host cho phép
|
|
488
|
+
concurrency nhiều hơn, giảm để tiết kiệm token. Chế độ whole-file (≤ 3 UC) không bao giờ chạm cap.
|
|
489
489
|
|
|
490
|
-
Spawn
|
|
491
|
-
|
|
492
|
-
|
|
493
|
-
dimension
|
|
490
|
+
Spawn các sub-agent đã chọn bằng Agent tool (gửi trong một message duy nhất để chúng
|
|
491
|
+
chạy đồng thời). Mỗi sub-agent nhận một **context window mới** và quét phạm vi của nó
|
|
492
|
+
chỉ qua **một** dimension duy nhất — độ phủ sâu hơn một session phải tung hứng mọi
|
|
493
|
+
dimension cùng lúc (tránh lost-in-the-middle).
|
|
494
494
|
|
|
495
|
-
|
|
495
|
+
Template prompt cho sub-agent (điền vào các ngoặc):
|
|
496
496
|
|
|
497
497
|
```
|
|
498
498
|
You are a {DIMENSION_NAME} reviewer. Read the full target file at {target_file}.
|
|
@@ -508,20 +508,20 @@ Return a JSON array of findings, each:
|
|
|
508
508
|
Return [] if this dimension is clean. Return ONLY the JSON array.
|
|
509
509
|
```
|
|
510
510
|
|
|
511
|
-
|
|
511
|
+
Gom mảng findings của mọi sub-agent vào một danh sách hợp nhất `ALL_FINDINGS`.
|
|
512
512
|
|
|
513
513
|
---
|
|
514
514
|
|
|
515
|
-
## Phase 2 —
|
|
515
|
+
## Phase 2 — Vòng lặp hội tụ critic độ-đầy-đủ
|
|
516
516
|
|
|
517
|
-
|
|
518
|
-
|
|
517
|
+
Đây là bước chống đập-chuột-chũi. Lặp cho tới khi **hai vòng liên tiếp thêm 0 finding
|
|
518
|
+
mới**, hoặc tới cap cứng **3 vòng**, cái nào đến trước:
|
|
519
519
|
|
|
520
|
-
1. Spawn
|
|
521
|
-
-
|
|
522
|
-
-
|
|
523
|
-
-
|
|
524
|
-
Prompt
|
|
520
|
+
1. Spawn một sub-agent **completeness-critic** bằng Agent tool. Cho nó:
|
|
521
|
+
- toàn bộ target file (`{target_file}`),
|
|
522
|
+
- danh sách `ALL_FINDINGS` hiện tại (để nó biết những gì đã được ghi nhận),
|
|
523
|
+
- cùng context gọn đó.
|
|
524
|
+
Prompt nó:
|
|
525
525
|
```
|
|
526
526
|
Here is a document and a list of issues already found. Read the WHOLE document.
|
|
527
527
|
List ONLY real, additional issues NOT already in the list — gaps, ambiguities,
|
|
@@ -530,142 +530,140 @@ findings**, or a hard cap of **3 rounds**, whichever comes first:
|
|
|
530
530
|
Do NOT repeat anything already listed. Return the same finding JSON shape, or [] if
|
|
531
531
|
nothing new.
|
|
532
532
|
```
|
|
533
|
-
2.
|
|
534
|
-
3.
|
|
535
|
-
4.
|
|
533
|
+
2. Thêm bất kỳ finding thực sự mới (chưa có trong `ALL_FINDINGS`) vào danh sách.
|
|
534
|
+
3. Nếu vòng này trả 0 finding mới → tăng bộ đếm dry-round; ngược lại reset về 0.
|
|
535
|
+
4. Dừng khi bộ đếm dry-round đạt 2, hoặc sau tổng cộng 3 vòng.
|
|
536
536
|
|
|
537
|
-
|
|
537
|
+
Ghi lại `convergence_rounds` (số vòng critic đã chạy) cho report.
|
|
538
538
|
|
|
539
539
|
---
|
|
540
540
|
|
|
541
|
-
## Phase 3 — Dedup,
|
|
542
|
-
|
|
543
|
-
|
|
544
|
-
|
|
545
|
-
|
|
546
|
-
|
|
547
|
-
1. **
|
|
548
|
-
`section` + `uc_id`
|
|
549
|
-
|
|
550
|
-
2. **
|
|
551
|
-
|
|
552
|
-
|
|
553
|
-
-
|
|
554
|
-
|
|
555
|
-
|
|
556
|
-
|
|
557
|
-
|
|
558
|
-
|
|
559
|
-
|
|
560
|
-
|
|
561
|
-
|
|
562
|
-
|
|
563
|
-
|
|
564
|
-
|
|
565
|
-
|
|
566
|
-
In the command's final report, add one line:
|
|
541
|
+
## Phase 3 — Dedup, giải quyết xung đột, merge
|
|
542
|
+
|
|
543
|
+
Các sub-agent chạy **mù với nhau** (độc lập = độ phủ đa dạng). Chúng không bao giờ
|
|
544
|
+
trao đổi hay điều hoà giữa chúng — mọi xử lý trùng/xung đột diễn ra **ở đây trong
|
|
545
|
+
orchestrator**, nơi thấy toàn bộ tập findings.
|
|
546
|
+
|
|
547
|
+
1. **Khử trùng lặp** `ALL_FINDINGS`: hai finding là trùng nếu cùng nhắm tới cùng
|
|
548
|
+
`section` + `uc_id` và mô tả cùng một vấn đề gốc. Giữ cái có `suggestion`
|
|
549
|
+
phong phú hơn; nếu khác nhau về severity, giữ severity **cao hơn**.
|
|
550
|
+
2. **Giải quyết xung đột** — nhóm các finding còn lại theo `section` + `uc_id` và kiểm tra
|
|
551
|
+
mâu thuẫn (hai finding có `suggestion` không thể cùng áp dụng, hoặc đề xuất sửa ngược nhau cho cùng một chỗ):
|
|
552
|
+
- Nếu hai đề xuất có thể **merge** thành một bản sửa mạch lạc → merge thành một finding duy nhất.
|
|
553
|
+
- Nếu chúng **loại trừ lẫn nhau** → phát ra **một** finding nêu cả hai phương án
|
|
554
|
+
và set `auto_fixable: false` với `status: "needs_discussion"` (PRD) /
|
|
555
|
+
`status: "pending"` (review) để con người chọn — không bao giờ âm thầm bỏ một bên.
|
|
556
|
+
- Nếu một finding bị **vô hiệu** bởi finding khác (vd một finding cấu trúc nói một section
|
|
557
|
+
bị thiếu, nhưng một finding khác trích dẫn nội dung từ chính section đó) → bỏ cái không hợp lệ.
|
|
558
|
+
3. **Sắp xếp** theo severity (critical → major → minor), rồi theo thứ tự `section` trong file.
|
|
559
|
+
4. **Gán ID ổn định** `F001, F002, …` theo thứ tự đã sắp đó.
|
|
560
|
+
5. Map `dimension` của mỗi finding vào field schema của lệnh
|
|
561
|
+
(`lens` cho `/refine-prd`; `check_id` cho `/review-context`).
|
|
562
|
+
6. Ghi **một** file findings duy nhất theo FINDINGS SCHEMA mà lệnh định nghĩa.
|
|
563
|
+
|
|
564
|
+
Trong report cuối của lệnh, thêm một dòng:
|
|
567
565
|
```
|
|
568
|
-
Convergence: {convergence_rounds} critic
|
|
566
|
+
Convergence: {convergence_rounds} vòng critic — file findings đã đầy đủ; chạy lại sẽ lòi ra 0 vấn đề mới.
|
|
569
567
|
```
|
|
570
568
|
|
|
571
569
|
|
|
572
|
-
**
|
|
573
|
-
- **DIMENSIONS** =
|
|
574
|
-
- **
|
|
575
|
-
-
|
|
570
|
+
**Các check dưới đây map vào quy trình như sau:**
|
|
571
|
+
- **DIMENSIONS** = các nhóm check theo mode phát hiện được — PRD: `P1, P2, P4, P5`; BDD: `B1, B2, B3, B4, B5, B6`. Fan out một sub-agent cho mỗi nhóm check, mỗi cái quét toàn bộ target file chỉ cho nhóm đó.
|
|
572
|
+
- **Check do orchestrator chạy (không fan out):** `P0` (umbrella routing) và `P3` (xung đột cross-PRD) cần config / context của PRD khác — orchestrator tự chạy chúng **trước** fan-out và thêm kết quả vào `ALL_FINDINGS`.
|
|
573
|
+
- Vòng lặp completeness-critic (Phase 2) đảm bảo file findings đầy đủ trong một lần chạy — chạy lại `/review-context` sẽ lòi ra **0 finding mới**. Map mỗi dimension vào field `check_id` của schema dưới đây.
|
|
576
574
|
|
|
577
575
|
---
|
|
578
576
|
|
|
579
577
|
## PRD Review Mode
|
|
580
578
|
|
|
581
|
-
### P0 — Umbrella Routing Check (
|
|
579
|
+
### P0 — Umbrella Routing Check (chỉ chế độ umbrella)
|
|
582
580
|
|
|
583
|
-
*
|
|
581
|
+
*Bỏ qua hoàn toàn check này nếu `setup.mode` không phải `"umbrella"` (tức không có section `services` trong project-context.yaml).*
|
|
584
582
|
|
|
585
|
-
|
|
583
|
+
Khi `setup.mode = umbrella`, PRD phải có metadata routing đúng để context-loader Bước 1.5 có thể đưa output sinh ra tới đúng service submodule. Chạy các check này **trước P1–P5**:
|
|
586
584
|
|
|
587
|
-
**P0.1 — `@trace.domain`
|
|
588
|
-
-
|
|
589
|
-
-
|
|
590
|
-
- `finding`: "`@trace.domain`
|
|
591
|
-
- `suggestion`: "
|
|
592
|
-
- `auto_fixable: false` — PO
|
|
585
|
+
**P0.1 — `@trace.domain` có mặt**
|
|
586
|
+
- Đọc block frontmatter của PRD (các dòng bắt đầu bằng `@trace.` ở đầu file)
|
|
587
|
+
- Nếu `@trace.domain` **vắng mặt** → finding **critical**:
|
|
588
|
+
- `finding`: "`@trace.domain` đang thiếu. Umbrella routing của team dev phụ thuộc field này để đưa BDD và code output tới đúng service submodule."
|
|
589
|
+
- `suggestion`: "Thêm `@trace.domain: {domain}` vào frontmatter của PRD. Dùng một trong các domain key được định nghĩa trong section services của `project-context.yaml` của umbrella."
|
|
590
|
+
- `auto_fixable: false` — PO phải confirm tên domain đúng
|
|
593
591
|
|
|
594
|
-
**P0.2 — `@trace.domain`
|
|
595
|
-
-
|
|
596
|
-
-
|
|
597
|
-
- `finding`: "`@trace.domain: {value}`
|
|
598
|
-
- `suggestion`: "
|
|
592
|
+
**P0.2 — `@trace.domain` khớp một service key**
|
|
593
|
+
- Nếu `@trace.domain` có mặt, kiểm tra giá trị của nó có khớp key nào trong section `services` của project-context.yaml không
|
|
594
|
+
- Nếu **không khớp** → finding **critical**:
|
|
595
|
+
- `finding`: "`@trace.domain: {value}` không khớp key nào trong config `services` của umbrella. Routing sẽ fallback về path mặc định và BDD có thể được sinh sai chỗ."
|
|
596
|
+
- `suggestion`: "Hoặc cập nhật `@trace.domain` cho khớp một service key có sẵn ({list known keys}), hoặc thêm entry mới vào `services` trong project-context.yaml cho domain `{value}`."
|
|
599
597
|
- `auto_fixable: false`
|
|
600
|
-
-
|
|
601
|
-
- `finding`: "
|
|
602
|
-
- `suggestion`: "
|
|
598
|
+
- Nếu section `services` chưa được cấu hình (rỗng/placeholder) → finding **major**:
|
|
599
|
+
- `finding`: "Section `services` của umbrella chưa được cấu hình. Không thể kiểm chứng domain routing."
|
|
600
|
+
- `suggestion`: "Cập nhật section services trong `.agent/project-context.yaml` với mapping domain-to-submodule trước khi sinh BDD."
|
|
603
601
|
- `auto_fixable: false`
|
|
604
602
|
|
|
605
|
-
**P0.3 — `@trace.status
|
|
606
|
-
-
|
|
607
|
-
-
|
|
608
|
-
- `finding`: "PRD status
|
|
609
|
-
- `suggestion`: "PO/SA
|
|
603
|
+
**P0.3 — Check `@trace.status`**
|
|
604
|
+
- Nếu `@trace.status` vắng mặt → **minor**, `auto_fixable: true` (thêm `@trace.status: draft`)
|
|
605
|
+
- Nếu `@trace.status: draft` → **major**, `auto_fixable: false`:
|
|
606
|
+
- `finding`: "PRD status là `draft`. Team dev không nên sinh BDD từ một PRD chưa được duyệt."
|
|
607
|
+
- `suggestion`: "PO/SA nên review và cập nhật status thành `approved` trước khi team dev tiếp tục."
|
|
610
608
|
|
|
611
|
-
> **P0
|
|
609
|
+
> **P0 là một gate check:** Nếu P0.1 hoặc P0.2 cho finding critical, hiển thị cảnh báo trước khi tiếp tục:
|
|
612
610
|
> ```
|
|
613
|
-
> ⚠️ ROUTING WARNING: @trace.domain
|
|
614
|
-
> BDD/code
|
|
615
|
-
>
|
|
611
|
+
> ⚠️ ROUTING WARNING: phát hiện vấn đề @trace.domain.
|
|
612
|
+
> BDD/code sinh từ PRD này có thể rơi vào sai service submodule.
|
|
613
|
+
> Giải quyết các finding P0 trước khi chạy /generate-bdd.
|
|
616
614
|
> ```
|
|
617
|
-
>
|
|
615
|
+
> Rồi tiếp tục với P1–P5 (đừng abort — PO có thể đang review PRD giai đoạn sớm).
|
|
618
616
|
|
|
619
617
|
### P1 — Terminology Check (Business Dictionary)
|
|
620
618
|
|
|
621
|
-
|
|
622
|
-
|
|
619
|
+
Nạp `{paths.business_dictionary}`.
|
|
620
|
+
Quét toàn bộ PRD tìm vấn đề thuật ngữ:
|
|
623
621
|
|
|
624
|
-
1. **Banned terms** —
|
|
625
|
-
→ Severity: **critical**. AI
|
|
622
|
+
1. **Banned terms** — mọi lần xuất hiện của một term trong §Banned Terms:
|
|
623
|
+
→ Severity: **critical**. AI có thể auto-fix khi `--resume`.
|
|
626
624
|
|
|
627
|
-
2. **
|
|
628
|
-
→ Severity: **major**.
|
|
625
|
+
2. **Dùng không nhất quán** — cùng một khái niệm được đặt tên khác nhau giữa các section:
|
|
626
|
+
→ Severity: **major**. Gắn cờ cả hai chỗ. Người quyết định dạng chuẩn trong note Review Board.
|
|
629
627
|
|
|
630
|
-
3. **
|
|
631
|
-
→ Severity: **minor**.
|
|
628
|
+
3. **Business term chưa liệt kê** — các term quan trọng vắng trong dictionary:
|
|
629
|
+
→ Severity: **minor**. Đề xuất thêm vào `business-dictionary.md`.
|
|
632
630
|
|
|
633
631
|
### P2 — Ambiguity Check
|
|
634
632
|
|
|
635
|
-
|
|
633
|
+
Quét mỗi AC và BR tìm:
|
|
636
634
|
|
|
637
|
-
|
|
|
635
|
+
| Tín hiệu | Ví dụ | Severity |
|
|
638
636
|
|--------|---------|----------|
|
|
639
|
-
|
|
|
640
|
-
|
|
|
641
|
-
|
|
|
642
|
-
|
|
|
643
|
-
|
|
|
637
|
+
| Định lượng mơ hồ | "nhanh", "lớn", "hợp lý", "mau" | Critical |
|
|
638
|
+
| Thiếu actor | "hệ thống nên" mà không nêu trigger | Major |
|
|
639
|
+
| Tham chiếu chưa định nghĩa | "{SomeThing}" được dùng nhưng chưa định nghĩa trong PRD này | Major |
|
|
640
|
+
| Thiếu luồng âm | AC chỉ mô tả happy path nhưng BR có điều kiện lỗi | Minor |
|
|
641
|
+
| Câu bị động giấu actor | "Invoice is created" — ai tạo? | Minor |
|
|
644
642
|
|
|
645
|
-
→ AI
|
|
643
|
+
→ AI không thể auto-fix finding P2. Người phải viết bản fix trong note "Modify" của Review Board.
|
|
646
644
|
|
|
647
645
|
### P3 — Domain Conflict Check
|
|
648
646
|
|
|
649
|
-
|
|
650
|
-
|
|
651
|
-
|
|
647
|
+
Liệt kê tất cả PRD khác trong `{paths.specs_dir}/{domain}/*/*.md` (file `.md` ở gốc mỗi feature folder = PRD).
|
|
648
|
+
Với mỗi PRD, kiểm tra xem PRD này có mâu thuẫn với một BR đã định nghĩa không (cùng trigger, khác outcome)
|
|
649
|
+
hoặc định nghĩa lại field/status transition của một entity khác đi.
|
|
652
650
|
|
|
653
|
-
→ Severity: **critical**.
|
|
651
|
+
→ Severity: **critical**. Người quyết định PRD nào đúng. Bắt buộc có note.
|
|
654
652
|
|
|
655
653
|
### P4 — Structural Completeness
|
|
656
654
|
|
|
657
|
-
- [ ] Metadata
|
|
658
|
-
- [ ]
|
|
659
|
-
- [ ]
|
|
660
|
-
- [ ] `## Changelog`
|
|
661
|
-
- [ ]
|
|
655
|
+
- [ ] Block Metadata: Version, Author, Date, Status
|
|
656
|
+
- [ ] Ít nhất 1 UC với heading `#### {TICKET-ID}-UC{N}:`
|
|
657
|
+
- [ ] Mỗi UC có: Description, Actors, Preconditions, Acceptance Criteria, Business Rules
|
|
658
|
+
- [ ] Có section `## Changelog`
|
|
659
|
+
- [ ] Không còn giá trị `{{PLACEHOLDER}}` chưa điền
|
|
662
660
|
|
|
663
|
-
→
|
|
661
|
+
→ Section thiếu: **major**. AI có thể thêm skeleton khi `--resume` nếu được chấp nhận.
|
|
664
662
|
|
|
665
|
-
### P5 — Custom Criteria (
|
|
663
|
+
### P5 — Custom Criteria (tuỳ chọn)
|
|
666
664
|
|
|
667
|
-
|
|
668
|
-
|
|
665
|
+
Nếu `$ARGUMENTS` chứa tiêu chí bổ sung sau path file, đánh giá chúng và tạo
|
|
666
|
+
finding với `check_id: "P5"` và severity phù hợp.
|
|
669
667
|
|
|
670
668
|
---
|
|
671
669
|
|
|
@@ -673,8 +671,8 @@ findings with `check_id: "P5"` and severity as appropriate.
|
|
|
673
671
|
|
|
674
672
|
### B1 — PRD Coverage Check
|
|
675
673
|
|
|
676
|
-
|
|
677
|
-
Map
|
|
674
|
+
Nạp PRD được tham chiếu bởi `# @trace.prd:` trong header file feature.
|
|
675
|
+
Map mọi AC và mọi BR (gồm cả sub-bullet) sang scenario:
|
|
678
676
|
|
|
679
677
|
```
|
|
680
678
|
AC1 ({short text}) → SC1, SC2 ✅
|
|
@@ -683,64 +681,64 @@ BR1 ({short text}) → SC1 ✅
|
|
|
683
681
|
BR2 ({short text}) → MISSING ❌
|
|
684
682
|
```
|
|
685
683
|
|
|
686
|
-
→
|
|
687
|
-
|
|
684
|
+
→ Mỗi AC/BR thiếu coverage: finding **critical**.
|
|
685
|
+
Nếu được chấp nhận trong Review Board, `--resume` sinh scenario còn thiếu.
|
|
688
686
|
|
|
689
687
|
### B2 — Terminology & Entity Check
|
|
690
688
|
|
|
691
|
-
|
|
689
|
+
Dùng `{paths.business_dictionary}` và `{paths.core_entities}`:
|
|
692
690
|
|
|
693
|
-
1. **Banned terms
|
|
694
|
-
2. **
|
|
695
|
-
3. **
|
|
696
|
-
4. **
|
|
691
|
+
1. **Banned terms trong steps** → **critical**, auto-fixable khi `--resume`
|
|
692
|
+
2. **Tên entity không chuẩn** → **major**, auto-fixable
|
|
693
|
+
3. **Tên field không chuẩn trong data table** → **major**, auto-fixable
|
|
694
|
+
4. **Sample data trông kỹ thuật** (UUID, `item_123`) → **minor**, auto-fixable
|
|
697
695
|
|
|
698
696
|
### B3 — Gherkin Rules Check (R1–R10)
|
|
699
697
|
|
|
700
698
|
| Rule | Check | Auto-fixable? |
|
|
701
699
|
|------|-------|---------------|
|
|
702
|
-
| R1 |
|
|
703
|
-
| R2 |
|
|
704
|
-
| R3 |
|
|
705
|
-
| R4 |
|
|
706
|
-
| R5 |
|
|
707
|
-
| R6 | `Then`
|
|
708
|
-
| R7 |
|
|
709
|
-
| R8 |
|
|
710
|
-
| R9 | Data
|
|
711
|
-
| R10 | Cross-UC reference
|
|
712
|
-
|
|
713
|
-
→ R3, R7, R9, R10
|
|
700
|
+
| R1 | Mỗi scenario có Given + When + Then | No — cần thiết kế lại scenario |
|
|
701
|
+
| R2 | Không chained `When … Then … When` | No — cần thiết kế lại |
|
|
702
|
+
| R3 | Không UI selector / API path / tech term trong steps | Yes — thay bằng cách diễn đạt nghiệp vụ |
|
|
703
|
+
| R4 | Tên scenario là một business outcome | No — cần người đổi tên |
|
|
704
|
+
| R5 | Khai báo WHAT, không phải mệnh lệnh HOW | No — cần viết lại |
|
|
705
|
+
| R6 | `Then` khẳng định business outcome quan sát được | No — cần thiết kế lại |
|
|
706
|
+
| R7 | Giá trị cụ thể, không phải "valid data" | Yes — thay bằng giá trị thực tế |
|
|
707
|
+
| R8 | Mỗi scenario chạy độc lập được | No — cần thiết kế lại |
|
|
708
|
+
| R9 | Data table đủ cột cho Then | Yes — thêm cột còn thiếu |
|
|
709
|
+
| R10 | Cross-UC reference dùng cách diễn đạt navigation + Note | Yes — thêm comment Note |
|
|
710
|
+
|
|
711
|
+
→ Vi phạm R3, R7, R9, R10: auto-fixable. Còn lại cần người hướng dẫn qua note Review Board.
|
|
714
712
|
|
|
715
713
|
### B4 — Compliance Checks (C.1–C.5)
|
|
716
714
|
|
|
717
|
-
- [ ] C.1 Wireframe Coverage:
|
|
718
|
-
- [ ] C.2 PRD Traceability:
|
|
719
|
-
- [ ] C.3 Business Dictionary
|
|
720
|
-
- [ ] C.4 Banned Terms: 0 banned
|
|
721
|
-
- [ ] C.5 NHÓM Grouping:
|
|
715
|
+
- [ ] C.1 Wireframe Coverage: mỗi component/action của màn hình có ≥1 SC → một finding cho mỗi mục thiếu
|
|
716
|
+
- [ ] C.2 PRD Traceability: đã phủ đầy đủ bởi B1 — KHÔNG tạo finding mới ở đây; dedup với finding B1
|
|
717
|
+
- [ ] C.3 Term Business Dictionary được dùng → giống B2
|
|
718
|
+
- [ ] C.4 Banned Terms: 0 banned term → **critical**, auto-fixable
|
|
719
|
+
- [ ] C.5 NHÓM Grouping: nếu ≥3 SC, gom nhóm theo business theme → **major**, auto-fixable
|
|
722
720
|
|
|
723
721
|
### B5 — Metadata & Structural Check
|
|
724
722
|
|
|
725
|
-
- [ ]
|
|
726
|
-
-
|
|
727
|
-
- [ ]
|
|
728
|
-
- [ ] Coverage Matrix
|
|
729
|
-
- [ ] Pre-merge Checklist
|
|
723
|
+
- [ ] Header file có đủ các field `@trace.*` bắt buộc — kiểm tra từng cái tường minh: `@trace.id`, `@trace.title`, `@trace.revision`, `@trace.domain`, `@trace.service`, `@trace.module`, `@trace.status`, `@trace.author`, `@trace.created_at`, `@trace.prd`, `@trace.prd_version`, `@trace.bdd_version`, `@trace.business_rules`, `@trace.dataset` → **minor** cho mỗi field thiếu, auto-fixable
|
|
724
|
+
- Lưu ý: `@trace.revision` luôn là `1` (field tĩnh — xem generate-bdd.tmpl). Chỉ kiểm tra có mặt; KHÔNG gắn cờ giá trị là stale.
|
|
725
|
+
- [ ] Mỗi scenario có `# @trace.scenario`, `# @trace.sc_version`, `# @trace.business_rules`, `# Side-effects:` → **minor**, auto-fixable
|
|
726
|
+
- [ ] Coverage Matrix ở cuối file → **major**, auto-fixable (AI sinh lại)
|
|
727
|
+
- [ ] Pre-merge Checklist ở cuối file → **minor**, auto-fixable
|
|
730
728
|
|
|
731
729
|
### B6 — Side-effect Completeness
|
|
732
730
|
|
|
733
|
-
|
|
734
|
-
- `# Side-effects:`
|
|
735
|
-
- `Then`
|
|
731
|
+
Với mỗi scenario `@happy`:
|
|
732
|
+
- Comment `# Side-effects:` liệt kê tất cả side effect quan sát được
|
|
733
|
+
- Block `Then` có `And <side-effect>` cho mỗi side effect đã liệt kê
|
|
736
734
|
|
|
737
|
-
→
|
|
735
|
+
→ Thiếu assertion side-effect: **major**, auto-fixable.
|
|
738
736
|
|
|
739
737
|
---
|
|
740
738
|
|
|
741
|
-
##
|
|
739
|
+
## Ghi File Findings
|
|
742
740
|
|
|
743
|
-
|
|
741
|
+
Sau khi chạy hết các check, ghi `{paths.refinement_dir}/{slug}-review-*-findings.yaml`:
|
|
744
742
|
|
|
745
743
|
```yaml
|
|
746
744
|
source_file: "{absolute path to reviewed file}"
|
|
@@ -750,14 +748,14 @@ status: "pending_review"
|
|
|
750
748
|
|
|
751
749
|
findings:
|
|
752
750
|
- id: "F001"
|
|
753
|
-
check_id: "P1" # P1-P5
|
|
751
|
+
check_id: "P1" # P1-P5 cho PRD; B1-B6 cho BDD
|
|
754
752
|
severity: "critical" # critical | major | minor
|
|
755
|
-
section: "{section
|
|
756
|
-
uc_id: "{UC-ID
|
|
757
|
-
quote: "{
|
|
758
|
-
finding: "{
|
|
759
|
-
suggestion: "{
|
|
760
|
-
auto_fixable: true # true = AI
|
|
753
|
+
section: "{section hoặc scenario ID nơi tìm thấy lỗi}"
|
|
754
|
+
uc_id: "{UC-ID mà finding này thuộc về — PRD: UC heading; BDD: @trace.id; \"\" nếu global}"
|
|
755
|
+
quote: "{trích đoạn nguyên văn copy CHÍNH XÁC từ file đang review tại vị trí lỗi, ≤120 ký tự}"
|
|
756
|
+
finding: "{mô tả rõ ràng vấn đề}"
|
|
757
|
+
suggestion: "{bản fix cụ thể, hành động được — AI sẽ áp dụng khi --resume nếu được chấp nhận}"
|
|
758
|
+
auto_fixable: true # true = AI áp dụng được; false = người phải viết note trong Review Board
|
|
761
759
|
status: "pending" # pending | accepted | modified | rejected | deferred
|
|
762
760
|
|
|
763
761
|
summary:
|
|
@@ -768,54 +766,54 @@ summary:
|
|
|
768
766
|
recommendation: "APPROVED | NEEDS_REVISION | BLOCKED"
|
|
769
767
|
```
|
|
770
768
|
|
|
771
|
-
> **
|
|
772
|
-
>
|
|
773
|
-
>
|
|
774
|
-
>
|
|
775
|
-
>
|
|
769
|
+
> **Field định vị (`quote` + `uc_id`) — bắt buộc cho source-jump của Review Board.**
|
|
770
|
+
> Với mỗi finding, copy một đoạn `quote` **nguyên văn** thẳng từ file đang review tại đúng
|
|
771
|
+
> chỗ lỗi xảy ra — KHÔNG diễn giải lại; nó được so khớp với tài liệu để định vị dòng.
|
|
772
|
+
> Đặt `uc_id` là Use Case sở hữu (`@trace.id` cho BDD, UC heading cho PRD; `""` nếu global).
|
|
773
|
+
> Hai field này cho phép reviewer click một finding trong Review Board và nhảy tới đúng vị trí nguồn.
|
|
776
774
|
|
|
777
|
-
## Post-Analysis Routing
|
|
775
|
+
## Định tuyến sau phân tích (Post-Analysis Routing)
|
|
778
776
|
|
|
779
|
-
|
|
777
|
+
Sau khi chạy hết các check và ghi file findings:
|
|
780
778
|
|
|
781
|
-
**
|
|
779
|
+
**Nếu có flag `--fix`** → nhảy tới Fix Mode (áp dụng ngay các finding `auto_fixable: true`).
|
|
782
780
|
|
|
783
|
-
**
|
|
781
|
+
**Nếu không có flag** → in Report bên dưới và dừng.
|
|
784
782
|
|
|
785
783
|
## Report
|
|
786
784
|
|
|
787
|
-
# Report Footer —
|
|
785
|
+
# Report Footer — Định dạng output chuẩn cho mọi lệnh
|
|
788
786
|
|
|
789
|
-
|
|
787
|
+
Mọi report của lệnh phải kết thúc bằng section footer chuẩn này.
|
|
790
788
|
|
|
791
789
|
## Status Badge
|
|
792
790
|
|
|
793
|
-
|
|
794
|
-
- `✅ Complete` —
|
|
795
|
-
- `❌ Failed` —
|
|
796
|
-
- `⚠️ Warnings` —
|
|
791
|
+
Chọn một theo kết quả:
|
|
792
|
+
- `✅ Complete` — mọi bước thành công, không có vấn đề
|
|
793
|
+
- `❌ Failed` — lệnh không hoàn thành được do lỗi chặn
|
|
794
|
+
- `⚠️ Warnings` — hoàn thành nhưng có vấn đề không chặn, nên review lại
|
|
797
795
|
|
|
798
796
|
## Output Artifacts
|
|
799
797
|
|
|
800
|
-
|
|
798
|
+
Liệt kê mọi file được tạo hoặc sửa bởi lệnh này:
|
|
801
799
|
```
|
|
802
800
|
Output Artifacts:
|
|
803
|
-
{created|updated} {file-path} ({
|
|
804
|
-
{created|updated} {file-path} ({
|
|
801
|
+
{created|updated} {file-path} ({mô tả ngắn})
|
|
802
|
+
{created|updated} {file-path} ({mô tả ngắn})
|
|
805
803
|
```
|
|
806
804
|
|
|
807
|
-
|
|
805
|
+
Nếu không ghi file nào (vd: lệnh review hoặc phân tích) → ghi `Output Artifacts: none (read-only)`.
|
|
808
806
|
|
|
809
807
|
## Pipeline Position
|
|
810
808
|
|
|
811
|
-
|
|
812
|
-
|
|
809
|
+
In một sơ đồ pipeline một dòng, đánh dấu phase của lệnh HIỆN TẠI bằng `◀ bạn ở đây`,
|
|
810
|
+
để người dùng luôn thấy lệnh này nằm ở đâu trong luồng end-to-end:
|
|
813
811
|
|
|
814
812
|
```
|
|
815
813
|
Discovery → PRD → [Design Spec] → BDD → Tech Design → Code → Dev Self-Check → QC → Trace Audit
|
|
816
814
|
```
|
|
817
815
|
|
|
818
|
-
|
|
816
|
+
Tìm lệnh hiện tại trong bảng phase dưới đây và đánh dấu **phase của nó** trong sơ đồ trên:
|
|
819
817
|
|
|
820
818
|
| Phase | Commands |
|
|
821
819
|
|-------|----------|
|
|
@@ -829,219 +827,219 @@ Find the current command in this phase legend and mark **its** phase in the map
|
|
|
829
827
|
| QC | `/qc-analyze` · `/qc-plan` · `/qc-design-test` · `/qc-review` · `/qc-run-test` · `/qc-report` |
|
|
830
828
|
| Trace Audit | `/validate-traces` |
|
|
831
829
|
|
|
832
|
-
|
|
830
|
+
Với **lệnh review**, thêm vòng review 3 bước và đánh dấu bước hiện tại, vd:
|
|
833
831
|
`Vòng review: [① phân tích ◀] → ② Review Board → ③ --resume`.
|
|
834
832
|
|
|
835
|
-
**
|
|
836
|
-
`/report-bug`, `/propose-scenario`, `/generate-spec-manifest`)
|
|
837
|
-
**
|
|
833
|
+
**Lệnh xuyên suốt** (`/sync`, `/update-framework`, `/fix-bug`, `/debug`, `/learn`,
|
|
834
|
+
`/report-bug`, `/propose-scenario`, `/generate-spec-manifest`) nằm ngoài pipeline tuyến tính —
|
|
835
|
+
**bỏ hẳn dòng Pipeline** cho các lệnh này (đừng cố nhét chúng vào sơ đồ).
|
|
838
836
|
|
|
839
|
-
##
|
|
837
|
+
## Gợi ý lệnh tiếp theo
|
|
840
838
|
|
|
841
|
-
|
|
839
|
+
Gợi ý lệnh kế tiếp hợp lý theo phase của workflow:
|
|
842
840
|
|
|
843
|
-
|
|
|
841
|
+
| Lệnh hiện tại | Gợi ý lệnh tiếp theo |
|
|
844
842
|
|-------------------------|-----------------------------------------------|
|
|
845
|
-
| /setup-ai-first | `/define-product`
|
|
843
|
+
| /setup-ai-first | `/define-product` để bắt đầu feature đầu tiên |
|
|
846
844
|
| /define-product | `/generate-prd {product-definition-file}` |
|
|
847
|
-
| /generate-prd | `/refine-prd {prd-file}`
|
|
848
|
-
| /refine-prd |
|
|
849
|
-
| /review-context (PRD) | FE/App: `/generate-design-spec {prd-file}` (
|
|
850
|
-
| /generate-design-spec | Designer review →
|
|
851
|
-
| /generate-bdd | `/review-context {feature-file}`
|
|
852
|
-
| /review-context (BDD) | `/generate-tech-docs {UC-ID}`
|
|
853
|
-
| /qc-analyze | `/qc-plan {UC-ID}` (
|
|
845
|
+
| /generate-prd | `/refine-prd {prd-file}` rồi `/review-context {prd-file}` |
|
|
846
|
+
| /refine-prd | Mở Review Board → cập nhật PRD → `/review-context {prd-file}` |
|
|
847
|
+
| /review-context (PRD) | FE/App: `/generate-design-spec {prd-file}` (rồi BDD sau khi sign-off); BE: `/generate-bdd {prd-file}` trực tiếp; sửa PRD nếu NEEDS_FIX |
|
|
848
|
+
| /generate-design-spec | Designer review → xác nhận link Figma → PO + Designer sign-off → `/generate-bdd {prd-file}` |
|
|
849
|
+
| /generate-bdd | `/review-context {feature-file}` để kiểm tra độ phủ |
|
|
850
|
+
| /review-context (BDD) | `/generate-tech-docs {UC-ID}` nếu APPROVED; sinh lại nếu NEEDS_FIX |
|
|
851
|
+
| /qc-analyze | `/qc-plan {UC-ID}` (xử lý các gap blocker 🔴 trước) |
|
|
854
852
|
| /qc-plan | `/qc-design-test {UC-ID}` |
|
|
855
|
-
| /qc-design-test | `/qc-review {UC-ID}` (test-case
|
|
856
|
-
| /qc-review (test-case) | `/qc-run-test {UC-ID}`
|
|
857
|
-
| /qc-run-test | `/qc-report {UC-ID}`
|
|
858
|
-
| /qc-review (script) | `/qc-report {UC-ID}`
|
|
859
|
-
| /qc-report | `/validate-traces {UC-ID}`
|
|
853
|
+
| /qc-design-test | `/qc-review {UC-ID}` (review test-case) |
|
|
854
|
+
| /qc-review (test-case) | `/qc-run-test {UC-ID}` nếu APPROVED; sửa TC nếu NEEDS_FIX |
|
|
855
|
+
| /qc-run-test | `/qc-report {UC-ID}` rồi `/qc-review {UC-ID}` (review script) |
|
|
856
|
+
| /qc-review (script) | `/qc-report {UC-ID}` rồi tạo PR nếu APPROVED |
|
|
857
|
+
| /qc-report | `/validate-traces {UC-ID}` để làm mới Living Docs (qc_status) |
|
|
860
858
|
| /generate-tech-docs | `/review-tech-docs {tech-design-file}` |
|
|
861
|
-
| /review-tech-docs | `/generate-code {feature-file}`
|
|
862
|
-
| /generate-code |
|
|
859
|
+
| /review-tech-docs | `/generate-code {feature-file}` nếu APPROVED; sửa doc nếu NEEDS_FIX |
|
|
860
|
+
| /generate-code | Lần gen đầu → `/review-code {UC-ID}`; gen lại → `/dev-gen-test {UC-ID}` |
|
|
863
861
|
| /dev-gen-test | `/dev-run-test {UC-ID}` |
|
|
864
862
|
| /dev-run-test (passing) | `/review-code {UC-ID}` |
|
|
865
|
-
| /dev-run-test (failing) | `/fix-bug {ticket-id}`
|
|
866
|
-
| /review-code | `/dev-smoke-test {UC-ID}`
|
|
867
|
-
| /dev-smoke-test |
|
|
868
|
-
| /validate-traces | DRIFT/UNTRACKED → `/generate-code {UC-ID}`; GAP → `/dev-gen-test {UC-ID}`;
|
|
869
|
-
| /fix-bug |
|
|
870
|
-
| /debug | `/fix-bug {ticket-id}`
|
|
871
|
-
| /report-bug |
|
|
872
|
-
| /propose-scenario |
|
|
873
|
-
| /learn |
|
|
874
|
-
| /sync | `/validate-traces`
|
|
875
|
-
| /update-framework | Review `git diff .agent/`, commit; `/sync`
|
|
876
|
-
|
|
877
|
-
|
|
863
|
+
| /dev-run-test (failing) | `/fix-bug {ticket-id}` hoặc `/debug {error}` |
|
|
864
|
+
| /review-code | `/dev-smoke-test {UC-ID}` hoặc tạo PR |
|
|
865
|
+
| /dev-smoke-test | Tạo PR và link tới ticket |
|
|
866
|
+
| /validate-traces | DRIFT/UNTRACKED → `/generate-code {UC-ID}`; GAP → `/dev-gen-test {UC-ID}`; tất cả OK → tạo PR |
|
|
867
|
+
| /fix-bug | Tạo PR và link tới ticket |
|
|
868
|
+
| /debug | `/fix-bug {ticket-id}` nếu cần sửa |
|
|
869
|
+
| /report-bug | Gửi cho dev (`/fix-bug {BUG-ID}`); nếu thiếu coverage → `/propose-scenario {UC-ID}` |
|
|
870
|
+
| /propose-scenario | Báo PO/Dev review proposal trong `feedback/bdd-proposals/` |
|
|
871
|
+
| /learn | Tiếp tục làm việc — lesson áp dụng ở lệnh kế tiếp |
|
|
872
|
+
| /sync | `/validate-traces` để xem độ phủ đầy đủ; xử lý mọi `📥 tester feedback` được nêu |
|
|
873
|
+
| /update-framework | Review `git diff .agent/`, commit; `/sync` để đồng bộ nội dung dự án |
|
|
874
|
+
|
|
875
|
+
Định dạng footer như sau:
|
|
878
876
|
```
|
|
879
877
|
---
|
|
880
878
|
Status : {badge}
|
|
881
|
-
{Output Artifacts
|
|
879
|
+
{khối Output Artifacts}
|
|
882
880
|
Pipeline : Discovery → PRD → [BDD ◀ bạn ở đây] → Tech Design → Code → Dev Self-Check → QC → Trace Audit
|
|
883
|
-
(review
|
|
884
|
-
Next : {
|
|
881
|
+
(lệnh review) Vòng review: [① phân tích ◀] → ② Review Board → ③ --resume
|
|
882
|
+
Next : {lệnh gợi ý kèm ví dụ tham số}
|
|
885
883
|
```
|
|
886
|
-
*(
|
|
884
|
+
*(Bỏ dòng `Pipeline` cho các lệnh xuyên suốt liệt kê ở trên.)*
|
|
887
885
|
|
|
888
886
|
|
|
889
887
|
```
|
|
890
|
-
/review-context
|
|
888
|
+
/review-context Hoàn tất — {target file}
|
|
891
889
|
Mode: {PRD | BDD}
|
|
892
890
|
Findings: {total} | 🔴 Critical: {N} | 🟡 Major: {N} | 🟢 Minor: {N}
|
|
893
891
|
Auto-fixable: {N} | Needs human decision: {N}
|
|
894
892
|
|
|
895
|
-
|
|
893
|
+
File findings:
|
|
896
894
|
{If PRD}: {paths.refinement_dir}/{prd-slug}-review-context-findings.yaml
|
|
897
895
|
{If BDD}: {paths.refinement_dir}/{uc-id}-review-bdd-findings.yaml
|
|
898
896
|
|
|
899
|
-
|
|
897
|
+
Lựa chọn tiếp theo:
|
|
900
898
|
A) Quick fix : /review-context --fix {target-file}
|
|
901
|
-
→
|
|
902
|
-
B) Review Board:
|
|
899
|
+
→ áp dụng ngay mọi finding auto-fixable
|
|
900
|
+
B) Review Board: mở file findings → accept/modify/reject
|
|
903
901
|
→ /review-context --resume {target-file}
|
|
904
902
|
```
|
|
905
903
|
|
|
906
904
|
---
|
|
907
905
|
|
|
908
|
-
## Fix Mode —
|
|
906
|
+
## Fix Mode — Áp dụng ngay các Finding Auto-Fixable
|
|
909
907
|
|
|
910
|
-
*
|
|
911
|
-
*
|
|
908
|
+
*Kích hoạt khi `$ARGUMENTS` chứa `--fix`.*
|
|
909
|
+
*Ví dụ: `/review-context --fix specs/payment/process-payment/bdd/PAY-001.feature`*
|
|
912
910
|
|
|
913
|
-
|
|
914
|
-
`auto_fixable: true`
|
|
911
|
+
Mode này chạy toàn bộ phân tích (giống mặc định), rồi áp dụng ngay mọi finding
|
|
912
|
+
`auto_fixable: true` mà không qua Review Board.
|
|
915
913
|
|
|
916
|
-
|
|
917
|
-
|
|
918
|
-
|
|
914
|
+
Dùng cho: dọn BDD, fix thuật ngữ, gap metadata — bất cứ thứ gì AI có thể sửa an toàn
|
|
915
|
+
mà không cần phán đoán của con người. Finding cần quyết định của con người vẫn được ghi
|
|
916
|
+
vào file findings như thường và để `status: pending`.
|
|
919
917
|
|
|
920
|
-
### Phase 1 —
|
|
918
|
+
### Phase 1 — Chạy phân tích
|
|
921
919
|
|
|
922
|
-
|
|
923
|
-
|
|
920
|
+
Chạy hết các check qua **Quy trình Review** (fan-out + completeness loop) đúng như mode mặc định.
|
|
921
|
+
Ghi file findings với tất cả `status: "pending"` như thường.
|
|
924
922
|
|
|
925
|
-
### Phase 2 —
|
|
923
|
+
### Phase 2 — Áp dụng các finding auto-fixable
|
|
926
924
|
|
|
927
|
-
|
|
925
|
+
Với mỗi finding có `auto_fixable: true`, theo thứ tự (critical → major → minor):
|
|
928
926
|
|
|
929
|
-
**
|
|
927
|
+
**Với file PRD:**
|
|
930
928
|
|
|
931
|
-
| check_id |
|
|
929
|
+
| check_id | Áp dụng gì |
|
|
932
930
|
|----------|--------------|
|
|
933
|
-
| P0.3 (
|
|
934
|
-
| P1 (Banned term) |
|
|
935
|
-
| P4 (Structure) |
|
|
931
|
+
| P0.3 (Thiếu status) | Thêm `@trace.status: draft` vào frontmatter |
|
|
932
|
+
| P1 (Banned term) | Thay mọi lần xuất hiện banned term bằng canonical term |
|
|
933
|
+
| P4 (Structure) | Thêm skeleton section/metadata còn thiếu |
|
|
936
934
|
|
|
937
|
-
**
|
|
935
|
+
**Với file BDD:**
|
|
938
936
|
|
|
939
|
-
| check_id |
|
|
937
|
+
| check_id | Áp dụng gì |
|
|
940
938
|
|----------|--------------|
|
|
941
|
-
| B2 (Terminology) |
|
|
942
|
-
| B3 R3 |
|
|
943
|
-
| B3 R7 |
|
|
944
|
-
| B3 R9 |
|
|
945
|
-
| B3 R10 |
|
|
946
|
-
| B4 C4 | Fix banned
|
|
947
|
-
| B4 C5 |
|
|
948
|
-
| B5 |
|
|
949
|
-
| B6 |
|
|
950
|
-
|
|
951
|
-
|
|
939
|
+
| B2 (Terminology) | Thay banned term, fix tên entity/field, fix sample data kỹ thuật |
|
|
940
|
+
| B3 R3 | Thay tech term/UI selector bằng cách diễn đạt nghiệp vụ |
|
|
941
|
+
| B3 R7 | Thay bằng giá trị thực tế cụ thể |
|
|
942
|
+
| B3 R9 | Thêm cột data table còn thiếu |
|
|
943
|
+
| B3 R10 | Thêm comment Note navigation cross-UC |
|
|
944
|
+
| B4 C4 | Fix banned term trong tag |
|
|
945
|
+
| B4 C5 | Thêm NHÓM grouping nếu ≥3 SC |
|
|
946
|
+
| B5 | Thêm @trace header còn thiếu, sinh lại Coverage Matrix / Pre-merge Checklist |
|
|
947
|
+
| B6 | Thêm `And <side-effect>` còn thiếu vào block Then |
|
|
948
|
+
|
|
949
|
+
Sau khi áp dụng mỗi finding, đánh dấu nó `status: "applied_automatically"` trong file findings.
|
|
952
950
|
|
|
953
951
|
### Phase 3 — Version bump
|
|
954
952
|
|
|
955
|
-
- **PRD**:
|
|
953
|
+
- **PRD**: nếu ≥1 finding được áp dụng → bump version **minor** (auto-fix chỉ áp dụng thay banned-term P1 và thêm skeleton P4 — không bao giờ thay đổi cấu trúc UC hay nội dung BR, nên minor bump luôn đúng), thêm entry Changelog:
|
|
956
954
|
`Auto-fix: applied {N} auto-fixable review-context findings`
|
|
957
|
-
- **BDD**:
|
|
955
|
+
- **BDD**: nếu ≥1 finding được áp dụng → tăng `@trace.bdd_version` lên 0.1
|
|
958
956
|
|
|
959
957
|
### Phase 4 — Report
|
|
960
958
|
|
|
961
959
|
```
|
|
962
|
-
/review-context --fix
|
|
960
|
+
/review-context --fix Đã áp dụng — {target file}
|
|
963
961
|
Mode: {PRD | BDD}
|
|
964
962
|
|
|
965
963
|
Auto-fixed : {N} findings ({critical} critical, {major} major, {minor} minor)
|
|
966
|
-
- {change 1
|
|
967
|
-
- {change 2
|
|
964
|
+
- {tóm tắt change 1}
|
|
965
|
+
- {tóm tắt change 2}
|
|
968
966
|
|
|
969
|
-
|
|
970
|
-
- F00X [{severity}] {finding
|
|
967
|
+
Còn pending (cần quyết định của con người): {N}
|
|
968
|
+
- F00X [{severity}] {tóm tắt finding} ← mở file findings trong Review Board
|
|
971
969
|
|
|
972
970
|
{If PRD}: Version bumped: {old} → {new}
|
|
973
971
|
{If BDD}: bdd_version: {old} → {new}
|
|
974
972
|
|
|
975
|
-
|
|
973
|
+
File findings:
|
|
976
974
|
{If PRD}: {paths.refinement_dir}/{prd-slug}-review-context-findings.yaml
|
|
977
975
|
{If BDD}: {paths.refinement_dir}/{uc-id}-review-bdd-findings.yaml
|
|
978
|
-
|
|
976
|
+
Chạy lại /review-context {file} để xác nhận 0 finding critical còn lại.
|
|
979
977
|
```
|
|
980
978
|
|
|
981
|
-
|
|
979
|
+
Nếu 0 finding nào auto-fixable → in:
|
|
982
980
|
```
|
|
983
|
-
|
|
984
|
-
|
|
981
|
+
Không có gì để auto-fix. Cả {N} finding đều cần quyết định của con người.
|
|
982
|
+
Mở file findings trong Review Board → rồi chạy: /review-context --resume {file}
|
|
985
983
|
```
|
|
986
984
|
|
|
987
985
|
---
|
|
988
986
|
|
|
989
|
-
## Resume Mode —
|
|
987
|
+
## Resume Mode — Áp dụng các Finding được chấp nhận
|
|
990
988
|
|
|
991
|
-
*
|
|
992
|
-
*
|
|
989
|
+
*Kích hoạt khi `$ARGUMENTS` chứa `--resume`.*
|
|
990
|
+
*Ví dụ: `/review-context --resume specs/payment/process-payment/PAY01-process-payment.md`*
|
|
993
991
|
|
|
994
|
-
### Phase 1 —
|
|
992
|
+
### Phase 1 — Đọc các finding được chấp nhận
|
|
995
993
|
|
|
996
|
-
1.
|
|
994
|
+
1. Suy ra tên file findings từ target file dùng cùng quy tắc như Detect Review Mode:
|
|
997
995
|
- PRD: `{paths.refinement_dir}/{prd-slug}-review-context-findings.yaml`
|
|
998
996
|
- BDD: `{paths.refinement_dir}/{uc-id}-review-bdd-findings.yaml`
|
|
999
|
-
2.
|
|
1000
|
-
3.
|
|
1001
|
-
4.
|
|
997
|
+
2. Đọc file findings.
|
|
998
|
+
3. Gom các finding có `status: "accepted"` hoặc `status: "modified"`.
|
|
999
|
+
4. Nếu không có → báo "No accepted findings. File unchanged." và dừng.
|
|
1002
1000
|
|
|
1003
|
-
### Phase 2 —
|
|
1001
|
+
### Phase 2 — Áp dụng fix
|
|
1004
1002
|
|
|
1005
|
-
|
|
1003
|
+
Áp dụng theo thứ tự: critical → major → minor.
|
|
1006
1004
|
|
|
1007
|
-
**
|
|
1008
|
-
| check_id |
|
|
1005
|
+
**Với finding PRD:**
|
|
1006
|
+
| check_id | Làm gì |
|
|
1009
1007
|
|----------|-----------|
|
|
1010
|
-
| P0.3 (
|
|
1011
|
-
| P1 (Banned term) |
|
|
1012
|
-
| P2 (Ambiguity) |
|
|
1013
|
-
| P3 (Conflict) |
|
|
1014
|
-
| P4 (Structure) |
|
|
1015
|
-
| P5 (Custom) |
|
|
1008
|
+
| P0.3 (Thiếu status) | Thêm `@trace.status: draft` vào frontmatter |
|
|
1009
|
+
| P1 (Banned term) | Thay mọi lần xuất hiện banned term bằng canonical term |
|
|
1010
|
+
| P2 (Ambiguity) | Áp dụng fix nêu trong `suggestion` hoặc note `modified` |
|
|
1011
|
+
| P3 (Conflict) | Áp dụng cách giải quyết nêu trong note modified |
|
|
1012
|
+
| P4 (Structure) | Thêm section/metadata field còn thiếu |
|
|
1013
|
+
| P5 (Custom) | Áp dụng như nêu trong suggestion/note |
|
|
1016
1014
|
|
|
1017
|
-
→
|
|
1015
|
+
→ Sau khi áp dụng, bump version PRD (minor) và thêm entry Changelog.
|
|
1018
1016
|
|
|
1019
|
-
**
|
|
1020
|
-
| check_id |
|
|
1017
|
+
**Với finding BDD:**
|
|
1018
|
+
| check_id | Làm gì |
|
|
1021
1019
|
|----------|-----------|
|
|
1022
|
-
| B1 (Coverage gap) |
|
|
1023
|
-
| B2 (Terminology) |
|
|
1024
|
-
| B3 (Gherkin rule) |
|
|
1025
|
-
| B4 (Compliance) |
|
|
1026
|
-
| B5 (Metadata) |
|
|
1027
|
-
| B6 (Side effects) |
|
|
1020
|
+
| B1 (Coverage gap) | Sinh scenario mới cho AC/BR chưa phủ và chèn vào đúng NHÓM |
|
|
1021
|
+
| B2 (Terminology) | Thay banned term, fix tên entity/field |
|
|
1022
|
+
| B3 (Gherkin rule) | Áp dụng fix theo từng rule (thay tech term, thêm giá trị cụ thể, v.v.) |
|
|
1023
|
+
| B4 (Compliance) | Thêm NHÓM grouping, fix tag @trace |
|
|
1024
|
+
| B5 (Metadata) | Thêm @trace header còn thiếu, sinh lại Coverage Matrix / Pre-merge Checklist |
|
|
1025
|
+
| B6 (Side effects) | Thêm `And <side-effect>` còn thiếu vào block Then |
|
|
1028
1026
|
|
|
1029
|
-
→
|
|
1030
|
-
→
|
|
1027
|
+
→ Sau khi áp dụng, tăng `@trace.bdd_version` trong header file lên 0.1.
|
|
1028
|
+
→ Đồng thời cập nhật `{paths.trace_dir}/{domain}/{prd-slug}/{UC-ID}.tsv`: đặt cột `bdd_version` thành giá trị `@trace.bdd_version` mới cho mọi row, và đặt `last_updated` thành ngày hôm nay.
|
|
1031
1029
|
|
|
1032
1030
|
### Phase 3 — Report
|
|
1033
1031
|
|
|
1034
1032
|
```
|
|
1035
|
-
/review-context --resume
|
|
1033
|
+
/review-context --resume Đã áp dụng — {target file}
|
|
1036
1034
|
Applied : {N} findings ({critical} critical, {major} major, {minor} minor)
|
|
1037
1035
|
Skipped : {N} rejected/deferred
|
|
1038
1036
|
|
|
1039
1037
|
Changes:
|
|
1040
|
-
- {change 1
|
|
1041
|
-
- {change 2
|
|
1038
|
+
- {tóm tắt change 1}
|
|
1039
|
+
- {tóm tắt change 2}
|
|
1042
1040
|
|
|
1043
1041
|
{If PRD}: Version bumped: {old} → {new}
|
|
1044
1042
|
{If BDD}: bdd_version: {old} → {new}
|
|
1045
1043
|
|
|
1046
|
-
|
|
1044
|
+
Chạy lại /review-context {file} để xác nhận 0 finding critical còn lại.
|
|
1047
1045
|
```
|