@urateam/core 0.1.4 → 0.1.8
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.
- package/dist/__tests__/audit/audit-e2e.test.d.ts +2 -0
- package/dist/__tests__/audit/audit-e2e.test.d.ts.map +1 -0
- package/dist/__tests__/audit/audit-e2e.test.js +56 -0
- package/dist/__tests__/audit/audit-e2e.test.js.map +1 -0
- package/dist/__tests__/audit/csv.test.d.ts +2 -0
- package/dist/__tests__/audit/csv.test.d.ts.map +1 -0
- package/dist/__tests__/audit/csv.test.js +134 -0
- package/dist/__tests__/audit/csv.test.js.map +1 -0
- package/dist/__tests__/audit/events.test.d.ts +2 -0
- package/dist/__tests__/audit/events.test.d.ts.map +1 -0
- package/dist/__tests__/audit/events.test.js +45 -0
- package/dist/__tests__/audit/events.test.js.map +1 -0
- package/dist/__tests__/audit/policy-events.test.d.ts +2 -0
- package/dist/__tests__/audit/policy-events.test.d.ts.map +1 -0
- package/dist/__tests__/audit/policy-events.test.js +49 -0
- package/dist/__tests__/audit/policy-events.test.js.map +1 -0
- package/dist/__tests__/audit/projection.test.d.ts +2 -0
- package/dist/__tests__/audit/projection.test.d.ts.map +1 -0
- package/dist/__tests__/audit/projection.test.js +76 -0
- package/dist/__tests__/audit/projection.test.js.map +1 -0
- package/dist/__tests__/audit/rbac-events.test.d.ts +2 -0
- package/dist/__tests__/audit/rbac-events.test.d.ts.map +1 -0
- package/dist/__tests__/audit/rbac-events.test.js +56 -0
- package/dist/__tests__/audit/rbac-events.test.js.map +1 -0
- package/dist/__tests__/audit/reader.test.d.ts +2 -0
- package/dist/__tests__/audit/reader.test.d.ts.map +1 -0
- package/dist/__tests__/audit/reader.test.js +133 -0
- package/dist/__tests__/audit/reader.test.js.map +1 -0
- package/dist/__tests__/audit/retention.test.d.ts +2 -0
- package/dist/__tests__/audit/retention.test.d.ts.map +1 -0
- package/dist/__tests__/audit/retention.test.js +48 -0
- package/dist/__tests__/audit/retention.test.js.map +1 -0
- package/dist/__tests__/audit/sso-events.test.d.ts +2 -0
- package/dist/__tests__/audit/sso-events.test.d.ts.map +1 -0
- package/dist/__tests__/audit/sso-events.test.js +49 -0
- package/dist/__tests__/audit/sso-events.test.js.map +1 -0
- package/dist/__tests__/audit/writer.test.d.ts +2 -0
- package/dist/__tests__/audit/writer.test.d.ts.map +1 -0
- package/dist/__tests__/audit/writer.test.js +67 -0
- package/dist/__tests__/audit/writer.test.js.map +1 -0
- package/dist/__tests__/audit-immutability.test.d.ts +2 -0
- package/dist/__tests__/audit-immutability.test.d.ts.map +1 -0
- package/dist/__tests__/audit-immutability.test.js +61 -0
- package/dist/__tests__/audit-immutability.test.js.map +1 -0
- package/dist/__tests__/audit-types.test.d.ts +2 -0
- package/dist/__tests__/audit-types.test.d.ts.map +1 -0
- package/dist/__tests__/audit-types.test.js +35 -0
- package/dist/__tests__/audit-types.test.js.map +1 -0
- package/dist/__tests__/auth/session-store.test.d.ts +2 -0
- package/dist/__tests__/auth/session-store.test.d.ts.map +1 -0
- package/dist/__tests__/auth/session-store.test.js +65 -0
- package/dist/__tests__/auth/session-store.test.js.map +1 -0
- package/dist/__tests__/auth/sso-config.test.d.ts +2 -0
- package/dist/__tests__/auth/sso-config.test.d.ts.map +1 -0
- package/dist/__tests__/auth/sso-config.test.js +75 -0
- package/dist/__tests__/auth/sso-config.test.js.map +1 -0
- package/dist/__tests__/auth/sso-feature-flag.test.d.ts +2 -0
- package/dist/__tests__/auth/sso-feature-flag.test.d.ts.map +1 -0
- package/dist/__tests__/auth/sso-feature-flag.test.js +33 -0
- package/dist/__tests__/auth/sso-feature-flag.test.js.map +1 -0
- package/dist/__tests__/auth/user-role-round-trip.test.d.ts +2 -0
- package/dist/__tests__/auth/user-role-round-trip.test.d.ts.map +1 -0
- package/dist/__tests__/auth/user-role-round-trip.test.js +30 -0
- package/dist/__tests__/auth/user-role-round-trip.test.js.map +1 -0
- package/dist/__tests__/auth/user-store.test.d.ts +2 -0
- package/dist/__tests__/auth/user-store.test.d.ts.map +1 -0
- package/dist/__tests__/auth/user-store.test.js +119 -0
- package/dist/__tests__/auth/user-store.test.js.map +1 -0
- package/dist/__tests__/auth/workos-client.test.d.ts +2 -0
- package/dist/__tests__/auth/workos-client.test.d.ts.map +1 -0
- package/dist/__tests__/auth/workos-client.test.js +17 -0
- package/dist/__tests__/auth/workos-client.test.js.map +1 -0
- package/dist/__tests__/auto-merge.test.js +2 -1
- package/dist/__tests__/auto-merge.test.js.map +1 -1
- package/dist/__tests__/cost/aggregate.test.d.ts +2 -0
- package/dist/__tests__/cost/aggregate.test.d.ts.map +1 -0
- package/dist/__tests__/cost/aggregate.test.js +358 -0
- package/dist/__tests__/cost/aggregate.test.js.map +1 -0
- package/dist/__tests__/cost/csv.test.d.ts +2 -0
- package/dist/__tests__/cost/csv.test.d.ts.map +1 -0
- package/dist/__tests__/cost/csv.test.js +109 -0
- package/dist/__tests__/cost/csv.test.js.map +1 -0
- package/dist/__tests__/cost/per-run.test.d.ts +2 -0
- package/dist/__tests__/cost/per-run.test.d.ts.map +1 -0
- package/dist/__tests__/cost/per-run.test.js +64 -0
- package/dist/__tests__/cost/per-run.test.js.map +1 -0
- package/dist/__tests__/cost/rates.test.d.ts +2 -0
- package/dist/__tests__/cost/rates.test.d.ts.map +1 -0
- package/dist/__tests__/cost/rates.test.js +47 -0
- package/dist/__tests__/cost/rates.test.js.map +1 -0
- package/dist/__tests__/cost/rollup.test.d.ts +2 -0
- package/dist/__tests__/cost/rollup.test.d.ts.map +1 -0
- package/dist/__tests__/cost/rollup.test.js +187 -0
- package/dist/__tests__/cost/rollup.test.js.map +1 -0
- package/dist/__tests__/cost-defensive.test.d.ts +2 -0
- package/dist/__tests__/cost-defensive.test.d.ts.map +1 -0
- package/dist/__tests__/cost-defensive.test.js +123 -0
- package/dist/__tests__/cost-defensive.test.js.map +1 -0
- package/dist/__tests__/cost-integration.test.d.ts +2 -0
- package/dist/__tests__/cost-integration.test.d.ts.map +1 -0
- package/dist/__tests__/cost-integration.test.js +80 -0
- package/dist/__tests__/cost-integration.test.js.map +1 -0
- package/dist/__tests__/cost-license.test.d.ts +2 -0
- package/dist/__tests__/cost-license.test.d.ts.map +1 -0
- package/dist/__tests__/cost-license.test.js +23 -0
- package/dist/__tests__/cost-license.test.js.map +1 -0
- package/dist/__tests__/cost-types.test.d.ts +2 -0
- package/dist/__tests__/cost-types.test.d.ts.map +1 -0
- package/dist/__tests__/cost-types.test.js +65 -0
- package/dist/__tests__/cost-types.test.js.map +1 -0
- package/dist/__tests__/db-audit-schema.test.d.ts +2 -0
- package/dist/__tests__/db-audit-schema.test.d.ts.map +1 -0
- package/dist/__tests__/db-audit-schema.test.js +35 -0
- package/dist/__tests__/db-audit-schema.test.js.map +1 -0
- package/dist/__tests__/db-cost-rollups-schema.test.d.ts +2 -0
- package/dist/__tests__/db-cost-rollups-schema.test.d.ts.map +1 -0
- package/dist/__tests__/db-cost-rollups-schema.test.js +28 -0
- package/dist/__tests__/db-cost-rollups-schema.test.js.map +1 -0
- package/dist/__tests__/db-rbac-schema.test.d.ts +2 -0
- package/dist/__tests__/db-rbac-schema.test.d.ts.map +1 -0
- package/dist/__tests__/db-rbac-schema.test.js +32 -0
- package/dist/__tests__/db-rbac-schema.test.js.map +1 -0
- package/dist/__tests__/db-sso-schema.test.d.ts +2 -0
- package/dist/__tests__/db-sso-schema.test.d.ts.map +1 -0
- package/dist/__tests__/db-sso-schema.test.js +48 -0
- package/dist/__tests__/db-sso-schema.test.js.map +1 -0
- package/dist/__tests__/helpers/license.d.ts +3 -0
- package/dist/__tests__/helpers/license.d.ts.map +1 -0
- package/dist/__tests__/helpers/license.js +74 -0
- package/dist/__tests__/helpers/license.js.map +1 -0
- package/dist/__tests__/license-audit-event.test.d.ts +2 -0
- package/dist/__tests__/license-audit-event.test.d.ts.map +1 -0
- package/dist/__tests__/license-audit-event.test.js +60 -0
- package/dist/__tests__/license-audit-event.test.js.map +1 -0
- package/dist/__tests__/license-end-to-end.test.d.ts +2 -0
- package/dist/__tests__/license-end-to-end.test.d.ts.map +1 -0
- package/dist/__tests__/license-end-to-end.test.js +75 -0
- package/dist/__tests__/license-end-to-end.test.js.map +1 -0
- package/dist/__tests__/license-public-key.test.d.ts +2 -0
- package/dist/__tests__/license-public-key.test.d.ts.map +1 -0
- package/dist/__tests__/license-public-key.test.js +18 -0
- package/dist/__tests__/license-public-key.test.js.map +1 -0
- package/dist/__tests__/license.test.js +195 -32
- package/dist/__tests__/license.test.js.map +1 -1
- package/dist/__tests__/pm-action-audit-events.test.d.ts +2 -0
- package/dist/__tests__/pm-action-audit-events.test.d.ts.map +1 -0
- package/dist/__tests__/pm-action-audit-events.test.js +185 -0
- package/dist/__tests__/pm-action-audit-events.test.js.map +1 -0
- package/dist/__tests__/pm-approvals.test.js +1 -1
- package/dist/__tests__/pm-approvals.test.js.map +1 -1
- package/dist/__tests__/pm-audit-retention-step.test.d.ts +2 -0
- package/dist/__tests__/pm-audit-retention-step.test.d.ts.map +1 -0
- package/dist/__tests__/pm-audit-retention-step.test.js +126 -0
- package/dist/__tests__/pm-audit-retention-step.test.js.map +1 -0
- package/dist/__tests__/pm-budget-alerts.test.d.ts +2 -0
- package/dist/__tests__/pm-budget-alerts.test.d.ts.map +1 -0
- package/dist/__tests__/pm-budget-alerts.test.js +128 -0
- package/dist/__tests__/pm-budget-alerts.test.js.map +1 -0
- package/dist/__tests__/pm-budget-refused-event.test.d.ts +2 -0
- package/dist/__tests__/pm-budget-refused-event.test.d.ts.map +1 -0
- package/dist/__tests__/pm-budget-refused-event.test.js +141 -0
- package/dist/__tests__/pm-budget-refused-event.test.js.map +1 -0
- package/dist/__tests__/pm-budget.test.js +138 -42
- package/dist/__tests__/pm-budget.test.js.map +1 -1
- package/dist/__tests__/pm-cost-rollup-step.test.d.ts +2 -0
- package/dist/__tests__/pm-cost-rollup-step.test.d.ts.map +1 -0
- package/dist/__tests__/pm-cost-rollup-step.test.js +113 -0
- package/dist/__tests__/pm-cost-rollup-step.test.js.map +1 -0
- package/dist/__tests__/pm-scheduler.test.js +48 -21
- package/dist/__tests__/pm-scheduler.test.js.map +1 -1
- package/dist/__tests__/pm-slack.test.js +37 -0
- package/dist/__tests__/pm-slack.test.js.map +1 -1
- package/dist/__tests__/pm-sso-prune-step.test.d.ts +2 -0
- package/dist/__tests__/pm-sso-prune-step.test.d.ts.map +1 -0
- package/dist/__tests__/pm-sso-prune-step.test.js +103 -0
- package/dist/__tests__/pm-sso-prune-step.test.js.map +1 -0
- package/dist/__tests__/pm-types.test.js +130 -1
- package/dist/__tests__/pm-types.test.js.map +1 -1
- package/dist/__tests__/policy/auto-merge-reviewer-gate.test.d.ts +2 -0
- package/dist/__tests__/policy/auto-merge-reviewer-gate.test.d.ts.map +1 -0
- package/dist/__tests__/policy/auto-merge-reviewer-gate.test.js +50 -0
- package/dist/__tests__/policy/auto-merge-reviewer-gate.test.js.map +1 -0
- package/dist/__tests__/policy/cost-gate.test.d.ts +2 -0
- package/dist/__tests__/policy/cost-gate.test.d.ts.map +1 -0
- package/dist/__tests__/policy/cost-gate.test.js +27 -0
- package/dist/__tests__/policy/cost-gate.test.js.map +1 -0
- package/dist/__tests__/policy/evaluate.test.d.ts +2 -0
- package/dist/__tests__/policy/evaluate.test.d.ts.map +1 -0
- package/dist/__tests__/policy/evaluate.test.js +105 -0
- package/dist/__tests__/policy/evaluate.test.js.map +1 -0
- package/dist/__tests__/policy/override.test.d.ts +2 -0
- package/dist/__tests__/policy/override.test.d.ts.map +1 -0
- package/dist/__tests__/policy/override.test.js +26 -0
- package/dist/__tests__/policy/override.test.js.map +1 -0
- package/dist/__tests__/policy/path-gate.test.d.ts +2 -0
- package/dist/__tests__/policy/path-gate.test.d.ts.map +1 -0
- package/dist/__tests__/policy/path-gate.test.js +31 -0
- package/dist/__tests__/policy/path-gate.test.js.map +1 -0
- package/dist/__tests__/policy/pr-reviewer-wiring.test.d.ts +2 -0
- package/dist/__tests__/policy/pr-reviewer-wiring.test.d.ts.map +1 -0
- package/dist/__tests__/policy/pr-reviewer-wiring.test.js +24 -0
- package/dist/__tests__/policy/pr-reviewer-wiring.test.js.map +1 -0
- package/dist/__tests__/policy/reviewer-gate.test.d.ts +2 -0
- package/dist/__tests__/policy/reviewer-gate.test.d.ts.map +1 -0
- package/dist/__tests__/policy/reviewer-gate.test.js +125 -0
- package/dist/__tests__/policy/reviewer-gate.test.js.map +1 -0
- package/dist/__tests__/policy-license.test.d.ts +2 -0
- package/dist/__tests__/policy-license.test.d.ts.map +1 -0
- package/dist/__tests__/policy-license.test.js +31 -0
- package/dist/__tests__/policy-license.test.js.map +1 -0
- package/dist/__tests__/policy-types.test.d.ts +2 -0
- package/dist/__tests__/policy-types.test.d.ts.map +1 -0
- package/dist/__tests__/policy-types.test.js +49 -0
- package/dist/__tests__/policy-types.test.js.map +1 -0
- package/dist/__tests__/ralph-gate.test.js +6 -4
- package/dist/__tests__/ralph-gate.test.js.map +1 -1
- package/dist/__tests__/rbac/bootstrap.test.d.ts +2 -0
- package/dist/__tests__/rbac/bootstrap.test.d.ts.map +1 -0
- package/dist/__tests__/rbac/bootstrap.test.js +62 -0
- package/dist/__tests__/rbac/bootstrap.test.js.map +1 -0
- package/dist/__tests__/rbac/matrix.test.d.ts +2 -0
- package/dist/__tests__/rbac/matrix.test.d.ts.map +1 -0
- package/dist/__tests__/rbac/matrix.test.js +62 -0
- package/dist/__tests__/rbac/matrix.test.js.map +1 -0
- package/dist/__tests__/rbac/user-role-store.test.d.ts +2 -0
- package/dist/__tests__/rbac/user-role-store.test.d.ts.map +1 -0
- package/dist/__tests__/rbac/user-role-store.test.js +78 -0
- package/dist/__tests__/rbac/user-role-store.test.js.map +1 -0
- package/dist/__tests__/rbac-defensive.test.d.ts +2 -0
- package/dist/__tests__/rbac-defensive.test.d.ts.map +1 -0
- package/dist/__tests__/rbac-defensive.test.js +29 -0
- package/dist/__tests__/rbac-defensive.test.js.map +1 -0
- package/dist/__tests__/rbac-integration.test.d.ts +2 -0
- package/dist/__tests__/rbac-integration.test.d.ts.map +1 -0
- package/dist/__tests__/rbac-integration.test.js +58 -0
- package/dist/__tests__/rbac-integration.test.js.map +1 -0
- package/dist/__tests__/rbac-license.test.d.ts +2 -0
- package/dist/__tests__/rbac-license.test.d.ts.map +1 -0
- package/dist/__tests__/rbac-license.test.js +27 -0
- package/dist/__tests__/rbac-license.test.js.map +1 -0
- package/dist/__tests__/reproduce-bec113-pagination-warning.test.js +14 -2
- package/dist/__tests__/reproduce-bec113-pagination-warning.test.js.map +1 -1
- package/dist/__tests__/reproduce-bec62.test.js +10 -3
- package/dist/__tests__/reproduce-bec62.test.js.map +1 -1
- package/dist/__tests__/reproduce-bec91-stuck-in-progress.test.js +24 -4
- package/dist/__tests__/reproduce-bec91-stuck-in-progress.test.js.map +1 -1
- package/dist/__tests__/server.test.js +46 -2
- package/dist/__tests__/server.test.js.map +1 -1
- package/dist/__tests__/sso-license.test.d.ts +2 -0
- package/dist/__tests__/sso-license.test.d.ts.map +1 -0
- package/dist/__tests__/sso-license.test.js +49 -0
- package/dist/__tests__/sso-license.test.js.map +1 -0
- package/dist/__tests__/start-todo.test.js +5 -0
- package/dist/__tests__/start-todo.test.js.map +1 -1
- package/dist/__tests__/webhook-handler.test.js +138 -0
- package/dist/__tests__/webhook-handler.test.js.map +1 -1
- package/dist/audit/config-fingerprint.d.ts +10 -0
- package/dist/audit/config-fingerprint.d.ts.map +1 -0
- package/dist/audit/config-fingerprint.js +20 -0
- package/dist/audit/config-fingerprint.js.map +1 -0
- package/dist/audit/csv.d.ts +8 -0
- package/dist/audit/csv.d.ts.map +1 -0
- package/dist/audit/csv.js +56 -0
- package/dist/audit/csv.js.map +1 -0
- package/dist/audit/events.d.ts +107 -0
- package/dist/audit/events.d.ts.map +1 -0
- package/dist/audit/events.js +221 -0
- package/dist/audit/events.js.map +1 -0
- package/dist/audit/index.d.ts +7 -0
- package/dist/audit/index.d.ts.map +1 -0
- package/dist/audit/index.js +7 -0
- package/dist/audit/index.js.map +1 -0
- package/dist/audit/projection.d.ts +38 -0
- package/dist/audit/projection.d.ts.map +1 -0
- package/dist/audit/projection.js +109 -0
- package/dist/audit/projection.js.map +1 -0
- package/dist/audit/reader.d.ts +42 -0
- package/dist/audit/reader.d.ts.map +1 -0
- package/dist/audit/reader.js +243 -0
- package/dist/audit/reader.js.map +1 -0
- package/dist/audit/retention.d.ts +12 -0
- package/dist/audit/retention.d.ts.map +1 -0
- package/dist/audit/retention.js +23 -0
- package/dist/audit/retention.js.map +1 -0
- package/dist/audit/writer.d.ts +26 -0
- package/dist/audit/writer.d.ts.map +1 -0
- package/dist/audit/writer.js +54 -0
- package/dist/audit/writer.js.map +1 -0
- package/dist/auth/index.d.ts +5 -0
- package/dist/auth/index.d.ts.map +1 -0
- package/dist/auth/index.js +5 -0
- package/dist/auth/index.js.map +1 -0
- package/dist/auth/session-store.d.ts +17 -0
- package/dist/auth/session-store.d.ts.map +1 -0
- package/dist/auth/session-store.js +59 -0
- package/dist/auth/session-store.js.map +1 -0
- package/dist/auth/sso-config.d.ts +14 -0
- package/dist/auth/sso-config.d.ts.map +1 -0
- package/dist/auth/sso-config.js +50 -0
- package/dist/auth/sso-config.js.map +1 -0
- package/dist/auth/user-store.d.ts +33 -0
- package/dist/auth/user-store.d.ts.map +1 -0
- package/dist/auth/user-store.js +71 -0
- package/dist/auth/user-store.js.map +1 -0
- package/dist/auth/workos-client.d.ts +46 -0
- package/dist/auth/workos-client.d.ts.map +1 -0
- package/dist/auth/workos-client.js +66 -0
- package/dist/auth/workos-client.js.map +1 -0
- package/dist/cost/aggregate.d.ts +49 -0
- package/dist/cost/aggregate.d.ts.map +1 -0
- package/dist/cost/aggregate.js +364 -0
- package/dist/cost/aggregate.js.map +1 -0
- package/dist/cost/csv.d.ts +12 -0
- package/dist/cost/csv.d.ts.map +1 -0
- package/dist/cost/csv.js +79 -0
- package/dist/cost/csv.js.map +1 -0
- package/dist/cost/index.d.ts +7 -0
- package/dist/cost/index.d.ts.map +1 -0
- package/dist/cost/index.js +7 -0
- package/dist/cost/index.js.map +1 -0
- package/dist/cost/per-run.d.ts +31 -0
- package/dist/cost/per-run.d.ts.map +1 -0
- package/dist/cost/per-run.js +35 -0
- package/dist/cost/per-run.js.map +1 -0
- package/dist/cost/rates.d.ts +27 -0
- package/dist/cost/rates.d.ts.map +1 -0
- package/dist/cost/rates.js +26 -0
- package/dist/cost/rates.js.map +1 -0
- package/dist/cost/rollup.d.ts +17 -0
- package/dist/cost/rollup.d.ts.map +1 -0
- package/dist/cost/rollup.js +164 -0
- package/dist/cost/rollup.js.map +1 -0
- package/dist/cost/types.d.ts +54 -0
- package/dist/cost/types.d.ts.map +1 -0
- package/dist/cost/types.js +2 -0
- package/dist/cost/types.js.map +1 -0
- package/dist/db/client.d.ts.map +1 -1
- package/dist/db/client.js +79 -1
- package/dist/db/client.js.map +1 -1
- package/dist/db/migrations/postgres/006_spend_caps.sql +14 -0
- package/dist/db/migrations/postgres/007_audit_events.sql +19 -0
- package/dist/db/migrations/postgres/008_cost_rollups.sql +19 -0
- package/dist/db/migrations/postgres/008_sso.sql +20 -0
- package/dist/db/migrations/sqlite/005_spend_caps.sql +16 -0
- package/dist/db/migrations/sqlite/006_audit_events.sql +19 -0
- package/dist/db/migrations/sqlite/007_cost_rollups.sql +19 -0
- package/dist/db/migrations/sqlite/007_sso.sql +20 -0
- package/dist/db/schema.d.ts +802 -0
- package/dist/db/schema.d.ts.map +1 -1
- package/dist/db/schema.js +80 -1
- package/dist/db/schema.js.map +1 -1
- package/dist/index.d.ts +7 -1
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +8 -1
- package/dist/index.js.map +1 -1
- package/dist/license-public-key.d.ts +15 -0
- package/dist/license-public-key.d.ts.map +1 -0
- package/dist/license-public-key.js +15 -0
- package/dist/license-public-key.js.map +1 -0
- package/dist/license.d.ts +30 -5
- package/dist/license.d.ts.map +1 -1
- package/dist/license.js +142 -12
- package/dist/license.js.map +1 -1
- package/dist/pipeline/runner.d.ts +1 -7
- package/dist/pipeline/runner.d.ts.map +1 -1
- package/dist/pipeline/runner.js +168 -21
- package/dist/pipeline/runner.js.map +1 -1
- package/dist/pm/actions/promote.d.ts +3 -0
- package/dist/pm/actions/promote.d.ts.map +1 -1
- package/dist/pm/actions/promote.js +12 -0
- package/dist/pm/actions/promote.js.map +1 -1
- package/dist/pm/actions/resolve-approvals.d.ts.map +1 -1
- package/dist/pm/actions/resolve-approvals.js +16 -0
- package/dist/pm/actions/resolve-approvals.js.map +1 -1
- package/dist/pm/actions/start-todo.d.ts +3 -0
- package/dist/pm/actions/start-todo.d.ts.map +1 -1
- package/dist/pm/actions/start-todo.js +5 -1
- package/dist/pm/actions/start-todo.js.map +1 -1
- package/dist/pm/actions/triage.d.ts +3 -0
- package/dist/pm/actions/triage.d.ts.map +1 -1
- package/dist/pm/actions/triage.js +8 -0
- package/dist/pm/actions/triage.js.map +1 -1
- package/dist/pm/budget-alerts.d.ts +18 -0
- package/dist/pm/budget-alerts.d.ts.map +1 -0
- package/dist/pm/budget-alerts.js +100 -0
- package/dist/pm/budget-alerts.js.map +1 -0
- package/dist/pm/budget.d.ts +28 -5
- package/dist/pm/budget.d.ts.map +1 -1
- package/dist/pm/budget.js +115 -39
- package/dist/pm/budget.js.map +1 -1
- package/dist/pm/scheduler.d.ts +8 -4
- package/dist/pm/scheduler.d.ts.map +1 -1
- package/dist/pm/scheduler.js +128 -5
- package/dist/pm/scheduler.js.map +1 -1
- package/dist/pm/slack.d.ts.map +1 -1
- package/dist/pm/slack.js +11 -0
- package/dist/pm/slack.js.map +1 -1
- package/dist/pm/types.d.ts +36 -0
- package/dist/pm/types.d.ts.map +1 -1
- package/dist/pm/types.js +12 -0
- package/dist/pm/types.js.map +1 -1
- package/dist/policy/cost-gate.d.ts +3 -0
- package/dist/policy/cost-gate.d.ts.map +1 -0
- package/dist/policy/cost-gate.js +11 -0
- package/dist/policy/cost-gate.js.map +1 -0
- package/dist/policy/evaluate.d.ts +32 -0
- package/dist/policy/evaluate.d.ts.map +1 -0
- package/dist/policy/evaluate.js +61 -0
- package/dist/policy/evaluate.js.map +1 -0
- package/dist/policy/index.d.ts +7 -0
- package/dist/policy/index.d.ts.map +1 -0
- package/dist/policy/index.js +7 -0
- package/dist/policy/index.js.map +1 -0
- package/dist/policy/override.d.ts +23 -0
- package/dist/policy/override.d.ts.map +1 -0
- package/dist/policy/override.js +26 -0
- package/dist/policy/override.js.map +1 -0
- package/dist/policy/path-gate.d.ts +3 -0
- package/dist/policy/path-gate.d.ts.map +1 -0
- package/dist/policy/path-gate.js +20 -0
- package/dist/policy/path-gate.js.map +1 -0
- package/dist/policy/reviewer-gate.d.ts +45 -0
- package/dist/policy/reviewer-gate.d.ts.map +1 -0
- package/dist/policy/reviewer-gate.js +77 -0
- package/dist/policy/reviewer-gate.js.map +1 -0
- package/dist/policy/types.d.ts +7 -0
- package/dist/policy/types.d.ts.map +1 -0
- package/dist/policy/types.js +2 -0
- package/dist/policy/types.js.map +1 -0
- package/dist/rbac/errors.d.ts +7 -0
- package/dist/rbac/errors.d.ts.map +1 -0
- package/dist/rbac/errors.js +13 -0
- package/dist/rbac/errors.js.map +1 -0
- package/dist/rbac/index.d.ts +5 -0
- package/dist/rbac/index.d.ts.map +1 -0
- package/dist/rbac/index.js +5 -0
- package/dist/rbac/index.js.map +1 -0
- package/dist/rbac/matrix.d.ts +35 -0
- package/dist/rbac/matrix.d.ts.map +1 -0
- package/dist/rbac/matrix.js +42 -0
- package/dist/rbac/matrix.js.map +1 -0
- package/dist/rbac/types.d.ts +2 -0
- package/dist/rbac/types.d.ts.map +1 -0
- package/dist/rbac/types.js +2 -0
- package/dist/rbac/types.js.map +1 -0
- package/dist/rbac/user-role-store.d.ts +27 -0
- package/dist/rbac/user-role-store.d.ts.map +1 -0
- package/dist/rbac/user-role-store.js +148 -0
- package/dist/rbac/user-role-store.js.map +1 -0
- package/dist/repo/git.d.ts +9 -0
- package/dist/repo/git.d.ts.map +1 -1
- package/dist/repo/git.js +11 -0
- package/dist/repo/git.js.map +1 -1
- package/dist/repo/github.d.ts +4 -0
- package/dist/repo/github.d.ts.map +1 -1
- package/dist/repo/github.js +27 -0
- package/dist/repo/github.js.map +1 -1
- package/dist/repo/gitlab.d.ts +4 -0
- package/dist/repo/gitlab.d.ts.map +1 -1
- package/dist/repo/gitlab.js +7 -0
- package/dist/repo/gitlab.js.map +1 -1
- package/dist/server.d.ts +6 -0
- package/dist/server.d.ts.map +1 -1
- package/dist/server.js +16 -10
- package/dist/server.js.map +1 -1
- package/dist/types.d.ts +155 -0
- package/dist/types.d.ts.map +1 -1
- package/dist/types.js +83 -0
- package/dist/types.js.map +1 -1
- package/dist/util/glob.d.ts +7 -0
- package/dist/util/glob.d.ts.map +1 -0
- package/dist/util/glob.js +20 -0
- package/dist/util/glob.js.map +1 -0
- package/dist/webhook/handler.d.ts +3 -0
- package/dist/webhook/handler.d.ts.map +1 -1
- package/dist/webhook/handler.js +30 -1
- package/dist/webhook/handler.js.map +1 -1
- package/package.json +2 -1
|
@@ -0,0 +1,126 @@
|
|
|
1
|
+
import { describe, it, expect, vi, beforeEach, afterEach } from "vitest";
|
|
2
|
+
import { createPmScheduler } from "../pm/scheduler.js";
|
|
3
|
+
import { createDb } from "../db/client.js";
|
|
4
|
+
import { auditEvents } from "../db/schema.js";
|
|
5
|
+
import { installTestProLicense, restoreLicense } from "./helpers/license.js";
|
|
6
|
+
function stubActions() {
|
|
7
|
+
return {
|
|
8
|
+
evaluateBudget: vi.fn().mockResolvedValue({
|
|
9
|
+
scopes: [
|
|
10
|
+
{
|
|
11
|
+
scope: { kind: "global" },
|
|
12
|
+
scopeLabel: "global",
|
|
13
|
+
limit: 100,
|
|
14
|
+
used: 0,
|
|
15
|
+
percent: 0,
|
|
16
|
+
tier: "ok",
|
|
17
|
+
},
|
|
18
|
+
],
|
|
19
|
+
worstTier: "ok",
|
|
20
|
+
promoteBlocked: false,
|
|
21
|
+
activeCount: 0,
|
|
22
|
+
}),
|
|
23
|
+
recoverRetriableRuns: vi.fn().mockResolvedValue({ recovered: [], exhausted: [] }),
|
|
24
|
+
recoverStuckInProgressIssues: vi.fn().mockResolvedValue([]),
|
|
25
|
+
triageNewIssues: vi.fn().mockResolvedValue([]),
|
|
26
|
+
resolveApprovals: vi.fn().mockResolvedValue({ resolved: 0, stillPending: 0 }),
|
|
27
|
+
promoteReadyIssues: vi.fn().mockResolvedValue([]),
|
|
28
|
+
deprioritizeStaleIssues: vi.fn().mockResolvedValue([]),
|
|
29
|
+
cancelAbandonedIssues: vi.fn().mockResolvedValue([]),
|
|
30
|
+
postDigest: vi.fn().mockResolvedValue(undefined),
|
|
31
|
+
getActiveFileMaps: vi.fn().mockResolvedValue(new Map()),
|
|
32
|
+
predictConflict: vi.fn().mockResolvedValue({ overlapRisk: "none", likelyFiles: [], reasoning: "" }),
|
|
33
|
+
};
|
|
34
|
+
}
|
|
35
|
+
function baseConfig(extra = {}) {
|
|
36
|
+
return {
|
|
37
|
+
enabled: true,
|
|
38
|
+
cronIntervalMs: 1800000,
|
|
39
|
+
triageBatchSize: 3,
|
|
40
|
+
maxInFlight: 3,
|
|
41
|
+
dailyTokenBudget: 100,
|
|
42
|
+
slackChannelId: "C123",
|
|
43
|
+
teamIds: ["team-1"],
|
|
44
|
+
...extra,
|
|
45
|
+
};
|
|
46
|
+
}
|
|
47
|
+
describe("pm tick audit retention sweep", () => {
|
|
48
|
+
beforeEach(async () => {
|
|
49
|
+
await installTestProLicense("enterprise");
|
|
50
|
+
});
|
|
51
|
+
afterEach(async () => {
|
|
52
|
+
await restoreLicense();
|
|
53
|
+
});
|
|
54
|
+
it("deletes audit_events older than retentionDays", async () => {
|
|
55
|
+
const db = await createDb({ connectionString: ":memory:" });
|
|
56
|
+
const oldTs = new Date(Date.now() - 400 * 86400000);
|
|
57
|
+
const freshTs = new Date();
|
|
58
|
+
await db.insert(auditEvents).values([
|
|
59
|
+
{
|
|
60
|
+
id: "old",
|
|
61
|
+
timestamp: oldTs,
|
|
62
|
+
eventType: "pm.issue_promoted",
|
|
63
|
+
actor: "pm-agent",
|
|
64
|
+
actorType: "pm-agent",
|
|
65
|
+
payload: "{}",
|
|
66
|
+
},
|
|
67
|
+
{
|
|
68
|
+
id: "fresh",
|
|
69
|
+
timestamp: freshTs,
|
|
70
|
+
eventType: "pm.issue_promoted",
|
|
71
|
+
actor: "pm-agent",
|
|
72
|
+
actorType: "pm-agent",
|
|
73
|
+
payload: "{}",
|
|
74
|
+
},
|
|
75
|
+
]);
|
|
76
|
+
const scheduler = createPmScheduler({
|
|
77
|
+
config: baseConfig({ auditLog: { retentionDays: 365 } }),
|
|
78
|
+
db: db,
|
|
79
|
+
linearApiKey: "",
|
|
80
|
+
slackBotToken: "",
|
|
81
|
+
actions: stubActions(),
|
|
82
|
+
});
|
|
83
|
+
await scheduler.tick();
|
|
84
|
+
const rows = await db.select().from(auditEvents);
|
|
85
|
+
const ids = rows.map((r) => r.id);
|
|
86
|
+
expect(ids).not.toContain("old");
|
|
87
|
+
expect(ids).toContain("fresh");
|
|
88
|
+
});
|
|
89
|
+
it("uses default retention (365d) when config.auditLog is unset", async () => {
|
|
90
|
+
const db = await createDb({ connectionString: ":memory:" });
|
|
91
|
+
await db.insert(auditEvents).values([
|
|
92
|
+
{
|
|
93
|
+
id: "ancient",
|
|
94
|
+
timestamp: new Date(Date.now() - 400 * 86400000),
|
|
95
|
+
eventType: "pm.issue_promoted",
|
|
96
|
+
actor: "pm-agent",
|
|
97
|
+
actorType: "pm-agent",
|
|
98
|
+
payload: "{}",
|
|
99
|
+
},
|
|
100
|
+
]);
|
|
101
|
+
const scheduler = createPmScheduler({
|
|
102
|
+
config: baseConfig(),
|
|
103
|
+
db: db,
|
|
104
|
+
linearApiKey: "",
|
|
105
|
+
slackBotToken: "",
|
|
106
|
+
actions: stubActions(),
|
|
107
|
+
});
|
|
108
|
+
await scheduler.tick();
|
|
109
|
+
const rows = await db.select().from(auditEvents);
|
|
110
|
+
expect(rows.find((r) => r.id === "ancient")).toBeUndefined();
|
|
111
|
+
});
|
|
112
|
+
it("tick does not throw when retention sweep fails", async () => {
|
|
113
|
+
const db = await createDb({ connectionString: ":memory:" });
|
|
114
|
+
// Break the audit_events table so pruneAuditLog throws.
|
|
115
|
+
await db.run?.("DROP TABLE audit_events");
|
|
116
|
+
const scheduler = createPmScheduler({
|
|
117
|
+
config: baseConfig({ auditLog: { retentionDays: 30 } }),
|
|
118
|
+
db: db,
|
|
119
|
+
linearApiKey: "",
|
|
120
|
+
slackBotToken: "",
|
|
121
|
+
actions: stubActions(),
|
|
122
|
+
});
|
|
123
|
+
await expect(scheduler.tick()).resolves.toBeUndefined();
|
|
124
|
+
});
|
|
125
|
+
});
|
|
126
|
+
//# sourceMappingURL=pm-audit-retention-step.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"pm-audit-retention-step.test.js","sourceRoot":"","sources":["../../src/__tests__/pm-audit-retention-step.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"pm-budget-alerts.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/pm-budget-alerts.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,128 @@
|
|
|
1
|
+
import { describe, it, expect, beforeEach, vi } from "vitest";
|
|
2
|
+
import Database from "better-sqlite3";
|
|
3
|
+
import { drizzle } from "drizzle-orm/better-sqlite3";
|
|
4
|
+
import * as schema from "../db/schema.js";
|
|
5
|
+
import { maybeFireAlerts } from "../pm/budget-alerts.js";
|
|
6
|
+
import { getCreateTablesDDL } from "../db/client.js";
|
|
7
|
+
import { _setSchemaDriver } from "../db/schema.js";
|
|
8
|
+
function makeDb() {
|
|
9
|
+
_setSchemaDriver("sqlite");
|
|
10
|
+
const sqlite = new Database(":memory:");
|
|
11
|
+
sqlite.exec(getCreateTablesDDL("sqlite"));
|
|
12
|
+
return drizzle(sqlite, { schema });
|
|
13
|
+
}
|
|
14
|
+
function scopeAt(kind, id, percent) {
|
|
15
|
+
return {
|
|
16
|
+
scope: kind === "global"
|
|
17
|
+
? { kind: "global" }
|
|
18
|
+
: kind === "team"
|
|
19
|
+
? { kind: "team", teamId: id }
|
|
20
|
+
: { kind: "repo", repoUrl: id },
|
|
21
|
+
scopeLabel: kind === "global" ? "global" : `${kind} ${id}`,
|
|
22
|
+
limit: 1_000_000,
|
|
23
|
+
used: Math.floor(1_000_000 * (percent / 100)),
|
|
24
|
+
percent,
|
|
25
|
+
tier: percent >= 100
|
|
26
|
+
? "blocked-100"
|
|
27
|
+
: percent >= 80
|
|
28
|
+
? "warn-80"
|
|
29
|
+
: percent >= 50
|
|
30
|
+
? "warn-50"
|
|
31
|
+
: "ok",
|
|
32
|
+
};
|
|
33
|
+
}
|
|
34
|
+
function evaluationWith(scopes) {
|
|
35
|
+
const tierRank = { ok: 0, "warn-50": 1, "warn-80": 2, "blocked-100": 3 };
|
|
36
|
+
let worst = "ok";
|
|
37
|
+
for (const s of scopes) {
|
|
38
|
+
if (tierRank[s.tier] > tierRank[worst])
|
|
39
|
+
worst = s.tier;
|
|
40
|
+
}
|
|
41
|
+
return {
|
|
42
|
+
scopes,
|
|
43
|
+
worstTier: worst,
|
|
44
|
+
promoteBlocked: worst === "blocked-100",
|
|
45
|
+
activeCount: 0,
|
|
46
|
+
};
|
|
47
|
+
}
|
|
48
|
+
describe("maybeFireAlerts", () => {
|
|
49
|
+
let db;
|
|
50
|
+
let postSlack;
|
|
51
|
+
beforeEach(() => {
|
|
52
|
+
db = makeDb();
|
|
53
|
+
postSlack = vi.fn().mockResolvedValue(undefined);
|
|
54
|
+
});
|
|
55
|
+
it("skips scopes at tier ok", async () => {
|
|
56
|
+
const evaluation = evaluationWith([scopeAt("global", "", 10)]);
|
|
57
|
+
await maybeFireAlerts(evaluation, db, postSlack, "C_TEST");
|
|
58
|
+
expect(postSlack).not.toHaveBeenCalled();
|
|
59
|
+
});
|
|
60
|
+
it("fires a message at 50%", async () => {
|
|
61
|
+
const evaluation = evaluationWith([scopeAt("global", "", 55)]);
|
|
62
|
+
await maybeFireAlerts(evaluation, db, postSlack, "C_TEST");
|
|
63
|
+
expect(postSlack).toHaveBeenCalledTimes(1);
|
|
64
|
+
const [channel, blocks] = postSlack.mock.calls[0];
|
|
65
|
+
expect(channel).toBe("C_TEST");
|
|
66
|
+
const json = JSON.stringify(blocks);
|
|
67
|
+
expect(json).toContain("global");
|
|
68
|
+
expect(json).toContain("55");
|
|
69
|
+
});
|
|
70
|
+
it("fires 50 and 80 when a scope is at 80%", async () => {
|
|
71
|
+
const evaluation = evaluationWith([scopeAt("team", "team-a", 82)]);
|
|
72
|
+
await maybeFireAlerts(evaluation, db, postSlack, "C_TEST");
|
|
73
|
+
expect(postSlack).toHaveBeenCalledTimes(2);
|
|
74
|
+
});
|
|
75
|
+
it("fires 50, 80, and 100 when a scope is blocked", async () => {
|
|
76
|
+
const evaluation = evaluationWith([scopeAt("repo", "r", 105)]);
|
|
77
|
+
await maybeFireAlerts(evaluation, db, postSlack, "C_TEST");
|
|
78
|
+
expect(postSlack).toHaveBeenCalledTimes(3);
|
|
79
|
+
const joined = postSlack.mock.calls.map((c) => JSON.stringify(c[1])).join("\n");
|
|
80
|
+
expect(joined).toContain("blocked");
|
|
81
|
+
});
|
|
82
|
+
it("dedup: same threshold same day fires exactly once", async () => {
|
|
83
|
+
const evaluation = evaluationWith([scopeAt("global", "", 60)]);
|
|
84
|
+
await maybeFireAlerts(evaluation, db, postSlack, "C_TEST");
|
|
85
|
+
await maybeFireAlerts(evaluation, db, postSlack, "C_TEST");
|
|
86
|
+
expect(postSlack).toHaveBeenCalledTimes(1);
|
|
87
|
+
});
|
|
88
|
+
it("dedup: different scopes same threshold fire separately", async () => {
|
|
89
|
+
const evaluation = evaluationWith([
|
|
90
|
+
scopeAt("team", "team-a", 60),
|
|
91
|
+
scopeAt("team", "team-b", 60),
|
|
92
|
+
]);
|
|
93
|
+
await maybeFireAlerts(evaluation, db, postSlack, "C_TEST");
|
|
94
|
+
expect(postSlack).toHaveBeenCalledTimes(2);
|
|
95
|
+
});
|
|
96
|
+
it("dedup: threshold escalation from 50 to 80 fires only the new one on second call", async () => {
|
|
97
|
+
const first = evaluationWith([scopeAt("global", "", 55)]);
|
|
98
|
+
await maybeFireAlerts(first, db, postSlack, "C_TEST");
|
|
99
|
+
expect(postSlack).toHaveBeenCalledTimes(1);
|
|
100
|
+
const second = evaluationWith([scopeAt("global", "", 85)]);
|
|
101
|
+
await maybeFireAlerts(second, db, postSlack, "C_TEST");
|
|
102
|
+
expect(postSlack).toHaveBeenCalledTimes(2);
|
|
103
|
+
});
|
|
104
|
+
it("compensating delete: a failed Slack post leaves no dedup row, so next call retries", async () => {
|
|
105
|
+
const evaluation = evaluationWith([scopeAt("global", "", 55)]);
|
|
106
|
+
// First call: postSlack throws, row should be rolled back
|
|
107
|
+
const failingPost = vi
|
|
108
|
+
.fn()
|
|
109
|
+
.mockRejectedValueOnce(new Error("slack is down"));
|
|
110
|
+
await maybeFireAlerts(evaluation, db, failingPost, "C_TEST");
|
|
111
|
+
expect(failingPost).toHaveBeenCalledTimes(1);
|
|
112
|
+
// Second call with a working post: should re-fire because the row was rolled back
|
|
113
|
+
await maybeFireAlerts(evaluation, db, postSlack, "C_TEST");
|
|
114
|
+
expect(postSlack).toHaveBeenCalledTimes(1);
|
|
115
|
+
});
|
|
116
|
+
it("DB insert failure is swallowed and does NOT call Slack", async () => {
|
|
117
|
+
const evaluation = evaluationWith([scopeAt("global", "", 55)]);
|
|
118
|
+
// Construct a db whose .insert() throws
|
|
119
|
+
const brokenDb = {
|
|
120
|
+
insert: () => {
|
|
121
|
+
throw new Error("db is down");
|
|
122
|
+
},
|
|
123
|
+
};
|
|
124
|
+
await maybeFireAlerts(evaluation, brokenDb, postSlack, "C_TEST");
|
|
125
|
+
expect(postSlack).not.toHaveBeenCalled();
|
|
126
|
+
});
|
|
127
|
+
});
|
|
128
|
+
//# sourceMappingURL=pm-budget-alerts.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"pm-budget-alerts.test.js","sourceRoot":"","sources":["../../src/__tests__/pm-budget-alerts.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"pm-budget-refused-event.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/pm-budget-refused-event.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,141 @@
|
|
|
1
|
+
import { describe, it, expect, vi, beforeEach, afterEach } from "vitest";
|
|
2
|
+
import { createPmScheduler } from "../pm/scheduler.js";
|
|
3
|
+
import { createDb } from "../db/client.js";
|
|
4
|
+
import { auditEvents } from "../db/schema.js";
|
|
5
|
+
import { installTestProLicense, restoreLicense } from "./helpers/license.js";
|
|
6
|
+
function mixedBlockedEvaluation() {
|
|
7
|
+
return {
|
|
8
|
+
scopes: [
|
|
9
|
+
{
|
|
10
|
+
scope: { kind: "global" },
|
|
11
|
+
scopeLabel: "global",
|
|
12
|
+
limit: 100,
|
|
13
|
+
used: 100,
|
|
14
|
+
percent: 100,
|
|
15
|
+
tier: "blocked-100",
|
|
16
|
+
},
|
|
17
|
+
{
|
|
18
|
+
scope: { kind: "team", teamId: "T1" },
|
|
19
|
+
scopeLabel: "team T1",
|
|
20
|
+
limit: 50,
|
|
21
|
+
used: 60,
|
|
22
|
+
percent: 120,
|
|
23
|
+
tier: "blocked-100",
|
|
24
|
+
},
|
|
25
|
+
{
|
|
26
|
+
scope: { kind: "repo", repoUrl: "https://github.com/x/y" },
|
|
27
|
+
scopeLabel: "repo x",
|
|
28
|
+
limit: 100,
|
|
29
|
+
used: 10,
|
|
30
|
+
percent: 10,
|
|
31
|
+
tier: "ok",
|
|
32
|
+
},
|
|
33
|
+
],
|
|
34
|
+
worstTier: "blocked-100",
|
|
35
|
+
promoteBlocked: true,
|
|
36
|
+
blockReason: "global at 100%",
|
|
37
|
+
activeCount: 0,
|
|
38
|
+
};
|
|
39
|
+
}
|
|
40
|
+
describe("budget.run_refused audit event", () => {
|
|
41
|
+
beforeEach(async () => {
|
|
42
|
+
await installTestProLicense("enterprise");
|
|
43
|
+
});
|
|
44
|
+
afterEach(async () => {
|
|
45
|
+
await restoreLicense();
|
|
46
|
+
});
|
|
47
|
+
it("writes one event per blocked scope from evaluateBudget", async () => {
|
|
48
|
+
const db = await createDb({ connectionString: ":memory:" });
|
|
49
|
+
const mockActions = {
|
|
50
|
+
evaluateBudget: vi.fn().mockResolvedValue(mixedBlockedEvaluation()),
|
|
51
|
+
recoverRetriableRuns: vi.fn().mockResolvedValue({ recovered: [], exhausted: [] }),
|
|
52
|
+
recoverStuckInProgressIssues: vi.fn().mockResolvedValue([]),
|
|
53
|
+
triageNewIssues: vi.fn().mockResolvedValue([]),
|
|
54
|
+
resolveApprovals: vi.fn().mockResolvedValue({ resolved: 0, stillPending: 0 }),
|
|
55
|
+
promoteReadyIssues: vi.fn().mockResolvedValue([]),
|
|
56
|
+
deprioritizeStaleIssues: vi.fn().mockResolvedValue([]),
|
|
57
|
+
cancelAbandonedIssues: vi.fn().mockResolvedValue([]),
|
|
58
|
+
postDigest: vi.fn().mockResolvedValue(undefined),
|
|
59
|
+
getActiveFileMaps: vi.fn().mockResolvedValue(new Map()),
|
|
60
|
+
predictConflict: vi.fn().mockResolvedValue({ overlapRisk: "none", likelyFiles: [], reasoning: "" }),
|
|
61
|
+
};
|
|
62
|
+
const scheduler = createPmScheduler({
|
|
63
|
+
config: {
|
|
64
|
+
enabled: true,
|
|
65
|
+
cronIntervalMs: 1800000,
|
|
66
|
+
triageBatchSize: 3,
|
|
67
|
+
maxInFlight: 3,
|
|
68
|
+
dailyTokenBudget: 100,
|
|
69
|
+
slackChannelId: "C123",
|
|
70
|
+
teamIds: ["team-1"],
|
|
71
|
+
},
|
|
72
|
+
db: db,
|
|
73
|
+
linearApiKey: "",
|
|
74
|
+
slackBotToken: "",
|
|
75
|
+
actions: mockActions,
|
|
76
|
+
});
|
|
77
|
+
await scheduler.tick();
|
|
78
|
+
const rows = await db.select().from(auditEvents);
|
|
79
|
+
const refused = rows.filter((r) => r.eventType === "budget.run_refused");
|
|
80
|
+
expect(refused).toHaveLength(2);
|
|
81
|
+
const scopes = refused.map((r) => r.scope).sort();
|
|
82
|
+
expect(scopes).toEqual(["global", "team:T1"]);
|
|
83
|
+
for (const row of refused) {
|
|
84
|
+
const payload = JSON.parse(row.payload);
|
|
85
|
+
expect(payload).toHaveProperty("scopeType");
|
|
86
|
+
expect(payload).toHaveProperty("tokensUsed");
|
|
87
|
+
expect(payload).toHaveProperty("limit");
|
|
88
|
+
expect(payload).toHaveProperty("utilization");
|
|
89
|
+
}
|
|
90
|
+
});
|
|
91
|
+
it("does not write audit events when no scopes are blocked", async () => {
|
|
92
|
+
const db = await createDb({ connectionString: ":memory:" });
|
|
93
|
+
const mockActions = {
|
|
94
|
+
evaluateBudget: vi.fn().mockResolvedValue({
|
|
95
|
+
scopes: [
|
|
96
|
+
{
|
|
97
|
+
scope: { kind: "global" },
|
|
98
|
+
scopeLabel: "global",
|
|
99
|
+
limit: 100,
|
|
100
|
+
used: 10,
|
|
101
|
+
percent: 10,
|
|
102
|
+
tier: "ok",
|
|
103
|
+
},
|
|
104
|
+
],
|
|
105
|
+
worstTier: "ok",
|
|
106
|
+
promoteBlocked: false,
|
|
107
|
+
activeCount: 0,
|
|
108
|
+
}),
|
|
109
|
+
recoverRetriableRuns: vi.fn().mockResolvedValue({ recovered: [], exhausted: [] }),
|
|
110
|
+
recoverStuckInProgressIssues: vi.fn().mockResolvedValue([]),
|
|
111
|
+
triageNewIssues: vi.fn().mockResolvedValue([]),
|
|
112
|
+
resolveApprovals: vi.fn().mockResolvedValue({ resolved: 0, stillPending: 0 }),
|
|
113
|
+
promoteReadyIssues: vi.fn().mockResolvedValue([]),
|
|
114
|
+
deprioritizeStaleIssues: vi.fn().mockResolvedValue([]),
|
|
115
|
+
cancelAbandonedIssues: vi.fn().mockResolvedValue([]),
|
|
116
|
+
postDigest: vi.fn().mockResolvedValue(undefined),
|
|
117
|
+
getActiveFileMaps: vi.fn().mockResolvedValue(new Map()),
|
|
118
|
+
predictConflict: vi.fn().mockResolvedValue({ overlapRisk: "none", likelyFiles: [], reasoning: "" }),
|
|
119
|
+
};
|
|
120
|
+
const scheduler = createPmScheduler({
|
|
121
|
+
config: {
|
|
122
|
+
enabled: true,
|
|
123
|
+
cronIntervalMs: 1800000,
|
|
124
|
+
triageBatchSize: 3,
|
|
125
|
+
maxInFlight: 3,
|
|
126
|
+
dailyTokenBudget: 100,
|
|
127
|
+
slackChannelId: "C123",
|
|
128
|
+
teamIds: ["team-1"],
|
|
129
|
+
},
|
|
130
|
+
db: db,
|
|
131
|
+
linearApiKey: "",
|
|
132
|
+
slackBotToken: "",
|
|
133
|
+
actions: mockActions,
|
|
134
|
+
});
|
|
135
|
+
await scheduler.tick();
|
|
136
|
+
const rows = await db.select().from(auditEvents);
|
|
137
|
+
const refused = rows.filter((r) => r.eventType === "budget.run_refused");
|
|
138
|
+
expect(refused).toHaveLength(0);
|
|
139
|
+
});
|
|
140
|
+
});
|
|
141
|
+
//# sourceMappingURL=pm-budget-refused-event.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"pm-budget-refused-event.test.js","sourceRoot":"","sources":["../../src/__tests__/pm-budget-refused-event.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -1,65 +1,161 @@
|
|
|
1
1
|
import { describe, it, expect } from "vitest";
|
|
2
|
-
import {
|
|
3
|
-
/**
|
|
4
|
-
* Mock DB that simulates a Drizzle query builder chain.
|
|
5
|
-
* budget.ts now uses Drizzle select().from().where() instead of raw SQL.
|
|
6
|
-
*/
|
|
2
|
+
import { evaluateBudget } from "../pm/budget.js";
|
|
7
3
|
function mockDb(rows) {
|
|
8
|
-
const mapped = rows.length > 0 ? {
|
|
9
|
-
totalIn: rows[0].totalIn ?? 0,
|
|
10
|
-
totalOut: rows[0].totalOut ?? 0,
|
|
11
|
-
activeCount: rows[0].activeCount ?? 0,
|
|
12
|
-
} : { totalIn: 0, totalOut: 0, activeCount: 0 };
|
|
13
4
|
const chain = {
|
|
14
5
|
select: () => chain,
|
|
15
6
|
from: () => chain,
|
|
16
|
-
where: () =>
|
|
7
|
+
where: () => chain,
|
|
8
|
+
groupBy: () => Promise.resolve(rows),
|
|
17
9
|
};
|
|
18
10
|
return chain;
|
|
19
11
|
}
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
12
|
+
function baseConfig(overrides = {}) {
|
|
13
|
+
return {
|
|
14
|
+
enabled: true,
|
|
15
|
+
dailyTokenBudget: 5_000_000,
|
|
16
|
+
slackChannelId: "C_TEST",
|
|
17
|
+
teamIds: ["team-a"],
|
|
18
|
+
maxInFlight: 3,
|
|
19
|
+
cronIntervalMs: 1_800_000,
|
|
20
|
+
triageBatchSize: 3,
|
|
21
|
+
stuckIssueRecovery: true,
|
|
22
|
+
stuckIssueTargetState: "Backlog",
|
|
23
|
+
stuckIssueMaxPerTick: 5,
|
|
24
|
+
...overrides,
|
|
25
|
+
};
|
|
26
|
+
}
|
|
27
|
+
describe("evaluateBudget", () => {
|
|
28
|
+
it("returns ok for empty spend with default config", async () => {
|
|
29
|
+
const db = mockDb([]);
|
|
30
|
+
const result = await evaluateBudget({ db, config: baseConfig() });
|
|
31
|
+
expect(result.worstTier).toBe("ok");
|
|
28
32
|
expect(result.promoteBlocked).toBe(false);
|
|
33
|
+
expect(result.scopes).toHaveLength(1);
|
|
34
|
+
expect(result.scopes[0].scope.kind).toBe("global");
|
|
35
|
+
expect(result.scopes[0].used).toBe(0);
|
|
36
|
+
expect(result.scopes[0].percent).toBe(0);
|
|
37
|
+
expect(result.activeCount).toBe(0);
|
|
38
|
+
});
|
|
39
|
+
it("computes global scope from rows", async () => {
|
|
40
|
+
const db = mockDb([
|
|
41
|
+
{ linearTeamId: "team-a", repoUrl: "github.com/org/repo", totalTokens: 2_500_000, activeCount: 1 },
|
|
42
|
+
]);
|
|
43
|
+
const result = await evaluateBudget({ db, config: baseConfig() });
|
|
44
|
+
const global = result.scopes.find((s) => s.scope.kind === "global");
|
|
45
|
+
expect(global.used).toBe(2_500_000);
|
|
46
|
+
expect(global.percent).toBe(50);
|
|
47
|
+
expect(global.tier).toBe("warn-50");
|
|
29
48
|
expect(result.activeCount).toBe(1);
|
|
30
|
-
expect(result.tokenSpendPercent).toBeLessThan(80);
|
|
31
49
|
});
|
|
32
|
-
it("
|
|
33
|
-
const
|
|
34
|
-
|
|
50
|
+
it("tier transitions: 0/50/80/100", async () => {
|
|
51
|
+
const cases = [
|
|
52
|
+
[0, "ok"],
|
|
53
|
+
[49, "ok"],
|
|
54
|
+
[50, "warn-50"],
|
|
55
|
+
[79, "warn-50"],
|
|
56
|
+
[80, "warn-80"],
|
|
57
|
+
[99, "warn-80"],
|
|
58
|
+
[100, "blocked-100"],
|
|
59
|
+
[150, "blocked-100"],
|
|
60
|
+
];
|
|
61
|
+
for (const [percent, expected] of cases) {
|
|
62
|
+
const used = (5_000_000 * percent) / 100;
|
|
63
|
+
const db = mockDb([
|
|
64
|
+
{ linearTeamId: "team-a", repoUrl: "r", totalTokens: used, activeCount: 0 },
|
|
65
|
+
]);
|
|
66
|
+
const result = await evaluateBudget({ db, config: baseConfig() });
|
|
67
|
+
const global = result.scopes.find((s) => s.scope.kind === "global");
|
|
68
|
+
expect({ percent, tier: global.tier }).toEqual({ percent, tier: expected });
|
|
69
|
+
}
|
|
70
|
+
});
|
|
71
|
+
it("per-team scope uses perTeam override", async () => {
|
|
72
|
+
const db = mockDb([
|
|
73
|
+
{ linearTeamId: "team-a", repoUrl: "r", totalTokens: 1_600_000, activeCount: 0 },
|
|
74
|
+
]);
|
|
75
|
+
const result = await evaluateBudget({
|
|
35
76
|
db,
|
|
36
|
-
|
|
37
|
-
|
|
77
|
+
config: baseConfig({
|
|
78
|
+
budgets: { perTeam: { "team-a": 2_000_000 } },
|
|
79
|
+
}),
|
|
38
80
|
});
|
|
39
|
-
|
|
40
|
-
expect(
|
|
81
|
+
const teamScope = result.scopes.find((s) => s.scope.kind === "team" && s.scope.teamId === "team-a");
|
|
82
|
+
expect(teamScope).toBeDefined();
|
|
83
|
+
expect(teamScope.limit).toBe(2_000_000);
|
|
84
|
+
expect(teamScope.percent).toBe(80);
|
|
85
|
+
expect(teamScope.tier).toBe("warn-80");
|
|
86
|
+
});
|
|
87
|
+
it("per-team scope falls back to budgets.default when team not in perTeam", async () => {
|
|
88
|
+
const db = mockDb([
|
|
89
|
+
{ linearTeamId: "team-z", repoUrl: "r", totalTokens: 500_000, activeCount: 0 },
|
|
90
|
+
]);
|
|
91
|
+
const result = await evaluateBudget({
|
|
92
|
+
db,
|
|
93
|
+
config: baseConfig({
|
|
94
|
+
budgets: {
|
|
95
|
+
default: 1_000_000,
|
|
96
|
+
perTeam: { "team-a": 500_000 },
|
|
97
|
+
},
|
|
98
|
+
}),
|
|
99
|
+
});
|
|
100
|
+
const teamScope = result.scopes.find((s) => s.scope.kind === "team" && s.scope.teamId === "team-z");
|
|
101
|
+
expect(teamScope.limit).toBe(1_000_000);
|
|
102
|
+
expect(teamScope.percent).toBe(50);
|
|
103
|
+
});
|
|
104
|
+
it("per-repo scope uses perRepo override", async () => {
|
|
105
|
+
const db = mockDb([
|
|
106
|
+
{ linearTeamId: "team-a", repoUrl: "github.com/org/secret", totalTokens: 900_000, activeCount: 0 },
|
|
107
|
+
]);
|
|
108
|
+
const result = await evaluateBudget({
|
|
109
|
+
db,
|
|
110
|
+
config: baseConfig({
|
|
111
|
+
budgets: { perRepo: { "github.com/org/secret": 1_000_000 } },
|
|
112
|
+
}),
|
|
113
|
+
});
|
|
114
|
+
const repoScope = result.scopes.find((s) => s.scope.kind === "repo" && s.scope.repoUrl === "github.com/org/secret");
|
|
115
|
+
expect(repoScope.limit).toBe(1_000_000);
|
|
116
|
+
expect(repoScope.percent).toBe(90);
|
|
117
|
+
expect(repoScope.tier).toBe("warn-80");
|
|
41
118
|
});
|
|
42
|
-
it("
|
|
43
|
-
const db = mockDb([
|
|
44
|
-
|
|
119
|
+
it("both per-team and per-repo can apply — worstTier is the max", async () => {
|
|
120
|
+
const db = mockDb([
|
|
121
|
+
{ linearTeamId: "team-a", repoUrl: "github.com/org/secret", totalTokens: 5_200_000, activeCount: 1 },
|
|
122
|
+
]);
|
|
123
|
+
const result = await evaluateBudget({
|
|
45
124
|
db,
|
|
46
|
-
|
|
47
|
-
|
|
125
|
+
config: baseConfig({
|
|
126
|
+
dailyTokenBudget: 20_000_000, // keep global well below 100% so only repo blocks
|
|
127
|
+
budgets: {
|
|
128
|
+
perTeam: { "team-a": 10_000_000 }, // 52% — warn-50
|
|
129
|
+
perRepo: { "github.com/org/secret": 5_000_000 }, // 104% — blocked-100
|
|
130
|
+
},
|
|
131
|
+
}),
|
|
48
132
|
});
|
|
133
|
+
expect(result.worstTier).toBe("blocked-100");
|
|
49
134
|
expect(result.promoteBlocked).toBe(true);
|
|
50
|
-
expect(result.
|
|
51
|
-
expect(result.tokenSpendPercent).toBeGreaterThanOrEqual(80);
|
|
135
|
+
expect(result.blockReason).toContain("github.com/org/secret");
|
|
52
136
|
});
|
|
53
|
-
it("
|
|
54
|
-
const db = mockDb([
|
|
55
|
-
|
|
137
|
+
it("legacy rows with NULL linearTeamId contribute to global only", async () => {
|
|
138
|
+
const db = mockDb([
|
|
139
|
+
{ linearTeamId: null, repoUrl: "r1", totalTokens: 1_000_000, activeCount: 0 },
|
|
140
|
+
{ linearTeamId: "team-a", repoUrl: "r1", totalTokens: 500_000, activeCount: 0 },
|
|
141
|
+
]);
|
|
142
|
+
const result = await evaluateBudget({
|
|
56
143
|
db,
|
|
57
|
-
|
|
58
|
-
dailyTokenBudget: 5000000,
|
|
144
|
+
config: baseConfig({ budgets: { perTeam: { "team-a": 2_000_000 } } }),
|
|
59
145
|
});
|
|
60
|
-
|
|
61
|
-
expect(
|
|
62
|
-
|
|
146
|
+
const global = result.scopes.find((s) => s.scope.kind === "global");
|
|
147
|
+
expect(global.used).toBe(1_500_000);
|
|
148
|
+
const teamScope = result.scopes.find((s) => s.scope.kind === "team" && s.scope.teamId === "team-a");
|
|
149
|
+
expect(teamScope.used).toBe(500_000);
|
|
150
|
+
});
|
|
151
|
+
it("blocks when any scope is at 100%", async () => {
|
|
152
|
+
const db = mockDb([
|
|
153
|
+
{ linearTeamId: "team-a", repoUrl: "r", totalTokens: 5_000_000, activeCount: 0 },
|
|
154
|
+
]);
|
|
155
|
+
const result = await evaluateBudget({ db, config: baseConfig() });
|
|
156
|
+
expect(result.promoteBlocked).toBe(true);
|
|
157
|
+
expect(result.blockReason).toBeDefined();
|
|
158
|
+
expect(result.blockReason).toContain("global");
|
|
63
159
|
});
|
|
64
160
|
});
|
|
65
161
|
//# sourceMappingURL=pm-budget.test.js.map
|