Some checks are pending
Bidi Control Character Guard / bidi-control-guard (push) Waiting to run
Circular Dependency Check / Check for new circular dependencies (push) Waiting to run
Citus Migration Smoke / Combined migrations on single-node Citus (push) Waiting to run
E2E Fresh Install Tests / fresh-install-e2e (push) Waiting to run
ext-v2 guardrails / Run ext-v2 guard and ESLint (push) Waiting to run
Integration Tests / Check for relevant changes (push) Waiting to run
Integration Tests / ${{ (github.event_name == 'schedule' || github.event.inputs.suite == 'full') && 'Full integration suite' || 'Tier-1 integration subset' }} (push) Blocked by required conditions
Mobile checks / Mobile lint + typecheck (push) Waiting to run
Mobile checks / Mobile unit tests (push) Waiting to run
Mobile checks / Mobile dependency audit (report) (push) Waiting to run
Mobile checks / Mobile reproducibility checks (push) Waiting to run
Secrets guard (env backups) / Ensure no tracked env backup files (push) Waiting to run
Temporal Readiness / fast-readiness (push) Waiting to run
Temporal Readiness / docker-parity (push) Waiting to run
TypeScript Type Check / Nx affected typecheck (push) Waiting to run
Unit Tests / Skipped-test budget (push) Waiting to run
Unit Tests / Nx affected unit tests (push) Waiting to run
Unit Tests / Server unit coverage (informational) (push) Waiting to run
Validate Tenant Management Schema / Check for relevant changes (push) Waiting to run
Validate Tenant Management Schema / Validate Tenant Management Schema (push) Blocked by required conditions
EE Workflows Build Guard / ee-workflows-build-guard (push) Waiting to run
Excluded: .git, node_modules, secrets/, compose.env, assemblyscript tgz Source: /opt/alga-psa on psa.joliet.tech
9.1 KiB
9.1 KiB
Scratchpad — Unapproved Time Blocks Recurring Invoices
- Plan slug:
unapproved-time-blocks-recurring-invoices - Created:
2026-04-12
What This Is
Working notes for making recurring invoice approval blockers explicit in Automatic Invoices and enforcing the same rule server-side during generation.
Decisions
- (2026-04-12) A recurring invoice window is blocked in full when it contains at least one uninvoiced billable time entry for that window whose
approval_status !== 'APPROVED'. - (2026-04-12) Blocked windows belong in a dedicated Needs Approval section above Ready to Invoice.
- (2026-04-12) The blocked row should show
X unapproved entries, not blocked hours, because count is more actionable for approvers. - (2026-04-12) Mixed-charge windows are blocked entirely; no partial invoice should be created for fixed or other non-time charges while related billable time is still unapproved.
- (2026-04-12) The generation path must re-check approval blockers immediately before invoice creation so stale UI state cannot bypass the rule.
Discoveries / Constraints
- (2026-04-12) Reviewed billing-engine selection queries indicate contract-hourly and unresolved/non-contract time selection already filter
time_entries.approval_status = 'APPROVED'. Relevant file:packages/billing/src/lib/billing/billingEngine.ts. - (2026-04-12)
rolloverUnapprovedTime(...)explicitly targetsDRAFT,SUBMITTED, andCHANGES_REQUESTEDentries after recurring invoice generation, which reinforces that unapproved time is conceptually expected to stay out of billed time selection. Relevant files:packages/billing/src/lib/billing/billingEngine.ts,packages/billing/src/actions/invoiceGeneration.ts. - (2026-04-12)
AutomaticInvoices.tsxalready understands grouped recurring candidates and generic blocked states (canGenerate/blockedReason), so approval blockers can likely fit the existing grouped row model rather than requiring a brand new page architecture. - (2026-04-12)
packages/billing/src/actions/billingAndTax.tsappears to be the primary recurring due-work shaping path and likely the right place to compute approval-blocker metadata for the UI. - (2026-04-12)
packages/types/src/interfaces/recurringTiming.interfaces.tsalready carries grouped due-work row and candidate metadata and likely needs approval-blocker count/reason fields. - (2026-04-12) During review, no obvious source-linkage population path for
invoice_time_entries/invoice_usage_recordswas found in the inspected invoice creation path. That looks adjacent to, but not required for, this blocker-focused change and should stay out of scope unless implementation depends on it.
Commands / Runbooks
- (2026-04-12) Scaffolded this plan with:
python3 /Users/roberisaacs/.codex/skills/alga-plan/scripts/scaffold_plan.py "Unapproved Time Blocks Recurring Invoices" --slug unapproved-time-blocks-recurring-invoices
- (2026-04-12) Validate the plan with:
python3 /Users/roberisaacs/.codex/skills/alga-plan/scripts/validate_plan.py ee/docs/plans/2026-04-12-unapproved-time-blocks-recurring-invoices
- (2026-04-12) Design doc written to:
docs/plans/2026-04-12-unapproved-time-blocks-recurring-invoices-design.md
Links / References
- Design doc:
docs/plans/2026-04-12-unapproved-time-blocks-recurring-invoices-design.md - Automatic invoices UI:
packages/billing/src/components/billing-dashboard/AutomaticInvoices.tsx - Recurring due-work shaping:
packages/billing/src/actions/billingAndTax.ts - Recurring generation:
packages/billing/src/actions/invoiceGeneration.ts - Recurring billing runs:
packages/billing/src/actions/recurringBillingRunActions.ts - Billing engine selection logic:
packages/billing/src/lib/billing/billingEngine.ts - Due-work interfaces:
packages/types/src/interfaces/recurringTiming.interfaces.ts - Related grouped automatic-invoices plan:
ee/docs/plans/2026-03-20-grouped-automatic-invoices-selection/
Open Questions
- What is the exact target screen for
Review Approvals, and can we deep-link/filter by client and invoice window in the first pass?
Implementation Log (2026-04-12)
- Added shared recurring approval-blocker detection helper:
packages/billing/src/actions/recurringApprovalBlockers.ts- Centralizes blocker count logic for contract-hourly windows and unresolved/non-contract time selections.
- Mirrors contract-hourly billing semantics more closely by including client/work-item scoping, configured service matching, and uniquely assignable unassigned time.
- Treats
approval_status IS NULLas non-approved, excludesinvoiced=true, and returns counts by recurring execution identity key.
- Applied blocker metadata during due-work shaping:
packages/billing/src/actions/billingAndTax.tsgetAvailableRecurringDueWork(...)now computes and applies approval-blocker counts before pagination.- Candidates and members now carry blocker metadata and approval-specific blocked reason text.
- Extended recurring due-work interfaces for blocker metadata:
packages/types/src/interfaces/recurringTiming.interfaces.ts- Added
approvalBlockedEntryCounton rows/candidates andhasApprovalBlockerson candidates.
- Added server-side recurring generation guard:
packages/billing/src/actions/invoiceGeneration.ts- Re-resolves selector-input windows to matching recurring rows, re-checks approval blockers immediately before invoice creation, and throws
Blocked until approval: X unapproved entries.when blocked. - Fixed pre-check query regression by joining
recurring_service_periods -> contract_lines -> contractsand filtering bycontracts.owner_client_id.
- Ensured grouped recurring runs continue when one target is blocked:
packages/billing/src/actions/recurringBillingRunActions.ts- Preserves per-target failure handling while continuing unrelated eligible targets.
- Adjusted helper invocations to avoid passing explicit
undefinedbridge args.
- Added Automatic Invoices Needs Approval UX:
packages/billing/src/components/billing-dashboard/AutomaticInvoices.tsx- New Needs Approval section above Ready to Invoice.
- Blocked groups moved out of ready table.
- Needs Approval rows show client, service period, invoice window,
X unapproved entries, andReview Approvalslink to/msp/time-sheet-approvalswith query params. - Ready selection/preview/generate semantics remain for clean windows.
Test Coverage Added / Updated
- Integration (DB-backed):
server/src/test/integration/billingInvoiceTiming.integration.test.ts- Added/updated:
T001/T004approval-blocked contract-hourly window with uninvoiced non-approved time.- Added parity coverage for uniquely assignable unassigned time on the final included service-period day and for
approval_status = NULLbehaving as non-approved. T002unrelated non-approved time outside window does not block.T003/T008/T017mixed-charge window blocked in full; direct generation rejects.T009/T011stale-ready server rejection and approval-cleared transition back to ready.
- Added helper options in
createApprovedTimeEntryForContractLine(...)for nullableapprovalStatus, nullablecontractLineId, andinvoiced.
- UI unit:
server/src/test/unit/billing/automaticInvoices.recurringDueWork.ui.test.tsx- Added Needs Approval rendering/assertions (section order, row content, review link, non-selectable/non-generatable behavior).
- Added explicit
T007coverage marker for preserving ready-list grouped behavior.
- Server generation unit:
server/src/test/unit/billing/invoiceGeneration.selectorInputGenerate.test.ts- Added
T008/T009guard test validating pre-generation blocker rejection.
- Recurring run unit:
server/src/test/unit/billing/recurringBillingRunActions.test.ts- Added
T010coverage ensuring blocked target failure does not stop unrelated eligible target.
- Docs/copy unit:
server/src/test/unit/docs/unapprovedTimeBlocksRecurringInvoices.copy.test.ts- Added
T012assertions for runbook + in-product copy.
Validation Commands Run
pnpm -s vitest run src/test/unit/billing/automaticInvoices.recurringDueWork.ui.test.tsx --coverage.enabled=falsepnpm -s vitest run src/test/unit/billing/invoiceGeneration.selectorInputGenerate.test.ts --coverage.enabled=falsepnpm -s vitest run src/test/unit/billing/recurringBillingRunActions.test.ts src/test/unit/docs/unapprovedTimeBlocksRecurringInvoices.copy.test.ts --coverage.enabled=falsepnpm -s vitest run src/test/integration/billingInvoiceTiming.integration.test.ts -t "T001/T004|parity: recurring due-work blocks uniquely assignable unassigned hourly time|T002: recurring due-work does not block|T003/T008/T017|T009/T011" --coverage.enabled=false
Gotchas / Notes
- Running the full
billingInvoiceTiming.integration.test.tssuite currently includes pre-existing failures unrelated to this approval-blocker scope; validation for this change set uses targeted DB-backed cases tied to plan IDs. - One unit fixture (
invoiceGeneration.selectorInputGenerate) required explicit recurring-service-period row fields (owner_client_id, matching invoice window) so the new pre-generation blocker query can resolve rows in the mocked environment.