@edupia-tutor/spec-driven-docs 0.14.4 → 0.14.5

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.
@@ -416,6 +416,47 @@ Sau khi hoàn thành tất cả các bước, bạn đã nạp:
416
416
  Tiếp tục sang bước kế tiếp của lệnh đang gọi.
417
417
 
418
418
 
419
+ ---
420
+
421
+ ## Ngôn ngữ nghiệp vụ *(áp khi viết/áp fix — gồm cả Resume Mode Phase 2)*
422
+ # Business Language Guard — chặn thuật ngữ kỹ thuật rò vào tài liệu nghiệp vụ
423
+
424
+ Tài liệu nghiệp vụ (PRD, product-definition) mô tả **WHAT** — chỉ ngôn ngữ nghiệp vụ. Guard này chạy **mỗi khi viết hoặc sửa** prose (gen mới, áp fix `--resume`, hiệu chỉnh): **quét và xử lý** các thuật ngữ kỹ thuật/UI phổ thông bên dưới **trước khi ghi**.
425
+
426
+ > Guard này là **baseline framework**, chạy **song song** với Banned Terms của `business-dictionary.md` (cơ chế dictionary giữ nguyên; project vẫn bổ sung term đặc thù vào đó). Khi cả hai cùng áp, ưu tiên bản chuẩn của dictionary nếu có.
427
+
428
+ ## Bản đồ xử lý (3 nhóm)
429
+
430
+ **Nhóm 1 — Tương tác/triển khai → DIỄN ĐẠT LẠI sang nghiệp vụ (giữ nguyên nghĩa):**
431
+
432
+ | Kỹ thuật/UI | Cách nói nghiệp vụ |
433
+ |---|---|
434
+ | re-render / render lại / reload / refresh (màn) | "hiển thị lại {tên màn}" |
435
+ | timeout | "quá thời gian chờ" |
436
+ | lỗi mạng / network error | "lỗi kết nối" |
437
+ | UI / giao diện (khi chỉ một màn) | "màn" / "màn hình" |
438
+ | click / tap | "bấm" / "chọn" |
439
+ | popup / modal (nếu chỉ là khái niệm hiển thị) | "hộp thoại" / "thông báo" |
440
+ | disable / enable (nút) | "khoá" / "mở" thao tác |
441
+ | redirect / navigate | "chuyển tới {màn}" |
442
+
443
+ **Nhóm 2 — Visual thuần → CHUYỂN Design Spec (bỏ khỏi PRD, ghi nhận lại):**
444
+ `spinner`, `loading indicator`, `animation`, `fade/slide`, màu sắc, font, layout pixel, micro-interaction → *"Chi tiết visual này thuộc Design Spec — ghi nhận để tạo Design Spec sau."*
445
+
446
+ **Nhóm 3 — Backend/contract thuần → BỎ khỏi PRD (thuộc Tech Docs):**
447
+ `API`, `endpoint`, `token/JWT`, `HTTP status`, tên class/bảng/cột DB, query, payload, header.
448
+ *(Ngoại lệ DUY NHẤT: Appendix "Existing API Contract" khi `API Source: existing` — xem Platform Strategy.)*
449
+
450
+ ## Quy tắc áp dụng
451
+ - Quét toàn bộ text sắp ghi (User Story, AC, BR, Business Logic, Scope, Edge Cases, Assumptions…).
452
+ - Nhóm 1 → thay tại chỗ, giữ nguyên nghĩa nghiệp vụ. **Đồng bộ cách diễn đạt** với chỗ đã có sẵn trong cùng tài liệu (vd nếu "quá thời gian chờ" đã dùng ở một BR → dùng nhất quán ở mọi nơi).
453
+ - Nhóm 2 → gỡ khỏi prose nghiệp vụ + nhắc chuyển Design Spec.
454
+ - Nhóm 3 → gỡ khỏi PRD (trừ ngoại lệ brownfield).
455
+ - Nếu term không có trong bản đồ nhưng rõ ràng là tên kỹ thuật/triển khai → vẫn diễn đạt lại theo tinh thần Nhóm 1, đừng để lọt.
456
+
457
+ **Checklist (dùng ở Quality Checklist của lệnh):** 0 thuật ngữ kỹ thuật/UI (re-render, UI, timeout, spinner, API/endpoint/token…) trong prose nghiệp vụ — đã diễn đạt lại (Nhóm 1) / chuyển Design Spec (Nhóm 2) / bỏ về Tech Docs (Nhóm 3).
458
+
459
+
419
460
  ---
420
461
 
421
462
  ## Quy trình Review
@@ -427,10 +468,12 @@ dừng ở mức "đủ" findings, nên mỗi vòng review sau lại lòi ra v
427
468
  fan out song song theo các chiều review, rồi lặp một critic độ-đầy-đủ cho tới khi một
428
469
  vòng không sinh thêm gì mới, *trước khi* ghi file findings.
429
470
 
430
- Lệnh gọi cung cấp hai thứ:
471
+ Lệnh gọi cung cấp hai thứ bắt buộc + hai tuỳ chọn:
431
472
  - **DIMENSIONS** — danh sách các chiều review để fan out
432
473
  (`/refine-prd` → 4 lăng kính; `/review-context` → các P-check hoặc B-check).
433
474
  - **FINDINGS SCHEMA** — dạng YAML mà mỗi finding phải theo (định nghĩa trong lệnh).
475
+ - **GRANULARITY** *(tuỳ chọn, mặc định `auto`)* — `auto`: chọn độ mịn fan-out theo bảng ngưỡng kích thước ở Phase 1 (hành vi cũ). `per-uc`: **LUÔN** fan-out theo từng UC, **bỏ qua ngưỡng** — dùng cho review cần độ đầy đủ cao (`/refine-prd` truyền cái này để lần đầu đã quét sâu). Lệnh không truyền → `auto` → hành vi không đổi.
476
+ - **CHANGED_SCOPE** *(tuỳ chọn)* — danh sách UC/section đã thay đổi (review **delta**). Nếu được truyền, Phase 1 chỉ fan-out trên các phạm vi này + PRD-global; Phase 2 critic vẫn quét **toàn doc** làm lưới an toàn. Không truyền → quét toàn bộ như thường.
434
477
 
435
478
  > **Bỏ qua ở chế độ sub-agent:** Nếu Gate Bước 0 đã set `_agent_mode: true`, toàn bộ
436
479
  > quy trình này bị **bỏ qua** — orchestrator đã chạy sẵn một dimension/UC cho mỗi
@@ -442,8 +485,11 @@ Lệnh gọi cung cấp hai thứ:
442
485
 
443
486
  **Bao nhiêu sub-agent:** *số lượng* agent không phải là đòn bẩy độ đầy đủ — bề rộng được
444
487
  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 đề),
445
- còn *độ sâu* thuộc về vòng lặp critic ở Phase 2. Chọn **độ mịn fan-out**
446
- theo kích thước target, tái dùng ngưỡng của `steps/spawn-agent.md`:
488
+ còn *độ sâu* thuộc về vòng lặp critic ở Phase 2.
489
+
490
+ **Nếu `GRANULARITY = per-uc`:** **bỏ qua bảng ngưỡng dưới đây**, luôn dùng độ mịn **DIMENSION × phạm vi UC** (kể cả PRD nhỏ) — đảm bảo quét sâu, không bỏ sót ngay lần đầu. (Cái giá: nhiều agent hơn cho PRD nhỏ — chấp nhận để lần đầu đầy đủ.)
491
+
492
+ **Nếu `GRANULARITY = auto`** (mặc định): chọn **độ mịn fan-out** theo kích thước target, tái dùng ngưỡng của `steps/spawn-agent.md`:
447
493
 
448
494
  | Kích thước target | Độ mịn | Số agent |
449
495
  |-------------|-------------|-------------|
@@ -466,6 +512,7 @@ UC duy nhất — chính là điều ngăn bỏ sót trên các PRD lớn.
466
512
  agent). Giới hạn mỗi wave ở **`AGENT_CAP = 12`** agent và gom batch các phạm vi UC cho vừa:
467
513
 
