[CLAUDE] Workflow: State machine 5 trạng thái — Trả lại = Phase riêng
All checks were successful
Deploy SOLUTION_ERP / build-deploy (push) Successful in 3m17s
All checks were successful
Deploy SOLUTION_ERP / build-deploy (push) Successful in 3m17s
Session 17 spec: chốt 5 trạng thái phiếu PE/HĐ/Budget theo state diagram:
Nháp ─trình──► Đã gửi duyệt ─approve cấp cuối──► Đã duyệt (terminal)
├─ Trả lại ────────► Trả lại
└─ Từ chối ────────► Từ chối (terminal)
Trả lại ──Drafter sửa+gửi lại──► Đã gửi duyệt (chạy LẠI từ đầu)
Khác Mig 21 (Session 16): bỏ smart-reject jump-back. Trả lại = Phase
RIÊNG (TraLai=98), không revert về DangSoanThao + không jump-back step.
Drafter từ TraLai gửi lại như case Nháp — workflow chạy lại từ Cấp 1
Bước 1 (Option A diagram chốt với user).
BE Domain:
- ContractPhase + TraLai = 98
- BudgetPhase + TraLai = 98
- PurchaseEvaluationPhase: TraLai=98 đổi từ [LEGACY deprecated] thành
primary state. Comment update enum docs cho cả 3.
BE Policy (PE/HĐ/Budget):
- Reject transitions trỏ về TraLai (thay DangSoanThao)
- Mirror entry transitions: TraLai → next phase (cho Drafter resubmit)
- ActivePhases thêm TraLai
- FromDefinition mirror: TraLai → step.Phase + reject → TraLai
- DefaultSla cho TraLai = same as DangSoanThao
BE Service (PE + Contract):
- Reject branch: target=TuChoi giữ; else set Phase=TraLai, clear
CurrentWorkflowStepIndex=null
- Bỏ ResumeAfterReject branch + RejectedAtStepIndex/RejectedFromPhase
assignment (DB column giữ deprecated cho data cũ)
- Drafter trình branch: từ DangSoanThao HOẶC TraLai → ChoDuyet, init
CurrentWorkflowStepIndex=0 (cùng entry point, chạy lại từ đầu)
- Notification: TraLai when fromPhase=ChoDuyet → "bị trả lại"
- Budget Handler: simplify reject → TraLai, bỏ smart-reject + isResuming
BE Tests update:
- WorkflowPolicyTests: Standard_RejectFromCCM → TraLai (rename + assert)
+ Standard_TraLai_To_DangGopY_Allowed_For_Drafter (new)
- PurchaseEvaluationPolicyTests: BothPolicies_RejectFromCCM → TraLai
+ BothPolicies_TraLai_To_ChoPurchasing_AllowedForDrafter (new theory)
- BudgetPolicyTests: Default_CostControl_ChoCCM_To_TraLai (rename)
+ ActivePhases All6States (was All5) + NextPhasesFrom_TraLai (new)
+ NextPhasesFrom_ChoCEO_Includes_DaDuyet_And_TraLai (rename)
- 77 → 81 test pass (+4 tests TraLai entry point)
FE rename "Bản nháp" → "Nháp" (cả 2 app + types):
- types/purchaseEvaluation.ts: PurchaseEvaluationPhaseLabel 1=Nháp,
10=Đã gửi duyệt. PeDisplayStatus.BanNhap → Nhap (key + value).
PhaseLabel/Color cho TraLai update active.
- types/contracts.ts: +ChoDuyet=10, +TraLai=98 const + label/color.
Phase 2 'Đang soạn thảo' → 'Nháp'.
- types/budget.ts: +TraLai=98 const + label/color. Phase 1 → 'Nháp'.
- PeListPanel + PurchaseEvaluationsListPage filter dropdown: Nhap +
TraLai map đúng phase value.
BE label maps update consistent:
- ContractExcelExporter PhaseLabel: DangSoanThao → "Nháp" + add ChoDuyet/
TraLai entries.
- PeWorkflowAdminFeatures + WorkflowAdminFeatures PhaseLabels: same.
Verify: dotnet test 81 pass · npm build × 2 app pass · BE 0 error.
Field RejectedAtStepIndex/RejectedFromPhase giữ DB column (nullable,
không set value mới). Cleanup migration sau.
This commit is contained in:
@ -1,14 +1,17 @@
|
||||
namespace SolutionErp.Domain.Budgets;
|
||||
|
||||
// State machine ngân sách — đơn giản 3 bước duyệt + 2 terminal.
|
||||
// DangSoanThao → ChoCCM → ChoCEO → DaDuyet
|
||||
// Bất kỳ phase duyệt → DangSoanThao (reject)
|
||||
// DangSoanThao → TuChoi
|
||||
// State machine ngân sách — Session 17 spec mới (5 trạng thái mirror PE/HĐ):
|
||||
// DangSoanThao (Nháp) → ChoCCM (Drafter trình)
|
||||
// TraLai (Trả lại) → ChoCCM (Drafter sửa+gửi lại, chạy từ đầu)
|
||||
// ChoCCM/ChoCEO → next phase OR TraLai OR TuChoi
|
||||
// ChoCEO → DaDuyet (terminal)
|
||||
// DangSoanThao/TraLai → TuChoi (Drafter huỷ)
|
||||
public enum BudgetPhase
|
||||
{
|
||||
DangSoanThao = 1,
|
||||
ChoCCM = 2,
|
||||
ChoCEO = 3,
|
||||
DaDuyet = 4,
|
||||
TraLai = 98, // Phase riêng (không revert DangSoanThao)
|
||||
TuChoi = 99,
|
||||
}
|
||||
|
||||
@ -31,12 +31,15 @@ public static class BudgetPolicies
|
||||
private static readonly Dictionary<BudgetPhase, TimeSpan?> DefaultSla = new()
|
||||
{
|
||||
[BudgetPhase.DangSoanThao] = TimeSpan.FromDays(5),
|
||||
[BudgetPhase.TraLai] = TimeSpan.FromDays(5),
|
||||
[BudgetPhase.ChoCCM] = TimeSpan.FromDays(3),
|
||||
[BudgetPhase.ChoCEO] = TimeSpan.FromDays(2),
|
||||
[BudgetPhase.DaDuyet] = null,
|
||||
[BudgetPhase.TuChoi] = null,
|
||||
};
|
||||
|
||||
// Session 17 spec: Reject = về TraLai (Phase riêng). Drafter từ TraLai
|
||||
// gửi lại = entry point thứ 2 (mirror DangSoanThao → ChoCCM).
|
||||
public static readonly BudgetPolicy Default = new(
|
||||
Name: "Default",
|
||||
Description: "Quy trình ngân sách 3-step (Drafter → CCM → CEO).",
|
||||
@ -44,17 +47,22 @@ public static class BudgetPolicies
|
||||
{
|
||||
[(BudgetPhase.DangSoanThao, BudgetPhase.ChoCCM)] = [AppRoles.Drafter, AppRoles.DeptManager],
|
||||
[(BudgetPhase.DangSoanThao, BudgetPhase.TuChoi)] = [AppRoles.Drafter, AppRoles.DeptManager],
|
||||
[(BudgetPhase.TraLai, BudgetPhase.ChoCCM)] = [AppRoles.Drafter, AppRoles.DeptManager],
|
||||
[(BudgetPhase.TraLai, BudgetPhase.TuChoi)] = [AppRoles.Drafter, AppRoles.DeptManager],
|
||||
|
||||
[(BudgetPhase.ChoCCM, BudgetPhase.ChoCEO)] = [AppRoles.CostControl],
|
||||
[(BudgetPhase.ChoCCM, BudgetPhase.DangSoanThao)] = [AppRoles.CostControl],
|
||||
[(BudgetPhase.ChoCCM, BudgetPhase.TraLai)] = [AppRoles.CostControl],
|
||||
[(BudgetPhase.ChoCCM, BudgetPhase.TuChoi)] = [AppRoles.CostControl],
|
||||
|
||||
[(BudgetPhase.ChoCEO, BudgetPhase.DaDuyet)] = [AppRoles.Director, AppRoles.AuthorizedSigner],
|
||||
[(BudgetPhase.ChoCEO, BudgetPhase.DangSoanThao)] = [AppRoles.Director, AppRoles.AuthorizedSigner],
|
||||
[(BudgetPhase.ChoCEO, BudgetPhase.TraLai)] = [AppRoles.Director, AppRoles.AuthorizedSigner],
|
||||
[(BudgetPhase.ChoCEO, BudgetPhase.TuChoi)] = [AppRoles.Director, AppRoles.AuthorizedSigner],
|
||||
},
|
||||
PhaseSla: DefaultSla,
|
||||
ActivePhases:
|
||||
[
|
||||
BudgetPhase.DangSoanThao,
|
||||
BudgetPhase.TraLai,
|
||||
BudgetPhase.ChoCCM,
|
||||
BudgetPhase.ChoCEO,
|
||||
BudgetPhase.DaDuyet,
|
||||
|
||||
Reference in New Issue
Block a user