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
116 lines
9.1 KiB
Markdown
116 lines
9.1 KiB
Markdown
# 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 targets `DRAFT`, `SUBMITTED`, and `CHANGES_REQUESTED` entries 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.tsx` already 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.ts` appears 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.ts` already 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_records` was 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 NULL` as non-approved, excludes `invoiced=true`, and returns counts by recurring execution identity key.
|
|
- Applied blocker metadata during due-work shaping:
|
|
- `packages/billing/src/actions/billingAndTax.ts`
|
|
- `getAvailableRecurringDueWork(...)` 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 `approvalBlockedEntryCount` on rows/candidates and `hasApprovalBlockers` on 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 -> contracts` and filtering by `contracts.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 `undefined` bridge 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`, and `Review Approvals` link to `/msp/time-sheet-approvals` with 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/T004` approval-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 = NULL` behaving as non-approved.
|
|
- `T002` unrelated non-approved time outside window does not block.
|
|
- `T003/T008/T017` mixed-charge window blocked in full; direct generation rejects.
|
|
- `T009/T011` stale-ready server rejection and approval-cleared transition back to ready.
|
|
- Added helper options in `createApprovedTimeEntryForContractLine(...)` for nullable `approvalStatus`, nullable `contractLineId`, and `invoiced`.
|
|
- 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 `T007` coverage marker for preserving ready-list grouped behavior.
|
|
- Server generation unit:
|
|
- `server/src/test/unit/billing/invoiceGeneration.selectorInputGenerate.test.ts`
|
|
- Added `T008/T009` guard test validating pre-generation blocker rejection.
|
|
- Recurring run unit:
|
|
- `server/src/test/unit/billing/recurringBillingRunActions.test.ts`
|
|
- Added `T010` coverage ensuring blocked target failure does not stop unrelated eligible target.
|
|
- Docs/copy unit:
|
|
- `server/src/test/unit/docs/unapprovedTimeBlocksRecurringInvoices.copy.test.ts`
|
|
- Added `T012` assertions for runbook + in-product copy.
|
|
|
|
## Validation Commands Run
|
|
|
|
- `pnpm -s vitest run src/test/unit/billing/automaticInvoices.recurringDueWork.ui.test.tsx --coverage.enabled=false`
|
|
- `pnpm -s vitest run src/test/unit/billing/invoiceGeneration.selectorInputGenerate.test.ts --coverage.enabled=false`
|
|
- `pnpm -s vitest run src/test/unit/billing/recurringBillingRunActions.test.ts src/test/unit/docs/unapprovedTimeBlocksRecurringInvoices.copy.test.ts --coverage.enabled=false`
|
|
- `pnpm -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.ts` suite 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.
|