468
514
  1. Dựng danh sách phạm vi = `[UC1, UC2, …, UCn, PRD-global]` (độ dài `UCs + 1`).
515
+ - **Nếu `CHANGED_SCOPE` được truyền (review delta):** danh sách phạm vi = `[các UC trong CHANGED_SCOPE] + [PRD-global]` (chỉ các UC đã đổi + global), KHÔNG phải tất cả UC. Số agent tụt theo đó.
469
516
  2. Tính số-phạm-vi-mỗi-bucket: `groups = max(1, floor(AGENT_CAP / dimensions))`.
470
517
  - Nếu `groups ≥ UCs + 1` → không cần batch, chạy một agent cho mỗi `DIMENSION × scope`.
471
518
  - 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
@@ -507,6 +554,8 @@ Gom mảng findings của mọi sub-agent vào một danh sách hợp nhất `AL
507
554
  Đâ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
508
555
  mới**, hoặc tới cap cứng **3 vòng**, cái nào đến trước:
509
556
 
557
+ > **Lưu ý delta:** kể cả khi `CHANGED_SCOPE` giới hạn Phase 1 vào các UC đã đổi, completeness-critic ở Phase 2 **vẫn đọc TOÀN bộ doc** — đây là lưới an toàn bắt các vấn đề mà một fix ở UC đã đổi có thể làm lộ ra ở chỗ khác.
558
+
510
559
  1. Spawn một sub-agent **completeness-critic** bằng Agent tool. Cho nó:
511
560
  - toàn bộ target file (`{target_file}`),
512
561
  - danh sách findings đã ghi nhận dưới dạng **slim JSON** — chỉ 3 fields cốt lõi
@@ -571,6 +620,19 @@ Convergence: {convergence_rounds} vòng critic — file findings đã đầy đ
571
620
 
572
621
  Chạy review qua **Quy trình Review** ở trên (`steps/review-fanout.md`).
573
622
 
623
+ **Tham số truyền vào Quy trình Review:**
624
+ - `GRANULARITY = per-uc` — LUÔN fan-out theo từng UC (bỏ ngưỡng cả-file), để **ngay lần đầu đã lòi phần nhiều issue**, không dồn sang lần sau.
625
+ - `CHANGED_SCOPE` — xác định theo chế độ full/delta ngay dưới đây.
626
+
627
+ **Chọn full vs delta** *(mặc định: lần đầu FULL, lần sau DELTA)*:
628
+ 1. Tách `--full` khỏi `$ARGUMENTS` nếu có.
629
+ 2. Kiểm tra `{paths.refinement_dir}/{prd-slug}-findings.yaml`:
630
+ - **Không tồn tại** (lần đầu review PRD này) → **FULL**: KHÔNG truyền `CHANGED_SCOPE`.
631
+ - **Tồn tại** + có `--full` → **FULL**: bỏ qua findings cũ, không truyền `CHANGED_SCOPE` (ép quét lại toàn bộ).
632
+ - **Tồn tại** + KHÔNG có `--full` → so `prd_version` trong findings cũ với `| **Version** |` của PRD hiện tại:
633
+ - **Bằng nhau** (PRD chưa đổi từ lần review trước) → DỪNG, báo: `"PRD chưa đổi từ v{X} (lần review gần nhất). Không có gì để review lại — dùng --full nếu vẫn muốn quét toàn bộ."`
634
+ - **Khác** (đã có `--resume` bump version) → **DELTA**: `CHANGED_SCOPE` = { `uc_id`/`section` của các finding `status: applied` trong findings cũ } ∪ { UC có trong PRD hiện tại nhưng chưa từng xuất hiện ở findings cũ }. Truyền `CHANGED_SCOPE` này vào Quy trình Review.
635
+
574
636
  **DIMENSIONS** = 4 lăng kính dưới đây — fan out một sub-agent cho mỗi lăng kính, mỗi cái quét
575
637
  toàn bộ PRD qua đúng lăng kính của nó:
576
638
 
@@ -600,6 +662,7 @@ Ghi `{paths.refinement_dir}/{prd-slug}-findings.yaml`:
600
662
 
601
663
  ```yaml
602
664
  prd_source: "{paths.specs_dir}/{domain}/{prd-slug}/{TICKET-ID}-{prd-slug}.md"
665
+ prd_version: "{đọc | **Version** | từ metadata PRD lúc sinh findings — dùng để chọn full/delta lần chạy sau}"
603
666
  generated_at: "{ISO datetime}"
604
667
  status: "pending_review"
605
668
 
@@ -612,6 +675,11 @@ findings:
612
675
  quote: "{trích đoạn nguyên văn copy CHÍNH XÁC từ PRD tại vị trí lỗi, ≤120 ký tự}"
613
676
  finding: "{mô tả gap hoặc vấn đề}"
614
677
  suggestion: "{đề xuất cải thiện cụ thể, hành động được}"
678
+ resolution_edge_cases: # CHỈ điền cho critical/major; minor → để [] (bỏ qua)
679
+ # Phân tích bậc-hai (advisory, KHÔNG chặn): nếu áp `suggestion` này thì có thể đẻ ra
680
+ # edge case / side-effect gì — path lỗi mới, va chạm với BR/UC khác, trạng thái biên,
681
+ # hệ luỵ cross-section. PO đọc để cân nhắc trước khi accept; nếu muốn xử lý → tạo finding mới.
682
+ - "{edge case có thể phát sinh nếu chốt phương án này}"
615
683
  auto_fixable: false
616
684
  # true = AI tự tin cao vào suggestion này; Review Board có thể hiển thị nút "quick accept"
617
685
  # false = cần human đọc kỹ và ghi quyết định trước khi accept
@@ -641,6 +709,14 @@ summary:
641
709
  > `uc_id` là Use Case sở hữu (hoặc `""` cho finding PRD-global). Hai field này cho phép reviewer click một
642
710
  > finding trong Review Board và nhảy thẳng tới đúng vị trí trong PRD nguồn.
643
711
 
712
+ > **`resolution_edge_cases` — phân tích bậc-hai (chỉ critical/major).**
713
+ > Với mỗi finding `critical`/`major`, sau khi viết `suggestion`, nghĩ tiếp: *nếu PO chốt phương án này thì
714
+ > đẻ ra edge case / side-effect gì?* (path lỗi mới, va chạm BR/UC khác, trạng thái biên, hệ luỵ cross-section).
715
+ > Ghi vào `resolution_edge_cases` để PO **thấy trước khi accept**. Đây là **advisory** — KHÔNG chặn, KHÔNG tự
716
+ > tạo finding; PO đọc rồi quyết. Finding `minor` → để `[]`.
717
+ > *(Phần phản hồi cho phương án PO **tự sửa** (`modified`) đến ở vòng sau: sau `--resume`, lần `/refine-prd`
718
+ > kế chạy delta sẽ quét lại UC đã đổi + critic toàn-doc → tự lòi edge case mà phương án đó tạo ra.)*
719
+
644
720
  ## Report
645
721
 
646
722
  # Report Footer — Định dạng output chuẩn cho mọi lệnh
@@ -802,6 +878,7 @@ Nếu có finding nào có `status: "needs_discussion"`, thêm warning block sau
802
878
  Với mỗi finding được chấp nhận, theo thứ tự severity (critical → major → minor):
803
879
  - Đi tới section PRD được chỉ định bởi `finding.section`.
804
880
  - Áp dụng `finding.suggestion`. Với finding `status: "modified"`, người đã sửa sẵn `finding.suggestion` trong Review Board — giá trị đã sửa đó CHÍNH LÀ bản fix cần áp dụng.
881
+ - **Chạy Business Language Guard trên text mới TRƯỚC khi ghi** (xem section "Ngôn ngữ nghiệp vụ"): diễn đạt lại / gỡ thuật ngữ kỹ thuật-UI để fix không tự kéo theo term kỹ thuật vào PRD.
805
882
  - Không thay đổi bất kỳ section nào không được tham chiếu bởi một finding được chấp nhận.
806
883
 
807
884
  ### Phase 2.5 — Cập nhật file findings
@@ -839,6 +916,8 @@ Changes :
839
916
  - {change 1}
