@urateam/core 0.1.3 → 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.
Files changed (478) hide show
  1. package/dist/__tests__/audit/audit-e2e.test.d.ts +2 -0
  2. package/dist/__tests__/audit/audit-e2e.test.d.ts.map +1 -0
  3. package/dist/__tests__/audit/audit-e2e.test.js +56 -0
  4. package/dist/__tests__/audit/audit-e2e.test.js.map +1 -0
  5. package/dist/__tests__/audit/csv.test.d.ts +2 -0
  6. package/dist/__tests__/audit/csv.test.d.ts.map +1 -0
  7. package/dist/__tests__/audit/csv.test.js +134 -0
  8. package/dist/__tests__/audit/csv.test.js.map +1 -0
  9. package/dist/__tests__/audit/events.test.d.ts +2 -0
  10. package/dist/__tests__/audit/events.test.d.ts.map +1 -0
  11. package/dist/__tests__/audit/events.test.js +45 -0
  12. package/dist/__tests__/audit/events.test.js.map +1 -0
  13. package/dist/__tests__/audit/policy-events.test.d.ts +2 -0
  14. package/dist/__tests__/audit/policy-events.test.d.ts.map +1 -0
  15. package/dist/__tests__/audit/policy-events.test.js +49 -0
  16. package/dist/__tests__/audit/policy-events.test.js.map +1 -0
  17. package/dist/__tests__/audit/projection.test.d.ts +2 -0
  18. package/dist/__tests__/audit/projection.test.d.ts.map +1 -0
  19. package/dist/__tests__/audit/projection.test.js +76 -0
  20. package/dist/__tests__/audit/projection.test.js.map +1 -0
  21. package/dist/__tests__/audit/rbac-events.test.d.ts +2 -0
  22. package/dist/__tests__/audit/rbac-events.test.d.ts.map +1 -0
  23. package/dist/__tests__/audit/rbac-events.test.js +56 -0
  24. package/dist/__tests__/audit/rbac-events.test.js.map +1 -0
  25. package/dist/__tests__/audit/reader.test.d.ts +2 -0
  26. package/dist/__tests__/audit/reader.test.d.ts.map +1 -0
  27. package/dist/__tests__/audit/reader.test.js +133 -0
  28. package/dist/__tests__/audit/reader.test.js.map +1 -0
  29. package/dist/__tests__/audit/retention.test.d.ts +2 -0
  30. package/dist/__tests__/audit/retention.test.d.ts.map +1 -0
  31. package/dist/__tests__/audit/retention.test.js +48 -0
  32. package/dist/__tests__/audit/retention.test.js.map +1 -0
  33. package/dist/__tests__/audit/sso-events.test.d.ts +2 -0
  34. package/dist/__tests__/audit/sso-events.test.d.ts.map +1 -0
  35. package/dist/__tests__/audit/sso-events.test.js +49 -0
  36. package/dist/__tests__/audit/sso-events.test.js.map +1 -0
  37. package/dist/__tests__/audit/writer.test.d.ts +2 -0
  38. package/dist/__tests__/audit/writer.test.d.ts.map +1 -0
  39. package/dist/__tests__/audit/writer.test.js +67 -0
  40. package/dist/__tests__/audit/writer.test.js.map +1 -0
  41. package/dist/__tests__/audit-immutability.test.d.ts +2 -0
  42. package/dist/__tests__/audit-immutability.test.d.ts.map +1 -0
  43. package/dist/__tests__/audit-immutability.test.js +61 -0
  44. package/dist/__tests__/audit-immutability.test.js.map +1 -0
  45. package/dist/__tests__/audit-types.test.d.ts +2 -0
  46. package/dist/__tests__/audit-types.test.d.ts.map +1 -0
  47. package/dist/__tests__/audit-types.test.js +35 -0
  48. package/dist/__tests__/audit-types.test.js.map +1 -0
  49. package/dist/__tests__/auth/session-store.test.d.ts +2 -0
  50. package/dist/__tests__/auth/session-store.test.d.ts.map +1 -0
  51. package/dist/__tests__/auth/session-store.test.js +65 -0
  52. package/dist/__tests__/auth/session-store.test.js.map +1 -0
  53. package/dist/__tests__/auth/sso-config.test.d.ts +2 -0
  54. package/dist/__tests__/auth/sso-config.test.d.ts.map +1 -0
  55. package/dist/__tests__/auth/sso-config.test.js +75 -0
  56. package/dist/__tests__/auth/sso-config.test.js.map +1 -0
  57. package/dist/__tests__/auth/sso-feature-flag.test.d.ts +2 -0
  58. package/dist/__tests__/auth/sso-feature-flag.test.d.ts.map +1 -0
  59. package/dist/__tests__/auth/sso-feature-flag.test.js +33 -0
  60. package/dist/__tests__/auth/sso-feature-flag.test.js.map +1 -0
  61. package/dist/__tests__/auth/user-role-round-trip.test.d.ts +2 -0
  62. package/dist/__tests__/auth/user-role-round-trip.test.d.ts.map +1 -0
  63. package/dist/__tests__/auth/user-role-round-trip.test.js +30 -0
  64. package/dist/__tests__/auth/user-role-round-trip.test.js.map +1 -0
  65. package/dist/__tests__/auth/user-store.test.d.ts +2 -0
  66. package/dist/__tests__/auth/user-store.test.d.ts.map +1 -0
  67. package/dist/__tests__/auth/user-store.test.js +119 -0
  68. package/dist/__tests__/auth/user-store.test.js.map +1 -0
  69. package/dist/__tests__/auth/workos-client.test.d.ts +2 -0
  70. package/dist/__tests__/auth/workos-client.test.d.ts.map +1 -0
  71. package/dist/__tests__/auth/workos-client.test.js +17 -0
  72. package/dist/__tests__/auth/workos-client.test.js.map +1 -0
  73. package/dist/__tests__/auto-merge.test.js +2 -1
  74. package/dist/__tests__/auto-merge.test.js.map +1 -1
  75. package/dist/__tests__/cost/aggregate.test.d.ts +2 -0
  76. package/dist/__tests__/cost/aggregate.test.d.ts.map +1 -0
  77. package/dist/__tests__/cost/aggregate.test.js +358 -0
  78. package/dist/__tests__/cost/aggregate.test.js.map +1 -0
  79. package/dist/__tests__/cost/csv.test.d.ts +2 -0
  80. package/dist/__tests__/cost/csv.test.d.ts.map +1 -0
  81. package/dist/__tests__/cost/csv.test.js +109 -0
  82. package/dist/__tests__/cost/csv.test.js.map +1 -0
  83. package/dist/__tests__/cost/per-run.test.d.ts +2 -0
  84. package/dist/__tests__/cost/per-run.test.d.ts.map +1 -0
  85. package/dist/__tests__/cost/per-run.test.js +64 -0
  86. package/dist/__tests__/cost/per-run.test.js.map +1 -0
  87. package/dist/__tests__/cost/rates.test.d.ts +2 -0
  88. package/dist/__tests__/cost/rates.test.d.ts.map +1 -0
  89. package/dist/__tests__/cost/rates.test.js +47 -0
  90. package/dist/__tests__/cost/rates.test.js.map +1 -0
  91. package/dist/__tests__/cost/rollup.test.d.ts +2 -0
  92. package/dist/__tests__/cost/rollup.test.d.ts.map +1 -0
  93. package/dist/__tests__/cost/rollup.test.js +187 -0
  94. package/dist/__tests__/cost/rollup.test.js.map +1 -0
  95. package/dist/__tests__/cost-defensive.test.d.ts +2 -0
  96. package/dist/__tests__/cost-defensive.test.d.ts.map +1 -0
  97. package/dist/__tests__/cost-defensive.test.js +123 -0
  98. package/dist/__tests__/cost-defensive.test.js.map +1 -0
  99. package/dist/__tests__/cost-integration.test.d.ts +2 -0
  100. package/dist/__tests__/cost-integration.test.d.ts.map +1 -0
  101. package/dist/__tests__/cost-integration.test.js +80 -0
  102. package/dist/__tests__/cost-integration.test.js.map +1 -0
  103. package/dist/__tests__/cost-license.test.d.ts +2 -0
  104. package/dist/__tests__/cost-license.test.d.ts.map +1 -0
  105. package/dist/__tests__/cost-license.test.js +23 -0
  106. package/dist/__tests__/cost-license.test.js.map +1 -0
  107. package/dist/__tests__/cost-types.test.d.ts +2 -0
  108. package/dist/__tests__/cost-types.test.d.ts.map +1 -0
  109. package/dist/__tests__/cost-types.test.js +65 -0
  110. package/dist/__tests__/cost-types.test.js.map +1 -0
  111. package/dist/__tests__/db-audit-schema.test.d.ts +2 -0
  112. package/dist/__tests__/db-audit-schema.test.d.ts.map +1 -0
  113. package/dist/__tests__/db-audit-schema.test.js +35 -0
  114. package/dist/__tests__/db-audit-schema.test.js.map +1 -0
  115. package/dist/__tests__/db-cost-rollups-schema.test.d.ts +2 -0
  116. package/dist/__tests__/db-cost-rollups-schema.test.d.ts.map +1 -0
  117. package/dist/__tests__/db-cost-rollups-schema.test.js +28 -0
  118. package/dist/__tests__/db-cost-rollups-schema.test.js.map +1 -0
  119. package/dist/__tests__/db-rbac-schema.test.d.ts +2 -0
  120. package/dist/__tests__/db-rbac-schema.test.d.ts.map +1 -0
  121. package/dist/__tests__/db-rbac-schema.test.js +32 -0
  122. package/dist/__tests__/db-rbac-schema.test.js.map +1 -0
  123. package/dist/__tests__/db-sso-schema.test.d.ts +2 -0
  124. package/dist/__tests__/db-sso-schema.test.d.ts.map +1 -0
  125. package/dist/__tests__/db-sso-schema.test.js +48 -0
  126. package/dist/__tests__/db-sso-schema.test.js.map +1 -0
  127. package/dist/__tests__/helpers/license.d.ts +3 -0
  128. package/dist/__tests__/helpers/license.d.ts.map +1 -0
  129. package/dist/__tests__/helpers/license.js +74 -0
  130. package/dist/__tests__/helpers/license.js.map +1 -0
  131. package/dist/__tests__/license-audit-event.test.d.ts +2 -0
  132. package/dist/__tests__/license-audit-event.test.d.ts.map +1 -0
  133. package/dist/__tests__/license-audit-event.test.js +60 -0
  134. package/dist/__tests__/license-audit-event.test.js.map +1 -0
  135. package/dist/__tests__/license-end-to-end.test.d.ts +2 -0
  136. package/dist/__tests__/license-end-to-end.test.d.ts.map +1 -0
  137. package/dist/__tests__/license-end-to-end.test.js +75 -0
  138. package/dist/__tests__/license-end-to-end.test.js.map +1 -0
  139. package/dist/__tests__/license-public-key.test.d.ts +2 -0
  140. package/dist/__tests__/license-public-key.test.d.ts.map +1 -0
  141. package/dist/__tests__/license-public-key.test.js +18 -0
  142. package/dist/__tests__/license-public-key.test.js.map +1 -0
  143. package/dist/__tests__/license.test.js +195 -32
  144. package/dist/__tests__/license.test.js.map +1 -1
  145. package/dist/__tests__/pm-action-audit-events.test.d.ts +2 -0
  146. package/dist/__tests__/pm-action-audit-events.test.d.ts.map +1 -0
  147. package/dist/__tests__/pm-action-audit-events.test.js +185 -0
  148. package/dist/__tests__/pm-action-audit-events.test.js.map +1 -0
  149. package/dist/__tests__/pm-approvals.test.js +1 -1
  150. package/dist/__tests__/pm-approvals.test.js.map +1 -1
  151. package/dist/__tests__/pm-audit-retention-step.test.d.ts +2 -0
  152. package/dist/__tests__/pm-audit-retention-step.test.d.ts.map +1 -0
  153. package/dist/__tests__/pm-audit-retention-step.test.js +126 -0
  154. package/dist/__tests__/pm-audit-retention-step.test.js.map +1 -0
  155. package/dist/__tests__/pm-budget-alerts.test.d.ts +2 -0
  156. package/dist/__tests__/pm-budget-alerts.test.d.ts.map +1 -0
  157. package/dist/__tests__/pm-budget-alerts.test.js +128 -0
  158. package/dist/__tests__/pm-budget-alerts.test.js.map +1 -0
  159. package/dist/__tests__/pm-budget-refused-event.test.d.ts +2 -0
  160. package/dist/__tests__/pm-budget-refused-event.test.d.ts.map +1 -0
  161. package/dist/__tests__/pm-budget-refused-event.test.js +141 -0
  162. package/dist/__tests__/pm-budget-refused-event.test.js.map +1 -0
  163. package/dist/__tests__/pm-budget.test.js +138 -42
  164. package/dist/__tests__/pm-budget.test.js.map +1 -1
  165. package/dist/__tests__/pm-cost-rollup-step.test.d.ts +2 -0
  166. package/dist/__tests__/pm-cost-rollup-step.test.d.ts.map +1 -0
  167. package/dist/__tests__/pm-cost-rollup-step.test.js +113 -0
  168. package/dist/__tests__/pm-cost-rollup-step.test.js.map +1 -0
  169. package/dist/__tests__/pm-scheduler.test.js +48 -21
  170. package/dist/__tests__/pm-scheduler.test.js.map +1 -1
  171. package/dist/__tests__/pm-slack.test.js +37 -0
  172. package/dist/__tests__/pm-slack.test.js.map +1 -1
  173. package/dist/__tests__/pm-sso-prune-step.test.d.ts +2 -0
  174. package/dist/__tests__/pm-sso-prune-step.test.d.ts.map +1 -0
  175. package/dist/__tests__/pm-sso-prune-step.test.js +103 -0
  176. package/dist/__tests__/pm-sso-prune-step.test.js.map +1 -0
  177. package/dist/__tests__/pm-types.test.js +130 -1
  178. package/dist/__tests__/pm-types.test.js.map +1 -1
  179. package/dist/__tests__/policy/auto-merge-reviewer-gate.test.d.ts +2 -0
  180. package/dist/__tests__/policy/auto-merge-reviewer-gate.test.d.ts.map +1 -0
  181. package/dist/__tests__/policy/auto-merge-reviewer-gate.test.js +50 -0
  182. package/dist/__tests__/policy/auto-merge-reviewer-gate.test.js.map +1 -0
  183. package/dist/__tests__/policy/cost-gate.test.d.ts +2 -0
  184. package/dist/__tests__/policy/cost-gate.test.d.ts.map +1 -0
  185. package/dist/__tests__/policy/cost-gate.test.js +27 -0
  186. package/dist/__tests__/policy/cost-gate.test.js.map +1 -0
  187. package/dist/__tests__/policy/evaluate.test.d.ts +2 -0
  188. package/dist/__tests__/policy/evaluate.test.d.ts.map +1 -0
  189. package/dist/__tests__/policy/evaluate.test.js +105 -0
  190. package/dist/__tests__/policy/evaluate.test.js.map +1 -0
  191. package/dist/__tests__/policy/override.test.d.ts +2 -0
  192. package/dist/__tests__/policy/override.test.d.ts.map +1 -0
  193. package/dist/__tests__/policy/override.test.js +26 -0
  194. package/dist/__tests__/policy/override.test.js.map +1 -0
  195. package/dist/__tests__/policy/path-gate.test.d.ts +2 -0
  196. package/dist/__tests__/policy/path-gate.test.d.ts.map +1 -0
  197. package/dist/__tests__/policy/path-gate.test.js +31 -0
  198. package/dist/__tests__/policy/path-gate.test.js.map +1 -0
  199. package/dist/__tests__/policy/pr-reviewer-wiring.test.d.ts +2 -0
  200. package/dist/__tests__/policy/pr-reviewer-wiring.test.d.ts.map +1 -0
  201. package/dist/__tests__/policy/pr-reviewer-wiring.test.js +24 -0
  202. package/dist/__tests__/policy/pr-reviewer-wiring.test.js.map +1 -0
  203. package/dist/__tests__/policy/reviewer-gate.test.d.ts +2 -0
  204. package/dist/__tests__/policy/reviewer-gate.test.d.ts.map +1 -0
  205. package/dist/__tests__/policy/reviewer-gate.test.js +125 -0
  206. package/dist/__tests__/policy/reviewer-gate.test.js.map +1 -0
  207. package/dist/__tests__/policy-license.test.d.ts +2 -0
  208. package/dist/__tests__/policy-license.test.d.ts.map +1 -0
  209. package/dist/__tests__/policy-license.test.js +31 -0
  210. package/dist/__tests__/policy-license.test.js.map +1 -0
  211. package/dist/__tests__/policy-types.test.d.ts +2 -0
  212. package/dist/__tests__/policy-types.test.d.ts.map +1 -0
  213. package/dist/__tests__/policy-types.test.js +49 -0
  214. package/dist/__tests__/policy-types.test.js.map +1 -0
  215. package/dist/__tests__/ralph-gate.test.js +6 -4
  216. package/dist/__tests__/ralph-gate.test.js.map +1 -1
  217. package/dist/__tests__/rbac/bootstrap.test.d.ts +2 -0
  218. package/dist/__tests__/rbac/bootstrap.test.d.ts.map +1 -0
  219. package/dist/__tests__/rbac/bootstrap.test.js +62 -0
  220. package/dist/__tests__/rbac/bootstrap.test.js.map +1 -0
  221. package/dist/__tests__/rbac/matrix.test.d.ts +2 -0
  222. package/dist/__tests__/rbac/matrix.test.d.ts.map +1 -0
  223. package/dist/__tests__/rbac/matrix.test.js +62 -0
  224. package/dist/__tests__/rbac/matrix.test.js.map +1 -0
  225. package/dist/__tests__/rbac/user-role-store.test.d.ts +2 -0
  226. package/dist/__tests__/rbac/user-role-store.test.d.ts.map +1 -0
  227. package/dist/__tests__/rbac/user-role-store.test.js +78 -0
  228. package/dist/__tests__/rbac/user-role-store.test.js.map +1 -0
  229. package/dist/__tests__/rbac-defensive.test.d.ts +2 -0
  230. package/dist/__tests__/rbac-defensive.test.d.ts.map +1 -0
  231. package/dist/__tests__/rbac-defensive.test.js +29 -0
  232. package/dist/__tests__/rbac-defensive.test.js.map +1 -0
  233. package/dist/__tests__/rbac-integration.test.d.ts +2 -0
  234. package/dist/__tests__/rbac-integration.test.d.ts.map +1 -0
  235. package/dist/__tests__/rbac-integration.test.js +58 -0
  236. package/dist/__tests__/rbac-integration.test.js.map +1 -0
  237. package/dist/__tests__/rbac-license.test.d.ts +2 -0
  238. package/dist/__tests__/rbac-license.test.d.ts.map +1 -0
  239. package/dist/__tests__/rbac-license.test.js +27 -0
  240. package/dist/__tests__/rbac-license.test.js.map +1 -0
  241. package/dist/__tests__/reproduce-bec113-pagination-warning.test.js +14 -2
  242. package/dist/__tests__/reproduce-bec113-pagination-warning.test.js.map +1 -1
  243. package/dist/__tests__/reproduce-bec62.test.js +10 -3
  244. package/dist/__tests__/reproduce-bec62.test.js.map +1 -1
  245. package/dist/__tests__/reproduce-bec91-stuck-in-progress.test.js +24 -4
  246. package/dist/__tests__/reproduce-bec91-stuck-in-progress.test.js.map +1 -1
  247. package/dist/__tests__/server.test.js +46 -2
  248. package/dist/__tests__/server.test.js.map +1 -1
  249. package/dist/__tests__/sso-license.test.d.ts +2 -0
  250. package/dist/__tests__/sso-license.test.d.ts.map +1 -0
  251. package/dist/__tests__/sso-license.test.js +49 -0
  252. package/dist/__tests__/sso-license.test.js.map +1 -0
  253. package/dist/__tests__/start-todo.test.js +5 -0
  254. package/dist/__tests__/start-todo.test.js.map +1 -1
  255. package/dist/__tests__/webhook-handler.test.js +138 -0
  256. package/dist/__tests__/webhook-handler.test.js.map +1 -1
  257. package/dist/audit/config-fingerprint.d.ts +10 -0
  258. package/dist/audit/config-fingerprint.d.ts.map +1 -0
  259. package/dist/audit/config-fingerprint.js +20 -0
  260. package/dist/audit/config-fingerprint.js.map +1 -0
  261. package/dist/audit/csv.d.ts +8 -0
  262. package/dist/audit/csv.d.ts.map +1 -0
  263. package/dist/audit/csv.js +56 -0
  264. package/dist/audit/csv.js.map +1 -0
  265. package/dist/audit/events.d.ts +107 -0
  266. package/dist/audit/events.d.ts.map +1 -0
  267. package/dist/audit/events.js +221 -0
  268. package/dist/audit/events.js.map +1 -0
  269. package/dist/audit/index.d.ts +7 -0
  270. package/dist/audit/index.d.ts.map +1 -0
  271. package/dist/audit/index.js +7 -0
  272. package/dist/audit/index.js.map +1 -0
  273. package/dist/audit/projection.d.ts +38 -0
  274. package/dist/audit/projection.d.ts.map +1 -0
  275. package/dist/audit/projection.js +109 -0
  276. package/dist/audit/projection.js.map +1 -0
  277. package/dist/audit/reader.d.ts +42 -0
  278. package/dist/audit/reader.d.ts.map +1 -0
  279. package/dist/audit/reader.js +243 -0
  280. package/dist/audit/reader.js.map +1 -0
  281. package/dist/audit/retention.d.ts +12 -0
  282. package/dist/audit/retention.d.ts.map +1 -0
  283. package/dist/audit/retention.js +23 -0
  284. package/dist/audit/retention.js.map +1 -0
  285. package/dist/audit/writer.d.ts +26 -0
  286. package/dist/audit/writer.d.ts.map +1 -0
  287. package/dist/audit/writer.js +54 -0
  288. package/dist/audit/writer.js.map +1 -0
  289. package/dist/auth/index.d.ts +5 -0
  290. package/dist/auth/index.d.ts.map +1 -0
  291. package/dist/auth/index.js +5 -0
  292. package/dist/auth/index.js.map +1 -0
  293. package/dist/auth/session-store.d.ts +17 -0
  294. package/dist/auth/session-store.d.ts.map +1 -0
  295. package/dist/auth/session-store.js +59 -0
  296. package/dist/auth/session-store.js.map +1 -0
  297. package/dist/auth/sso-config.d.ts +14 -0
  298. package/dist/auth/sso-config.d.ts.map +1 -0
  299. package/dist/auth/sso-config.js +50 -0
  300. package/dist/auth/sso-config.js.map +1 -0
  301. package/dist/auth/user-store.d.ts +33 -0
  302. package/dist/auth/user-store.d.ts.map +1 -0
  303. package/dist/auth/user-store.js +71 -0
  304. package/dist/auth/user-store.js.map +1 -0
  305. package/dist/auth/workos-client.d.ts +46 -0
  306. package/dist/auth/workos-client.d.ts.map +1 -0
  307. package/dist/auth/workos-client.js +66 -0
  308. package/dist/auth/workos-client.js.map +1 -0
  309. package/dist/cost/aggregate.d.ts +49 -0
  310. package/dist/cost/aggregate.d.ts.map +1 -0
  311. package/dist/cost/aggregate.js +364 -0
  312. package/dist/cost/aggregate.js.map +1 -0
  313. package/dist/cost/csv.d.ts +12 -0
  314. package/dist/cost/csv.d.ts.map +1 -0
  315. package/dist/cost/csv.js +79 -0
  316. package/dist/cost/csv.js.map +1 -0
  317. package/dist/cost/index.d.ts +7 -0
  318. package/dist/cost/index.d.ts.map +1 -0
  319. package/dist/cost/index.js +7 -0
  320. package/dist/cost/index.js.map +1 -0
  321. package/dist/cost/per-run.d.ts +31 -0
  322. package/dist/cost/per-run.d.ts.map +1 -0
  323. package/dist/cost/per-run.js +35 -0
  324. package/dist/cost/per-run.js.map +1 -0
  325. package/dist/cost/rates.d.ts +27 -0
  326. package/dist/cost/rates.d.ts.map +1 -0
  327. package/dist/cost/rates.js +26 -0
  328. package/dist/cost/rates.js.map +1 -0
  329. package/dist/cost/rollup.d.ts +17 -0
  330. package/dist/cost/rollup.d.ts.map +1 -0
  331. package/dist/cost/rollup.js +164 -0
  332. package/dist/cost/rollup.js.map +1 -0
  333. package/dist/cost/types.d.ts +54 -0
  334. package/dist/cost/types.d.ts.map +1 -0
  335. package/dist/cost/types.js +2 -0
  336. package/dist/cost/types.js.map +1 -0
  337. package/dist/db/client.d.ts.map +1 -1
  338. package/dist/db/client.js +79 -1
  339. package/dist/db/client.js.map +1 -1
  340. package/dist/db/migrations/postgres/006_spend_caps.sql +14 -0
  341. package/dist/db/migrations/postgres/007_audit_events.sql +19 -0
  342. package/dist/db/migrations/postgres/008_cost_rollups.sql +19 -0
  343. package/dist/db/migrations/postgres/008_sso.sql +20 -0
  344. package/dist/db/migrations/sqlite/005_spend_caps.sql +16 -0
  345. package/dist/db/migrations/sqlite/006_audit_events.sql +19 -0
  346. package/dist/db/migrations/sqlite/007_cost_rollups.sql +19 -0
  347. package/dist/db/migrations/sqlite/007_sso.sql +20 -0
  348. package/dist/db/schema.d.ts +802 -0
  349. package/dist/db/schema.d.ts.map +1 -1
  350. package/dist/db/schema.js +80 -1
  351. package/dist/db/schema.js.map +1 -1
  352. package/dist/index.d.ts +7 -1
  353. package/dist/index.d.ts.map +1 -1
  354. package/dist/index.js +8 -1
  355. package/dist/index.js.map +1 -1
  356. package/dist/license-public-key.d.ts +15 -0
  357. package/dist/license-public-key.d.ts.map +1 -0
  358. package/dist/license-public-key.js +15 -0
  359. package/dist/license-public-key.js.map +1 -0
  360. package/dist/license.d.ts +30 -5
  361. package/dist/license.d.ts.map +1 -1
  362. package/dist/license.js +142 -12
  363. package/dist/license.js.map +1 -1
  364. package/dist/pipeline/runner.d.ts +1 -7
  365. package/dist/pipeline/runner.d.ts.map +1 -1
  366. package/dist/pipeline/runner.js +168 -21
  367. package/dist/pipeline/runner.js.map +1 -1
  368. package/dist/pm/actions/promote.d.ts +3 -0
  369. package/dist/pm/actions/promote.d.ts.map +1 -1
  370. package/dist/pm/actions/promote.js +12 -0
  371. package/dist/pm/actions/promote.js.map +1 -1
  372. package/dist/pm/actions/resolve-approvals.d.ts.map +1 -1
  373. package/dist/pm/actions/resolve-approvals.js +16 -0
  374. package/dist/pm/actions/resolve-approvals.js.map +1 -1
  375. package/dist/pm/actions/start-todo.d.ts +3 -0
  376. package/dist/pm/actions/start-todo.d.ts.map +1 -1
  377. package/dist/pm/actions/start-todo.js +5 -1
  378. package/dist/pm/actions/start-todo.js.map +1 -1
  379. package/dist/pm/actions/triage.d.ts +3 -0
  380. package/dist/pm/actions/triage.d.ts.map +1 -1
  381. package/dist/pm/actions/triage.js +8 -0
  382. package/dist/pm/actions/triage.js.map +1 -1
  383. package/dist/pm/budget-alerts.d.ts +18 -0
  384. package/dist/pm/budget-alerts.d.ts.map +1 -0
  385. package/dist/pm/budget-alerts.js +100 -0
  386. package/dist/pm/budget-alerts.js.map +1 -0
  387. package/dist/pm/budget.d.ts +28 -5
  388. package/dist/pm/budget.d.ts.map +1 -1
  389. package/dist/pm/budget.js +115 -39
  390. package/dist/pm/budget.js.map +1 -1
  391. package/dist/pm/scheduler.d.ts +8 -4
  392. package/dist/pm/scheduler.d.ts.map +1 -1
  393. package/dist/pm/scheduler.js +128 -5
  394. package/dist/pm/scheduler.js.map +1 -1
  395. package/dist/pm/slack.d.ts.map +1 -1
  396. package/dist/pm/slack.js +11 -0
  397. package/dist/pm/slack.js.map +1 -1
  398. package/dist/pm/types.d.ts +36 -0
  399. package/dist/pm/types.d.ts.map +1 -1
  400. package/dist/pm/types.js +12 -0
  401. package/dist/pm/types.js.map +1 -1
  402. package/dist/policy/cost-gate.d.ts +3 -0
  403. package/dist/policy/cost-gate.d.ts.map +1 -0
  404. package/dist/policy/cost-gate.js +11 -0
  405. package/dist/policy/cost-gate.js.map +1 -0
  406. package/dist/policy/evaluate.d.ts +32 -0
  407. package/dist/policy/evaluate.d.ts.map +1 -0
  408. package/dist/policy/evaluate.js +61 -0
  409. package/dist/policy/evaluate.js.map +1 -0
  410. package/dist/policy/index.d.ts +7 -0
  411. package/dist/policy/index.d.ts.map +1 -0
  412. package/dist/policy/index.js +7 -0
  413. package/dist/policy/index.js.map +1 -0
  414. package/dist/policy/override.d.ts +23 -0
  415. package/dist/policy/override.d.ts.map +1 -0
  416. package/dist/policy/override.js +26 -0
  417. package/dist/policy/override.js.map +1 -0
  418. package/dist/policy/path-gate.d.ts +3 -0
  419. package/dist/policy/path-gate.d.ts.map +1 -0
  420. package/dist/policy/path-gate.js +20 -0
  421. package/dist/policy/path-gate.js.map +1 -0
  422. package/dist/policy/reviewer-gate.d.ts +45 -0
  423. package/dist/policy/reviewer-gate.d.ts.map +1 -0
  424. package/dist/policy/reviewer-gate.js +77 -0
  425. package/dist/policy/reviewer-gate.js.map +1 -0
  426. package/dist/policy/types.d.ts +7 -0
  427. package/dist/policy/types.d.ts.map +1 -0
  428. package/dist/policy/types.js +2 -0
  429. package/dist/policy/types.js.map +1 -0
  430. package/dist/rbac/errors.d.ts +7 -0
  431. package/dist/rbac/errors.d.ts.map +1 -0
  432. package/dist/rbac/errors.js +13 -0
  433. package/dist/rbac/errors.js.map +1 -0
  434. package/dist/rbac/index.d.ts +5 -0
  435. package/dist/rbac/index.d.ts.map +1 -0
  436. package/dist/rbac/index.js +5 -0
  437. package/dist/rbac/index.js.map +1 -0
  438. package/dist/rbac/matrix.d.ts +35 -0
  439. package/dist/rbac/matrix.d.ts.map +1 -0
  440. package/dist/rbac/matrix.js +42 -0
  441. package/dist/rbac/matrix.js.map +1 -0
  442. package/dist/rbac/types.d.ts +2 -0
  443. package/dist/rbac/types.d.ts.map +1 -0
  444. package/dist/rbac/types.js +2 -0
  445. package/dist/rbac/types.js.map +1 -0
  446. package/dist/rbac/user-role-store.d.ts +27 -0
  447. package/dist/rbac/user-role-store.d.ts.map +1 -0
  448. package/dist/rbac/user-role-store.js +148 -0
  449. package/dist/rbac/user-role-store.js.map +1 -0
  450. package/dist/repo/git.d.ts +9 -0
  451. package/dist/repo/git.d.ts.map +1 -1
  452. package/dist/repo/git.js +11 -0
  453. package/dist/repo/git.js.map +1 -1
  454. package/dist/repo/github.d.ts +4 -0
  455. package/dist/repo/github.d.ts.map +1 -1
  456. package/dist/repo/github.js +27 -0
  457. package/dist/repo/github.js.map +1 -1
  458. package/dist/repo/gitlab.d.ts +4 -0
  459. package/dist/repo/gitlab.d.ts.map +1 -1
  460. package/dist/repo/gitlab.js +7 -0
  461. package/dist/repo/gitlab.js.map +1 -1
  462. package/dist/server.d.ts +6 -0
  463. package/dist/server.d.ts.map +1 -1
  464. package/dist/server.js +16 -10
  465. package/dist/server.js.map +1 -1
  466. package/dist/types.d.ts +155 -0
  467. package/dist/types.d.ts.map +1 -1
  468. package/dist/types.js +83 -0
  469. package/dist/types.js.map +1 -1
  470. package/dist/util/glob.d.ts +7 -0
  471. package/dist/util/glob.d.ts.map +1 -0
  472. package/dist/util/glob.js +20 -0
  473. package/dist/util/glob.js.map +1 -0
  474. package/dist/webhook/handler.d.ts +3 -0
  475. package/dist/webhook/handler.d.ts.map +1 -1
  476. package/dist/webhook/handler.js +30 -1
  477. package/dist/webhook/handler.js.map +1 -1
  478. 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,2 @@
