PSA/shared/billingClients/clientContractStatus.ts
Hermes 284313f908
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
Initial import of AlgaPSA codebase from PSA server
Excluded: .git, node_modules, secrets/, compose.env, assemblyscript tgz

Source: /opt/alga-psa on psa.joliet.tech
2026-06-22 16:12:17 -05:00

70 lines
1.6 KiB
TypeScript

import type { ContractStatus } from '@alga-psa/types';
export type ClientAssignmentStatus = Extract<
ContractStatus,
'draft' | 'active' | 'terminated' | 'expired'
>;
type DeriveClientContractStatusParams = {
isActive: boolean;
startDate: string | null | undefined;
endDate: string | null | undefined;
now?: string | Date;
};
const normalizeDateOnly = (value: string | null | undefined): string | null => {
if (typeof value !== 'string') {
return null;
}
const trimmed = value.trim();
if (!trimmed) {
return null;
}
if (/^\d{4}-\d{2}-\d{2}$/.test(trimmed)) {
return trimmed;
}
return trimmed.includes('T') ? trimmed.slice(0, 10) : null;
};
const normalizeNowDateOnly = (value?: string | Date): string => {
if (value instanceof Date) {
return value.toISOString().slice(0, 10);
}
const normalized = normalizeDateOnly(value);
if (normalized) {
return normalized;
}
return new Date().toISOString().slice(0, 10);
};
export function deriveClientContractStatus(
params: DeriveClientContractStatusParams
): ClientAssignmentStatus {
const startDate = normalizeDateOnly(params.startDate);
const endDate = normalizeDateOnly(params.endDate);
const nowDate = normalizeNowDateOnly(params.now);
if (startDate && startDate > nowDate) {
return 'draft';
}
if (!params.isActive) {
return 'terminated';
}
if (endDate && endDate < nowDate) {
return 'expired';
}
return 'active';
}
export function isLiveClientContractStatus(status: ClientAssignmentStatus): boolean {
return status === 'active' || status === 'expired' || status === 'terminated';
}