840
917
  - {change 2}
841
918
 
919
+ 💡 Vừa áp {N} phương án (gồm bản PO tự sửa) → chạy lại /refine-prd {prd-file}
920
+ (delta tự động) để lòi edge case mà các phương án vừa áp có thể tạo ra.
842
921
  ⚠️ BDD có thể đã lỗi thời. Chạy:
843
922
  /review-context {prd-file} ← kiểm tra chất lượng PRD trước
844
923
  → /generate-bdd {prd-file}
@@ -23,6 +23,11 @@ Nếu `$ARGUMENTS` không chứa `--resume` → tiếp tục luồng review bìn
23
23
 
24
24
  ---
25
25
 
26
+ ## Ngôn ngữ nghiệp vụ *(áp khi viết/áp fix — gồm cả Resume Mode Phase 2)*
27
+ {{include:steps/business-language.md}}
28
+
29
+ ---
30
+
26
31
  ## Quy trình Review
27
32
  {{include:steps/review-fanout.md}}
28
33
 
@@ -32,6 +37,19 @@ Nếu `$ARGUMENTS` không chứa `--resume` → tiếp tục luồng review bìn
32
37
 
33
38
  Chạy review qua **Quy trình Review** ở trên (`steps/review-fanout.md`).
34
39
 
40
+ **Tham số truyền vào Quy trình Review:**
41
+ - `GRANULARITY = per-uc` — LUÔN fan-out theo từng UC (bỏ ngưỡng cả-file), để **ngay lần đầu đã lòi phần nhiều issue**, không dồn sang lần sau.
42
+ - `CHANGED_SCOPE` — xác định theo chế độ full/delta ngay dưới đây.
43
+
44
+ **Chọn full vs delta** *(mặc định: lần đầu FULL, lần sau DELTA)*:
45
+ 1. Tách `--full` khỏi `$ARGUMENTS` nếu có.
46
+ 2. Kiểm tra `{paths.refinement_dir}/{prd-slug}-findings.yaml`:
47
+ - **Không tồn tại** (lần đầu review PRD này) → **FULL**: KHÔNG truyền `CHANGED_SCOPE`.
48
+ - **Tồn tại** + có `--full` → **FULL**: bỏ qua findings cũ, không truyền `CHANGED_SCOPE` (ép quét lại toàn bộ).
49
+ - **Tồn tại** + KHÔNG có `--full` → so `prd_version` trong findings cũ với `| **Version** |` của PRD hiện tại:
50
+ - **Bằng nhau** (PRD chưa đổi từ lần review trước) → DỪNG, báo: `"PRD chưa đổi từ v{X} (lần review gần nhất). Không có gì để review lại — dùng --full nếu vẫn muốn quét toàn bộ."`
51
+ - **Khác** (đã có `--resume` bump version) → **DELTA**: `CHANGED_SCOPE` = { `uc_id`/`section` của các finding `status: applied` trong findings cũ } ∪ { UC có trong PRD hiện tại nhưng chưa từng xuất hiện ở findings cũ }. Truyền `CHANGED_SCOPE` này vào Quy trình Review.
52
+
35
53
  **DIMENSIONS** = 4 lăng kính dưới đây — fan out một sub-agent cho mỗi lăng kính, mỗi cái quét
36
54
  toàn bộ PRD qua đúng lăng kính của nó:
37
55
 
@@ -61,6 +79,7 @@ Ghi `{paths.refinement_dir}/{prd-slug}-findings.yaml`:
61
79
 
62
80
  ```yaml
63
81
  prd_source: "{paths.specs_dir}/{domain}/{prd-slug}/{TICKET-ID}-{prd-slug}.md"
82
+ prd_version: "{đọc | **Version** | từ metadata PRD lúc sinh findings — dùng để chọn full/delta lần chạy sau}"
64
83
  generated_at: "{ISO datetime}"
65
84
  status: "pending_review"
66
85
 
@@ -73,6 +92,11 @@ findings:
73
92
  quote: "{trích đoạn nguyên văn copy CHÍNH XÁC từ PRD tại vị trí lỗi, ≤120 ký tự}"
74
93
  finding: "{mô tả gap hoặc vấn đề}"
75
94
  suggestion: "{đề xuất cải thiện cụ thể, hành động được}"
95
+ resolution_edge_cases: # CHỈ điền cho critical/major; minor → để [] (bỏ qua)
96
+ # Phân tích bậc-hai (advisory, KHÔNG chặn): nếu áp `suggestion` này thì có thể đẻ ra
97
+ # edge case / side-effect gì — path lỗi mới, va chạm với BR/UC khác, trạng thái biên,
98
+ # hệ luỵ cross-section. PO đọc để cân nhắc trước khi accept; nếu muốn xử lý → tạo finding mới.
99
+ - "{edge case có thể phát sinh nếu chốt phương án này}"
76
100
  auto_fixable: false
77
101
  # true = AI tự tin cao vào suggestion này; Review Board có thể hiển thị nút "quick accept"
78
102
  # false = cần human đọc kỹ và ghi quyết định trước khi accept
@@ -102,6 +126,14 @@ summary:
102
126
  > `uc_id` là Use Case sở hữu (hoặc `""` cho finding PRD-global). Hai field này cho phép reviewer click một
103
127
  > finding trong Review Board và nhảy thẳng tới đúng vị trí trong PRD nguồn.
104
128
 
129
+ > **`resolution_edge_cases` — phân tích bậc-hai (chỉ critical/major).**
130
+ > Với mỗi finding `critical`/`major`, sau khi viết `suggestion`, nghĩ tiếp: *nếu PO chốt phương án này thì
131
+ > đẻ ra edge case / side-effect gì?* (path lỗi mới, va chạm BR/UC khác, trạng thái biên, hệ luỵ cross-section).
132
+ > Ghi vào `resolution_edge_cases` để PO **thấy trước khi accept**. Đây là **advisory** — KHÔNG chặn, KHÔNG tự
133
+ > tạo finding; PO đọc rồi quyết. Finding `minor` → để `[]`.
134
+ > *(Phần phản hồi cho phương án PO **tự sửa** (`modified`) đến ở vòng sau: sau `--resume`, lần `/refine-prd`
135
+ > kế chạy delta sẽ quét lại UC đã đổi + critic toàn-doc → tự lòi edge case mà phương án đó tạo ra.)*
136
+
105
137
  ## Report
106
138
 
107
139
  {{include:steps/report-footer.md}}
@@ -163,6 +195,7 @@ Nếu có finding nào có `status: "needs_discussion"`, thêm warning block sau
163
195
  Với mỗi finding được chấp nhận, theo thứ tự severity (critical → major → minor):
164
196
  - Đi tới section PRD được chỉ định bởi `finding.section`.
165
197
  - Áp dụng `finding.suggestion`. Với finding `status: "modified"`, người đã sửa sẵn `finding.suggestion` trong Review Board — giá trị đã sửa đó CHÍNH LÀ bản fix cần áp dụng.
198
+ - **Chạy Business Language Guard trên text mới TRƯỚC khi ghi** (xem section "Ngôn ngữ nghiệp vụ"): diễn đạt lại / gỡ thuật ngữ kỹ thuật-UI để fix không tự kéo theo term kỹ thuật vào PRD.
166
199
  - Không thay đổi bất kỳ section nào không được tham chiếu bởi một finding được chấp nhận.
167
200
 
168
201
  ### Phase 2.5 — Cập nhật file findings
@@ -200,6 +233,8 @@ Changes :
200
233
  - {change 1}
201
234
  - {change 2}
202
235
 
236
+ 💡 Vừa áp {N} phương án (gồm bản PO tự sửa) → chạy lại /refine-prd {prd-file}
237
+ (delta tự động) để lòi edge case mà các phương án vừa áp có thể tạo ra.
203
238
  ⚠️ BDD có thể đã lỗi thời. Chạy:
204
239
  /review-context {prd-file} ← kiểm tra chất lượng PRD trước
205
240
  → /generate-bdd {prd-file}
@@ -438,10 +438,12 @@ dừng ở mức "đủ" findings, nên mỗi vòng review sau lại lòi ra v
438
438
  fan out song song theo các chiều review, rồi lặp một critic độ-đầy-đủ cho tới khi một
439
439
  vòng không sinh thêm gì mới, *trước khi* ghi file findings.
440
440
 
441
- Lệnh gọi cung cấp hai thứ:
441
+ Lệnh gọi cung cấp hai thứ bắt buộc + hai tuỳ chọn:
442
442
  - **DIMENSIONS** — danh sách các chiều review để fan out
443
443
  (`/refine-prd` → 4 lăng kính; `/review-context` → các P-check hoặc B-check).
444
444
  - **FINDINGS SCHEMA** — dạng YAML mà mỗi finding phải theo (định nghĩa trong lệnh).
445
+ - **GRANULARITY** *(tuỳ chọn, mặc định `auto`)* — `auto`: chọn độ mịn fan-out theo bảng ngưỡng kích thước ở Phase 1 (hành vi cũ). `per-uc`: **LUÔN** fan-out theo từng UC, **bỏ qua ngưỡng** — dùng cho review cần độ đầy đủ cao (`/refine-prd` truyền cái này để lần đầu đã quét sâu). Lệnh không truyền → `auto` → hành vi không đổi.
446
+ - **CHANGED_SCOPE** *(tuỳ chọn)* — danh sách UC/section đã thay đổi (review **delta**). Nếu được truyền, Phase 1 chỉ fan-out trên các phạm vi này + PRD-global; Phase 2 critic vẫn quét **toàn doc** làm lưới an toàn. Không truyền → quét toàn bộ như thường.
445
447
 
446
448
  > **Bỏ qua ở chế độ sub-agent:** Nếu Gate Bước 0 đã set `_agent_mode: true`, toàn bộ
447
449
  > quy trình này bị **bỏ qua** — orchestrator đã chạy sẵn một dimension/UC cho mỗi
@@ -453,8 +455,11 @@ Lệnh gọi cung cấp hai thứ:
453
455
 
454
456
  **Bao nhiêu sub-agent:** *số lượng* agent không phải là đòn bẩy độ đầy đủ — bề rộng được
455
457
  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 đề),
456
- còn *độ sâu* thuộc về vòng lặp critic ở Phase 2. Chọn **độ mịn fan-out**
457
- theo kích thước target, tái dùng ngưỡng của `steps/spawn-agent.md`:
458
+ còn *độ sâu* thuộc về vòng lặp critic ở Phase 2.
459
+
460
+ **Nếu `GRANULARITY = per-uc`:** **bỏ qua bảng ngưỡng dưới đây**, luôn dùng độ mịn **DIMENSION × phạm vi UC** (kể cả PRD nhỏ) — đảm bảo quét sâu, không bỏ sót ngay lần đầu. (Cái giá: nhiều agent hơn cho PRD nhỏ — chấp nhận để lần đầu đầy đủ.)
461
+
462
+ **Nếu `GRANULARITY = auto`** (mặc định): chọn **độ mịn fan-out** theo kích thước target, tái dùng ngưỡng của `steps/spawn-agent.md`:
458
463
 
459
464
  | Kích thước target | Độ mịn | Số agent |
460
465
  |-------------|-------------|-------------|
@@ -477,6 +482,7 @@ UC duy nhất — chính là điều ngăn bỏ sót trên các PRD lớn.
477
482
  agent). Giới hạn mỗi wave ở **`AGENT_CAP = 12`** agent và gom batch các phạm vi UC cho vừa:
478
483
 
479
484
  1. Dựng danh sách phạm vi = `[UC1, UC2, …, UCn, PRD-global]` (độ dài `UCs + 1`).
485
+ - **Nếu `CHANGED_SCOPE` được truyền (review delta):** danh sách phạm vi = `[các UC trong CHANGED_SCOPE] + [PRD-global]` (chỉ các UC đã đổi + global), KHÔNG phải tất cả UC. Số agent tụt theo đó.
480
486
  2. Tính số-phạm-vi-mỗi-bucket: `groups = max(1, floor(AGENT_CAP / dimensions))`.
481
487
  - Nếu `groups ≥ UCs + 1` → không cần batch, chạy một agent cho mỗi `DIMENSION × scope`.
482
488
  - 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
@@ -518,6 +524,8 @@ Gom mảng findings của mọi sub-agent vào một danh sách hợp nhất `AL
518
524
  Đâ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