1
+ export {};
2
+ //# sourceMappingURL=pm-budget-alerts.test.d.ts.map
@@ -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,2 @@
1
+ export {};
2
+ //# sourceMappingURL=pm-budget-refused-event.test.d.ts.map
@@ -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 { checkBudgetGuards } from "../pm/budget.js";
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: () => Promise.resolve([mapped]),
7
+ where: () => chain,
8
+ groupBy: () => Promise.resolve(rows),
17
9
  };
18
10
  return chain;
19
11
  }
20
- describe("checkBudgetGuards", () => {
21
- it("returns promoteBlocked=false when under limits", async () => {
22
- const db = mockDb([{ totalIn: 100000, totalOut: 50000, activeCount: 1 }]);
23
- const result = await checkBudgetGuards({
24
- db,
25
- maxInFlight: 3,
26
- dailyTokenBudget: 5000000,
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("blocks when activeCount >= maxInFlight", async () => {
33
- const db = mockDb([{ totalIn: 100000, totalOut: 50000, activeCount: 3 }]);
34
- const result = await checkBudgetGuards({
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
- maxInFlight: 3,
37
- dailyTokenBudget: 5000000,
77
+ config: baseConfig({
78
+ budgets: { perTeam: { "team-a": 2_000_000 } },
79
+ }),
38
80
  });
39
- expect(result.promoteBlocked).toBe(true);
40
- expect(result.reason).toContain("maxInFlight");
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("blocks when token spend >= 80%", async () => {
43
- const db = mockDb([{ totalIn: 3000000, totalOut: 1100000, activeCount: 1 }]);
44
- const result = await checkBudgetGuards({
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
- maxInFlight: 3,
47
- dailyTokenBudget: 5000000,
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.reason).toContain("token budget");
51
- expect(result.tokenSpendPercent).toBeGreaterThanOrEqual(80);
135
+ expect(result.blockReason).toContain("github.com/org/secret");
52
136
  });
53
- it("handles empty DB rows gracefully", async () => {
54
- const db = mockDb([]);
55
- const result = await checkBudgetGuards({
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
- maxInFlight: 3,
58
- dailyTokenBudget: 5000000,
144
+ config: baseConfig({ budgets: { perTeam: { "team-a": 2_000_000 } } }),
59
145
  });
60
- expect(result.promoteBlocked).toBe(false);
61
- expect(result.activeCount).toBe(0);
62
- expect(result.dailyTokensUsed).toBe(0);
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