519
525
  mới**, hoặc tới cap cứng **3 vòng**, cái nào đến trước:
520
526
 
527
+ > **Lưu ý delta:** kể cả khi `CHANGED_SCOPE` giới hạn Phase 1 vào các UC đã đổi, completeness-critic ở Phase 2 **vẫn đọc TOÀN bộ doc** — đây là lưới an toàn bắt các vấn đề mà một fix ở UC đã đổi có thể làm lộ ra ở chỗ khác.
528
+
521
529
  1. Spawn một sub-agent **completeness-critic** bằng Agent tool. Cho nó:
522
530
  - toàn bộ target file (`{target_file}`),
523
531
  - danh sách findings đã ghi nhận dưới dạng **slim JSON** — chỉ 3 fields cốt lõi
@@ -1 +1 @@
1
- 0.14.4
1
+ 0.14.5
@@ -403,6 +403,47 @@ Sau khi hoàn thành tất cả các bước, bạn đã nạp:
403
403
  Tiếp tục sang bước kế tiếp của lệnh đang gọi.
404
404
 
405
405
 
406
+ ---
407
+
408
+ ## Ngôn ngữ nghiệp vụ *(áp khi viết BR / AC / Business Logic / Scope)*
409
+ # Business Language Guard — chặn thuật ngữ kỹ thuật rò vào tài liệu nghiệp vụ
410
+
411
+ Tài liệu nghiệp vụ (PRD, product-definition) mô tả **WHAT** — chỉ ngôn ngữ nghiệp vụ. Guard này chạy **mỗi khi viết hoặc sửa** prose (gen mới, áp fix `--resume`, hiệu chỉnh): **quét và xử lý** các thuật ngữ kỹ thuật/UI phổ thông bên dưới **trước khi ghi**.
412
+
413
+ > Guard này là **baseline framework**, chạy **song song** với Banned Terms của `business-dictionary.md` (cơ chế dictionary giữ nguyên; project vẫn bổ sung term đặc thù vào đó). Khi cả hai cùng áp, ưu tiên bản chuẩn của dictionary nếu có.
414
+
415
+ ## Bản đồ xử lý (3 nhóm)
416
+
417
+ **Nhóm 1 — Tương tác/triển khai → DIỄN ĐẠT LẠI sang nghiệp vụ (giữ nguyên nghĩa):**
418
+
419
+ | Kỹ thuật/UI | Cách nói nghiệp vụ |
420
+ |---|---|
421
+ | re-render / render lại / reload / refresh (màn) | "hiển thị lại {tên màn}" |
422
+ | timeout | "quá thời gian chờ" |
423
+ | lỗi mạng / network error | "lỗi kết nối" |
424
+ | UI / giao diện (khi chỉ một màn) | "màn" / "màn hình" |
425
+ | click / tap | "bấm" / "chọn" |
426
+ | popup / modal (nếu chỉ là khái niệm hiển thị) | "hộp thoại" / "thông báo" |
427
+ | disable / enable (nút) | "khoá" / "mở" thao tác |
428
+ | redirect / navigate | "chuyển tới {màn}" |
429
+
430
+ **Nhóm 2 — Visual thuần → CHUYỂN Design Spec (bỏ khỏi PRD, ghi nhận lại):**
431
+ `spinner`, `loading indicator`, `animation`, `fade/slide`, màu sắc, font, layout pixel, micro-interaction → *"Chi tiết visual này thuộc Design Spec — ghi nhận để tạo Design Spec sau."*
432
+
433
+ **Nhóm 3 — Backend/contract thuần → BỎ khỏi PRD (thuộc Tech Docs):**
434
+ `API`, `endpoint`, `token/JWT`, `HTTP status`, tên class/bảng/cột DB, query, payload, header.
435
+ *(Ngoại lệ DUY NHẤT: Appendix "Existing API Contract" khi `API Source: existing` — xem Platform Strategy.)*
436
+
437
+ ## Quy tắc áp dụng
438
+ - Quét toàn bộ text sắp ghi (User Story, AC, BR, Business Logic, Scope, Edge Cases, Assumptions…).
439
+ - Nhóm 1 → thay tại chỗ, giữ nguyên nghĩa nghiệp vụ. **Đồng bộ cách diễn đạt** với chỗ đã có sẵn trong cùng tài liệu (vd nếu "quá thời gian chờ" đã dùng ở một BR → dùng nhất quán ở mọi nơi).
440
+ - Nhóm 2 → gỡ khỏi prose nghiệp vụ + nhắc chuyển Design Spec.
441
+ - Nhóm 3 → gỡ khỏi PRD (trừ ngoại lệ brownfield).
442
+ - Nếu term không có trong bản đồ nhưng rõ ràng là tên kỹ thuật/triển khai → vẫn diễn đạt lại theo tinh thần Nhóm 1, đừng để lọt.
443
+
444
+ **Checklist (dùng ở Quality Checklist của lệnh):** 0 thuật ngữ kỹ thuật/UI (re-render, UI, timeout, spinner, API/endpoint/token…) trong prose nghiệp vụ — đã diễn đạt lại (Nhóm 1) / chuyển Design Spec (Nhóm 2) / bỏ về Tech Docs (Nhóm 3).
445
+
446
+
406
447
  ---
407
448
 
408
449
  ## Phase 0 — Knowledge Sync *(AI tự điền — bối cảnh hệ thống, KHÔNG phải yêu cầu nghiệp vụ; không cần input PO)*
@@ -457,8 +498,12 @@ Hỏi:
457
498
  ```
458
499
  | Bước | Hành động | Trạng thái/Kết quả nghiệp vụ | Ghi chú |
459
500
  ```
460
- 3. **Exit Point**: Kết quả cuối khi flow hoàn thành là gì?
461
- 4. **Edge Cases**: Các kịch bản thất bại nghiệp vụ ngoài happy path? (input thiếu, điều kiện không thoả, thao tác đồng thời, phụ thuộc không sẵn sàng → kết quả nghiệp vụ kỳ vọng)
501
+ 3. **Màn hình & thành phần chính** *(mức nghiệp vụ — KHÔNG pixel/layout/màu)*: Feature gồm những **màn hình / bước giao diện** chính nào? Mỗi màn có **thành phần & hành động chính** gì (vd: ô nhập, nút, danh sách → bấm ra kết quả gì)? Đây nguồn cho Wireframe của PRD và độ phủ BDD — nếu PO không chắc, AI gợi ý rồi PO xác nhận.
502
+ ```
503
+ | Màn hình | Thành phần chính | Hành động → kết quả nghiệp vụ |
504
+ ```
505
+ 4. **Exit Point**: Kết quả cuối khi flow hoàn thành là gì?
506
+ 5. **Edge Cases**: Các kịch bản thất bại nghiệp vụ ngoài happy path? (input thiếu, điều kiện không thoả, thao tác đồng thời, phụ thuộc không sẵn sàng → kết quả nghiệp vụ kỳ vọng)
462
507
 
463
508
  Xác nhận → ghi `✅ PO xác nhận: Có` → tiếp tục.
464
509
 
@@ -551,6 +596,10 @@ Ghi `{paths.product_definitions_dir}/{TICKET-ID}-{slug}.md` theo `templates/prod
551
596
  > viết thường, chỉ a-z 0-9 và dấu `-`. Đây là định danh feature-package dùng xuyên suốt pipeline —
552
597
  > PRD, BDD, tech-docs, design-spec, trace của feature này đều kế thừa **nguyên văn** `slug` này
553
598
  > (`/generate-prd` đặt `prd-slug = slug`). Sinh một lần ở đây, các bước sau KHÔNG tái sinh.
599
+ >
600
+ > **Ranh giới tên file:** tên file = `{TICKET-ID}-{slug}.md`. TICKET-ID **được phép chứa `-`**
601
+ > (vd Jira key `LOYAL-29` → `LOYAL-29-loyalty-points.md`). Khi tách lại slug, các bước sau strip
602
+ > **đúng chuỗi TICKET-ID** ở đầu, KHÔNG tách theo dấu `-` đầu tiên.
554
603
 
555
604
  Điền tất cả section bằng dữ liệu thu thập qua các phase.
556
605
 
@@ -567,17 +567,21 @@ Từ vựng step của System BDD (luôn dùng — bất kể từ vựng FE/App
567
567
 
568
568
  *Bỏ qua section này nếu đang chạy spec repo mode.*
569
569
 
570
- Đọc bảng Metadata của PRD tìm row `| **Service** |` `| **Module** |`.
570
+ Routing service **domain-keyed** và **context-loader (Bước 1.5) đã phân giải sẵn** từ `@trace.domain`/Domain của PRD KHÔNG re-resolve đây, chỉ dùng lại các biến đã set:
571
+ - `active_service` = `services.{domain}.path` (path submodule, vd `user-service/`) — hoặc `"unresolved"` nếu domain không khớp entry nào, hoặc bỏ trống ở single-service.
572
+ - `active_module` = module của service (`services.{domain}.module`, đã override `tech_stack.module` ở Bước 1.5) — dùng cho từ vựng bên dưới.
571
573
 
572
- | Điều kiện | Hành động |
574
+ Chỉ cần kiểm tra trạng thái đã phân giải:
575
+
576
+ | Trạng thái (từ context-loader) | Hành động |
573
577
  |---|---|
574
- | PRD có row `Service` giá trị không phải "default" | Dùng làm `active_service` + `active_module`. Nạp catalog cho module đó. |
575
- | PRD khôngrow `Service` `services` được định nghĩa trong project-context.yaml | Hỏi: "BDD này dành cho service nào?" (liệt service, chờ chọn) |
576
- | Dự án single-service (không có mảng `services`) | `active_service = "default"`, `active_module = tech_stack.module` |
578
+ | `active_service` đã phân giải thành path service | Tiếp tục với `active_module` đã set. |
579
+ | `active_service = "unresolved"` (section `services` nhưng domain PRD không khớp entry nào) | **DỪNG**, báo: "Domain `{domain}` của PRD không khớp service nào trong `services:` của project-context.yaml bổ sung mapping rồi chạy lại." (Không đoán/hỏi tay — domain là khoá định danh, lệch là lỗi cấu hình cần sửa ở SoT.) |
580
+ | Single-service (không có section `services`) | `active_module = tech_stack.module` (đã set ở Bước 6.5). Tiếp tục. |
581
+
582
+ **Output path (umbrella mode):** `{paths.specs_dir}/{domain}/{prd-slug}/bdd/{TICKET-ID}-UC{N}-{slug}.feature`
577
583
 
578
- **Output path (umbrella mode):**
579
- - `active_service = "default"` → `{paths.specs_dir}/{domain}/{prd-slug}/bdd/{TICKET-ID}-UC{N}-{slug}.feature`
580
- - `active_service ≠ "default"` → `{paths.specs_dir}/{domain}/{prd-slug}/bdd/{active_service}/{TICKET-ID}-UC{N}-{slug}.feature`
584
+ *(Không thêm subfolder theo service: feature-package đã domain-scoped sẵn ở `{domain}/`, mà service route 1-1 theo domain — nên thêm subfolder service sẽ chỉ lặp lại domain. `active_service` chỉ dùng cho `service_root`/từ vựng, KHÔNG vào path spec.)*
581
585
 
582
586
  **Từ vựng theo platform** — điều chỉnh cách viết step BDD theo `active_module`:
583
587
 
@@ -627,7 +631,7 @@ Trước khi sinh, kiểm tra các file `.feature` có sẵn cho PRD này:
627
631
 
628
632
  1. Phân giải search path theo mode:
629
633
  - **Spec repo mode**: `{paths.specs_dir}/{domain}/{prd-slug}/bdd/{active_platform}/{TICKET-ID}-UC*.feature`
630
- - **Umbrella mode**: `{paths.specs_dir}/{domain}/{prd-slug}/bdd/{TICKET-ID}-UC*.feature` (hoặc với `{active_service}/` nếu multi-service)
634
+ - **Umbrella mode**: `{paths.specs_dir}/{domain}/{prd-slug}/bdd/{TICKET-ID}-UC*.feature`
631
635
  2. Đọc `| **Version** |` hiện tại của PRD từ metadata (vd: `1.2`).
632
636
 
633
637
  **Nếu không có file feature nào** → gen mới, tiếp tục bình thường. Dùng version PRD làm `@trace.prd_version`.
@@ -732,8 +736,7 @@ CHECKPOINT: "Outline này đúng chưa? Bạn muốn thêm hay bớt SC nào kh
732
736
 
733
737
  **Output path theo mode:**
734
738
  - **Spec repo mode**: `{paths.specs_dir}/{domain}/{prd-slug}/bdd/{active_platform}/{TICKET-ID}-UC{N}-{slug}.feature`
735
- - **Umbrella mode (single-service)**: `{paths.specs_dir}/{domain}/{prd-slug}/bdd/{TICKET-ID}-UC{N}-{slug}.feature`
736
- - **Umbrella mode (multi-service)**: `{paths.specs_dir}/{domain}/{prd-slug}/bdd/{active_service}/{TICKET-ID}-UC{N}-{slug}.feature`
739
+ - **Umbrella mode**: `{paths.specs_dir}/{domain}/{prd-slug}/bdd/{TICKET-ID}-UC{N}-{slug}.feature` *(service route 1-1 theo domain → không thêm subfolder service)*
737
740
 
738
741
  Với mỗi UC, ghi vào path đã phân giải ở trên. Dùng từ vựng cho active platform (từ Platform Selection hoặc Service Detection).
739
742
 
@@ -405,9 +405,17 @@ Tiếp tục sang bước kế tiếp của lệnh đang gọi.
405
405
 
406
406
  *Context bổ sung cho lệnh này: Đọc toàn bộ file product-definition. Trích xuất: **TICKET-ID**, **domain**, **tên feature**, **tên PO** (field `PO` ở Metadata), dữ liệu Phase 1-7, và bảng **Chuẩn hoá thuật ngữ** (Terminology Map ở Phase 0 — các cặp `thuật ngữ PO → thuật ngữ chuẩn`). Lưu bản đồ này để áp dụng ở bước Quy tắc thuật ngữ. **Nếu Phase 7 (Validation Report) còn `Xung đột phát hiện` / `Mục còn thiếu` khác "None" → diễn đạt lại mỗi mục thành một câu hỏi theo format `Q… — [AI DRAFT]` của section "Giả định AI" (nêu rõ độ vênh + cần PO chốt điều gì), thay vì dán thô. AI không tự quyết thay PO.***
407
407
 
408
+ **Map "Phụ thuộc liên service"** (Phase 1 câu 8 của product-definition) → **§1c "Phụ thuộc liên service"** của PRD, giữ nguyên mức nghiệp vụ (cần dữ liệu/năng lực gì, từ ai, vì sao — KHÔNG thêm chi tiết API/kỹ thuật). Nếu discovery ghi "Không có" → §1c ghi "Không có".
409
+
410
+ **Map "Màn hình & thành phần chính"** (Phase 2 của product-definition) → **§4b Wireframe** của PRD: mỗi màn hình thành một Screen với Components/Actions tương ứng. Đây là nguồn coverage cho `/generate-bdd` (C.1). Nếu phải **suy thêm** Screen/component mà discovery chưa nêu → đánh dấu rõ `*(AI đề xuất — PO review)*` cạnh phần đó, đừng để âm thầm thành chân lý coverage.
411
+
408
412
  **Phân giải PO** (đọc trước, hỏi sau): lấy `PO` từ Metadata của product-definition. Nếu đã có giá trị → dùng luôn, KHÔNG hỏi lại. Nếu trống/thiếu → hỏi PO ngay ở CHECKPOINT dưới. Điền giá trị này vào field `PO` của PRD Metadata.
409
413
 
410
- **Phân giải `slug`** (kế thừa, KHÔNG tái sinh): lấy `slug` từ tên file product-definition đoạn sau `{TICKET-ID}-` (vd `LOYAL29-loyalty-points.md` `slug = loyalty-points`). Đặt `prd-slug = slug` này cho feature-package PRD, để folder `{specs_dir}/{domain}/{prd-slug}/` khớp 1-1 với product-definition. KHÔNG tự bịa slug mới nếu tên file không theo dạng `{TICKET-ID}-{slug}.md`, dừng hỏi người dùng.
414
+ **Phân giải link tracker (hỏi cho MỌI ticket):** `TICKET-ID` **định danh nội bộ** PO tự đặt lúc discovery **KHÔNG đảm bảo** một Jira key thật, nên **TUYỆT ĐỐI KHÔNG tự dựng URL** kiểu `{base}/browse/{TICKET-ID}` (sẽ ra link rác trỏ tới ticket không tồn tại). Field `Ticket` **luôn giữ TICKET-ID dạng plain text**; link Jira (nếu có) chỉ **đặt thêm bên cạnh**. **Luôn hỏi PO**: *"Ticket {TICKET-ID} link tracker thật (Jira/khác) không? Nếu có, dán vào."*
415
+ - PO **có** link → điền field `Ticket` dạng: `{TICKET-ID} ([Jira]({tracker_url}))` — plain ID đứng trước, link trong ngoặc bên cạnh.
416
+ - PO **không** có / để trống → điền plain text `{TICKET-ID}` (KHÔNG link, KHÔNG placeholder `{...}` lủng lẳng).
417
+
418
+ **Phân giải `slug`** (kế thừa, KHÔNG tái sinh): đọc `TICKET-ID` từ Metadata `Ticket` của product-definition, rồi lấy `slug` = phần tên file **sau khi strip đúng chuỗi `{TICKET-ID}-` ở đầu**. **KHÔNG tách theo dấu `-` đầu tiên** — vì TICKET-ID có thể chứa `-` (vd Jira key `LOYAL-29`): `LOYAL-29-loyalty-points.md` với TICKET-ID `LOYAL-29` → `slug = loyalty-points` (KHÔNG phải `29-loyalty-points`). Đặt `prd-slug = slug` này cho feature-package PRD, để folder `{specs_dir}/{domain}/{prd-slug}/` khớp 1-1 với product-definition. KHÔNG tự bịa slug mới — nếu tên file không bắt đầu bằng `{TICKET-ID}-`, dừng và hỏi người dùng.
411
419
 
412
420
  **Guard — discovery phải hoàn tất:** đọc `Status` và `Completed Phase` từ Metadata của product-definition.
413
421
  - Nếu `Status: completed` (Completed Phase = 7) → tiếp tục bình thường.
@@ -419,7 +427,24 @@ Tiếp tục sang bước kế tiếp của lệnh đang gọi.
419
427
  ```
420
428
  PRD là artifact ký duyệt — không sinh từ nguồn discovery chưa chốt.
421
429
 
422
- CHECKPOINT trước khi sinh: "Sinh PRD cho **{TICKET-ID} {feature}** (domain: {domain}), PO: **{tên PO hoặc hỏi nếu product-definition để trống}**. Nguồn: discovery hoàn tất (Phase 7/7). Tiếp tục? (Y/N)"
430
+ **Phân giải `API Source` (brownfield / greenfield / partner song song):** hỏi PO **một** câu để chốt **loại** nguồn API PO CHỈ chốt loại, **không bao giờ chi tiết contract**:
431
+
432
+ > "API của feature này thuộc loại nào?
433
+ > 1. **existing** — API đã chạy production, contract cố định
434
+ > 2. **greenfield** — mình tự thiết kế contract mới
435
+ > 3. **partner** — đối tác đang phát triển song song, contract chưa có"
436
+
437
+ - **(1) existing** → set Metadata `API Source: existing`. Xin PO **con trỏ nguồn** contract (1 trong: file openapi/swagger, URL swagger, path service/repo BE, hoặc doc đính kèm). AI **trích as-is** vào Appendix "Existing API Contract" (ghi rõ nguồn từng row — KHÔNG bịa, KHÔNG thiết kế mới). *Người điền detail là AI từ artifact thật, KHÔNG phải PO.*
438
+ - Nếu con trỏ **không truy cập được** từ context hiện tại → KHÔNG fabricate: ghi block `⛔ PENDING: contract chưa trích — nguồn: {pointer}; phải trích as-is trước /generate-bdd` vào section đó, vẫn giữ `API Source: existing`. Coi phần contract của PRD chưa hoàn tất tới khi bảng đủ.
439
+ - **(2) greenfield** → để trống `API Source`. **Xoá** Appendix "Existing API Contract" (template đã dặn). Contract sẽ được thiết kế ở `/generate-tech-docs`.
440
+ - **(3) partner song song** → ĐI LUỒNG greenfield (`API Source` để trống, **xoá** section Existing API Contract), **KHÔNG** set `existing`. Thêm:
441
+ - Ghi phụ thuộc partner vào **§1c "Phụ thuộc liên service"** của PRD (mức nghiệp vụ): cần contract/năng lực gì từ partner nào, vì sao.
442
+ - Một mục trong "Giả định AI": *"Contract do partner {X} phát triển song song; bản tech-docs sinh sau là ĐỀ XUẤT của ta để đàm phán (qua cổng T7 cross-team sign-off), chốt khi partner confirm — có thể thay đổi."*
443
+ - Lý do KHÔNG dùng `existing`: `existing` làm `/review-tech-docs` **skip T7** — nhưng partner song song chính là lúc CẦN T7 để đàm phán contract.
444
+
445
+ > **`API Source` là field có vòng đời:** có thể bắt đầu greenfield/partner rồi **chuyển thành `existing`** khi contract được chốt (vd partner áp contract của họ và ta phải theo). Khi đó: chạy `/refine-prd` đổi `API Source → existing` + điền Existing API Contract từ nguồn partner (trích as-is) + bump Version; `/generate-bdd` (Version drift) và `/generate-tech-docs` (reverse-document) sẽ tự realign system BDD + tech-doc theo contract mới. FE/App BDD phần lớn không vỡ nhờ luật declarative (assert outcome quan sát được, không assert shape JSON).
446
+
447
+ CHECKPOINT trước khi sinh: "Sinh PRD cho **{TICKET-ID} — {feature}** (domain: {domain}), PO: **{tên PO — hoặc hỏi nếu product-definition để trống}**. API Source: **{existing — nguồn {pointer} | greenfield | partner song song}**. Link tracker: **{URL — hoặc 'không có, dùng plain ID'}**. Nguồn: discovery hoàn tất (Phase 7/7). Tiếp tục? (Y/N)"
423
448
 
424
449
  ---
425
450
 
@@ -435,6 +460,45 @@ CHECKPOINT trước khi sinh: "Sinh PRD cho **{TICKET-ID} — {feature}** (domai
435
460
  Sau khi PO confirm → cập nhật `business-dictionary.md` (nếu đồng ý) → tiếp tục sinh.
436
461
  - Nếu không có banned term và tất cả thuật ngữ đều chuẩn → tiếp tục không gián đoạn.
437
462
 
463
+ ## Ngôn ngữ nghiệp vụ
464
+ # Business Language Guard — chặn thuật ngữ kỹ thuật rò vào tài liệu nghiệp vụ
465
+
466
+ Tài liệu nghiệp vụ (PRD, product-definition) mô tả **WHAT** — chỉ ngôn ngữ nghiệp vụ. Guard này chạy **mỗi khi viết hoặc sửa** prose (gen mới, áp fix `--resume`, hiệu chỉnh): **quét và xử lý** các thuật ngữ kỹ thuật/UI phổ thông bên dưới **trước khi ghi**.
467
+
468
+ > Guard này là **baseline framework**, chạy **song song** với Banned Terms của `business-dictionary.md` (cơ chế dictionary giữ nguyên; project vẫn bổ sung term đặc thù vào đó). Khi cả hai cùng áp, ưu tiên bản chuẩn của dictionary nếu có.
469
+
470
+ ## Bản đồ xử lý (3 nhóm)
471
+
472
+ **Nhóm 1 — Tương tác/triển khai → DIỄN ĐẠT LẠI sang nghiệp vụ (giữ nguyên nghĩa):**
473
+
474
+ | Kỹ thuật/UI | Cách nói nghiệp vụ |
475
+ |---|---|
476
+ | re-render / render lại / reload / refresh (màn) | "hiển thị lại {tên màn}" |
477
+ | timeout | "quá thời gian chờ" |
478
+ | lỗi mạng / network error | "lỗi kết nối" |
479
+ | UI / giao diện (khi chỉ một màn) | "màn" / "màn hình" |
480
+ | click / tap | "bấm" / "chọn" |
481
+ | popup / modal (nếu chỉ là khái niệm hiển thị) | "hộp thoại" / "thông báo" |
482
+ | disable / enable (nút) | "khoá" / "mở" thao tác |
483
+ | redirect / navigate | "chuyển tới {màn}" |
484
+
485
+ **Nhóm 2 — Visual thuần → CHUYỂN Design Spec (bỏ khỏi PRD, ghi nhận lại):**
486
+ `spinner`, `loading indicator`, `animation`, `fade/slide`, màu sắc, font, layout pixel, micro-interaction → *"Chi tiết visual này thuộc Design Spec — ghi nhận để tạo Design Spec sau."*
487
+
488
+ **Nhóm 3 — Backend/contract thuần → BỎ khỏi PRD (thuộc Tech Docs):**
489
+ `API`, `endpoint`, `token/JWT`, `HTTP status`, tên class/bảng/cột DB, query, payload, header.
490
+ *(Ngoại lệ DUY NHẤT: Appendix "Existing API Contract" khi `API Source: existing` — xem Platform Strategy.)*
491
+
492
+ ## Quy tắc áp dụng
493
+ - Quét toàn bộ text sắp ghi (User Story, AC, BR, Business Logic, Scope, Edge Cases, Assumptions…).
494
+ - Nhóm 1 → thay tại chỗ, giữ nguyên nghĩa nghiệp vụ. **Đồng bộ cách diễn đạt** với chỗ đã có sẵn trong cùng tài liệu (vd nếu "quá thời gian chờ" đã dùng ở một BR → dùng nhất quán ở mọi nơi).
495
+ - Nhóm 2 → gỡ khỏi prose nghiệp vụ + nhắc chuyển Design Spec.
496
+ - Nhóm 3 → gỡ khỏi PRD (trừ ngoại lệ brownfield).
497
+ - Nếu term không có trong bản đồ nhưng rõ ràng là tên kỹ thuật/triển khai → vẫn diễn đạt lại theo tinh thần Nhóm 1, đừng để lọt.
498
+
499
+ **Checklist (dùng ở Quality Checklist của lệnh):** 0 thuật ngữ kỹ thuật/UI (re-render, UI, timeout, spinner, API/endpoint/token…) trong prose nghiệp vụ — đã diễn đạt lại (Nhóm 1) / chuyển Design Spec (Nhóm 2) / bỏ về Tech Docs (Nhóm 3).
500
+
501
+
438
502
  ## Quy tắc Cross-Reference
439
503
 
440
504
  Bất kỳ chỗ nào nhắc tới một TICKET-ID khác trong PRD (pre-condition, BR, ghi chú, appendix) → **PHẢI** là inline link. Mỗi PRD sống trong feature-package riêng (`{paths.specs_dir}/{domain}/{prd-slug-khác}/`), nên link trỏ sang folder anh em:
@@ -449,6 +513,12 @@ Không bao giờ để TICKET-ID dạng plain text nếu file PRD tương ứng
449
513
  - **BR ID**: `{TICKET-ID}-UC{N}-BR{M}` — **M tăng liên tục trên toàn PRD** (KHÔNG reset theo từng UC).
450
514
  - Ví dụ: UC1 → BR1, BR2; UC2 → BR3, BR4 (KHÔNG phải BR1, BR2 lại từ đầu).
451
515
 
516
+ ## Traceability AC ↔ BR ↔ UC
517
+
518
+ - **Giữ link AC→BR từ discovery (KHÔNG vứt khi flatten):** Product Definition Phase 6 có cột "Bắt nguồn từ BR-{N}". Khi viết AC §2, **remap** mỗi `BR-{N}` (số discovery) sang **BR ID của PRD** (`{TICKET-ID}-UC{n}-BR{m}` sau khi đã phân BR vào UC) và gắn vào cuối AC: `_(BR: {ids})_`. Mỗi AC PHẢI có ≥1 ref BR.
519
+ - **Điền "AC liên quan" cho mỗi UC §3 (chiều ngược):** với mỗi UC, liệt kê các AC mà nó thoả.
520
+ - **Nhất quán 2 chiều (bắt buộc):** tập "AC liên quan" của UC{n} phải **đúng bằng** tập AC có ref BR trỏ về UC{n}. Vì BR ID đã chứa số UC, hai chiều này suy ra lẫn nhau — lệch là lỗi traceability, sửa trước khi ghi.
521
+
452
522
  ---
453
523
 
454
524
  ## Platform Strategy — PRD là tài liệu nghiệp vụ (không technical)
@@ -456,6 +526,8 @@ Không bao giờ để TICKET-ID dạng plain text nếu file PRD tương ứng
456
526
  PRD mô tả **WHAT** (yêu cầu nghiệp vụ) — không nhét chi tiết **kỹ thuật** (API, token, endpoint, HTTP status, tên class/bảng/cột DB, query). Những thứ đó thuộc **Tech Docs**.
457
527
  PRD **được phép** có User Flow + Wireframe ở mức nghiệp vụ (§4) để BA/QC/Dev cùng hình dung; chỉ **chi tiết visual** (màu sắc, layout pixel, animation, micro-interaction) mới thuộc **Design Spec**.
458
528
 
529
+ > **Ngoại lệ DUY NHẤT (brownfield):** khi `API Source: existing`, Appendix "Existing API Contract" ĐƯỢC PHÉP chứa chi tiết kỹ thuật (method, path, request/response shape, HTTP status, error code) — vì đó là contract **đã tồn tại**, AI chỉ **trích as-is** làm input cho trước, KHÔNG phải thiết kế mới. Ngoại lệ này CHỈ áp cho riêng section đó; AC, BR, và toàn bộ thân PRD vẫn thuần nghiệp vụ.
530
+
459
531
  | ✅ Viết trong PRD (nghiệp vụ) | ❌ KHÔNG viết trong PRD |
460
532
  |---|---|
461
533
  | "Đăng nhập thành công → truy cập được tính năng" | "API trả về JWT token" ← Tech Docs |
@@ -529,7 +601,7 @@ Ghi `{paths.specs_dir}/{domain}/{prd-slug}/{TICKET-ID}-{prd-slug}.md` theo cấu
529
601
  | **Domain** | {domain} |
530
602
  | **Created** | {date} |
531
603
  | **Updated** | {date} |
532
- | **Jira Ticket** | [{TICKET}-{N}]({ticket_url}) |
604
+ | **Ticket** | {TICKET}-{N}{ — nếu PO có link tracker thật, thêm bên cạnh: `{TICKET}-{N} ([Jira]({tracker_url}))`} |
533
605
  | **API Source** | *(để trống nếu greenfield — chỉ điền `existing` khi PRD bọc một API đã chạy production)* |
534
606
 
535
607
  ---
@@ -559,13 +631,21 @@ Ghi `{paths.specs_dir}/{domain}/{prd-slug}/{TICKET-ID}-{prd-slug}.md` theo cấu
559
631
  **Out of Scope** *(chỉ thêm khi có ranh giới cần nói rõ)*
560
632
  - {hạng mục ngoài phạm vi + lý do / chủ sở hữu}
561
633
 
634
+ ## c. Phụ thuộc liên service *(mức nghiệp vụ — KHÔNG mô tả API/event/kỹ thuật)*
635
+
636
+ > Kế thừa từ Product Definition Phase 1 ("Phụ thuộc liên service"). Nếu contract do đối tác phát triển song song (xem `API Source`), ghi phụ thuộc partner vào đây.
637
+
638
+ - {Cần {dữ liệu/năng lực} từ {feature/team/partner} — vì {lý do nghiệp vụ}} — hoặc "Không có"
639
+
562
640
  ---
563
641
 
564
642
  # 2. Acceptance Criteria
565
643
 
566
- **AC1:** {Tiêu chí nghiệm thu, văn xuôi, kiểm chứng được.}
644
+ > Mỗi AC kế thừa liên kết "Bắt nguồn từ BR" của Product Definition (Phase 6), remap sang BR ID của PRD. Vì BR ID đã chứa số UC nên ref BR truy ngược được tới đúng UC.
645
+
646
+ **AC1:** {Tiêu chí nghiệm thu, văn xuôi, kiểm chứng được.} _(BR: {TICKET}-{N}-UC{n}-BR{m})_
567
647
 
568
- **AC2:** {…}
648
+ **AC2:** {…} _(BR: {TICKET}-{N}-UC{n}-BR{m})_
569
649
 
570
650
  ---
571
651
 
@@ -583,6 +663,8 @@ Ghi `{paths.specs_dir}/{domain}/{prd-slug}/{TICKET-ID}-{prd-slug}.md` theo cấu
583
663
  **Post-condition:**
584
664
  - {kết quả sau 1}
585
665
 
666
+ **AC liên quan:** AC{x}, AC{y} *(các AC mà UC này thoả — phải đúng bằng tập AC có ref BR trỏ về UC này ở §2)*
667
+
586
668
  **Business Rule**
587
669
 
588
670
  | ID | Business Rule | Business Logic |
@@ -693,10 +775,14 @@ _(Nếu không có độ vênh: ghi "Không có — toàn bộ nội dung đã
693
775
 
694
776
  - [ ] Mọi AC đều testable (pass/fail rõ ràng), không có chi tiết kỹ thuật (API/token/DB) hay visual chi tiết (màu sắc, font, animation)
695
777
  - [ ] Mỗi UC có Actor / Description / Pre-condition / Post-condition
778
+ - [ ] §1c "Phụ thuộc liên service" có mặt: kế thừa từ discovery Phase 1 câu 8 (hoặc "Không có" nếu discovery trống); case partner song song có ghi phụ thuộc partner ở đây
696
779
  - [ ] Business Rule (WHAT) và Business Logic (HOW) nằm chung bảng 3 cột `ID | Business Rule | Business Logic`
697
780
  - [ ] BR ID tăng liên tục trên toàn PRD — không reset theo từng UC
781
+ - [ ] **Traceability AC↔BR↔UC**: mỗi AC §2 có ≥1 ref `_(BR: …)_`; mỗi UC §3 có dòng "AC liên quan"; và hai chiều khớp nhau (tập "AC liên quan" của UC = tập AC có ref BR trỏ về UC đó)
698
782
  - [ ] Mọi cross-reference TICKET-ID đều là inline link `[TICKET-ID](./file.md)`
699
783
  - [ ] Không có banned term (nếu dictionary tồn tại)
784
+ - [ ] **Ngôn ngữ nghiệp vụ**: 0 thuật ngữ kỹ thuật/UI trong prose (re-render→hiển thị lại, UI→màn, timeout→quá thời gian chờ…) — xem Business Language Guard
785
+ - [ ] **API Source nhất quán**: nếu `existing` → bảng "Existing API Contract" đủ method/path/request/response (hoặc mang block ⛔ PENDING kèm con trỏ nguồn); nếu greenfield/partner → section "Existing API Contract" đã bị xoá hẳn (không để bảng rỗng), và case partner có dependency + assumption ghi rõ
700
786
  - [ ] User Flow có cả luồng lỗi / luồng ngoại lệ
701
787
  - [ ] Wireframe phủ tất cả màn hình liên quan tới Use Case
702
788
  - [ ] **Định dạng (readability)**: User Story / AC / các field của UC (Actor, Description, Pre/Post-condition) viết dạng bullet `- **Label:** …` — mỗi ý MỘT dòng, KHÔNG viết các dòng liền nhau (sẽ bị dồn thành 1 đoạn khi render); có một dòng trống trước và sau mỗi bảng