@dealdesk/server 2026.527.0-canary.0

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 (1311) hide show
  1. package/LICENSE +21 -0
  2. package/dist/__tests__/helpers/embedded-postgres.d.ts +2 -0
  3. package/dist/__tests__/helpers/embedded-postgres.d.ts.map +1 -0
  4. package/dist/__tests__/helpers/embedded-postgres.js +2 -0
  5. package/dist/__tests__/helpers/embedded-postgres.js.map +1 -0
  6. package/dist/adapters/builtin-adapter-types.d.ts +5 -0
  7. package/dist/adapters/builtin-adapter-types.d.ts.map +1 -0
  8. package/dist/adapters/builtin-adapter-types.js +18 -0
  9. package/dist/adapters/builtin-adapter-types.js.map +1 -0
  10. package/dist/adapters/codex-models.d.ts +5 -0
  11. package/dist/adapters/codex-models.d.ts.map +1 -0
  12. package/dist/adapters/codex-models.js +105 -0
  13. package/dist/adapters/codex-models.js.map +1 -0
  14. package/dist/adapters/cursor-models.d.ts +13 -0
  15. package/dist/adapters/cursor-models.d.ts.map +1 -0
  16. package/dist/adapters/cursor-models.js +148 -0
  17. package/dist/adapters/cursor-models.js.map +1 -0
  18. package/dist/adapters/http/execute.d.ts +3 -0
  19. package/dist/adapters/http/execute.d.ts.map +1 -0
  20. package/dist/adapters/http/execute.js +51 -0
  21. package/dist/adapters/http/execute.js.map +1 -0
  22. package/dist/adapters/http/execute.test.d.ts +2 -0
  23. package/dist/adapters/http/execute.test.d.ts.map +1 -0
  24. package/dist/adapters/http/execute.test.js +40 -0
  25. package/dist/adapters/http/execute.test.js.map +1 -0
  26. package/dist/adapters/http/index.d.ts +3 -0
  27. package/dist/adapters/http/index.d.ts.map +1 -0
  28. package/dist/adapters/http/index.js +20 -0
  29. package/dist/adapters/http/index.js.map +1 -0
  30. package/dist/adapters/http/test.d.ts +3 -0
  31. package/dist/adapters/http/test.d.ts.map +1 -0
  32. package/dist/adapters/http/test.js +106 -0
  33. package/dist/adapters/http/test.js.map +1 -0
  34. package/dist/adapters/index.d.ts +4 -0
  35. package/dist/adapters/index.d.ts.map +1 -0
  36. package/dist/adapters/index.js +3 -0
  37. package/dist/adapters/index.js.map +1 -0
  38. package/dist/adapters/plugin-loader.d.ts +28 -0
  39. package/dist/adapters/plugin-loader.d.ts.map +1 -0
  40. package/dist/adapters/plugin-loader.js +196 -0
  41. package/dist/adapters/plugin-loader.js.map +1 -0
  42. package/dist/adapters/process/execute.d.ts +3 -0
  43. package/dist/adapters/process/execute.d.ts.map +1 -0
  44. package/dist/adapters/process/execute.js +70 -0
  45. package/dist/adapters/process/execute.js.map +1 -0
  46. package/dist/adapters/process/index.d.ts +3 -0
  47. package/dist/adapters/process/index.d.ts.map +1 -0
  48. package/dist/adapters/process/index.js +23 -0
  49. package/dist/adapters/process/index.js.map +1 -0
  50. package/dist/adapters/process/test.d.ts +3 -0
  51. package/dist/adapters/process/test.d.ts.map +1 -0
  52. package/dist/adapters/process/test.js +77 -0
  53. package/dist/adapters/process/test.js.map +1 -0
  54. package/dist/adapters/registry.d.ts +69 -0
  55. package/dist/adapters/registry.d.ts.map +1 -0
  56. package/dist/adapters/registry.js +589 -0
  57. package/dist/adapters/registry.js.map +1 -0
  58. package/dist/adapters/types.d.ts +2 -0
  59. package/dist/adapters/types.d.ts.map +1 -0
  60. package/dist/adapters/types.js +2 -0
  61. package/dist/adapters/types.js.map +1 -0
  62. package/dist/adapters/utils.d.ts +43 -0
  63. package/dist/adapters/utils.d.ts.map +1 -0
  64. package/dist/adapters/utils.js +52 -0
  65. package/dist/adapters/utils.js.map +1 -0
  66. package/dist/agent-auth-jwt.d.ts +14 -0
  67. package/dist/agent-auth-jwt.d.ts.map +1 -0
  68. package/dist/agent-auth-jwt.js +117 -0
  69. package/dist/agent-auth-jwt.js.map +1 -0
  70. package/dist/app.d.ts +43 -0
  71. package/dist/app.d.ts.map +1 -0
  72. package/dist/app.js +426 -0
  73. package/dist/app.js.map +1 -0
  74. package/dist/attachment-types.d.ts +23 -0
  75. package/dist/attachment-types.d.ts.map +1 -0
  76. package/dist/attachment-types.js +91 -0
  77. package/dist/attachment-types.js.map +1 -0
  78. package/dist/auth/better-auth.d.ts +33 -0
  79. package/dist/auth/better-auth.d.ts.map +1 -0
  80. package/dist/auth/better-auth.js +133 -0
  81. package/dist/auth/better-auth.js.map +1 -0
  82. package/dist/board-claim.d.ts +23 -0
  83. package/dist/board-claim.d.ts.map +1 -0
  84. package/dist/board-claim.js +115 -0
  85. package/dist/board-claim.js.map +1 -0
  86. package/dist/config-file.d.ts +3 -0
  87. package/dist/config-file.d.ts.map +1 -0
  88. package/dist/config-file.js +16 -0
  89. package/dist/config-file.js.map +1 -0
  90. package/dist/config.d.ts +50 -0
  91. package/dist/config.d.ts.map +1 -0
  92. package/dist/config.js +233 -0
  93. package/dist/config.js.map +1 -0
  94. package/dist/deal-desk/enrichment/__tests__/apollo-client.test.d.ts +2 -0
  95. package/dist/deal-desk/enrichment/__tests__/apollo-client.test.d.ts.map +1 -0
  96. package/dist/deal-desk/enrichment/__tests__/apollo-client.test.js +168 -0
  97. package/dist/deal-desk/enrichment/__tests__/apollo-client.test.js.map +1 -0
  98. package/dist/deal-desk/enrichment/__tests__/apollo-config.test.d.ts +2 -0
  99. package/dist/deal-desk/enrichment/__tests__/apollo-config.test.d.ts.map +1 -0
  100. package/dist/deal-desk/enrichment/__tests__/apollo-config.test.js +63 -0
  101. package/dist/deal-desk/enrichment/__tests__/apollo-config.test.js.map +1 -0
  102. package/dist/deal-desk/enrichment/__tests__/resolve-contact-email.test.d.ts +2 -0
  103. package/dist/deal-desk/enrichment/__tests__/resolve-contact-email.test.d.ts.map +1 -0
  104. package/dist/deal-desk/enrichment/__tests__/resolve-contact-email.test.js +127 -0
  105. package/dist/deal-desk/enrichment/__tests__/resolve-contact-email.test.js.map +1 -0
  106. package/dist/deal-desk/enrichment/apollo-client.d.ts +57 -0
  107. package/dist/deal-desk/enrichment/apollo-client.d.ts.map +1 -0
  108. package/dist/deal-desk/enrichment/apollo-client.js +263 -0
  109. package/dist/deal-desk/enrichment/apollo-client.js.map +1 -0
  110. package/dist/deal-desk/enrichment/apollo-config.d.ts +56 -0
  111. package/dist/deal-desk/enrichment/apollo-config.d.ts.map +1 -0
  112. package/dist/deal-desk/enrichment/apollo-config.js +58 -0
  113. package/dist/deal-desk/enrichment/apollo-config.js.map +1 -0
  114. package/dist/deal-desk/enrichment/resolve-contact-email.d.ts +62 -0
  115. package/dist/deal-desk/enrichment/resolve-contact-email.d.ts.map +1 -0
  116. package/dist/deal-desk/enrichment/resolve-contact-email.js +199 -0
  117. package/dist/deal-desk/enrichment/resolve-contact-email.js.map +1 -0
  118. package/dist/deal-desk/gmail/__tests__/client-config.test.d.ts +2 -0
  119. package/dist/deal-desk/gmail/__tests__/client-config.test.d.ts.map +1 -0
  120. package/dist/deal-desk/gmail/__tests__/client-config.test.js +73 -0
  121. package/dist/deal-desk/gmail/__tests__/client-config.test.js.map +1 -0
  122. package/dist/deal-desk/gmail/__tests__/oauth.test.d.ts +2 -0
  123. package/dist/deal-desk/gmail/__tests__/oauth.test.d.ts.map +1 -0
  124. package/dist/deal-desk/gmail/__tests__/oauth.test.js +38 -0
  125. package/dist/deal-desk/gmail/__tests__/oauth.test.js.map +1 -0
  126. package/dist/deal-desk/gmail/__tests__/redirect-uri.test.d.ts +2 -0
  127. package/dist/deal-desk/gmail/__tests__/redirect-uri.test.d.ts.map +1 -0
  128. package/dist/deal-desk/gmail/__tests__/redirect-uri.test.js +74 -0
  129. package/dist/deal-desk/gmail/__tests__/redirect-uri.test.js.map +1 -0
  130. package/dist/deal-desk/gmail/__tests__/send.test.d.ts +2 -0
  131. package/dist/deal-desk/gmail/__tests__/send.test.d.ts.map +1 -0
  132. package/dist/deal-desk/gmail/__tests__/send.test.js +44 -0
  133. package/dist/deal-desk/gmail/__tests__/send.test.js.map +1 -0
  134. package/dist/deal-desk/gmail/__tests__/tokens.test.d.ts +2 -0
  135. package/dist/deal-desk/gmail/__tests__/tokens.test.d.ts.map +1 -0
  136. package/dist/deal-desk/gmail/__tests__/tokens.test.js +69 -0
  137. package/dist/deal-desk/gmail/__tests__/tokens.test.js.map +1 -0
  138. package/dist/deal-desk/gmail/client-config.d.ts +51 -0
  139. package/dist/deal-desk/gmail/client-config.d.ts.map +1 -0
  140. package/dist/deal-desk/gmail/client-config.js +38 -0
  141. package/dist/deal-desk/gmail/client-config.js.map +1 -0
  142. package/dist/deal-desk/gmail/oauth.d.ts +26 -0
  143. package/dist/deal-desk/gmail/oauth.d.ts.map +1 -0
  144. package/dist/deal-desk/gmail/oauth.js +50 -0
  145. package/dist/deal-desk/gmail/oauth.js.map +1 -0
  146. package/dist/deal-desk/gmail/redirect-uri.d.ts +8 -0
  147. package/dist/deal-desk/gmail/redirect-uri.d.ts.map +1 -0
  148. package/dist/deal-desk/gmail/redirect-uri.js +83 -0
  149. package/dist/deal-desk/gmail/redirect-uri.js.map +1 -0
  150. package/dist/deal-desk/gmail/send.d.ts +19 -0
  151. package/dist/deal-desk/gmail/send.d.ts.map +1 -0
  152. package/dist/deal-desk/gmail/send.js +39 -0
  153. package/dist/deal-desk/gmail/send.js.map +1 -0
  154. package/dist/deal-desk/gmail/tokens.d.ts +50 -0
  155. package/dist/deal-desk/gmail/tokens.d.ts.map +1 -0
  156. package/dist/deal-desk/gmail/tokens.js +50 -0
  157. package/dist/deal-desk/gmail/tokens.js.map +1 -0
  158. package/dist/deal-desk/intermediary-campaign.d.ts +4 -0
  159. package/dist/deal-desk/intermediary-campaign.d.ts.map +1 -0
  160. package/dist/deal-desk/intermediary-campaign.js +26 -0
  161. package/dist/deal-desk/intermediary-campaign.js.map +1 -0
  162. package/dist/deal-desk/seeds/role-templates.d.ts +10 -0
  163. package/dist/deal-desk/seeds/role-templates.d.ts.map +1 -0
  164. package/dist/deal-desk/seeds/role-templates.js +110 -0
  165. package/dist/deal-desk/seeds/role-templates.js.map +1 -0
  166. package/dist/deal-desk/seeds/seed-role-templates.d.ts +3 -0
  167. package/dist/deal-desk/seeds/seed-role-templates.d.ts.map +1 -0
  168. package/dist/deal-desk/seeds/seed-role-templates.js +39 -0
  169. package/dist/deal-desk/seeds/seed-role-templates.js.map +1 -0
  170. package/dist/deal-desk/target-service.d.ts +131 -0
  171. package/dist/deal-desk/target-service.d.ts.map +1 -0
  172. package/dist/deal-desk/target-service.js +169 -0
  173. package/dist/deal-desk/target-service.js.map +1 -0
  174. package/dist/deal-desk/tools/__tests__/apollo-api-key.test.d.ts +2 -0
  175. package/dist/deal-desk/tools/__tests__/apollo-api-key.test.d.ts.map +1 -0
  176. package/dist/deal-desk/tools/__tests__/apollo-api-key.test.js +122 -0
  177. package/dist/deal-desk/tools/__tests__/apollo-api-key.test.js.map +1 -0
  178. package/dist/deal-desk/tools/__tests__/create-contact.test.d.ts +2 -0
  179. package/dist/deal-desk/tools/__tests__/create-contact.test.d.ts.map +1 -0
  180. package/dist/deal-desk/tools/__tests__/create-contact.test.js +73 -0
  181. package/dist/deal-desk/tools/__tests__/create-contact.test.js.map +1 -0
  182. package/dist/deal-desk/tools/__tests__/deal-desk-tools.test.d.ts +2 -0
  183. package/dist/deal-desk/tools/__tests__/deal-desk-tools.test.d.ts.map +1 -0
  184. package/dist/deal-desk/tools/__tests__/deal-desk-tools.test.js +193 -0
  185. package/dist/deal-desk/tools/__tests__/deal-desk-tools.test.js.map +1 -0
  186. package/dist/deal-desk/tools/__tests__/email-accounts.test.d.ts +2 -0
  187. package/dist/deal-desk/tools/__tests__/email-accounts.test.d.ts.map +1 -0
  188. package/dist/deal-desk/tools/__tests__/email-accounts.test.js +71 -0
  189. package/dist/deal-desk/tools/__tests__/email-accounts.test.js.map +1 -0
  190. package/dist/deal-desk/tools/__tests__/enrich-contact.test.d.ts +2 -0
  191. package/dist/deal-desk/tools/__tests__/enrich-contact.test.d.ts.map +1 -0
  192. package/dist/deal-desk/tools/__tests__/enrich-contact.test.js +183 -0
  193. package/dist/deal-desk/tools/__tests__/enrich-contact.test.js.map +1 -0
  194. package/dist/deal-desk/tools/__tests__/gmail-client-config.test.d.ts +2 -0
  195. package/dist/deal-desk/tools/__tests__/gmail-client-config.test.d.ts.map +1 -0
  196. package/dist/deal-desk/tools/__tests__/gmail-client-config.test.js +67 -0
  197. package/dist/deal-desk/tools/__tests__/gmail-client-config.test.js.map +1 -0
  198. package/dist/deal-desk/tools/__tests__/helpers/where-introspection.d.ts +2 -0
  199. package/dist/deal-desk/tools/__tests__/helpers/where-introspection.d.ts.map +1 -0
  200. package/dist/deal-desk/tools/__tests__/helpers/where-introspection.js +29 -0
  201. package/dist/deal-desk/tools/__tests__/helpers/where-introspection.js.map +1 -0
  202. package/dist/deal-desk/tools/__tests__/outreach-approve.test.d.ts +2 -0
  203. package/dist/deal-desk/tools/__tests__/outreach-approve.test.d.ts.map +1 -0
  204. package/dist/deal-desk/tools/__tests__/outreach-approve.test.js +258 -0
  205. package/dist/deal-desk/tools/__tests__/outreach-approve.test.js.map +1 -0
  206. package/dist/deal-desk/tools/__tests__/outreach-draft.test.d.ts +2 -0
  207. package/dist/deal-desk/tools/__tests__/outreach-draft.test.d.ts.map +1 -0
  208. package/dist/deal-desk/tools/__tests__/outreach-draft.test.js +115 -0
  209. package/dist/deal-desk/tools/__tests__/outreach-draft.test.js.map +1 -0
  210. package/dist/deal-desk/tools/__tests__/outreach-edit.test.d.ts +2 -0
  211. package/dist/deal-desk/tools/__tests__/outreach-edit.test.d.ts.map +1 -0
  212. package/dist/deal-desk/tools/__tests__/outreach-edit.test.js +169 -0
  213. package/dist/deal-desk/tools/__tests__/outreach-edit.test.js.map +1 -0
  214. package/dist/deal-desk/tools/__tests__/outreach-list-pending.test.d.ts +2 -0
  215. package/dist/deal-desk/tools/__tests__/outreach-list-pending.test.d.ts.map +1 -0
  216. package/dist/deal-desk/tools/__tests__/outreach-list-pending.test.js +59 -0
  217. package/dist/deal-desk/tools/__tests__/outreach-list-pending.test.js.map +1 -0
  218. package/dist/deal-desk/tools/__tests__/test-gmail-send.test.d.ts +2 -0
  219. package/dist/deal-desk/tools/__tests__/test-gmail-send.test.d.ts.map +1 -0
  220. package/dist/deal-desk/tools/__tests__/test-gmail-send.test.js +98 -0
  221. package/dist/deal-desk/tools/__tests__/test-gmail-send.test.js.map +1 -0
  222. package/dist/deal-desk/tools/apollo-api-key.d.ts +11 -0
  223. package/dist/deal-desk/tools/apollo-api-key.d.ts.map +1 -0
  224. package/dist/deal-desk/tools/apollo-api-key.js +70 -0
  225. package/dist/deal-desk/tools/apollo-api-key.js.map +1 -0
  226. package/dist/deal-desk/tools/create-contact.d.ts +28 -0
  227. package/dist/deal-desk/tools/create-contact.d.ts.map +1 -0
  228. package/dist/deal-desk/tools/create-contact.js +52 -0
  229. package/dist/deal-desk/tools/create-contact.js.map +1 -0
  230. package/dist/deal-desk/tools/create-intermediary.d.ts +62 -0
  231. package/dist/deal-desk/tools/create-intermediary.d.ts.map +1 -0
  232. package/dist/deal-desk/tools/create-intermediary.js +105 -0
  233. package/dist/deal-desk/tools/create-intermediary.js.map +1 -0
  234. package/dist/deal-desk/tools/create-target.d.ts +8 -0
  235. package/dist/deal-desk/tools/create-target.d.ts.map +1 -0
  236. package/dist/deal-desk/tools/create-target.js +56 -0
  237. package/dist/deal-desk/tools/create-target.js.map +1 -0
  238. package/dist/deal-desk/tools/enrich-contact.d.ts +22 -0
  239. package/dist/deal-desk/tools/enrich-contact.d.ts.map +1 -0
  240. package/dist/deal-desk/tools/enrich-contact.js +85 -0
  241. package/dist/deal-desk/tools/enrich-contact.js.map +1 -0
  242. package/dist/deal-desk/tools/get-target.d.ts +5 -0
  243. package/dist/deal-desk/tools/get-target.d.ts.map +1 -0
  244. package/dist/deal-desk/tools/get-target.js +33 -0
  245. package/dist/deal-desk/tools/get-target.js.map +1 -0
  246. package/dist/deal-desk/tools/gmail-client-config.d.ts +20 -0
  247. package/dist/deal-desk/tools/gmail-client-config.d.ts.map +1 -0
  248. package/dist/deal-desk/tools/gmail-client-config.js +43 -0
  249. package/dist/deal-desk/tools/gmail-client-config.js.map +1 -0
  250. package/dist/deal-desk/tools/index.d.ts +39 -0
  251. package/dist/deal-desk/tools/index.d.ts.map +1 -0
  252. package/dist/deal-desk/tools/index.js +208 -0
  253. package/dist/deal-desk/tools/index.js.map +1 -0
  254. package/dist/deal-desk/tools/intermediary-outreach-draft.d.ts +21 -0
  255. package/dist/deal-desk/tools/intermediary-outreach-draft.d.ts.map +1 -0
  256. package/dist/deal-desk/tools/intermediary-outreach-draft.js +59 -0
  257. package/dist/deal-desk/tools/intermediary-outreach-draft.js.map +1 -0
  258. package/dist/deal-desk/tools/list-intermediaries.d.ts +20 -0
  259. package/dist/deal-desk/tools/list-intermediaries.d.ts.map +1 -0
  260. package/dist/deal-desk/tools/list-intermediaries.js +74 -0
  261. package/dist/deal-desk/tools/list-intermediaries.js.map +1 -0
  262. package/dist/deal-desk/tools/list-targets.d.ts +20 -0
  263. package/dist/deal-desk/tools/list-targets.d.ts.map +1 -0
  264. package/dist/deal-desk/tools/list-targets.js +73 -0
  265. package/dist/deal-desk/tools/list-targets.js.map +1 -0
  266. package/dist/deal-desk/tools/outreach-approve.d.ts +18 -0
  267. package/dist/deal-desk/tools/outreach-approve.d.ts.map +1 -0
  268. package/dist/deal-desk/tools/outreach-approve.js +229 -0
  269. package/dist/deal-desk/tools/outreach-approve.js.map +1 -0
  270. package/dist/deal-desk/tools/outreach-draft.d.ts +32 -0
  271. package/dist/deal-desk/tools/outreach-draft.d.ts.map +1 -0
  272. package/dist/deal-desk/tools/outreach-draft.js +101 -0
  273. package/dist/deal-desk/tools/outreach-draft.js.map +1 -0
  274. package/dist/deal-desk/tools/outreach-edit.d.ts +6 -0
  275. package/dist/deal-desk/tools/outreach-edit.d.ts.map +1 -0
  276. package/dist/deal-desk/tools/outreach-edit.js +94 -0
  277. package/dist/deal-desk/tools/outreach-edit.js.map +1 -0
  278. package/dist/deal-desk/tools/outreach-list-pending.d.ts +4 -0
  279. package/dist/deal-desk/tools/outreach-list-pending.d.ts.map +1 -0
  280. package/dist/deal-desk/tools/outreach-list-pending.js +55 -0
  281. package/dist/deal-desk/tools/outreach-list-pending.js.map +1 -0
  282. package/dist/deal-desk/tools/record-intermediary-touch.d.ts +20 -0
  283. package/dist/deal-desk/tools/record-intermediary-touch.d.ts.map +1 -0
  284. package/dist/deal-desk/tools/record-intermediary-touch.js +80 -0
  285. package/dist/deal-desk/tools/record-intermediary-touch.js.map +1 -0
  286. package/dist/deal-desk/tools/test-gmail-send.d.ts +21 -0
  287. package/dist/deal-desk/tools/test-gmail-send.d.ts.map +1 -0
  288. package/dist/deal-desk/tools/test-gmail-send.js +65 -0
  289. package/dist/deal-desk/tools/test-gmail-send.js.map +1 -0
  290. package/dist/deal-desk/tools/update-target.d.ts +6 -0
  291. package/dist/deal-desk/tools/update-target.d.ts.map +1 -0
  292. package/dist/deal-desk/tools/update-target.js +43 -0
  293. package/dist/deal-desk/tools/update-target.js.map +1 -0
  294. package/dist/dev-runner-worktree.d.ts +15 -0
  295. package/dist/dev-runner-worktree.d.ts.map +1 -0
  296. package/dist/dev-runner-worktree.js +68 -0
  297. package/dist/dev-runner-worktree.js.map +1 -0
  298. package/dist/dev-server-status.d.ts +27 -0
  299. package/dist/dev-server-status.d.ts.map +1 -0
  300. package/dist/dev-server-status.js +74 -0
  301. package/dist/dev-server-status.js.map +1 -0
  302. package/dist/dev-watch-ignore.d.ts +2 -0
  303. package/dist/dev-watch-ignore.d.ts.map +1 -0
  304. package/dist/dev-watch-ignore.js +36 -0
  305. package/dist/dev-watch-ignore.js.map +1 -0
  306. package/dist/embedded-postgres-watchdog.d.ts +12 -0
  307. package/dist/embedded-postgres-watchdog.d.ts.map +1 -0
  308. package/dist/embedded-postgres-watchdog.js +21 -0
  309. package/dist/embedded-postgres-watchdog.js.map +1 -0
  310. package/dist/errors.d.ts +12 -0
  311. package/dist/errors.d.ts.map +1 -0
  312. package/dist/errors.js +28 -0
  313. package/dist/errors.js.map +1 -0
  314. package/dist/home-paths.d.ts +15 -0
  315. package/dist/home-paths.d.ts.map +1 -0
  316. package/dist/home-paths.js +48 -0
  317. package/dist/home-paths.js.map +1 -0
  318. package/dist/index.d.ts +10 -0
  319. package/dist/index.d.ts.map +1 -0
  320. package/dist/index.js +787 -0
  321. package/dist/index.js.map +1 -0
  322. package/dist/lib/join-request-dedupe.d.ts +11 -0
  323. package/dist/lib/join-request-dedupe.d.ts.map +1 -0
  324. package/dist/lib/join-request-dedupe.js +49 -0
  325. package/dist/lib/join-request-dedupe.js.map +1 -0
  326. package/dist/log-redaction.d.ts +11 -0
  327. package/dist/log-redaction.d.ts.map +1 -0
  328. package/dist/log-redaction.js +122 -0
  329. package/dist/log-redaction.js.map +1 -0
  330. package/dist/middleware/auth.d.ts +12 -0
  331. package/dist/middleware/auth.d.ts.map +1 -0
  332. package/dist/middleware/auth.js +302 -0
  333. package/dist/middleware/auth.js.map +1 -0
  334. package/dist/middleware/board-mutation-guard.d.ts +3 -0
  335. package/dist/middleware/board-mutation-guard.d.ts.map +1 -0
  336. package/dist/middleware/board-mutation-guard.js +67 -0
  337. package/dist/middleware/board-mutation-guard.js.map +1 -0
  338. package/dist/middleware/error-handler.d.ts +17 -0
  339. package/dist/middleware/error-handler.d.ts.map +1 -0
  340. package/dist/middleware/error-handler.js +45 -0
  341. package/dist/middleware/error-handler.js.map +1 -0
  342. package/dist/middleware/http-log-policy.d.ts +2 -0
  343. package/dist/middleware/http-log-policy.d.ts.map +1 -0
  344. package/dist/middleware/http-log-policy.js +52 -0
  345. package/dist/middleware/http-log-policy.js.map +1 -0
  346. package/dist/middleware/index.d.ts +4 -0
  347. package/dist/middleware/index.d.ts.map +1 -0
  348. package/dist/middleware/index.js +4 -0
  349. package/dist/middleware/index.js.map +1 -0
  350. package/dist/middleware/logger.d.ts +4 -0
  351. package/dist/middleware/logger.d.ts.map +1 -0
  352. package/dist/middleware/logger.js +92 -0
  353. package/dist/middleware/logger.js.map +1 -0
  354. package/dist/middleware/private-hostname-guard.d.ts +11 -0
  355. package/dist/middleware/private-hostname-guard.d.ts.map +1 -0
  356. package/dist/middleware/private-hostname-guard.js +78 -0
  357. package/dist/middleware/private-hostname-guard.js.map +1 -0
  358. package/dist/middleware/validate.d.ts +4 -0
  359. package/dist/middleware/validate.d.ts.map +1 -0
  360. package/dist/middleware/validate.js +7 -0
  361. package/dist/middleware/validate.js.map +1 -0
  362. package/dist/onboarding-assets/ceo/AGENTS.md +59 -0
  363. package/dist/onboarding-assets/ceo/HEARTBEAT.md +85 -0
  364. package/dist/onboarding-assets/ceo/SOUL.md +33 -0
  365. package/dist/onboarding-assets/ceo/TOOLS.md +3 -0
  366. package/dist/onboarding-assets/default/AGENTS.md +17 -0
  367. package/dist/paths.d.ts +3 -0
  368. package/dist/paths.d.ts.map +1 -0
  369. package/dist/paths.js +31 -0
  370. package/dist/paths.js.map +1 -0
  371. package/dist/realtime/live-events-ws.d.ts +28 -0
  372. package/dist/realtime/live-events-ws.d.ts.map +1 -0
  373. package/dist/realtime/live-events-ws.js +187 -0
  374. package/dist/realtime/live-events-ws.js.map +1 -0
  375. package/dist/redaction.d.ts +5 -0
  376. package/dist/redaction.d.ts.map +1 -0
  377. package/dist/redaction.js +98 -0
  378. package/dist/redaction.js.map +1 -0
  379. package/dist/routes/__tests__/deal-desk-thesis.test.d.ts +2 -0
  380. package/dist/routes/__tests__/deal-desk-thesis.test.d.ts.map +1 -0
  381. package/dist/routes/__tests__/deal-desk-thesis.test.js +53 -0
  382. package/dist/routes/__tests__/deal-desk-thesis.test.js.map +1 -0
  383. package/dist/routes/__tests__/gmail-oauth.test.d.ts +2 -0
  384. package/dist/routes/__tests__/gmail-oauth.test.d.ts.map +1 -0
  385. package/dist/routes/__tests__/gmail-oauth.test.js +99 -0
  386. package/dist/routes/__tests__/gmail-oauth.test.js.map +1 -0
  387. package/dist/routes/access.d.ts +82 -0
  388. package/dist/routes/access.d.ts.map +1 -0
  389. package/dist/routes/access.js +3411 -0
  390. package/dist/routes/access.js.map +1 -0
  391. package/dist/routes/activity.d.ts +3 -0
  392. package/dist/routes/activity.d.ts.map +1 -0
  393. package/dist/routes/activity.js +90 -0
  394. package/dist/routes/activity.js.map +1 -0
  395. package/dist/routes/adapters.d.ts +16 -0
  396. package/dist/routes/adapters.d.ts.map +1 -0
  397. package/dist/routes/adapters.js +527 -0
  398. package/dist/routes/adapters.js.map +1 -0
  399. package/dist/routes/agents.d.ts +6 -0
  400. package/dist/routes/agents.d.ts.map +1 -0
  401. package/dist/routes/agents.js +2765 -0
  402. package/dist/routes/agents.js.map +1 -0
  403. package/dist/routes/approvals.d.ts +6 -0
  404. package/dist/routes/approvals.d.ts.map +1 -0
  405. package/dist/routes/approvals.js +300 -0
  406. package/dist/routes/approvals.js.map +1 -0
  407. package/dist/routes/assets.d.ts +4 -0
  408. package/dist/routes/assets.d.ts.map +1 -0
  409. package/dist/routes/assets.js +309 -0
  410. package/dist/routes/assets.js.map +1 -0
  411. package/dist/routes/auth.d.ts +3 -0
  412. package/dist/routes/auth.d.ts.map +1 -0
  413. package/dist/routes/auth.js +82 -0
  414. package/dist/routes/auth.js.map +1 -0
  415. package/dist/routes/authz.d.ts +19 -0
  416. package/dist/routes/authz.d.ts.map +1 -0
  417. package/dist/routes/authz.js +75 -0
  418. package/dist/routes/authz.js.map +1 -0
  419. package/dist/routes/companies.d.ts +4 -0
  420. package/dist/routes/companies.d.ts.map +1 -0
  421. package/dist/routes/companies.js +359 -0
  422. package/dist/routes/companies.js.map +1 -0
  423. package/dist/routes/company-skills.d.ts +3 -0
  424. package/dist/routes/company-skills.d.ts.map +1 -0
  425. package/dist/routes/company-skills.js +258 -0
  426. package/dist/routes/company-skills.js.map +1 -0
  427. package/dist/routes/costs.d.ts +11 -0
  428. package/dist/routes/costs.d.ts.map +1 -0
  429. package/dist/routes/costs.js +307 -0
  430. package/dist/routes/costs.js.map +1 -0
  431. package/dist/routes/dashboard.d.ts +3 -0
  432. package/dist/routes/dashboard.d.ts.map +1 -0
  433. package/dist/routes/dashboard.js +15 -0
  434. package/dist/routes/dashboard.js.map +1 -0
  435. package/dist/routes/deal-desk.d.ts +83 -0
  436. package/dist/routes/deal-desk.d.ts.map +1 -0
  437. package/dist/routes/deal-desk.js +288 -0
  438. package/dist/routes/deal-desk.js.map +1 -0
  439. package/dist/routes/deal-desk.test.d.ts +2 -0
  440. package/dist/routes/deal-desk.test.d.ts.map +1 -0
  441. package/dist/routes/deal-desk.test.js +291 -0
  442. package/dist/routes/deal-desk.test.js.map +1 -0
  443. package/dist/routes/environment-selection.d.ts +13 -0
  444. package/dist/routes/environment-selection.d.ts.map +1 -0
  445. package/dist/routes/environment-selection.js +30 -0
  446. package/dist/routes/environment-selection.js.map +1 -0
  447. package/dist/routes/environments.d.ts +6 -0
  448. package/dist/routes/environments.d.ts.map +1 -0
  449. package/dist/routes/environments.js +408 -0
  450. package/dist/routes/environments.js.map +1 -0
  451. package/dist/routes/execution-workspaces.d.ts +3 -0
  452. package/dist/routes/execution-workspaces.d.ts.map +1 -0
  453. package/dist/routes/execution-workspaces.js +536 -0
  454. package/dist/routes/execution-workspaces.js.map +1 -0
  455. package/dist/routes/gmail-oauth.d.ts +14 -0
  456. package/dist/routes/gmail-oauth.d.ts.map +1 -0
  457. package/dist/routes/gmail-oauth.js +124 -0
  458. package/dist/routes/gmail-oauth.js.map +1 -0
  459. package/dist/routes/goals.d.ts +3 -0
  460. package/dist/routes/goals.d.ts.map +1 -0
  461. package/dist/routes/goals.js +101 -0
  462. package/dist/routes/goals.js.map +1 -0
  463. package/dist/routes/health.d.ts +9 -0
  464. package/dist/routes/health.d.ts.map +1 -0
  465. package/dist/routes/health.js +114 -0
  466. package/dist/routes/health.js.map +1 -0
  467. package/dist/routes/inbox-dismissals.d.ts +3 -0
  468. package/dist/routes/inbox-dismissals.d.ts.map +1 -0
  469. package/dist/routes/inbox-dismissals.js +58 -0
  470. package/dist/routes/inbox-dismissals.js.map +1 -0
  471. package/dist/routes/index.d.ts +21 -0
  472. package/dist/routes/index.d.ts.map +1 -0
  473. package/dist/routes/index.js +21 -0
  474. package/dist/routes/index.js.map +1 -0
  475. package/dist/routes/instance-database-backups.d.ts +15 -0
  476. package/dist/routes/instance-database-backups.d.ts.map +1 -0
  477. package/dist/routes/instance-database-backups.js +12 -0
  478. package/dist/routes/instance-database-backups.js.map +1 -0
  479. package/dist/routes/instance-settings.d.ts +3 -0
  480. package/dist/routes/instance-settings.d.ts.map +1 -0
  481. package/dist/routes/instance-settings.js +110 -0
  482. package/dist/routes/instance-settings.js.map +1 -0
  483. package/dist/routes/issue-tree-control.d.ts +3 -0
  484. package/dist/routes/issue-tree-control.d.ts.map +1 -0
  485. package/dist/routes/issue-tree-control.js +363 -0
  486. package/dist/routes/issue-tree-control.js.map +1 -0
  487. package/dist/routes/issues-checkout-wakeup.d.ts +9 -0
  488. package/dist/routes/issues-checkout-wakeup.d.ts.map +1 -0
  489. package/dist/routes/issues-checkout-wakeup.js +12 -0
  490. package/dist/routes/issues-checkout-wakeup.js.map +1 -0
  491. package/dist/routes/issues.d.ts +23 -0
  492. package/dist/routes/issues.d.ts.map +1 -0
  493. package/dist/routes/issues.js +4094 -0
  494. package/dist/routes/issues.js.map +1 -0
  495. package/dist/routes/llms.d.ts +3 -0
  496. package/dist/routes/llms.d.ts.map +1 -0
  497. package/dist/routes/llms.js +80 -0
  498. package/dist/routes/llms.js.map +1 -0
  499. package/dist/routes/org-chart-svg.d.ts +25 -0
  500. package/dist/routes/org-chart-svg.d.ts.map +1 -0
  501. package/dist/routes/org-chart-svg.js +656 -0
  502. package/dist/routes/org-chart-svg.js.map +1 -0
  503. package/dist/routes/plugin-ui-static.d.ts +69 -0
  504. package/dist/routes/plugin-ui-static.d.ts.map +1 -0
  505. package/dist/routes/plugin-ui-static.js +411 -0
  506. package/dist/routes/plugin-ui-static.js.map +1 -0
  507. package/dist/routes/plugins.d.ts +121 -0
  508. package/dist/routes/plugins.d.ts.map +1 -0
  509. package/dist/routes/plugins.js +2184 -0
  510. package/dist/routes/plugins.js.map +1 -0
  511. package/dist/routes/projects.d.ts +3 -0
  512. package/dist/routes/projects.d.ts.map +1 -0
  513. package/dist/routes/projects.js +572 -0
  514. package/dist/routes/projects.js.map +1 -0
  515. package/dist/routes/routines.d.ts +6 -0
  516. package/dist/routes/routines.d.ts.map +1 -0
  517. package/dist/routes/routines.js +417 -0
  518. package/dist/routes/routines.js.map +1 -0
  519. package/dist/routes/secrets.d.ts +3 -0
  520. package/dist/routes/secrets.d.ts.map +1 -0
  521. package/dist/routes/secrets.js +392 -0
  522. package/dist/routes/secrets.js.map +1 -0
  523. package/dist/routes/sidebar-badges.d.ts +3 -0
  524. package/dist/routes/sidebar-badges.d.ts.map +1 -0
  525. package/dist/routes/sidebar-badges.js +68 -0
  526. package/dist/routes/sidebar-badges.js.map +1 -0
  527. package/dist/routes/sidebar-preferences.d.ts +3 -0
  528. package/dist/routes/sidebar-preferences.d.ts.map +1 -0
  529. package/dist/routes/sidebar-preferences.js +63 -0
  530. package/dist/routes/sidebar-preferences.js.map +1 -0
  531. package/dist/routes/user-profiles.d.ts +3 -0
  532. package/dist/routes/user-profiles.d.ts.map +1 -0
  533. package/dist/routes/user-profiles.js +337 -0
  534. package/dist/routes/user-profiles.js.map +1 -0
  535. package/dist/routes/workspace-command-authz.d.ts +14 -0
  536. package/dist/routes/workspace-command-authz.d.ts.map +1 -0
  537. package/dist/routes/workspace-command-authz.js +83 -0
  538. package/dist/routes/workspace-command-authz.js.map +1 -0
  539. package/dist/routes/workspace-runtime-service-authz.d.ts +12 -0
  540. package/dist/routes/workspace-runtime-service-authz.d.ts.map +1 -0
  541. package/dist/routes/workspace-runtime-service-authz.js +96 -0
  542. package/dist/routes/workspace-runtime-service-authz.js.map +1 -0
  543. package/dist/runtime-api.d.ts +19 -0
  544. package/dist/runtime-api.d.ts.map +1 -0
  545. package/dist/runtime-api.js +137 -0
  546. package/dist/runtime-api.js.map +1 -0
  547. package/dist/secrets/aws-secrets-manager-provider.d.ts +87 -0
  548. package/dist/secrets/aws-secrets-manager-provider.d.ts.map +1 -0
  549. package/dist/secrets/aws-secrets-manager-provider.js +748 -0
  550. package/dist/secrets/aws-secrets-manager-provider.js.map +1 -0
  551. package/dist/secrets/configured-provider.d.ts +3 -0
  552. package/dist/secrets/configured-provider.d.ts.map +1 -0
  553. package/dist/secrets/configured-provider.js +8 -0
  554. package/dist/secrets/configured-provider.js.map +1 -0
  555. package/dist/secrets/external-stub-providers.d.ts +5 -0
  556. package/dist/secrets/external-stub-providers.d.ts.map +1 -0
  557. package/dist/secrets/external-stub-providers.js +71 -0
  558. package/dist/secrets/external-stub-providers.js.map +1 -0
  559. package/dist/secrets/local-encrypted-provider.d.ts +3 -0
  560. package/dist/secrets/local-encrypted-provider.d.ts.map +1 -0
  561. package/dist/secrets/local-encrypted-provider.js +244 -0
  562. package/dist/secrets/local-encrypted-provider.js.map +1 -0
  563. package/dist/secrets/provider-registry.d.ts +6 -0
  564. package/dist/secrets/provider-registry.d.ts.map +1 -0
  565. package/dist/secrets/provider-registry.js +24 -0
  566. package/dist/secrets/provider-registry.js.map +1 -0
  567. package/dist/secrets/types.d.ts +131 -0
  568. package/dist/secrets/types.d.ts.map +1 -0
  569. package/dist/secrets/types.js +36 -0
  570. package/dist/secrets/types.js.map +1 -0
  571. package/dist/services/access.d.ts +171 -0
  572. package/dist/services/access.d.ts.map +1 -0
  573. package/dist/services/access.js +522 -0
  574. package/dist/services/access.js.map +1 -0
  575. package/dist/services/activity-log.d.ts +19 -0
  576. package/dist/services/activity-log.d.ts.map +1 -0
  577. package/dist/services/activity-log.js +99 -0
  578. package/dist/services/activity-log.js.map +1 -0
  579. package/dist/services/activity.d.ts +462 -0
  580. package/dist/services/activity.d.ts.map +1 -0
  581. package/dist/services/activity.js +443 -0
  582. package/dist/services/activity.js.map +1 -0
  583. package/dist/services/adapter-plugin-store.d.ts +36 -0
  584. package/dist/services/adapter-plugin-store.d.ts.map +1 -0
  585. package/dist/services/adapter-plugin-store.js +154 -0
  586. package/dist/services/adapter-plugin-store.js.map +1 -0
  587. package/dist/services/agent-instructions.d.ts +91 -0
  588. package/dist/services/agent-instructions.d.ts.map +1 -0
  589. package/dist/services/agent-instructions.js +580 -0
  590. package/dist/services/agent-instructions.js.map +1 -0
  591. package/dist/services/agent-permissions.d.ts +6 -0
  592. package/dist/services/agent-permissions.d.ts.map +1 -0
  593. package/dist/services/agent-permissions.js +18 -0
  594. package/dist/services/agent-permissions.js.map +1 -0
  595. package/dist/services/agent-start-lock.d.ts +2 -0
  596. package/dist/services/agent-start-lock.d.ts.map +1 -0
  597. package/dist/services/agent-start-lock.js +43 -0
  598. package/dist/services/agent-start-lock.js.map +1 -0
  599. package/dist/services/agents.d.ts +2253 -0
  600. package/dist/services/agents.d.ts.map +1 -0
  601. package/dist/services/agents.js +609 -0
  602. package/dist/services/agents.js.map +1 -0
  603. package/dist/services/approvals.d.ts +546 -0
  604. package/dist/services/approvals.d.ts.map +1 -0
  605. package/dist/services/approvals.js +212 -0
  606. package/dist/services/approvals.js.map +1 -0
  607. package/dist/services/assets.d.ts +33 -0
  608. package/dist/services/assets.d.ts.map +1 -0
  609. package/dist/services/assets.js +17 -0
  610. package/dist/services/assets.js.map +1 -0
  611. package/dist/services/board-auth.d.ts +239 -0
  612. package/dist/services/board-auth.d.ts.map +1 -0
  613. package/dist/services/board-auth.js +300 -0
  614. package/dist/services/board-auth.js.map +1 -0
  615. package/dist/services/budgets.d.ts +38 -0
  616. package/dist/services/budgets.d.ts.map +1 -0
  617. package/dist/services/budgets.js +784 -0
  618. package/dist/services/budgets.js.map +1 -0
  619. package/dist/services/companies.d.ts +154 -0
  620. package/dist/services/companies.d.ts.map +1 -0
  621. package/dist/services/companies.js +286 -0
  622. package/dist/services/companies.js.map +1 -0
  623. package/dist/services/company-export-readme.d.ts +17 -0
  624. package/dist/services/company-export-readme.d.ts.map +1 -0
  625. package/dist/services/company-export-readme.js +148 -0
  626. package/dist/services/company-export-readme.js.map +1 -0
  627. package/dist/services/company-member-roles.d.ts +9 -0
  628. package/dist/services/company-member-roles.d.ts.map +1 -0
  629. package/dist/services/company-member-roles.js +46 -0
  630. package/dist/services/company-member-roles.js.map +1 -0
  631. package/dist/services/company-portability.d.ts +24 -0
  632. package/dist/services/company-portability.d.ts.map +1 -0
  633. package/dist/services/company-portability.js +4077 -0
  634. package/dist/services/company-portability.js.map +1 -0
  635. package/dist/services/company-search-rate-limit.d.ts +22 -0
  636. package/dist/services/company-search-rate-limit.d.ts.map +1 -0
  637. package/dist/services/company-search-rate-limit.js +38 -0
  638. package/dist/services/company-search-rate-limit.js.map +1 -0
  639. package/dist/services/company-search.d.ts +8 -0
  640. package/dist/services/company-search.d.ts.map +1 -0
  641. package/dist/services/company-search.js +626 -0
  642. package/dist/services/company-search.js.map +1 -0
  643. package/dist/services/company-skills.d.ts +77 -0
  644. package/dist/services/company-skills.d.ts.map +1 -0
  645. package/dist/services/company-skills.js +2131 -0
  646. package/dist/services/company-skills.js.map +1 -0
  647. package/dist/services/cost-backfill.d.ts +16 -0
  648. package/dist/services/cost-backfill.d.ts.map +1 -0
  649. package/dist/services/cost-backfill.js +115 -0
  650. package/dist/services/cost-backfill.js.map +1 -0
  651. package/dist/services/cost-ledger.d.ts +28 -0
  652. package/dist/services/cost-ledger.d.ts.map +1 -0
  653. package/dist/services/cost-ledger.js +140 -0
  654. package/dist/services/cost-ledger.js.map +1 -0
  655. package/dist/services/cost-ledger.test.d.ts +2 -0
  656. package/dist/services/cost-ledger.test.d.ts.map +1 -0
  657. package/dist/services/cost-ledger.test.js +50 -0
  658. package/dist/services/cost-ledger.test.js.map +1 -0
  659. package/dist/services/cost-pipeline-health.d.ts +4 -0
  660. package/dist/services/cost-pipeline-health.d.ts.map +1 -0
  661. package/dist/services/cost-pipeline-health.js +42 -0
  662. package/dist/services/cost-pipeline-health.js.map +1 -0
  663. package/dist/services/costs.d.ts +128 -0
  664. package/dist/services/costs.d.ts.map +1 -0
  665. package/dist/services/costs.js +412 -0
  666. package/dist/services/costs.js.map +1 -0
  667. package/dist/services/cron.d.ts +80 -0
  668. package/dist/services/cron.d.ts.map +1 -0
  669. package/dist/services/cron.js +300 -0
  670. package/dist/services/cron.js.map +1 -0
  671. package/dist/services/dashboard.d.ts +34 -0
  672. package/dist/services/dashboard.d.ts.map +1 -0
  673. package/dist/services/dashboard.js +142 -0
  674. package/dist/services/dashboard.js.map +1 -0
  675. package/dist/services/default-agent-instructions.d.ts +9 -0
  676. package/dist/services/default-agent-instructions.d.ts.map +1 -0
  677. package/dist/services/default-agent-instructions.js +20 -0
  678. package/dist/services/default-agent-instructions.js.map +1 -0
  679. package/dist/services/documents.d.ts +199 -0
  680. package/dist/services/documents.d.ts.map +1 -0
  681. package/dist/services/documents.js +411 -0
  682. package/dist/services/documents.js.map +1 -0
  683. package/dist/services/environment-config.d.ts +55 -0
  684. package/dist/services/environment-config.d.ts.map +1 -0
  685. package/dist/services/environment-config.js +441 -0
  686. package/dist/services/environment-config.js.map +1 -0
  687. package/dist/services/environment-execution-target.d.ts +21 -0
  688. package/dist/services/environment-execution-target.d.ts.map +1 -0
  689. package/dist/services/environment-execution-target.js +121 -0
  690. package/dist/services/environment-execution-target.js.map +1 -0
  691. package/dist/services/environment-probe.d.ts +9 -0
  692. package/dist/services/environment-probe.d.ts.map +1 -0
  693. package/dist/services/environment-probe.js +106 -0
  694. package/dist/services/environment-probe.js.map +1 -0
  695. package/dist/services/environment-run-orchestrator.d.ts +124 -0
  696. package/dist/services/environment-run-orchestrator.d.ts.map +1 -0
  697. package/dist/services/environment-run-orchestrator.js +392 -0
  698. package/dist/services/environment-run-orchestrator.js.map +1 -0
  699. package/dist/services/environment-runtime.d.ts +90 -0
  700. package/dist/services/environment-runtime.d.ts.map +1 -0
  701. package/dist/services/environment-runtime.js +968 -0
  702. package/dist/services/environment-runtime.js.map +1 -0
  703. package/dist/services/environments.d.ts +36 -0
  704. package/dist/services/environments.d.ts.map +1 -0
  705. package/dist/services/environments.js +260 -0
  706. package/dist/services/environments.js.map +1 -0
  707. package/dist/services/execution-workspace-policy.d.ts +30 -0
  708. package/dist/services/execution-workspace-policy.d.ts.map +1 -0
  709. package/dist/services/execution-workspace-policy.js +195 -0
  710. package/dist/services/execution-workspace-policy.js.map +1 -0
  711. package/dist/services/execution-workspaces.d.ts +30 -0
  712. package/dist/services/execution-workspaces.d.ts.map +1 -0
  713. package/dist/services/execution-workspaces.js +635 -0
  714. package/dist/services/execution-workspaces.js.map +1 -0
  715. package/dist/services/feedback-redaction.d.ts +23 -0
  716. package/dist/services/feedback-redaction.d.ts.map +1 -0
  717. package/dist/services/feedback-redaction.js +150 -0
  718. package/dist/services/feedback-redaction.js.map +1 -0
  719. package/dist/services/feedback-share-client.d.ts +9 -0
  720. package/dist/services/feedback-share-client.d.ts.map +1 -0
  721. package/dist/services/feedback-share-client.js +46 -0
  722. package/dist/services/feedback-share-client.js.map +1 -0
  723. package/dist/services/feedback.d.ts +93 -0
  724. package/dist/services/feedback.d.ts.map +1 -0
  725. package/dist/services/feedback.js +1717 -0
  726. package/dist/services/feedback.js.map +1 -0
  727. package/dist/services/finance.d.ts +93 -0
  728. package/dist/services/finance.d.ts.map +1 -0
  729. package/dist/services/finance.js +120 -0
  730. package/dist/services/finance.js.map +1 -0
  731. package/dist/services/github-fetch.d.ts +4 -0
  732. package/dist/services/github-fetch.d.ts.map +1 -0
  733. package/dist/services/github-fetch.js +23 -0
  734. package/dist/services/github-fetch.js.map +1 -0
  735. package/dist/services/goals.d.ts +433 -0
  736. package/dist/services/goals.d.ts.map +1 -0
  737. package/dist/services/goals.js +54 -0
  738. package/dist/services/goals.js.map +1 -0
  739. package/dist/services/heartbeat-run-summary.d.ts +7 -0
  740. package/dist/services/heartbeat-run-summary.d.ts.map +1 -0
  741. package/dist/services/heartbeat-run-summary.js +84 -0
  742. package/dist/services/heartbeat-run-summary.js.map +1 -0
  743. package/dist/services/heartbeat-stop-metadata.d.ts +28 -0
  744. package/dist/services/heartbeat-stop-metadata.d.ts.map +1 -0
  745. package/dist/services/heartbeat-stop-metadata.js +86 -0
  746. package/dist/services/heartbeat-stop-metadata.js.map +1 -0
  747. package/dist/services/heartbeat-stop-metadata.test.d.ts +2 -0
  748. package/dist/services/heartbeat-stop-metadata.test.d.ts.map +1 -0
  749. package/dist/services/heartbeat-stop-metadata.test.js +93 -0
  750. package/dist/services/heartbeat-stop-metadata.test.js.map +1 -0
  751. package/dist/services/heartbeat.d.ts +1546 -0
  752. package/dist/services/heartbeat.d.ts.map +1 -0
  753. package/dist/services/heartbeat.js +7854 -0
  754. package/dist/services/heartbeat.js.map +1 -0
  755. package/dist/services/hire-hook.d.ts +14 -0
  756. package/dist/services/hire-hook.d.ts.map +1 -0
  757. package/dist/services/hire-hook.js +85 -0
  758. package/dist/services/hire-hook.js.map +1 -0
  759. package/dist/services/inbox-dismissals.d.ts +22 -0
  760. package/dist/services/inbox-dismissals.d.ts.map +1 -0
  761. package/dist/services/inbox-dismissals.js +33 -0
  762. package/dist/services/inbox-dismissals.js.map +1 -0
  763. package/dist/services/index.d.ts +45 -0
  764. package/dist/services/index.d.ts.map +1 -0
  765. package/dist/services/index.js +45 -0
  766. package/dist/services/index.js.map +1 -0
  767. package/dist/services/instance-settings.d.ts +11 -0
  768. package/dist/services/instance-settings.d.ts.map +1 -0
  769. package/dist/services/instance-settings.js +138 -0
  770. package/dist/services/instance-settings.js.map +1 -0
  771. package/dist/services/invite-grants.d.ts +15 -0
  772. package/dist/services/invite-grants.d.ts.map +1 -0
  773. package/dist/services/invite-grants.js +50 -0
  774. package/dist/services/invite-grants.js.map +1 -0
  775. package/dist/services/issue-approvals.d.ts +56 -0
  776. package/dist/services/issue-approvals.d.ts.map +1 -0
  777. package/dist/services/issue-approvals.js +153 -0
  778. package/dist/services/issue-approvals.js.map +1 -0
  779. package/dist/services/issue-assignment-wakeup.d.ts +29 -0
  780. package/dist/services/issue-assignment-wakeup.d.ts.map +1 -0
  781. package/dist/services/issue-assignment-wakeup.js +22 -0
  782. package/dist/services/issue-assignment-wakeup.js.map +1 -0
  783. package/dist/services/issue-continuation-summary.d.ts +68 -0
  784. package/dist/services/issue-continuation-summary.d.ts.map +1 -0
  785. package/dist/services/issue-continuation-summary.js +222 -0
  786. package/dist/services/issue-continuation-summary.js.map +1 -0
  787. package/dist/services/issue-execution-policy.d.ts +93 -0
  788. package/dist/services/issue-execution-policy.d.ts.map +1 -0
  789. package/dist/services/issue-execution-policy.js +838 -0
  790. package/dist/services/issue-execution-policy.js.map +1 -0
  791. package/dist/services/issue-goal-fallback.d.ts +18 -0
  792. package/dist/services/issue-goal-fallback.d.ts.map +1 -0
  793. package/dist/services/issue-goal-fallback.js +33 -0
  794. package/dist/services/issue-goal-fallback.js.map +1 -0
  795. package/dist/services/issue-liveness.d.ts +3 -0
  796. package/dist/services/issue-liveness.d.ts.map +1 -0
  797. package/dist/services/issue-liveness.js +2 -0
  798. package/dist/services/issue-liveness.js.map +1 -0
  799. package/dist/services/issue-recovery-actions.d.ts +40 -0
  800. package/dist/services/issue-recovery-actions.d.ts.map +1 -0
  801. package/dist/services/issue-recovery-actions.js +204 -0
  802. package/dist/services/issue-recovery-actions.js.map +1 -0
  803. package/dist/services/issue-references.d.ts +21 -0
  804. package/dist/services/issue-references.d.ts.map +1 -0
  805. package/dist/services/issue-references.js +318 -0
  806. package/dist/services/issue-references.js.map +1 -0
  807. package/dist/services/issue-thread-interactions.d.ts +76 -0
  808. package/dist/services/issue-thread-interactions.d.ts.map +1 -0
  809. package/dist/services/issue-thread-interactions.js +923 -0
  810. package/dist/services/issue-thread-interactions.js.map +1 -0
  811. package/dist/services/issue-thread-interactions.test.d.ts +2 -0
  812. package/dist/services/issue-thread-interactions.test.d.ts.map +1 -0
  813. package/dist/services/issue-thread-interactions.test.js +195 -0
  814. package/dist/services/issue-thread-interactions.test.js.map +1 -0
  815. package/dist/services/issue-tree-control.d.ts +89 -0
  816. package/dist/services/issue-tree-control.d.ts.map +1 -0
  817. package/dist/services/issue-tree-control.js +933 -0
  818. package/dist/services/issue-tree-control.js.map +1 -0
  819. package/dist/services/issues.d.ts +744 -0
  820. package/dist/services/issues.d.ts.map +1 -0
  821. package/dist/services/issues.js +3374 -0
  822. package/dist/services/issues.js.map +1 -0
  823. package/dist/services/json-schema-secret-refs.d.ts +5 -0
  824. package/dist/services/json-schema-secret-refs.d.ts.map +1 -0
  825. package/dist/services/json-schema-secret-refs.js +67 -0
  826. package/dist/services/json-schema-secret-refs.js.map +1 -0
  827. package/dist/services/live-events.d.ts +17 -0
  828. package/dist/services/live-events.d.ts.map +1 -0
  829. package/dist/services/live-events.js +33 -0
  830. package/dist/services/live-events.js.map +1 -0
  831. package/dist/services/local-service-supervisor.d.ts +56 -0
  832. package/dist/services/local-service-supervisor.d.ts.map +1 -0
  833. package/dist/services/local-service-supervisor.js +284 -0
  834. package/dist/services/local-service-supervisor.js.map +1 -0
  835. package/dist/services/plugin-capability-validator.d.ts +108 -0
  836. package/dist/services/plugin-capability-validator.d.ts.map +1 -0
  837. package/dist/services/plugin-capability-validator.js +313 -0
  838. package/dist/services/plugin-capability-validator.js.map +1 -0
  839. package/dist/services/plugin-config-validator.d.ts +26 -0
  840. package/dist/services/plugin-config-validator.d.ts.map +1 -0
  841. package/dist/services/plugin-config-validator.js +41 -0
  842. package/dist/services/plugin-config-validator.js.map +1 -0
  843. package/dist/services/plugin-database.d.ts +49 -0
  844. package/dist/services/plugin-database.d.ts.map +1 -0
  845. package/dist/services/plugin-database.js +440 -0
  846. package/dist/services/plugin-database.js.map +1 -0
  847. package/dist/services/plugin-dev-watcher.d.ts +30 -0
  848. package/dist/services/plugin-dev-watcher.d.ts.map +1 -0
  849. package/dist/services/plugin-dev-watcher.js +246 -0
  850. package/dist/services/plugin-dev-watcher.js.map +1 -0
  851. package/dist/services/plugin-environment-driver.d.ts +126 -0
  852. package/dist/services/plugin-environment-driver.d.ts.map +1 -0
  853. package/dist/services/plugin-environment-driver.js +226 -0
  854. package/dist/services/plugin-environment-driver.js.map +1 -0
  855. package/dist/services/plugin-event-bus.d.ts +149 -0
  856. package/dist/services/plugin-event-bus.d.ts.map +1 -0
  857. package/dist/services/plugin-event-bus.js +258 -0
  858. package/dist/services/plugin-event-bus.js.map +1 -0
  859. package/dist/services/plugin-host-service-cleanup.d.ts +14 -0
  860. package/dist/services/plugin-host-service-cleanup.d.ts.map +1 -0
  861. package/dist/services/plugin-host-service-cleanup.js +37 -0
  862. package/dist/services/plugin-host-service-cleanup.js.map +1 -0
  863. package/dist/services/plugin-host-services.d.ts +17 -0
  864. package/dist/services/plugin-host-services.d.ts.map +1 -0
  865. package/dist/services/plugin-host-services.js +1867 -0
  866. package/dist/services/plugin-host-services.js.map +1 -0
  867. package/dist/services/plugin-job-coordinator.d.ts +81 -0
  868. package/dist/services/plugin-job-coordinator.d.ts.map +1 -0
  869. package/dist/services/plugin-job-coordinator.js +172 -0
  870. package/dist/services/plugin-job-coordinator.js.map +1 -0
  871. package/dist/services/plugin-job-scheduler.d.ts +163 -0
  872. package/dist/services/plugin-job-scheduler.d.ts.map +1 -0
  873. package/dist/services/plugin-job-scheduler.js +459 -0
  874. package/dist/services/plugin-job-scheduler.js.map +1 -0
  875. package/dist/services/plugin-job-store.d.ts +208 -0
  876. package/dist/services/plugin-job-store.d.ts.map +1 -0
  877. package/dist/services/plugin-job-store.js +350 -0
  878. package/dist/services/plugin-job-store.js.map +1 -0
  879. package/dist/services/plugin-lifecycle.d.ts +203 -0
  880. package/dist/services/plugin-lifecycle.d.ts.map +1 -0
  881. package/dist/services/plugin-lifecycle.js +476 -0
  882. package/dist/services/plugin-lifecycle.js.map +1 -0
  883. package/dist/services/plugin-loader.d.ts +445 -0
  884. package/dist/services/plugin-loader.d.ts.map +1 -0
  885. package/dist/services/plugin-loader.js +1273 -0
  886. package/dist/services/plugin-loader.js.map +1 -0
  887. package/dist/services/plugin-local-folders.d.ts +49 -0
  888. package/dist/services/plugin-local-folders.d.ts.map +1 -0
  889. package/dist/services/plugin-local-folders.js +506 -0
  890. package/dist/services/plugin-local-folders.js.map +1 -0
  891. package/dist/services/plugin-log-retention.d.ts +20 -0
  892. package/dist/services/plugin-log-retention.d.ts.map +1 -0
  893. package/dist/services/plugin-log-retention.js +63 -0
  894. package/dist/services/plugin-log-retention.js.map +1 -0
  895. package/dist/services/plugin-managed-agents.d.ts +15 -0
  896. package/dist/services/plugin-managed-agents.d.ts.map +1 -0
  897. package/dist/services/plugin-managed-agents.js +458 -0
  898. package/dist/services/plugin-managed-agents.js.map +1 -0
  899. package/dist/services/plugin-managed-routines.d.ts +41 -0
  900. package/dist/services/plugin-managed-routines.d.ts.map +1 -0
  901. package/dist/services/plugin-managed-routines.js +416 -0
  902. package/dist/services/plugin-managed-routines.js.map +1 -0
  903. package/dist/services/plugin-managed-skills.d.ts +14 -0
  904. package/dist/services/plugin-managed-skills.d.ts.map +1 -0
  905. package/dist/services/plugin-managed-skills.js +264 -0
  906. package/dist/services/plugin-managed-skills.js.map +1 -0
  907. package/dist/services/plugin-manifest-validator.d.ts +79 -0
  908. package/dist/services/plugin-manifest-validator.d.ts.map +1 -0
  909. package/dist/services/plugin-manifest-validator.js +84 -0
  910. package/dist/services/plugin-manifest-validator.js.map +1 -0
  911. package/dist/services/plugin-registry.d.ts +2550 -0
  912. package/dist/services/plugin-registry.d.ts.map +1 -0
  913. package/dist/services/plugin-registry.js +581 -0
  914. package/dist/services/plugin-registry.js.map +1 -0
  915. package/dist/services/plugin-runtime-sandbox.d.ts +40 -0
  916. package/dist/services/plugin-runtime-sandbox.d.ts.map +1 -0
  917. package/dist/services/plugin-runtime-sandbox.js +154 -0
  918. package/dist/services/plugin-runtime-sandbox.js.map +1 -0
  919. package/dist/services/plugin-secrets-handler.d.ts +83 -0
  920. package/dist/services/plugin-secrets-handler.d.ts.map +1 -0
  921. package/dist/services/plugin-secrets-handler.js +168 -0
  922. package/dist/services/plugin-secrets-handler.js.map +1 -0
  923. package/dist/services/plugin-state-store.d.ts +92 -0
  924. package/dist/services/plugin-state-store.d.ts.map +1 -0
  925. package/dist/services/plugin-state-store.js +190 -0
  926. package/dist/services/plugin-state-store.js.map +1 -0
  927. package/dist/services/plugin-stream-bus.d.ts +29 -0
  928. package/dist/services/plugin-stream-bus.d.ts.map +1 -0
  929. package/dist/services/plugin-stream-bus.js +48 -0
  930. package/dist/services/plugin-stream-bus.js.map +1 -0
  931. package/dist/services/plugin-tool-dispatcher.d.ts +180 -0
  932. package/dist/services/plugin-tool-dispatcher.d.ts.map +1 -0
  933. package/dist/services/plugin-tool-dispatcher.js +224 -0
  934. package/dist/services/plugin-tool-dispatcher.js.map +1 -0
  935. package/dist/services/plugin-tool-registry.d.ts +192 -0
  936. package/dist/services/plugin-tool-registry.d.ts.map +1 -0
  937. package/dist/services/plugin-tool-registry.js +224 -0
  938. package/dist/services/plugin-tool-registry.js.map +1 -0
  939. package/dist/services/plugin-worker-manager.d.ts +262 -0
  940. package/dist/services/plugin-worker-manager.d.ts.map +1 -0
  941. package/dist/services/plugin-worker-manager.js +842 -0
  942. package/dist/services/plugin-worker-manager.js.map +1 -0
  943. package/dist/services/productivity-review.d.ts +83 -0
  944. package/dist/services/productivity-review.d.ts.map +1 -0
  945. package/dist/services/productivity-review.js +652 -0
  946. package/dist/services/productivity-review.js.map +1 -0
  947. package/dist/services/project-workspace-runtime-config.d.ts +4 -0
  948. package/dist/services/project-workspace-runtime-config.d.ts.map +1 -0
  949. package/dist/services/project-workspace-runtime-config.js +54 -0
  950. package/dist/services/project-workspace-runtime-config.js.map +1 -0
  951. package/dist/services/projects.d.ts +99 -0
  952. package/dist/services/projects.d.ts.map +1 -0
  953. package/dist/services/projects.js +822 -0
  954. package/dist/services/projects.js.map +1 -0
  955. package/dist/services/quota-windows.d.ts +9 -0
  956. package/dist/services/quota-windows.d.ts.map +1 -0
  957. package/dist/services/quota-windows.js +56 -0
  958. package/dist/services/quota-windows.js.map +1 -0
  959. package/dist/services/recovery/index.d.ts +10 -0
  960. package/dist/services/recovery/index.d.ts.map +1 -0
  961. package/dist/services/recovery/index.js +6 -0
  962. package/dist/services/recovery/index.js.map +1 -0
  963. package/dist/services/recovery/issue-graph-liveness.d.ts +85 -0
  964. package/dist/services/recovery/issue-graph-liveness.d.ts.map +1 -0
  965. package/dist/services/recovery/issue-graph-liveness.js +356 -0
  966. package/dist/services/recovery/issue-graph-liveness.js.map +1 -0
  967. package/dist/services/recovery/model-profile-hint.d.ts +8 -0
  968. package/dist/services/recovery/model-profile-hint.d.ts.map +1 -0
  969. package/dist/services/recovery/model-profile-hint.js +11 -0
  970. package/dist/services/recovery/model-profile-hint.js.map +1 -0
  971. package/dist/services/recovery/origins.d.ts +36 -0
  972. package/dist/services/recovery/origins.d.ts.map +1 -0
  973. package/dist/services/recovery/origins.js +45 -0
  974. package/dist/services/recovery/origins.js.map +1 -0
  975. package/dist/services/recovery/pause-hold-guard.d.ts +6 -0
  976. package/dist/services/recovery/pause-hold-guard.d.ts.map +1 -0
  977. package/dist/services/recovery/pause-hold-guard.js +6 -0
  978. package/dist/services/recovery/pause-hold-guard.js.map +1 -0
  979. package/dist/services/recovery/run-liveness-continuations.d.ts +50 -0
  980. package/dist/services/recovery/run-liveness-continuations.d.ts.map +1 -0
  981. package/dist/services/recovery/run-liveness-continuations.js +117 -0
  982. package/dist/services/recovery/run-liveness-continuations.js.map +1 -0
  983. package/dist/services/recovery/service.d.ts +256 -0
  984. package/dist/services/recovery/service.d.ts.map +1 -0
  985. package/dist/services/recovery/service.js +2441 -0
  986. package/dist/services/recovery/service.js.map +1 -0
  987. package/dist/services/recovery/successful-run-handoff.d.ts +89 -0
  988. package/dist/services/recovery/successful-run-handoff.d.ts.map +1 -0
  989. package/dist/services/recovery/successful-run-handoff.js +304 -0
  990. package/dist/services/recovery/successful-run-handoff.js.map +1 -0
  991. package/dist/services/recovery/successful-run-handoff.test.d.ts +2 -0
  992. package/dist/services/recovery/successful-run-handoff.test.d.ts.map +1 -0
  993. package/dist/services/recovery/successful-run-handoff.test.js +270 -0
  994. package/dist/services/recovery/successful-run-handoff.test.js.map +1 -0
  995. package/dist/services/routines.d.ts +166 -0
  996. package/dist/services/routines.d.ts.map +1 -0
  997. package/dist/services/routines.js +1969 -0
  998. package/dist/services/routines.js.map +1 -0
  999. package/dist/services/run-continuations.d.ts +3 -0
  1000. package/dist/services/run-continuations.d.ts.map +1 -0
  1001. package/dist/services/run-continuations.js +2 -0
  1002. package/dist/services/run-continuations.js.map +1 -0
  1003. package/dist/services/run-liveness.d.ts +46 -0
  1004. package/dist/services/run-liveness.d.ts.map +1 -0
  1005. package/dist/services/run-liveness.js +275 -0
  1006. package/dist/services/run-liveness.js.map +1 -0
  1007. package/dist/services/run-log-store.d.ts +34 -0
  1008. package/dist/services/run-log-store.d.ts.map +1 -0
  1009. package/dist/services/run-log-store.js +111 -0
  1010. package/dist/services/run-log-store.js.map +1 -0
  1011. package/dist/services/sandbox-provider-runtime.d.ts +132 -0
  1012. package/dist/services/sandbox-provider-runtime.d.ts.map +1 -0
  1013. package/dist/services/sandbox-provider-runtime.js +216 -0
  1014. package/dist/services/sandbox-provider-runtime.js.map +1 -0
  1015. package/dist/services/secrets.d.ts +1962 -0
  1016. package/dist/services/secrets.d.ts.map +1 -0
  1017. package/dist/services/secrets.js +1686 -0
  1018. package/dist/services/secrets.js.map +1 -0
  1019. package/dist/services/sidebar-badges.d.ts +14 -0
  1020. package/dist/services/sidebar-badges.d.ts.map +1 -0
  1021. package/dist/services/sidebar-badges.js +48 -0
  1022. package/dist/services/sidebar-badges.js.map +1 -0
  1023. package/dist/services/sidebar-preferences.d.ts +9 -0
  1024. package/dist/services/sidebar-preferences.d.ts.map +1 -0
  1025. package/dist/services/sidebar-preferences.js +82 -0
  1026. package/dist/services/sidebar-preferences.js.map +1 -0
  1027. package/dist/services/work-products.d.ts +14 -0
  1028. package/dist/services/work-products.d.ts.map +1 -0
  1029. package/dist/services/work-products.js +100 -0
  1030. package/dist/services/work-products.js.map +1 -0
  1031. package/dist/services/workspace-operation-log-store.d.ts +33 -0
  1032. package/dist/services/workspace-operation-log-store.d.ts.map +1 -0
  1033. package/dist/services/workspace-operation-log-store.js +110 -0
  1034. package/dist/services/workspace-operation-log-store.js.map +1 -0
  1035. package/dist/services/workspace-operations.d.ts +44 -0
  1036. package/dist/services/workspace-operations.d.ts.map +1 -0
  1037. package/dist/services/workspace-operations.js +211 -0
  1038. package/dist/services/workspace-operations.js.map +1 -0
  1039. package/dist/services/workspace-realization.d.ts +33 -0
  1040. package/dist/services/workspace-realization.d.ts.map +1 -0
  1041. package/dist/services/workspace-realization.js +221 -0
  1042. package/dist/services/workspace-realization.js.map +1 -0
  1043. package/dist/services/workspace-runtime-read-model.d.ts +92 -0
  1044. package/dist/services/workspace-runtime-read-model.d.ts.map +1 -0
  1045. package/dist/services/workspace-runtime-read-model.js +67 -0
  1046. package/dist/services/workspace-runtime-read-model.js.map +1 -0
  1047. package/dist/services/workspace-runtime.d.ts +238 -0
  1048. package/dist/services/workspace-runtime.d.ts.map +1 -0
  1049. package/dist/services/workspace-runtime.js +2388 -0
  1050. package/dist/services/workspace-runtime.js.map +1 -0
  1051. package/dist/startup-banner.d.ts +32 -0
  1052. package/dist/startup-banner.d.ts.map +1 -0
  1053. package/dist/startup-banner.js +89 -0
  1054. package/dist/startup-banner.js.map +1 -0
  1055. package/dist/storage/index.d.ts +6 -0
  1056. package/dist/storage/index.d.ts.map +1 -0
  1057. package/dist/storage/index.js +29 -0
  1058. package/dist/storage/index.js.map +1 -0
  1059. package/dist/storage/local-disk-provider.d.ts +3 -0
  1060. package/dist/storage/local-disk-provider.d.ts.map +1 -0
  1061. package/dist/storage/local-disk-provider.js +79 -0
  1062. package/dist/storage/local-disk-provider.js.map +1 -0
  1063. package/dist/storage/provider-registry.d.ts +4 -0
  1064. package/dist/storage/provider-registry.d.ts.map +1 -0
  1065. package/dist/storage/provider-registry.js +15 -0
  1066. package/dist/storage/provider-registry.js.map +1 -0
  1067. package/dist/storage/s3-provider.d.ts +11 -0
  1068. package/dist/storage/s3-provider.d.ts.map +1 -0
  1069. package/dist/storage/s3-provider.js +123 -0
  1070. package/dist/storage/s3-provider.js.map +1 -0
  1071. package/dist/storage/service.d.ts +3 -0
  1072. package/dist/storage/service.d.ts.map +1 -0
  1073. package/dist/storage/service.js +120 -0
  1074. package/dist/storage/service.js.map +1 -0
  1075. package/dist/storage/types.d.ts +55 -0
  1076. package/dist/storage/types.d.ts.map +1 -0
  1077. package/dist/storage/types.js +2 -0
  1078. package/dist/storage/types.js.map +1 -0
  1079. package/dist/telemetry.d.ts +6 -0
  1080. package/dist/telemetry.d.ts.map +1 -0
  1081. package/dist/telemetry.js +20 -0
  1082. package/dist/telemetry.js.map +1 -0
  1083. package/dist/ui-branding.d.ts +13 -0
  1084. package/dist/ui-branding.d.ts.map +1 -0
  1085. package/dist/ui-branding.js +188 -0
  1086. package/dist/ui-branding.js.map +1 -0
  1087. package/dist/utils/coalesced-error-logger.d.ts +11 -0
  1088. package/dist/utils/coalesced-error-logger.d.ts.map +1 -0
  1089. package/dist/utils/coalesced-error-logger.js +21 -0
  1090. package/dist/utils/coalesced-error-logger.js.map +1 -0
  1091. package/dist/version.d.ts +2 -0
  1092. package/dist/version.d.ts.map +1 -0
  1093. package/dist/version.js +5 -0
  1094. package/dist/version.js.map +1 -0
  1095. package/dist/vite-html-renderer.d.ts +18 -0
  1096. package/dist/vite-html-renderer.d.ts.map +1 -0
  1097. package/dist/vite-html-renderer.js +61 -0
  1098. package/dist/vite-html-renderer.js.map +1 -0
  1099. package/dist/worktree-config.d.ts +19 -0
  1100. package/dist/worktree-config.d.ts.map +1 -0
  1101. package/dist/worktree-config.js +368 -0
  1102. package/dist/worktree-config.js.map +1 -0
  1103. package/package.json +95 -0
  1104. package/skills/deal-desk-outreach/SKILL.md +77 -0
  1105. package/skills/dealdesk/SKILL.md +109 -0
  1106. package/skills/dealdesk/references/api-reference.md +899 -0
  1107. package/skills/dealdesk/references/company-skills.md +193 -0
  1108. package/skills/dealdesk/references/issue-workspaces.md +80 -0
  1109. package/skills/dealdesk/references/routines.md +187 -0
  1110. package/skills/dealdesk/references/workflows.md +141 -0
  1111. package/skills/dealdesk-converting-plans-to-tasks/SKILL.md +110 -0
  1112. package/skills/dealdesk-create-agent/SKILL.md +124 -0
  1113. package/skills/dealdesk-create-agent/references/agent-instruction-templates.md +123 -0
  1114. package/skills/dealdesk-create-agent/references/agents/coder.md +64 -0
  1115. package/skills/dealdesk-create-agent/references/agents/qa.md +88 -0
  1116. package/skills/dealdesk-create-agent/references/agents/securityengineer.md +135 -0
  1117. package/skills/dealdesk-create-agent/references/agents/uxdesigner.md +115 -0
  1118. package/skills/dealdesk-create-agent/references/api-reference.md +110 -0
  1119. package/skills/dealdesk-create-agent/references/baseline-role-guide.md +168 -0
  1120. package/skills/dealdesk-create-agent/references/draft-review-checklist.md +95 -0
  1121. package/skills/dealdesk-create-plugin/SKILL.md +92 -0
  1122. package/skills/dealdesk-dev/SKILL.md +103 -0
  1123. package/skills/diagnose-why-work-stopped/SKILL.md +161 -0
  1124. package/skills/para-memory-files/SKILL.md +104 -0
  1125. package/skills/para-memory-files/references/schemas.md +35 -0
  1126. package/skills/terminal-bench-loop/SKILL.md +236 -0
  1127. package/ui-dist/android-chrome-192x192.png +0 -0
  1128. package/ui-dist/android-chrome-512x512.png +0 -0
  1129. package/ui-dist/apple-touch-icon.png +0 -0
  1130. package/ui-dist/assets/_basePickBy-CaQZTZgE.js +1 -0
  1131. package/ui-dist/assets/_baseUniq-DvA-iyvx.js +1 -0
  1132. package/ui-dist/assets/apl-B4CMkyY2.js +1 -0
  1133. package/ui-dist/assets/arc-_uwl-PL9.js +1 -0
  1134. package/ui-dist/assets/architectureDiagram-VXUJARFQ-hoEr3qym.js +36 -0
  1135. package/ui-dist/assets/asciiarmor-Df11BRmG.js +1 -0
  1136. package/ui-dist/assets/asn1-EdZsLKOL.js +1 -0
  1137. package/ui-dist/assets/asterisk-B-8jnY81.js +1 -0
  1138. package/ui-dist/assets/blockDiagram-VD42YOAC-BpNmbQ9P.js +122 -0
  1139. package/ui-dist/assets/brainfuck-C4LP7Hcl.js +1 -0
  1140. package/ui-dist/assets/c4Diagram-YG6GDRKO-BG14x8dM.js +10 -0
  1141. package/ui-dist/assets/channel-NmuAvKfd.js +1 -0
  1142. package/ui-dist/assets/chunk-4BX2VUAB-Cb1YcG81.js +1 -0
  1143. package/ui-dist/assets/chunk-55IACEB6-DkMTjrQj.js +1 -0
  1144. package/ui-dist/assets/chunk-B4BG7PRW-BD6QmYfE.js +165 -0
  1145. package/ui-dist/assets/chunk-DI55MBZ5-B_xkiL3H.js +220 -0
  1146. package/ui-dist/assets/chunk-FMBD7UC4-DC95tjc1.js +15 -0
  1147. package/ui-dist/assets/chunk-QN33PNHL-BkIJ2KXA.js +1 -0
  1148. package/ui-dist/assets/chunk-QZHKN3VN-BWUNV7If.js +1 -0
  1149. package/ui-dist/assets/chunk-TZMSLE5B-NQYnSIs1.js +1 -0
  1150. package/ui-dist/assets/classDiagram-2ON5EDUG-eS4kRH8R.js +1 -0
  1151. package/ui-dist/assets/classDiagram-v2-WZHVMYZB-eS4kRH8R.js +1 -0
  1152. package/ui-dist/assets/clike-B9uivgTg.js +1 -0
  1153. package/ui-dist/assets/clojure-BMjYHr_A.js +1 -0
  1154. package/ui-dist/assets/clone-CbA0swX6.js +1 -0
  1155. package/ui-dist/assets/cmake-BQqOBYOt.js +1 -0
  1156. package/ui-dist/assets/cobol-CWcv1MsR.js +1 -0
  1157. package/ui-dist/assets/coffeescript-S37ZYGWr.js +1 -0
  1158. package/ui-dist/assets/commonlisp-DBKNyK5s.js +1 -0
  1159. package/ui-dist/assets/cose-bilkent-S5V4N54A-BHo6DkZj.js +1 -0
  1160. package/ui-dist/assets/crystal-SjHAIU92.js +1 -0
  1161. package/ui-dist/assets/css-BnMrqG3P.js +1 -0
  1162. package/ui-dist/assets/cypher-C_CwsFkJ.js +1 -0
  1163. package/ui-dist/assets/cytoscape.esm-jbPEKk2Y.js +321 -0
  1164. package/ui-dist/assets/d-pRatUO7H.js +1 -0
  1165. package/ui-dist/assets/dagre-6UL2VRFP-CfzNlqfr.js +4 -0
  1166. package/ui-dist/assets/defaultLocale-DX6XiGOO.js +1 -0
  1167. package/ui-dist/assets/diagram-PSM6KHXK-DjPk4Vl5.js +24 -0
  1168. package/ui-dist/assets/diagram-QEK2KX5R-ClmlUcpT.js +43 -0
  1169. package/ui-dist/assets/diagram-S2PKOQOG-B68LEXtz.js +24 -0
  1170. package/ui-dist/assets/diff-DbItnlRl.js +1 -0
  1171. package/ui-dist/assets/dockerfile-BKs6k2Af.js +1 -0
  1172. package/ui-dist/assets/dtd-DF_7sFjM.js +1 -0
  1173. package/ui-dist/assets/dylan-DwRh75JA.js +1 -0
  1174. package/ui-dist/assets/ebnf-CDyGwa7X.js +1 -0
  1175. package/ui-dist/assets/ecl-Cabwm37j.js +1 -0
  1176. package/ui-dist/assets/eiffel-CnydiIhH.js +1 -0
  1177. package/ui-dist/assets/elm-vLlmbW-K.js +1 -0
  1178. package/ui-dist/assets/erDiagram-Q2GNP2WA-CJSnga5T.js +60 -0
  1179. package/ui-dist/assets/erlang-BNw1qcRV.js +1 -0
  1180. package/ui-dist/assets/factor-kuTfRLto.js +1 -0
  1181. package/ui-dist/assets/fcl-Kvtd6kyn.js +1 -0
  1182. package/ui-dist/assets/flowDiagram-NV44I4VS-Drmp4DHd.js +162 -0
  1183. package/ui-dist/assets/forth-Ffai-XNe.js +1 -0
  1184. package/ui-dist/assets/fortran-DYz_wnZ1.js +1 -0
  1185. package/ui-dist/assets/ganttDiagram-JELNMOA3-C8ffQ-2u.js +267 -0
  1186. package/ui-dist/assets/gas-Bneqetm1.js +1 -0
  1187. package/ui-dist/assets/gherkin-heZmZLOM.js +1 -0
  1188. package/ui-dist/assets/gitGraphDiagram-V2S2FVAM-CPjF6oPQ.js +65 -0
  1189. package/ui-dist/assets/graph-B2qUgzgo.js +1 -0
  1190. package/ui-dist/assets/groovy-D9Dt4D0W.js +1 -0
  1191. package/ui-dist/assets/haskell-Cw1EW3IL.js +1 -0
  1192. package/ui-dist/assets/haxe-H-WmDvRZ.js +1 -0
  1193. package/ui-dist/assets/http-DBlCnlav.js +1 -0
  1194. package/ui-dist/assets/idl-BEugSyMb.js +1 -0
  1195. package/ui-dist/assets/index-6t3ZNnLR.js +1 -0
  1196. package/ui-dist/assets/index-ARl1W2jj.js +1 -0
  1197. package/ui-dist/assets/index-B9T1xuwJ.js +13 -0
  1198. package/ui-dist/assets/index-BAhZurej.js +2 -0
  1199. package/ui-dist/assets/index-BJ9IksfS.js +1 -0
  1200. package/ui-dist/assets/index-BKv2WUK1.js +1 -0
  1201. package/ui-dist/assets/index-BT7YS1Ir.js +3 -0
  1202. package/ui-dist/assets/index-Bkj-OIiI.js +1 -0
  1203. package/ui-dist/assets/index-CNEquZvK.js +1 -0
  1204. package/ui-dist/assets/index-CPGHpqpR.js +1 -0
  1205. package/ui-dist/assets/index-CW2NmfmI.js +1 -0
  1206. package/ui-dist/assets/index-Cf8smU9e.js +1 -0
  1207. package/ui-dist/assets/index-CrrVEpWP.js +537 -0
  1208. package/ui-dist/assets/index-CtebvrC7.js +1 -0
  1209. package/ui-dist/assets/index-CuwS3-lF.js +1 -0
  1210. package/ui-dist/assets/index-D1D5GrGc.js +1 -0
  1211. package/ui-dist/assets/index-DLN9KwFF.css +1 -0
  1212. package/ui-dist/assets/index-DWfkKcGS.js +1 -0
  1213. package/ui-dist/assets/index-DXzbokce.js +6 -0
  1214. package/ui-dist/assets/index-DY03uqeZ.js +1 -0
  1215. package/ui-dist/assets/index-DnDdXZYl.js +1 -0
  1216. package/ui-dist/assets/index-DvRp33UU.js +1 -0
  1217. package/ui-dist/assets/index-FnWH4Q9P.js +7 -0
  1218. package/ui-dist/assets/index-O_WP0QsK.js +1 -0
  1219. package/ui-dist/assets/infoDiagram-HS3SLOUP-DLOW6p4c.js +2 -0
  1220. package/ui-dist/assets/init-Gi6I4Gst.js +1 -0
  1221. package/ui-dist/assets/javascript-iXu5QeM3.js +1 -0
  1222. package/ui-dist/assets/journeyDiagram-XKPGCS4Q-oYaaSMQP.js +139 -0
  1223. package/ui-dist/assets/julia-DuME0IfC.js +1 -0
  1224. package/ui-dist/assets/kanban-definition-3W4ZIXB7-BE7mgk6q.js +89 -0
  1225. package/ui-dist/assets/katex-B95LWT_Q.js +261 -0
  1226. package/ui-dist/assets/layout-Cq5BDWzM.js +1 -0
  1227. package/ui-dist/assets/linear-HnEHTC6F.js +1 -0
  1228. package/ui-dist/assets/livescript-BwQOo05w.js +1 -0
  1229. package/ui-dist/assets/lua-BgMRiT3U.js +1 -0
  1230. package/ui-dist/assets/mathematica-DTrFuWx2.js +1 -0
  1231. package/ui-dist/assets/mbox-CNhZ1qSd.js +1 -0
  1232. package/ui-dist/assets/mermaid.core-JMROPF3S.js +250 -0
  1233. package/ui-dist/assets/mindmap-definition-VGOIOE7T-DoeZEBQU.js +68 -0
  1234. package/ui-dist/assets/mirc-CjQqDB4T.js +1 -0
  1235. package/ui-dist/assets/mllike-CXdrOF99.js +1 -0
  1236. package/ui-dist/assets/modelica-Dc1JOy9r.js +1 -0
  1237. package/ui-dist/assets/mscgen-BA5vi2Kp.js +1 -0
  1238. package/ui-dist/assets/mumps-BT43cFF4.js +1 -0
  1239. package/ui-dist/assets/nginx-DdIZxoE0.js +1 -0
  1240. package/ui-dist/assets/nsis-LdVXkNf5.js +1 -0
  1241. package/ui-dist/assets/ntriples-BfvgReVJ.js +1 -0
  1242. package/ui-dist/assets/octave-Ck1zUtKM.js +1 -0
  1243. package/ui-dist/assets/ordinal-Cboi1Yqb.js +1 -0
  1244. package/ui-dist/assets/oz-BzwKVEFT.js +1 -0
  1245. package/ui-dist/assets/pascal--L3eBynH.js +1 -0
  1246. package/ui-dist/assets/perl-CdXCOZ3F.js +1 -0
  1247. package/ui-dist/assets/pieDiagram-ADFJNKIX-BtfKQ-xw.js +30 -0
  1248. package/ui-dist/assets/pig-CevX1Tat.js +1 -0
  1249. package/ui-dist/assets/powershell-CFHJl5sT.js +1 -0
  1250. package/ui-dist/assets/properties-C78fOPTZ.js +1 -0
  1251. package/ui-dist/assets/protobuf-ChK-085T.js +1 -0
  1252. package/ui-dist/assets/pug-DeIclll2.js +1 -0
  1253. package/ui-dist/assets/puppet-DMA9R1ak.js +1 -0
  1254. package/ui-dist/assets/python-BuPzkPfP.js +1 -0
  1255. package/ui-dist/assets/q-pXgVlZs6.js +1 -0
  1256. package/ui-dist/assets/quadrantDiagram-AYHSOK5B-CIrjVDBB.js +7 -0
  1257. package/ui-dist/assets/r-B6wPVr8A.js +1 -0
  1258. package/ui-dist/assets/requirementDiagram-UZGBJVZJ-CNQ5h-EW.js +64 -0
  1259. package/ui-dist/assets/rpm-CTu-6PCP.js +1 -0
  1260. package/ui-dist/assets/ruby-B2Rjki9n.js +1 -0
  1261. package/ui-dist/assets/sankeyDiagram-TZEHDZUN-CrOSPDz5.js +10 -0
  1262. package/ui-dist/assets/sas-B4kiWyti.js +1 -0
  1263. package/ui-dist/assets/scheme-C41bIUwD.js +1 -0
  1264. package/ui-dist/assets/sequenceDiagram-WL72ISMW-D96XuK7n.js +145 -0
  1265. package/ui-dist/assets/shell-CjFT_Tl9.js +1 -0
  1266. package/ui-dist/assets/sieve-C3Gn_uJK.js +1 -0
  1267. package/ui-dist/assets/simple-mode-GW_nhZxv.js +1 -0
  1268. package/ui-dist/assets/smalltalk-CnHTOXQT.js +1 -0
  1269. package/ui-dist/assets/solr-DehyRSwq.js +1 -0
  1270. package/ui-dist/assets/sparql-DkYu6x3z.js +1 -0
  1271. package/ui-dist/assets/spreadsheet-BCZA_wO0.js +1 -0
  1272. package/ui-dist/assets/sql-D0XecflT.js +1 -0
  1273. package/ui-dist/assets/stateDiagram-FKZM4ZOC-BOf9izkD.js +1 -0
  1274. package/ui-dist/assets/stateDiagram-v2-4FDKWEC3-DEhq019K.js +1 -0
  1275. package/ui-dist/assets/stex-C3f8Ysf7.js +1 -0
  1276. package/ui-dist/assets/stylus-B533Al4x.js +1 -0
  1277. package/ui-dist/assets/swift-BzpIVaGY.js +1 -0
  1278. package/ui-dist/assets/tcl-DVfN8rqt.js +1 -0
  1279. package/ui-dist/assets/textile-CnDTJFAw.js +1 -0
  1280. package/ui-dist/assets/tiddlywiki-DO-Gjzrf.js +1 -0
  1281. package/ui-dist/assets/tiki-DGYXhP31.js +1 -0
  1282. package/ui-dist/assets/timeline-definition-IT6M3QCI-_lAaYcUI.js +61 -0
  1283. package/ui-dist/assets/toml-Bm5Em-hy.js +1 -0
  1284. package/ui-dist/assets/treemap-GDKQZRPO-BS18z8ox.js +154 -0
  1285. package/ui-dist/assets/troff-wAsdV37c.js +1 -0
  1286. package/ui-dist/assets/ttcn-CfJYG6tj.js +1 -0
  1287. package/ui-dist/assets/ttcn-cfg-B9xdYoR4.js +1 -0
  1288. package/ui-dist/assets/turtle-B1tBg_DP.js +1 -0
  1289. package/ui-dist/assets/vb-CmGdzxic.js +1 -0
  1290. package/ui-dist/assets/vbscript-BuJXcnF6.js +1 -0
  1291. package/ui-dist/assets/velocity-D8B20fx6.js +1 -0
  1292. package/ui-dist/assets/verilog-C6RDOZhf.js +1 -0
  1293. package/ui-dist/assets/vhdl-lSbBsy5d.js +1 -0
  1294. package/ui-dist/assets/webidl-ZXfAyPTL.js +1 -0
  1295. package/ui-dist/assets/xquery-DzFWVndE.js +1 -0
  1296. package/ui-dist/assets/xychartDiagram-PRI3JC2R-X82nn6D0.js +7 -0
  1297. package/ui-dist/assets/yacas-BJ4BC0dw.js +1 -0
  1298. package/ui-dist/assets/z80-Hz9HOZM7.js +1 -0
  1299. package/ui-dist/brands/opencode-logo-dark-square.svg +18 -0
  1300. package/ui-dist/brands/opencode-logo-light-square.svg +18 -0
  1301. package/ui-dist/favicon-16x16.png +0 -0
  1302. package/ui-dist/favicon-32x32.png +0 -0
  1303. package/ui-dist/favicon.ico +0 -0
  1304. package/ui-dist/favicon.svg +10 -0
  1305. package/ui-dist/index.html +49 -0
  1306. package/ui-dist/site.webmanifest +30 -0
  1307. package/ui-dist/sw.js +42 -0
  1308. package/ui-dist/worktree-favicon-16x16.png +0 -0
  1309. package/ui-dist/worktree-favicon-32x32.png +0 -0
  1310. package/ui-dist/worktree-favicon.ico +0 -0
  1311. package/ui-dist/worktree-favicon.svg +10 -0
@@ -0,0 +1,1969 @@
1
+ import crypto from "node:crypto";
2
+ import { and, asc, desc, eq, inArray, isNotNull, isNull, lte, not, or, sql } from "drizzle-orm";
3
+ import { agents, companySecretBindings, companySecretVersions, companySecrets, executionWorkspaces, goals, heartbeatRuns, issueInboxArchives, issueReadStates, issues, pluginManagedResources, plugins, projects, routineRevisions, routineRuns, routines, routineTriggers, } from "@dealdesk/db";
4
+ import { WORKSPACE_BRANCH_ROUTINE_VARIABLE, getBuiltinRoutineVariableValues, extractRoutineVariableNames, interpolateRoutineTemplate, pluginOperationIssueOriginKind, stringifyRoutineVariableValue, syncRoutineVariablesWithTemplate, } from "@dealdesk/shared";
5
+ import { trackRoutineRun } from "@dealdesk/shared/telemetry";
6
+ import { conflict, forbidden, notFound, unauthorized, unprocessable } from "../errors.js";
7
+ import { logger } from "../middleware/logger.js";
8
+ import { getTelemetryClient } from "../telemetry.js";
9
+ import { getConfiguredSecretProvider } from "../secrets/configured-provider.js";
10
+ import { issueService } from "./issues.js";
11
+ import { secretService } from "./secrets.js";
12
+ import { getSecretProvider } from "../secrets/provider-registry.js";
13
+ import { parseCron, validateCron } from "./cron.js";
14
+ import { heartbeatService } from "./heartbeat.js";
15
+ import { queueIssueAssignmentWakeup } from "./issue-assignment-wakeup.js";
16
+ import { logActivity } from "./activity-log.js";
17
+ const OPEN_ISSUE_STATUSES = ["backlog", "todo", "in_progress", "in_review", "blocked"];
18
+ const LIVE_HEARTBEAT_RUN_STATUSES = ["queued", "running", "scheduled_retry"];
19
+ const TERMINAL_ISSUE_STATUSES = new Set(["done", "cancelled"]);
20
+ const MAX_CATCH_UP_RUNS = 25;
21
+ const MAX_ROUTINE_REVISIONS = 100;
22
+ const WEEKDAY_INDEX = {
23
+ Sun: 0,
24
+ Mon: 1,
25
+ Tue: 2,
26
+ Wed: 3,
27
+ Thu: 4,
28
+ Fri: 5,
29
+ Sat: 6,
30
+ };
31
+ function routineWebhookSecretConfigPath(secretId) {
32
+ return `webhookSecret:${secretId}`;
33
+ }
34
+ function assertTimeZone(timeZone) {
35
+ try {
36
+ new Intl.DateTimeFormat("en-US", { timeZone }).format(new Date());
37
+ }
38
+ catch {
39
+ throw unprocessable(`Invalid timezone: ${timeZone}`);
40
+ }
41
+ }
42
+ function floorToMinute(date) {
43
+ const copy = new Date(date.getTime());
44
+ copy.setUTCSeconds(0, 0);
45
+ return copy;
46
+ }
47
+ function getZonedMinuteParts(date, timeZone) {
48
+ const formatter = new Intl.DateTimeFormat("en-US", {
49
+ timeZone,
50
+ hour12: false,
51
+ year: "numeric",
52
+ month: "numeric",
53
+ day: "numeric",
54
+ hour: "numeric",
55
+ minute: "numeric",
56
+ weekday: "short",
57
+ });
58
+ const parts = formatter.formatToParts(date);
59
+ const map = Object.fromEntries(parts.map((part) => [part.type, part.value]));
60
+ const weekday = WEEKDAY_INDEX[map.weekday ?? ""];
61
+ if (weekday == null) {
62
+ throw new Error(`Unable to resolve weekday for timezone ${timeZone}`);
63
+ }
64
+ return {
65
+ year: Number(map.year),
66
+ month: Number(map.month),
67
+ day: Number(map.day),
68
+ hour: Number(map.hour),
69
+ minute: Number(map.minute),
70
+ weekday,
71
+ };
72
+ }
73
+ function matchesCronMinute(expression, timeZone, date) {
74
+ const cron = parseCron(expression);
75
+ const parts = getZonedMinuteParts(date, timeZone);
76
+ return (cron.minutes.includes(parts.minute) &&
77
+ cron.hours.includes(parts.hour) &&
78
+ cron.daysOfMonth.includes(parts.day) &&
79
+ cron.months.includes(parts.month) &&
80
+ cron.daysOfWeek.includes(parts.weekday));
81
+ }
82
+ function nextCronTickInTimeZone(expression, timeZone, after) {
83
+ const trimmed = expression.trim();
84
+ assertTimeZone(timeZone);
85
+ const error = validateCron(trimmed);
86
+ if (error) {
87
+ throw unprocessable(error);
88
+ }
89
+ const cursor = floorToMinute(after);
90
+ cursor.setUTCMinutes(cursor.getUTCMinutes() + 1);
91
+ const limit = 366 * 24 * 60 * 5;
92
+ for (let i = 0; i < limit; i += 1) {
93
+ if (matchesCronMinute(trimmed, timeZone, cursor)) {
94
+ return new Date(cursor.getTime());
95
+ }
96
+ cursor.setUTCMinutes(cursor.getUTCMinutes() + 1);
97
+ }
98
+ return null;
99
+ }
100
+ function nextResultText(status, issueId) {
101
+ if (status === "issue_created" && issueId)
102
+ return `Created execution issue ${issueId}`;
103
+ if (status === "coalesced")
104
+ return "Coalesced into an existing live execution issue";
105
+ if (status === "skipped")
106
+ return "Skipped because a live execution issue already exists";
107
+ if (status === "completed")
108
+ return "Execution issue completed";
109
+ if (status === "failed")
110
+ return "Execution failed";
111
+ return status;
112
+ }
113
+ function normalizeWebhookTimestampMs(rawTimestamp) {
114
+ const parsed = Number(rawTimestamp);
115
+ if (!Number.isFinite(parsed))
116
+ return null;
117
+ return parsed > 1e12 ? parsed : parsed * 1000;
118
+ }
119
+ function isPlainRecord(value) {
120
+ return typeof value === "object" && value !== null && !Array.isArray(value);
121
+ }
122
+ function parseBooleanVariableValue(name, raw) {
123
+ if (typeof raw === "boolean")
124
+ return raw;
125
+ if (typeof raw === "number" && (raw === 0 || raw === 1))
126
+ return raw === 1;
127
+ if (typeof raw === "string") {
128
+ const normalized = raw.trim().toLowerCase();
129
+ if (["true", "1", "yes", "y", "on"].includes(normalized))
130
+ return true;
131
+ if (["false", "0", "no", "n", "off"].includes(normalized))
132
+ return false;
133
+ }
134
+ throw unprocessable(`Variable "${name}" must be a boolean`);
135
+ }
136
+ function parseNumberVariableValue(name, raw) {
137
+ if (typeof raw === "number" && Number.isFinite(raw))
138
+ return raw;
139
+ if (typeof raw === "string" && raw.trim().length > 0) {
140
+ const parsed = Number(raw);
141
+ if (Number.isFinite(parsed))
142
+ return parsed;
143
+ }
144
+ throw unprocessable(`Variable "${name}" must be a number`);
145
+ }
146
+ function normalizeRoutineVariableValue(variable, raw) {
147
+ if (raw == null)
148
+ return null;
149
+ if (variable.type === "boolean")
150
+ return parseBooleanVariableValue(variable.name, raw);
151
+ if (variable.type === "number")
152
+ return parseNumberVariableValue(variable.name, raw);
153
+ const normalized = stringifyRoutineVariableValue(raw);
154
+ if (variable.type === "select") {
155
+ if (!variable.options.includes(normalized)) {
156
+ throw unprocessable(`Variable "${variable.name}" must match one of: ${variable.options.join(", ")}`);
157
+ }
158
+ }
159
+ return normalized;
160
+ }
161
+ function isMissingRoutineVariableValue(value) {
162
+ return value == null || (typeof value === "string" && value.trim().length === 0);
163
+ }
164
+ function assertRoutineVariableDefinitions(variables) {
165
+ for (const variable of variables) {
166
+ if (variable.defaultValue != null) {
167
+ normalizeRoutineVariableValue(variable, variable.defaultValue);
168
+ }
169
+ if (variable.type === "select" && variable.options.length === 0) {
170
+ throw unprocessable(`Variable "${variable.name}" must define at least one option`);
171
+ }
172
+ }
173
+ }
174
+ function sanitizeRoutineVariableInputs(variables) {
175
+ return (variables ?? []).map((variable) => ({
176
+ name: variable.name,
177
+ label: variable.label ?? null,
178
+ type: variable.type ?? "text",
179
+ defaultValue: variable.defaultValue ?? null,
180
+ required: variable.required ?? true,
181
+ options: variable.options ?? [],
182
+ }));
183
+ }
184
+ function assertScheduleCompatibleVariables(variables) {
185
+ const missingDefaults = variables
186
+ .filter((variable) => variable.required)
187
+ .filter((variable) => {
188
+ try {
189
+ return isMissingRoutineVariableValue(normalizeRoutineVariableValue(variable, variable.defaultValue));
190
+ }
191
+ catch {
192
+ return true;
193
+ }
194
+ })
195
+ .map((variable) => variable.name);
196
+ if (missingDefaults.length > 0) {
197
+ throw unprocessable(`Scheduled routines require defaults for required variables: ${missingDefaults.join(", ")}`);
198
+ }
199
+ }
200
+ function statusRequiresDefaultAgent(status) {
201
+ return status === "active";
202
+ }
203
+ function normalizeDraftRoutineStatus(status, assigneeAgentId) {
204
+ if (statusRequiresDefaultAgent(status) && !assigneeAgentId) {
205
+ return "paused";
206
+ }
207
+ return status;
208
+ }
209
+ function assertRoutineCanEnable(status, assigneeAgentId) {
210
+ if (statusRequiresDefaultAgent(status) && !assigneeAgentId) {
211
+ throw unprocessable("Default agent required");
212
+ }
213
+ }
214
+ function collectProvidedRoutineVariables(source, payload, variables) {
215
+ const nestedVariables = isPlainRecord(payload) && isPlainRecord(payload.variables) ? payload.variables : {};
216
+ const provided = {
217
+ ...(source === "webhook" && payload ? payload : {}),
218
+ ...nestedVariables,
219
+ ...(variables ?? {}),
220
+ };
221
+ delete provided.variables;
222
+ return provided;
223
+ }
224
+ function resolveRoutineVariableValues(variables, input) {
225
+ if (variables.length === 0)
226
+ return {};
227
+ const provided = collectProvidedRoutineVariables(input.source, input.payload, input.variables);
228
+ const automaticVariables = input.automaticVariables ?? {};
229
+ const resolved = {};
230
+ const missing = [];
231
+ for (const variable of variables) {
232
+ // Workspace-derived automatic values are authoritative for variables that
233
+ // DealDesk manages from execution context, so callers cannot override them.
234
+ const candidate = automaticVariables[variable.name] !== undefined
235
+ ? automaticVariables[variable.name]
236
+ : provided[variable.name] !== undefined
237
+ ? provided[variable.name]
238
+ : variable.defaultValue;
239
+ const normalized = normalizeRoutineVariableValue(variable, candidate);
240
+ if (normalized == null || (typeof normalized === "string" && normalized.trim().length === 0)) {
241
+ if (variable.required)
242
+ missing.push(variable.name);
243
+ continue;
244
+ }
245
+ resolved[variable.name] = normalized;
246
+ }
247
+ if (missing.length > 0) {
248
+ throw unprocessable(`Missing routine variables: ${missing.join(", ")}`);
249
+ }
250
+ return resolved;
251
+ }
252
+ function mergeRoutineRunPayload(payload, variables) {
253
+ if (Object.keys(variables).length === 0)
254
+ return payload ?? null;
255
+ if (!payload)
256
+ return { variables };
257
+ const existingVariables = isPlainRecord(payload.variables) ? payload.variables : {};
258
+ return {
259
+ ...payload,
260
+ variables: {
261
+ ...existingVariables,
262
+ ...variables,
263
+ },
264
+ };
265
+ }
266
+ function normalizeRoutineDispatchFingerprintValue(value) {
267
+ if (value === undefined)
268
+ return null;
269
+ if (value == null || typeof value === "string" || typeof value === "number" || typeof value === "boolean") {
270
+ return value;
271
+ }
272
+ if (value instanceof Date)
273
+ return value.toISOString();
274
+ if (Array.isArray(value))
275
+ return value.map((item) => normalizeRoutineDispatchFingerprintValue(item));
276
+ if (isPlainRecord(value)) {
277
+ return Object.fromEntries(Object.keys(value)
278
+ .sort()
279
+ .map((key) => [key, normalizeRoutineDispatchFingerprintValue(value[key])]));
280
+ }
281
+ return String(value);
282
+ }
283
+ function createRoutineDispatchFingerprint(input) {
284
+ const canonical = JSON.stringify(normalizeRoutineDispatchFingerprintValue(input));
285
+ return crypto.createHash("sha256").update(canonical).digest("hex");
286
+ }
287
+ function readManagedRoutineIssueTemplate(defaultsJson) {
288
+ const value = defaultsJson?.issueTemplate;
289
+ if (!isPlainRecord(value))
290
+ return null;
291
+ return {
292
+ surfaceVisibility: typeof value.surfaceVisibility === "string" ? value.surfaceVisibility : null,
293
+ originId: typeof value.originId === "string" && value.originId.trim() ? value.originId.trim() : null,
294
+ billingCode: typeof value.billingCode === "string" && value.billingCode.trim() ? value.billingCode.trim() : null,
295
+ };
296
+ }
297
+ function routineUsesWorkspaceBranch(routine) {
298
+ return (routine.variables ?? []).some((variable) => variable.name === WORKSPACE_BRANCH_ROUTINE_VARIABLE)
299
+ || extractRoutineVariableNames([routine.title, routine.description]).includes(WORKSPACE_BRANCH_ROUTINE_VARIABLE);
300
+ }
301
+ function routineRevisionSnapshotRoutine(routine) {
302
+ return {
303
+ id: routine.id,
304
+ companyId: routine.companyId,
305
+ projectId: routine.projectId,
306
+ goalId: routine.goalId,
307
+ parentIssueId: routine.parentIssueId,
308
+ title: routine.title,
309
+ description: routine.description,
310
+ assigneeAgentId: routine.assigneeAgentId,
311
+ priority: routine.priority,
312
+ status: routine.status,
313
+ concurrencyPolicy: routine.concurrencyPolicy,
314
+ catchUpPolicy: routine.catchUpPolicy,
315
+ variables: routine.variables ?? [],
316
+ };
317
+ }
318
+ function routineRevisionSnapshotTrigger(trigger) {
319
+ return {
320
+ id: trigger.id,
321
+ kind: trigger.kind,
322
+ label: trigger.label,
323
+ enabled: trigger.enabled,
324
+ cronExpression: trigger.cronExpression,
325
+ timezone: trigger.timezone,
326
+ publicId: trigger.publicId,
327
+ signingMode: trigger.signingMode,
328
+ replayWindowSec: trigger.replayWindowSec,
329
+ };
330
+ }
331
+ async function buildRoutineRevisionSnapshot(executor, routine) {
332
+ const triggers = await executor
333
+ .select()
334
+ .from(routineTriggers)
335
+ .where(and(eq(routineTriggers.companyId, routine.companyId), eq(routineTriggers.routineId, routine.id)))
336
+ .orderBy(asc(routineTriggers.createdAt), asc(routineTriggers.id));
337
+ return {
338
+ version: 1,
339
+ routine: routineRevisionSnapshotRoutine(routine),
340
+ triggers: triggers.map(routineRevisionSnapshotTrigger),
341
+ };
342
+ }
343
+ function canonicalSnapshot(value) {
344
+ return JSON.stringify(value);
345
+ }
346
+ function snapshotsMatch(left, right) {
347
+ return canonicalSnapshot(left) === canonicalSnapshot(right);
348
+ }
349
+ function routineCurrentFieldsMatch(left, right) {
350
+ return snapshotsMatch({ version: 1, routine: routineRevisionSnapshotRoutine(left), triggers: [] }, { version: 1, routine: routineRevisionSnapshotRoutine(right), triggers: [] });
351
+ }
352
+ function mapRoutineRevision(row) {
353
+ return {
354
+ ...row,
355
+ snapshot: row.snapshot,
356
+ };
357
+ }
358
+ export function routineService(db, deps = {}) {
359
+ const issueSvc = issueService(db);
360
+ const secretsSvc = secretService(db);
361
+ const heartbeat = deps.heartbeat ?? heartbeatService(db, {
362
+ pluginWorkerManager: deps.pluginWorkerManager,
363
+ });
364
+ async function getRoutineById(id) {
365
+ return db
366
+ .select()
367
+ .from(routines)
368
+ .where(eq(routines.id, id))
369
+ .then((rows) => rows[0] ?? null);
370
+ }
371
+ async function getManagedRoutineBinding(routine) {
372
+ return db
373
+ .select({
374
+ pluginKey: pluginManagedResources.pluginKey,
375
+ defaultsJson: pluginManagedResources.defaultsJson,
376
+ manifestJson: plugins.manifestJson,
377
+ })
378
+ .from(pluginManagedResources)
379
+ .innerJoin(plugins, eq(pluginManagedResources.pluginId, plugins.id))
380
+ .where(and(eq(pluginManagedResources.companyId, routine.companyId), eq(pluginManagedResources.resourceKind, "routine"), eq(pluginManagedResources.resourceId, routine.id)))
381
+ .then((rows) => rows[0] ?? null);
382
+ }
383
+ async function listManagedRoutineMetadata(routineIds) {
384
+ if (routineIds.length === 0)
385
+ return new Map();
386
+ const rows = await db
387
+ .select({
388
+ id: pluginManagedResources.id,
389
+ pluginId: pluginManagedResources.pluginId,
390
+ pluginKey: pluginManagedResources.pluginKey,
391
+ manifestJson: plugins.manifestJson,
392
+ resourceKey: pluginManagedResources.resourceKey,
393
+ resourceId: pluginManagedResources.resourceId,
394
+ defaultsJson: pluginManagedResources.defaultsJson,
395
+ createdAt: pluginManagedResources.createdAt,
396
+ updatedAt: pluginManagedResources.updatedAt,
397
+ })
398
+ .from(pluginManagedResources)
399
+ .innerJoin(plugins, eq(pluginManagedResources.pluginId, plugins.id))
400
+ .where(and(eq(pluginManagedResources.resourceKind, "routine"), inArray(pluginManagedResources.resourceId, routineIds)));
401
+ return new Map(rows.map((row) => [
402
+ row.resourceId,
403
+ {
404
+ id: row.id,
405
+ pluginId: row.pluginId,
406
+ pluginKey: row.pluginKey,
407
+ pluginDisplayName: row.manifestJson.displayName ?? row.pluginKey,
408
+ resourceKind: "routine",
409
+ resourceKey: row.resourceKey,
410
+ defaultsJson: row.defaultsJson,
411
+ createdAt: row.createdAt,
412
+ updatedAt: row.updatedAt,
413
+ },
414
+ ]));
415
+ }
416
+ async function getTriggerById(id) {
417
+ return db
418
+ .select()
419
+ .from(routineTriggers)
420
+ .where(eq(routineTriggers.id, id))
421
+ .then((rows) => rows[0] ?? null);
422
+ }
423
+ async function appendRoutineRevision(executor, routine, actor, options = {}) {
424
+ const snapshot = await buildRoutineRevisionSnapshot(executor, routine);
425
+ const nextRevisionNumber = routine.latestRevisionId ? routine.latestRevisionNumber + 1 : 1;
426
+ const now = new Date();
427
+ const [revision] = await executor
428
+ .insert(routineRevisions)
429
+ .values({
430
+ companyId: routine.companyId,
431
+ routineId: routine.id,
432
+ revisionNumber: nextRevisionNumber,
433
+ title: snapshot.routine.title,
434
+ description: snapshot.routine.description,
435
+ snapshot,
436
+ changeSummary: options.changeSummary ?? null,
437
+ restoredFromRevisionId: options.restoredFromRevisionId ?? null,
438
+ createdByAgentId: actor.agentId ?? null,
439
+ createdByUserId: actor.userId ?? null,
440
+ createdByRunId: actor.runId ?? null,
441
+ createdAt: now,
442
+ })
443
+ .returning();
444
+ const [updatedRoutine] = await executor
445
+ .update(routines)
446
+ .set({
447
+ latestRevisionId: revision.id,
448
+ latestRevisionNumber: nextRevisionNumber,
449
+ updatedAt: now,
450
+ })
451
+ .where(eq(routines.id, routine.id))
452
+ .returning();
453
+ return {
454
+ routine: updatedRoutine ?? { ...routine, latestRevisionId: revision.id, latestRevisionNumber: nextRevisionNumber, updatedAt: now },
455
+ revision: mapRoutineRevision(revision),
456
+ };
457
+ }
458
+ async function assertRoutineAccess(companyId, routineId) {
459
+ const routine = await getRoutineById(routineId);
460
+ if (!routine)
461
+ throw notFound("Routine not found");
462
+ if (routine.companyId !== companyId)
463
+ throw forbidden("Routine must belong to same company");
464
+ return routine;
465
+ }
466
+ async function assertAssignableAgent(companyId, agentId) {
467
+ if (!agentId)
468
+ return;
469
+ const agent = await db
470
+ .select({ id: agents.id, companyId: agents.companyId, status: agents.status })
471
+ .from(agents)
472
+ .where(eq(agents.id, agentId))
473
+ .then((rows) => rows[0] ?? null);
474
+ if (!agent)
475
+ throw notFound("Assignee agent not found");
476
+ if (agent.companyId !== companyId)
477
+ throw unprocessable("Assignee must belong to same company");
478
+ if (agent.status === "pending_approval")
479
+ throw conflict("Cannot assign routines to pending approval agents");
480
+ if (agent.status === "terminated")
481
+ throw conflict("Cannot assign routines to terminated agents");
482
+ }
483
+ async function assertRestorableAssignee(companyId, assigneeAgentId, actor) {
484
+ await assertAssignableAgent(companyId, assigneeAgentId);
485
+ if (actor.agentId && assigneeAgentId !== actor.agentId) {
486
+ throw forbidden("Agents can only restore routine revisions assigned to themselves");
487
+ }
488
+ }
489
+ async function assertProject(companyId, projectId) {
490
+ if (!projectId)
491
+ return;
492
+ const project = await db
493
+ .select({ id: projects.id, companyId: projects.companyId })
494
+ .from(projects)
495
+ .where(eq(projects.id, projectId))
496
+ .then((rows) => rows[0] ?? null);
497
+ if (!project)
498
+ throw notFound("Project not found");
499
+ if (project.companyId !== companyId)
500
+ throw unprocessable("Project must belong to same company");
501
+ }
502
+ async function assertGoal(companyId, goalId) {
503
+ const goal = await db
504
+ .select({ id: goals.id, companyId: goals.companyId })
505
+ .from(goals)
506
+ .where(eq(goals.id, goalId))
507
+ .then((rows) => rows[0] ?? null);
508
+ if (!goal)
509
+ throw notFound("Goal not found");
510
+ if (goal.companyId !== companyId)
511
+ throw unprocessable("Goal must belong to same company");
512
+ }
513
+ async function assertParentIssue(companyId, issueId) {
514
+ const parentIssue = await db
515
+ .select({ id: issues.id, companyId: issues.companyId })
516
+ .from(issues)
517
+ .where(eq(issues.id, issueId))
518
+ .then((rows) => rows[0] ?? null);
519
+ if (!parentIssue)
520
+ throw notFound("Parent issue not found");
521
+ if (parentIssue.companyId !== companyId)
522
+ throw unprocessable("Parent issue must belong to same company");
523
+ }
524
+ async function listTriggersForRoutineIds(companyId, routineIds) {
525
+ if (routineIds.length === 0)
526
+ return new Map();
527
+ const rows = await db
528
+ .select()
529
+ .from(routineTriggers)
530
+ .where(and(eq(routineTriggers.companyId, companyId), inArray(routineTriggers.routineId, routineIds)))
531
+ .orderBy(asc(routineTriggers.createdAt), asc(routineTriggers.id));
532
+ const map = new Map();
533
+ for (const row of rows) {
534
+ const list = map.get(row.routineId) ?? [];
535
+ list.push(row);
536
+ map.set(row.routineId, list);
537
+ }
538
+ return map;
539
+ }
540
+ async function listLatestRunByRoutineIds(companyId, routineIds) {
541
+ if (routineIds.length === 0)
542
+ return new Map();
543
+ const rows = await db
544
+ .selectDistinctOn([routineRuns.routineId], {
545
+ id: routineRuns.id,
546
+ companyId: routineRuns.companyId,
547
+ routineId: routineRuns.routineId,
548
+ triggerId: routineRuns.triggerId,
549
+ source: routineRuns.source,
550
+ status: routineRuns.status,
551
+ triggeredAt: routineRuns.triggeredAt,
552
+ idempotencyKey: routineRuns.idempotencyKey,
553
+ triggerPayload: routineRuns.triggerPayload,
554
+ dispatchFingerprint: routineRuns.dispatchFingerprint,
555
+ linkedIssueId: routineRuns.linkedIssueId,
556
+ coalescedIntoRunId: routineRuns.coalescedIntoRunId,
557
+ failureReason: routineRuns.failureReason,
558
+ completedAt: routineRuns.completedAt,
559
+ createdAt: routineRuns.createdAt,
560
+ updatedAt: routineRuns.updatedAt,
561
+ triggerKind: routineTriggers.kind,
562
+ triggerLabel: routineTriggers.label,
563
+ issueIdentifier: issues.identifier,
564
+ issueTitle: issues.title,
565
+ issueStatus: issues.status,
566
+ issuePriority: issues.priority,
567
+ issueUpdatedAt: issues.updatedAt,
568
+ })
569
+ .from(routineRuns)
570
+ .leftJoin(routineTriggers, eq(routineRuns.triggerId, routineTriggers.id))
571
+ .leftJoin(issues, eq(routineRuns.linkedIssueId, issues.id))
572
+ .where(and(eq(routineRuns.companyId, companyId), inArray(routineRuns.routineId, routineIds)))
573
+ .orderBy(routineRuns.routineId, desc(routineRuns.createdAt), desc(routineRuns.id));
574
+ const map = new Map();
575
+ for (const row of rows) {
576
+ map.set(row.routineId, {
577
+ id: row.id,
578
+ companyId: row.companyId,
579
+ routineId: row.routineId,
580
+ triggerId: row.triggerId,
581
+ source: row.source,
582
+ status: row.status,
583
+ triggeredAt: row.triggeredAt,
584
+ idempotencyKey: row.idempotencyKey,
585
+ triggerPayload: row.triggerPayload,
586
+ dispatchFingerprint: row.dispatchFingerprint,
587
+ linkedIssueId: row.linkedIssueId,
588
+ coalescedIntoRunId: row.coalescedIntoRunId,
589
+ failureReason: row.failureReason,
590
+ completedAt: row.completedAt,
591
+ createdAt: row.createdAt,
592
+ updatedAt: row.updatedAt,
593
+ linkedIssue: row.linkedIssueId
594
+ ? {
595
+ id: row.linkedIssueId,
596
+ identifier: row.issueIdentifier,
597
+ title: row.issueTitle ?? "Routine execution",
598
+ status: row.issueStatus ?? "todo",
599
+ priority: row.issuePriority ?? "medium",
600
+ updatedAt: row.issueUpdatedAt ?? row.updatedAt,
601
+ }
602
+ : null,
603
+ trigger: row.triggerId
604
+ ? {
605
+ id: row.triggerId,
606
+ kind: row.triggerKind,
607
+ label: row.triggerLabel,
608
+ }
609
+ : null,
610
+ });
611
+ }
612
+ return map;
613
+ }
614
+ async function listLiveIssueByRoutineIds(companyId, routineIds) {
615
+ if (routineIds.length === 0)
616
+ return new Map();
617
+ const executionBoundRows = await db
618
+ .selectDistinctOn([issues.originId], {
619
+ originId: issues.originId,
620
+ id: issues.id,
621
+ identifier: issues.identifier,
622
+ title: issues.title,
623
+ status: issues.status,
624
+ priority: issues.priority,
625
+ updatedAt: issues.updatedAt,
626
+ })
627
+ .from(issues)
628
+ .innerJoin(heartbeatRuns, and(eq(heartbeatRuns.id, issues.executionRunId), inArray(heartbeatRuns.status, LIVE_HEARTBEAT_RUN_STATUSES)))
629
+ .where(and(eq(issues.companyId, companyId), eq(issues.originKind, "routine_execution"), inArray(issues.originId, routineIds), inArray(issues.status, OPEN_ISSUE_STATUSES), isNull(issues.hiddenAt)))
630
+ .orderBy(issues.originId, desc(issues.updatedAt), desc(issues.createdAt));
631
+ const rowsByOriginId = new Map();
632
+ for (const row of executionBoundRows) {
633
+ if (!row.originId)
634
+ continue;
635
+ rowsByOriginId.set(row.originId, row);
636
+ }
637
+ const missingRoutineIds = routineIds.filter((routineId) => !rowsByOriginId.has(routineId));
638
+ if (missingRoutineIds.length > 0) {
639
+ const legacyRows = await db
640
+ .selectDistinctOn([issues.originId], {
641
+ originId: issues.originId,
642
+ id: issues.id,
643
+ identifier: issues.identifier,
644
+ title: issues.title,
645
+ status: issues.status,
646
+ priority: issues.priority,
647
+ updatedAt: issues.updatedAt,
648
+ })
649
+ .from(issues)
650
+ .innerJoin(heartbeatRuns, and(eq(heartbeatRuns.companyId, issues.companyId), inArray(heartbeatRuns.status, LIVE_HEARTBEAT_RUN_STATUSES), sql `${heartbeatRuns.contextSnapshot} ->> 'issueId' = cast(${issues.id} as text)`))
651
+ .where(and(eq(issues.companyId, companyId), eq(issues.originKind, "routine_execution"), inArray(issues.originId, missingRoutineIds), inArray(issues.status, OPEN_ISSUE_STATUSES), isNull(issues.hiddenAt)))
652
+ .orderBy(issues.originId, desc(issues.updatedAt), desc(issues.createdAt));
653
+ for (const row of legacyRows) {
654
+ if (!row.originId)
655
+ continue;
656
+ rowsByOriginId.set(row.originId, row);
657
+ }
658
+ }
659
+ const map = new Map();
660
+ for (const row of rowsByOriginId.values()) {
661
+ if (!row.originId)
662
+ continue;
663
+ map.set(row.originId, {
664
+ id: row.id,
665
+ identifier: row.identifier,
666
+ title: row.title,
667
+ status: row.status,
668
+ priority: row.priority,
669
+ updatedAt: row.updatedAt,
670
+ });
671
+ }
672
+ return map;
673
+ }
674
+ async function updateRoutineTouchedState(input, executor = db) {
675
+ await executor
676
+ .update(routines)
677
+ .set({
678
+ lastTriggeredAt: input.triggeredAt,
679
+ lastEnqueuedAt: input.issueId ? input.triggeredAt : undefined,
680
+ updatedAt: new Date(),
681
+ })
682
+ .where(eq(routines.id, input.routineId));
683
+ if (input.triggerId) {
684
+ await executor
685
+ .update(routineTriggers)
686
+ .set({
687
+ lastFiredAt: input.triggeredAt,
688
+ lastResult: nextResultText(input.status, input.issueId),
689
+ nextRunAt: input.nextRunAt === undefined ? undefined : input.nextRunAt,
690
+ updatedAt: new Date(),
691
+ })
692
+ .where(eq(routineTriggers.id, input.triggerId));
693
+ }
694
+ }
695
+ function routineExecutionFingerprintCondition(dispatchFingerprint) {
696
+ if (!dispatchFingerprint)
697
+ return null;
698
+ // The "default" arm preserves coalescing against pre-migration open issues.
699
+ // It becomes inert once those legacy routine execution issues drain out.
700
+ return or(eq(issues.originFingerprint, dispatchFingerprint), eq(issues.originFingerprint, "default"));
701
+ }
702
+ async function findLiveExecutionIssue(routine, executor = db, dispatchFingerprint, origin) {
703
+ const fingerprintCondition = routineExecutionFingerprintCondition(dispatchFingerprint);
704
+ const originKind = origin?.kind ?? "routine_execution";
705
+ const originId = origin?.id ?? routine.id;
706
+ const executionBoundIssue = await executor
707
+ .select()
708
+ .from(issues)
709
+ .innerJoin(heartbeatRuns, and(eq(heartbeatRuns.id, issues.executionRunId), inArray(heartbeatRuns.status, LIVE_HEARTBEAT_RUN_STATUSES)))
710
+ .where(and(eq(issues.companyId, routine.companyId), eq(issues.originKind, originKind), eq(issues.originId, originId), inArray(issues.status, OPEN_ISSUE_STATUSES), isNull(issues.hiddenAt), ...(fingerprintCondition ? [fingerprintCondition] : [])))
711
+ .orderBy(desc(issues.updatedAt), desc(issues.createdAt))
712
+ .limit(1)
713
+ .then((rows) => rows[0]?.issues ?? null);
714
+ if (executionBoundIssue)
715
+ return executionBoundIssue;
716
+ return executor
717
+ .select()
718
+ .from(issues)
719
+ .innerJoin(heartbeatRuns, and(eq(heartbeatRuns.companyId, issues.companyId), inArray(heartbeatRuns.status, LIVE_HEARTBEAT_RUN_STATUSES), sql `${heartbeatRuns.contextSnapshot} ->> 'issueId' = cast(${issues.id} as text)`))
720
+ .where(and(eq(issues.companyId, routine.companyId), eq(issues.originKind, originKind), eq(issues.originId, originId), inArray(issues.status, OPEN_ISSUE_STATUSES), isNull(issues.hiddenAt), ...(fingerprintCondition ? [fingerprintCondition] : [])))
721
+ .orderBy(desc(issues.updatedAt), desc(issues.createdAt))
722
+ .limit(1)
723
+ .then((rows) => rows[0]?.issues ?? null);
724
+ }
725
+ async function finalizeRun(runId, patch, executor = db) {
726
+ return executor
727
+ .update(routineRuns)
728
+ .set({
729
+ ...patch,
730
+ updatedAt: new Date(),
731
+ })
732
+ .where(eq(routineRuns.id, runId))
733
+ .returning()
734
+ .then((rows) => rows[0] ?? null);
735
+ }
736
+ async function createWebhookSecret(companyId, routineId, actor, executor) {
737
+ const secretValue = crypto.randomBytes(24).toString("hex");
738
+ const providerId = getConfiguredSecretProvider();
739
+ const input = {
740
+ name: `routine-${routineId}-${crypto.randomBytes(6).toString("hex")}`,
741
+ provider: providerId,
742
+ value: secretValue,
743
+ description: `Webhook auth for routine ${routineId}`,
744
+ };
745
+ const provider = getSecretProvider(input.provider);
746
+ const prepared = await provider.createSecret({
747
+ value: input.value,
748
+ externalRef: null,
749
+ context: {
750
+ companyId,
751
+ secretKey: input.name,
752
+ secretName: input.name,
753
+ version: 1,
754
+ },
755
+ });
756
+ const insertSecret = async (secretDb) => {
757
+ const secret = await secretDb
758
+ .insert(companySecrets)
759
+ .values({
760
+ companyId,
761
+ key: input.name,
762
+ name: input.name,
763
+ provider: input.provider,
764
+ status: "active",
765
+ managedMode: "dealdesk_managed",
766
+ externalRef: prepared.externalRef,
767
+ providerMetadata: null,
768
+ latestVersion: 1,
769
+ description: input.description,
770
+ lastRotatedAt: new Date(),
771
+ createdByAgentId: actor.agentId ?? null,
772
+ createdByUserId: actor.userId ?? null,
773
+ })
774
+ .returning()
775
+ .then((rows) => rows[0]);
776
+ await secretDb.insert(companySecretVersions).values({
777
+ secretId: secret.id,
778
+ version: 1,
779
+ material: prepared.material,
780
+ valueSha256: prepared.valueSha256,
781
+ fingerprintSha256: prepared.fingerprintSha256 ?? prepared.valueSha256,
782
+ providerVersionRef: prepared.providerVersionRef ?? null,
783
+ status: "current",
784
+ createdByAgentId: actor.agentId ?? null,
785
+ createdByUserId: actor.userId ?? null,
786
+ });
787
+ await secretDb.insert(companySecretBindings).values({
788
+ companyId,
789
+ secretId: secret.id,
790
+ targetType: "routine",
791
+ targetId: routineId,
792
+ configPath: routineWebhookSecretConfigPath(secret.id),
793
+ });
794
+ return secret;
795
+ };
796
+ const secret = executor
797
+ ? await insertSecret(executor)
798
+ : await db.transaction(async (tx) => insertSecret(tx));
799
+ return { secret, secretValue };
800
+ }
801
+ async function resolveTriggerSecret(trigger, companyId) {
802
+ if (!trigger.secretId)
803
+ throw notFound("Routine trigger secret not found");
804
+ const secret = await db
805
+ .select()
806
+ .from(companySecrets)
807
+ .where(eq(companySecrets.id, trigger.secretId))
808
+ .then((rows) => rows[0] ?? null);
809
+ if (!secret || secret.companyId !== companyId)
810
+ throw notFound("Routine trigger secret not found");
811
+ const value = await secretsSvc.resolveSecretValue(companyId, trigger.secretId, "latest", {
812
+ consumerType: "routine",
813
+ consumerId: trigger.routineId,
814
+ actorType: "system",
815
+ actorId: null,
816
+ configPath: routineWebhookSecretConfigPath(trigger.secretId),
817
+ });
818
+ return value;
819
+ }
820
+ async function touchIssueForUserInbox(executor, input) {
821
+ await executor
822
+ .insert(issueReadStates)
823
+ .values({
824
+ companyId: input.companyId,
825
+ issueId: input.issueId,
826
+ userId: input.userId,
827
+ lastReadAt: input.touchedAt,
828
+ updatedAt: input.touchedAt,
829
+ })
830
+ .onConflictDoUpdate({
831
+ target: [issueReadStates.companyId, issueReadStates.issueId, issueReadStates.userId],
832
+ set: {
833
+ lastReadAt: input.touchedAt,
834
+ updatedAt: input.touchedAt,
835
+ },
836
+ });
837
+ await executor
838
+ .delete(issueInboxArchives)
839
+ .where(and(eq(issueInboxArchives.companyId, input.companyId), eq(issueInboxArchives.issueId, input.issueId), eq(issueInboxArchives.userId, input.userId)));
840
+ }
841
+ async function dispatchRoutineRun(input) {
842
+ const projectId = input.projectId ?? input.routine.projectId ?? null;
843
+ const assigneeAgentId = input.assigneeAgentId ?? input.routine.assigneeAgentId ?? null;
844
+ if (!assigneeAgentId) {
845
+ throw unprocessable("Default agent required");
846
+ }
847
+ const automaticVariables = {};
848
+ if (input.executionWorkspaceId && routineUsesWorkspaceBranch(input.routine)) {
849
+ const workspace = await db
850
+ .select({
851
+ branchName: executionWorkspaces.branchName,
852
+ mode: executionWorkspaces.mode,
853
+ })
854
+ .from(executionWorkspaces)
855
+ .where(and(eq(executionWorkspaces.id, input.executionWorkspaceId), eq(executionWorkspaces.companyId, input.routine.companyId)))
856
+ .then((rows) => rows[0] ?? null);
857
+ const branchName = workspace?.branchName?.trim();
858
+ if (workspace && workspace.mode !== "shared_workspace" && branchName) {
859
+ automaticVariables[WORKSPACE_BRANCH_ROUTINE_VARIABLE] = branchName;
860
+ }
861
+ }
862
+ const resolvedVariables = resolveRoutineVariableValues(input.routine.variables ?? [], {
863
+ ...input,
864
+ automaticVariables,
865
+ });
866
+ const allVariables = { ...getBuiltinRoutineVariableValues(), ...automaticVariables, ...resolvedVariables };
867
+ const title = interpolateRoutineTemplate(input.routine.title, allVariables) ?? input.routine.title;
868
+ const description = interpolateRoutineTemplate(input.routine.description, allVariables);
869
+ const triggerPayload = mergeRoutineRunPayload(input.payload, { ...automaticVariables, ...resolvedVariables });
870
+ const managedRoutineBinding = await getManagedRoutineBinding(input.routine);
871
+ const managedIssueTemplate = readManagedRoutineIssueTemplate(managedRoutineBinding?.defaultsJson);
872
+ const issueOriginKind = managedIssueTemplate?.surfaceVisibility === "plugin_operation" && managedRoutineBinding
873
+ ? pluginOperationIssueOriginKind(managedRoutineBinding.pluginKey)
874
+ : "routine_execution";
875
+ const issueOriginId = managedIssueTemplate?.originId ?? input.routine.id;
876
+ const issueBillingCode = managedIssueTemplate?.billingCode ?? null;
877
+ const dispatchFingerprint = createRoutineDispatchFingerprint({
878
+ payload: triggerPayload,
879
+ projectId,
880
+ assigneeAgentId,
881
+ executionWorkspaceId: input.executionWorkspaceId ?? null,
882
+ executionWorkspacePreference: input.executionWorkspacePreference ?? null,
883
+ executionWorkspaceSettings: input.executionWorkspaceSettings ?? null,
884
+ title,
885
+ description,
886
+ });
887
+ const run = await db.transaction(async (tx) => {
888
+ const txDb = tx;
889
+ await tx.execute(sql `select id from ${routines} where ${routines.id} = ${input.routine.id} and ${routines.companyId} = ${input.routine.companyId} for update`);
890
+ if (input.idempotencyKey) {
891
+ const existing = await txDb
892
+ .select()
893
+ .from(routineRuns)
894
+ .where(and(eq(routineRuns.companyId, input.routine.companyId), eq(routineRuns.routineId, input.routine.id), eq(routineRuns.source, input.source), eq(routineRuns.idempotencyKey, input.idempotencyKey), input.trigger ? eq(routineRuns.triggerId, input.trigger.id) : isNull(routineRuns.triggerId)))
895
+ .orderBy(desc(routineRuns.createdAt))
896
+ .limit(1)
897
+ .then((rows) => rows[0] ?? null);
898
+ if (existing)
899
+ return existing;
900
+ }
901
+ const triggeredAt = new Date();
902
+ const manualRunnerUserId = input.source === "manual" ? input.actor?.userId ?? null : null;
903
+ const [createdRun] = await txDb
904
+ .insert(routineRuns)
905
+ .values({
906
+ companyId: input.routine.companyId,
907
+ routineId: input.routine.id,
908
+ triggerId: input.trigger?.id ?? null,
909
+ source: input.source,
910
+ status: "received",
911
+ triggeredAt,
912
+ idempotencyKey: input.idempotencyKey ?? null,
913
+ triggerPayload,
914
+ dispatchFingerprint,
915
+ })
916
+ .returning();
917
+ const nextRunAt = input.trigger?.kind === "schedule" && input.trigger.cronExpression && input.trigger.timezone
918
+ ? nextCronTickInTimeZone(input.trigger.cronExpression, input.trigger.timezone, triggeredAt)
919
+ : undefined;
920
+ let createdIssue = null;
921
+ try {
922
+ const activeIssue = await findLiveExecutionIssue(input.routine, txDb, dispatchFingerprint, {
923
+ kind: issueOriginKind,
924
+ id: issueOriginId,
925
+ });
926
+ if (activeIssue && input.routine.concurrencyPolicy !== "always_enqueue") {
927
+ const status = input.routine.concurrencyPolicy === "skip_if_active" ? "skipped" : "coalesced";
928
+ if (manualRunnerUserId) {
929
+ await touchIssueForUserInbox(txDb, {
930
+ companyId: input.routine.companyId,
931
+ issueId: activeIssue.id,
932
+ userId: manualRunnerUserId,
933
+ touchedAt: triggeredAt,
934
+ });
935
+ }
936
+ const updated = await finalizeRun(createdRun.id, {
937
+ status,
938
+ linkedIssueId: activeIssue.id,
939
+ coalescedIntoRunId: activeIssue.originRunId,
940
+ completedAt: triggeredAt,
941
+ }, txDb);
942
+ await updateRoutineTouchedState({
943
+ routineId: input.routine.id,
944
+ triggerId: input.trigger?.id ?? null,
945
+ triggeredAt,
946
+ status,
947
+ issueId: activeIssue.id,
948
+ nextRunAt,
949
+ }, txDb);
950
+ return updated ?? createdRun;
951
+ }
952
+ try {
953
+ createdIssue = await issueSvc.create(input.routine.companyId, {
954
+ projectId,
955
+ goalId: input.routine.goalId,
956
+ parentId: input.routine.parentIssueId,
957
+ title,
958
+ description,
959
+ status: "todo",
960
+ priority: input.routine.priority,
961
+ assigneeAgentId,
962
+ createdByAgentId: input.source === "manual" ? input.actor?.agentId ?? null : null,
963
+ createdByUserId: manualRunnerUserId,
964
+ originKind: issueOriginKind,
965
+ originId: issueOriginId,
966
+ originRunId: createdRun.id,
967
+ originFingerprint: dispatchFingerprint,
968
+ billingCode: issueBillingCode,
969
+ executionWorkspaceId: input.executionWorkspaceId ?? null,
970
+ executionWorkspacePreference: input.executionWorkspacePreference ?? null,
971
+ executionWorkspaceSettings: input.executionWorkspaceSettings ?? null,
972
+ });
973
+ }
974
+ catch (error) {
975
+ const isOpenExecutionConflict = !!error &&
976
+ typeof error === "object" &&
977
+ "code" in error &&
978
+ error.code === "23505" &&
979
+ "constraint" in error &&
980
+ error.constraint === "issues_open_routine_execution_uq";
981
+ if (!isOpenExecutionConflict || input.routine.concurrencyPolicy === "always_enqueue") {
982
+ throw error;
983
+ }
984
+ const existingIssue = await findLiveExecutionIssue(input.routine, txDb, dispatchFingerprint, {
985
+ kind: issueOriginKind,
986
+ id: issueOriginId,
987
+ });
988
+ if (!existingIssue)
989
+ throw error;
990
+ const status = input.routine.concurrencyPolicy === "skip_if_active" ? "skipped" : "coalesced";
991
+ if (manualRunnerUserId) {
992
+ await touchIssueForUserInbox(txDb, {
993
+ companyId: input.routine.companyId,
994
+ issueId: existingIssue.id,
995
+ userId: manualRunnerUserId,
996
+ touchedAt: triggeredAt,
997
+ });
998
+ }
999
+ const updated = await finalizeRun(createdRun.id, {
1000
+ status,
1001
+ linkedIssueId: existingIssue.id,
1002
+ coalescedIntoRunId: existingIssue.originRunId,
1003
+ completedAt: triggeredAt,
1004
+ }, txDb);
1005
+ await updateRoutineTouchedState({
1006
+ routineId: input.routine.id,
1007
+ triggerId: input.trigger?.id ?? null,
1008
+ triggeredAt,
1009
+ status,
1010
+ issueId: existingIssue.id,
1011
+ nextRunAt,
1012
+ }, txDb);
1013
+ return updated ?? createdRun;
1014
+ }
1015
+ // Keep the dispatch lock until the issue is linked to a queued heartbeat run.
1016
+ await queueIssueAssignmentWakeup({
1017
+ heartbeat,
1018
+ issue: createdIssue,
1019
+ reason: "issue_assigned",
1020
+ mutation: "create",
1021
+ contextSource: "routine.dispatch",
1022
+ requestedByActorType: input.source === "schedule" ? "system" : undefined,
1023
+ rethrowOnError: true,
1024
+ });
1025
+ const updated = await finalizeRun(createdRun.id, {
1026
+ status: "issue_created",
1027
+ linkedIssueId: createdIssue.id,
1028
+ }, txDb);
1029
+ await updateRoutineTouchedState({
1030
+ routineId: input.routine.id,
1031
+ triggerId: input.trigger?.id ?? null,
1032
+ triggeredAt,
1033
+ status: "issue_created",
1034
+ issueId: createdIssue.id,
1035
+ nextRunAt,
1036
+ }, txDb);
1037
+ return updated ?? createdRun;
1038
+ }
1039
+ catch (error) {
1040
+ if (createdIssue) {
1041
+ await txDb.delete(issues).where(eq(issues.id, createdIssue.id));
1042
+ }
1043
+ const failureReason = error instanceof Error ? error.message : String(error);
1044
+ const failed = await finalizeRun(createdRun.id, {
1045
+ status: "failed",
1046
+ failureReason,
1047
+ completedAt: new Date(),
1048
+ }, txDb);
1049
+ await updateRoutineTouchedState({
1050
+ routineId: input.routine.id,
1051
+ triggerId: input.trigger?.id ?? null,
1052
+ triggeredAt,
1053
+ status: "failed",
1054
+ nextRunAt,
1055
+ }, txDb);
1056
+ return failed ?? createdRun;
1057
+ }
1058
+ });
1059
+ if (input.source === "schedule" || input.source === "webhook") {
1060
+ const actorId = input.source === "schedule" ? "routine-scheduler" : "routine-webhook";
1061
+ try {
1062
+ await logActivity(db, {
1063
+ companyId: input.routine.companyId,
1064
+ actorType: "system",
1065
+ actorId,
1066
+ action: "routine.run_triggered",
1067
+ entityType: "routine_run",
1068
+ entityId: run.id,
1069
+ details: {
1070
+ routineId: input.routine.id,
1071
+ triggerId: input.trigger?.id ?? null,
1072
+ source: run.source,
1073
+ status: run.status,
1074
+ },
1075
+ });
1076
+ }
1077
+ catch (err) {
1078
+ logger.warn({ err, routineId: input.routine.id, runId: run.id }, "failed to log automated routine run");
1079
+ }
1080
+ }
1081
+ const telemetryClient = getTelemetryClient();
1082
+ if (telemetryClient) {
1083
+ trackRoutineRun(telemetryClient, {
1084
+ source: run.source,
1085
+ status: run.status,
1086
+ });
1087
+ }
1088
+ return run;
1089
+ }
1090
+ return {
1091
+ get: getRoutineById,
1092
+ getTrigger: getTriggerById,
1093
+ list: async (companyId, filters) => {
1094
+ const conditions = [eq(routines.companyId, companyId)];
1095
+ if (filters?.projectId)
1096
+ conditions.push(eq(routines.projectId, filters.projectId));
1097
+ const rows = await db
1098
+ .select()
1099
+ .from(routines)
1100
+ .where(and(...conditions))
1101
+ .orderBy(desc(routines.updatedAt), asc(routines.title));
1102
+ const routineIds = rows.map((row) => row.id);
1103
+ const [triggersByRoutine, latestRunByRoutine, activeIssueByRoutine, managedByRoutine] = await Promise.all([
1104
+ listTriggersForRoutineIds(companyId, routineIds),
1105
+ listLatestRunByRoutineIds(companyId, routineIds),
1106
+ listLiveIssueByRoutineIds(companyId, routineIds),
1107
+ listManagedRoutineMetadata(routineIds),
1108
+ ]);
1109
+ return rows.map((row) => ({
1110
+ ...row,
1111
+ managedByPlugin: managedByRoutine.get(row.id) ?? null,
1112
+ triggers: (triggersByRoutine.get(row.id) ?? []).map((trigger) => ({
1113
+ id: trigger.id,
1114
+ kind: trigger.kind,
1115
+ label: trigger.label,
1116
+ enabled: trigger.enabled,
1117
+ cronExpression: trigger.cronExpression,
1118
+ timezone: trigger.timezone,
1119
+ nextRunAt: trigger.nextRunAt,
1120
+ lastFiredAt: trigger.lastFiredAt,
1121
+ lastResult: trigger.lastResult,
1122
+ })),
1123
+ lastRun: latestRunByRoutine.get(row.id) ?? null,
1124
+ activeIssue: activeIssueByRoutine.get(row.id) ?? null,
1125
+ }));
1126
+ },
1127
+ getDetail: async (id) => {
1128
+ const row = await getRoutineById(id);
1129
+ if (!row)
1130
+ return null;
1131
+ const [project, assignee, parentIssue, triggers, recentRuns, activeIssue, managedByRoutine] = await Promise.all([
1132
+ row.projectId
1133
+ ? db.select().from(projects).where(eq(projects.id, row.projectId)).then((rows) => rows[0] ?? null)
1134
+ : null,
1135
+ row.assigneeAgentId
1136
+ ? db.select().from(agents).where(eq(agents.id, row.assigneeAgentId)).then((rows) => rows[0] ?? null)
1137
+ : null,
1138
+ row.parentIssueId ? issueSvc.getById(row.parentIssueId) : null,
1139
+ db.select().from(routineTriggers).where(eq(routineTriggers.routineId, row.id)).orderBy(asc(routineTriggers.createdAt)),
1140
+ db
1141
+ .select({
1142
+ id: routineRuns.id,
1143
+ companyId: routineRuns.companyId,
1144
+ routineId: routineRuns.routineId,
1145
+ triggerId: routineRuns.triggerId,
1146
+ source: routineRuns.source,
1147
+ status: routineRuns.status,
1148
+ triggeredAt: routineRuns.triggeredAt,
1149
+ idempotencyKey: routineRuns.idempotencyKey,
1150
+ triggerPayload: routineRuns.triggerPayload,
1151
+ dispatchFingerprint: routineRuns.dispatchFingerprint,
1152
+ linkedIssueId: routineRuns.linkedIssueId,
1153
+ coalescedIntoRunId: routineRuns.coalescedIntoRunId,
1154
+ failureReason: routineRuns.failureReason,
1155
+ completedAt: routineRuns.completedAt,
1156
+ createdAt: routineRuns.createdAt,
1157
+ updatedAt: routineRuns.updatedAt,
1158
+ triggerKind: routineTriggers.kind,
1159
+ triggerLabel: routineTriggers.label,
1160
+ issueIdentifier: issues.identifier,
1161
+ issueTitle: issues.title,
1162
+ issueStatus: issues.status,
1163
+ issuePriority: issues.priority,
1164
+ issueUpdatedAt: issues.updatedAt,
1165
+ })
1166
+ .from(routineRuns)
1167
+ .leftJoin(routineTriggers, eq(routineRuns.triggerId, routineTriggers.id))
1168
+ .leftJoin(issues, eq(routineRuns.linkedIssueId, issues.id))
1169
+ .where(eq(routineRuns.routineId, row.id))
1170
+ .orderBy(desc(routineRuns.createdAt))
1171
+ .limit(25)
1172
+ .then((runs) => runs.map((run) => ({
1173
+ id: run.id,
1174
+ companyId: run.companyId,
1175
+ routineId: run.routineId,
1176
+ triggerId: run.triggerId,
1177
+ source: run.source,
1178
+ status: run.status,
1179
+ triggeredAt: run.triggeredAt,
1180
+ idempotencyKey: run.idempotencyKey,
1181
+ triggerPayload: run.triggerPayload,
1182
+ dispatchFingerprint: run.dispatchFingerprint,
1183
+ linkedIssueId: run.linkedIssueId,
1184
+ coalescedIntoRunId: run.coalescedIntoRunId,
1185
+ failureReason: run.failureReason,
1186
+ completedAt: run.completedAt,
1187
+ createdAt: run.createdAt,
1188
+ updatedAt: run.updatedAt,
1189
+ linkedIssue: run.linkedIssueId
1190
+ ? {
1191
+ id: run.linkedIssueId,
1192
+ identifier: run.issueIdentifier,
1193
+ title: run.issueTitle ?? "Routine execution",
1194
+ status: run.issueStatus ?? "todo",
1195
+ priority: run.issuePriority ?? "medium",
1196
+ updatedAt: run.issueUpdatedAt ?? run.updatedAt,
1197
+ }
1198
+ : null,
1199
+ trigger: run.triggerId
1200
+ ? {
1201
+ id: run.triggerId,
1202
+ kind: run.triggerKind,
1203
+ label: run.triggerLabel,
1204
+ }
1205
+ : null,
1206
+ }))),
1207
+ findLiveExecutionIssue(row),
1208
+ listManagedRoutineMetadata([row.id]),
1209
+ ]);
1210
+ return {
1211
+ ...row,
1212
+ managedByPlugin: managedByRoutine.get(row.id) ?? null,
1213
+ project,
1214
+ assignee,
1215
+ parentIssue,
1216
+ triggers: triggers,
1217
+ recentRuns,
1218
+ activeIssue,
1219
+ };
1220
+ },
1221
+ create: async (companyId, input, actor) => {
1222
+ await assertProject(companyId, input.projectId ?? null);
1223
+ await assertAssignableAgent(companyId, input.assigneeAgentId ?? null);
1224
+ if (input.goalId)
1225
+ await assertGoal(companyId, input.goalId);
1226
+ if (input.parentIssueId)
1227
+ await assertParentIssue(companyId, input.parentIssueId);
1228
+ const variables = syncRoutineVariablesWithTemplate([input.title, input.description], sanitizeRoutineVariableInputs(input.variables));
1229
+ assertRoutineVariableDefinitions(variables);
1230
+ const status = normalizeDraftRoutineStatus(input.status, input.assigneeAgentId);
1231
+ return db.transaction(async (tx) => {
1232
+ const txDb = tx;
1233
+ const [created] = await txDb
1234
+ .insert(routines)
1235
+ .values({
1236
+ companyId,
1237
+ projectId: input.projectId ?? null,
1238
+ goalId: input.goalId ?? null,
1239
+ parentIssueId: input.parentIssueId ?? null,
1240
+ title: input.title,
1241
+ description: input.description ?? null,
1242
+ assigneeAgentId: input.assigneeAgentId ?? null,
1243
+ priority: input.priority,
1244
+ status,
1245
+ concurrencyPolicy: input.concurrencyPolicy,
1246
+ catchUpPolicy: input.catchUpPolicy,
1247
+ variables,
1248
+ createdByAgentId: actor.agentId ?? null,
1249
+ createdByUserId: actor.userId ?? null,
1250
+ updatedByAgentId: actor.agentId ?? null,
1251
+ updatedByUserId: actor.userId ?? null,
1252
+ })
1253
+ .returning();
1254
+ const { routine } = await appendRoutineRevision(txDb, created, actor, {
1255
+ changeSummary: "Created routine",
1256
+ });
1257
+ return routine;
1258
+ });
1259
+ },
1260
+ update: async (id, patch, actor) => {
1261
+ const existing = await getRoutineById(id);
1262
+ if (!existing)
1263
+ return null;
1264
+ const nextProjectId = patch.projectId === undefined ? existing.projectId : patch.projectId;
1265
+ const nextAssigneeAgentId = patch.assigneeAgentId === undefined ? existing.assigneeAgentId : patch.assigneeAgentId;
1266
+ const nextTitle = patch.title ?? existing.title;
1267
+ const nextDescription = patch.description === undefined ? existing.description : patch.description;
1268
+ const requestedStatus = patch.status ?? existing.status;
1269
+ if (patch.status === "active") {
1270
+ assertRoutineCanEnable(patch.status, nextAssigneeAgentId);
1271
+ }
1272
+ const nextStatus = patch.assigneeAgentId === undefined
1273
+ ? requestedStatus
1274
+ : normalizeDraftRoutineStatus(requestedStatus, nextAssigneeAgentId);
1275
+ const nextVariables = syncRoutineVariablesWithTemplate([nextTitle, nextDescription], patch.variables === undefined ? existing.variables : sanitizeRoutineVariableInputs(patch.variables));
1276
+ if (patch.projectId !== undefined)
1277
+ await assertProject(existing.companyId, nextProjectId);
1278
+ if (patch.assigneeAgentId !== undefined)
1279
+ await assertAssignableAgent(existing.companyId, nextAssigneeAgentId);
1280
+ if (patch.goalId)
1281
+ await assertGoal(existing.companyId, patch.goalId);
1282
+ if (patch.parentIssueId)
1283
+ await assertParentIssue(existing.companyId, patch.parentIssueId);
1284
+ assertRoutineVariableDefinitions(nextVariables);
1285
+ const enabledScheduleTriggers = await db
1286
+ .select({ id: routineTriggers.id })
1287
+ .from(routineTriggers)
1288
+ .where(and(eq(routineTriggers.routineId, existing.id), eq(routineTriggers.kind, "schedule"), eq(routineTriggers.enabled, true)))
1289
+ .limit(1)
1290
+ .then((rows) => rows.length > 0);
1291
+ if (enabledScheduleTriggers) {
1292
+ assertScheduleCompatibleVariables(nextVariables);
1293
+ }
1294
+ return db.transaction(async (tx) => {
1295
+ const txDb = tx;
1296
+ await tx.execute(sql `select id from ${routines} where ${routines.id} = ${id} for update`);
1297
+ const locked = await txDb
1298
+ .select()
1299
+ .from(routines)
1300
+ .where(eq(routines.id, id))
1301
+ .then((rows) => rows[0] ?? null);
1302
+ if (!locked)
1303
+ return null;
1304
+ if (patch.baseRevisionId && patch.baseRevisionId !== locked.latestRevisionId) {
1305
+ throw conflict("Routine was updated by someone else", {
1306
+ currentRevisionId: locked.latestRevisionId,
1307
+ });
1308
+ }
1309
+ const candidate = {
1310
+ ...locked,
1311
+ projectId: nextProjectId,
1312
+ goalId: patch.goalId === undefined ? locked.goalId : patch.goalId,
1313
+ parentIssueId: patch.parentIssueId === undefined ? locked.parentIssueId : patch.parentIssueId,
1314
+ title: nextTitle,
1315
+ description: nextDescription,
1316
+ assigneeAgentId: nextAssigneeAgentId,
1317
+ priority: patch.priority ?? locked.priority,
1318
+ status: nextStatus,
1319
+ concurrencyPolicy: patch.concurrencyPolicy ?? locked.concurrencyPolicy,
1320
+ catchUpPolicy: patch.catchUpPolicy ?? locked.catchUpPolicy,
1321
+ variables: nextVariables,
1322
+ updatedByAgentId: actor.agentId ?? null,
1323
+ updatedByUserId: actor.userId ?? null,
1324
+ };
1325
+ if (locked.latestRevisionId && routineCurrentFieldsMatch(locked, candidate)) {
1326
+ return locked;
1327
+ }
1328
+ const nextSnapshot = await buildRoutineRevisionSnapshot(txDb, candidate);
1329
+ if (locked.latestRevisionId) {
1330
+ const latestRevision = await txDb
1331
+ .select({ snapshot: routineRevisions.snapshot })
1332
+ .from(routineRevisions)
1333
+ .where(and(eq(routineRevisions.companyId, locked.companyId), eq(routineRevisions.routineId, locked.id), eq(routineRevisions.id, locked.latestRevisionId)))
1334
+ .then((rows) => rows[0] ?? null);
1335
+ if (latestRevision && snapshotsMatch(nextSnapshot, latestRevision.snapshot)) {
1336
+ return locked;
1337
+ }
1338
+ }
1339
+ const [updated] = await txDb
1340
+ .update(routines)
1341
+ .set({
1342
+ projectId: candidate.projectId,
1343
+ goalId: candidate.goalId,
1344
+ parentIssueId: candidate.parentIssueId,
1345
+ title: candidate.title,
1346
+ description: candidate.description,
1347
+ assigneeAgentId: candidate.assigneeAgentId,
1348
+ priority: candidate.priority,
1349
+ status: candidate.status,
1350
+ concurrencyPolicy: candidate.concurrencyPolicy,
1351
+ catchUpPolicy: candidate.catchUpPolicy,
1352
+ variables: candidate.variables,
1353
+ updatedByAgentId: actor.agentId ?? null,
1354
+ updatedByUserId: actor.userId ?? null,
1355
+ updatedAt: new Date(),
1356
+ })
1357
+ .where(eq(routines.id, id))
1358
+ .returning();
1359
+ if (!updated)
1360
+ return null;
1361
+ const { routine } = await appendRoutineRevision(txDb, updated, actor, {
1362
+ changeSummary: "Updated routine",
1363
+ });
1364
+ return routine;
1365
+ });
1366
+ },
1367
+ createTrigger: async (routineId, input, actor) => {
1368
+ const routine = await getRoutineById(routineId);
1369
+ if (!routine)
1370
+ throw notFound("Routine not found");
1371
+ let secretMaterial = null;
1372
+ let secretId = null;
1373
+ let publicId = null;
1374
+ let nextRunAt = null;
1375
+ if (input.kind === "schedule") {
1376
+ assertScheduleCompatibleVariables(routine.variables ?? []);
1377
+ const timeZone = input.timezone || "UTC";
1378
+ assertTimeZone(timeZone);
1379
+ const error = validateCron(input.cronExpression);
1380
+ if (error)
1381
+ throw unprocessable(error);
1382
+ nextRunAt = nextCronTickInTimeZone(input.cronExpression, timeZone, new Date());
1383
+ }
1384
+ if (input.kind === "webhook") {
1385
+ publicId = crypto.randomBytes(12).toString("hex");
1386
+ const created = await createWebhookSecret(routine.companyId, routine.id, actor);
1387
+ secretId = created.secret.id;
1388
+ secretMaterial = {
1389
+ webhookUrl: `${process.env.DEALDESK_API_URL}/api/routine-triggers/public/${publicId}/fire`,
1390
+ webhookSecret: created.secretValue,
1391
+ };
1392
+ }
1393
+ const { trigger, revision } = await db.transaction(async (tx) => {
1394
+ const txDb = tx;
1395
+ await tx.execute(sql `select id from ${routines} where ${routines.id} = ${routine.id} for update`);
1396
+ const [createdTrigger] = await txDb
1397
+ .insert(routineTriggers)
1398
+ .values({
1399
+ companyId: routine.companyId,
1400
+ routineId: routine.id,
1401
+ kind: input.kind,
1402
+ label: input.label ?? null,
1403
+ enabled: input.enabled ?? true,
1404
+ cronExpression: input.kind === "schedule" ? input.cronExpression : null,
1405
+ timezone: input.kind === "schedule" ? (input.timezone || "UTC") : null,
1406
+ nextRunAt,
1407
+ publicId,
1408
+ secretId,
1409
+ signingMode: input.kind === "webhook" ? input.signingMode : null,
1410
+ replayWindowSec: input.kind === "webhook" ? input.replayWindowSec : null,
1411
+ lastRotatedAt: input.kind === "webhook" ? new Date() : null,
1412
+ createdByAgentId: actor.agentId ?? null,
1413
+ createdByUserId: actor.userId ?? null,
1414
+ updatedByAgentId: actor.agentId ?? null,
1415
+ updatedByUserId: actor.userId ?? null,
1416
+ })
1417
+ .returning();
1418
+ const latestRoutine = await txDb.select().from(routines).where(eq(routines.id, routine.id)).then((rows) => rows[0] ?? routine);
1419
+ const appended = await appendRoutineRevision(txDb, latestRoutine, actor, {
1420
+ changeSummary: `Created ${input.kind} trigger`,
1421
+ });
1422
+ return { trigger: createdTrigger, revision: appended.revision };
1423
+ });
1424
+ return {
1425
+ trigger: trigger,
1426
+ secretMaterial,
1427
+ revision,
1428
+ };
1429
+ },
1430
+ updateTrigger: async (id, patch, actor) => {
1431
+ const existing = await getTriggerById(id);
1432
+ if (!existing)
1433
+ return null;
1434
+ let nextRunAt = existing.nextRunAt;
1435
+ let cronExpression = existing.cronExpression;
1436
+ let timezone = existing.timezone;
1437
+ if (existing.kind === "schedule") {
1438
+ const routine = await getRoutineById(existing.routineId);
1439
+ if (!routine)
1440
+ throw notFound("Routine not found");
1441
+ if (patch.cronExpression !== undefined) {
1442
+ if (patch.cronExpression == null)
1443
+ throw unprocessable("Scheduled triggers require cronExpression");
1444
+ const error = validateCron(patch.cronExpression);
1445
+ if (error)
1446
+ throw unprocessable(error);
1447
+ cronExpression = patch.cronExpression;
1448
+ }
1449
+ if (patch.timezone !== undefined) {
1450
+ if (patch.timezone == null)
1451
+ throw unprocessable("Scheduled triggers require timezone");
1452
+ assertTimeZone(patch.timezone);
1453
+ timezone = patch.timezone;
1454
+ }
1455
+ if (cronExpression && timezone) {
1456
+ nextRunAt = nextCronTickInTimeZone(cronExpression, timezone, new Date());
1457
+ }
1458
+ if ((patch.enabled ?? existing.enabled) === true) {
1459
+ assertScheduleCompatibleVariables(routine.variables ?? []);
1460
+ }
1461
+ }
1462
+ return db.transaction(async (tx) => {
1463
+ const txDb = tx;
1464
+ await tx.execute(sql `select id from ${routines} where ${routines.id} = ${existing.routineId} for update`);
1465
+ const [updated] = await txDb
1466
+ .update(routineTriggers)
1467
+ .set({
1468
+ label: patch.label === undefined ? existing.label : patch.label,
1469
+ enabled: patch.enabled ?? existing.enabled,
1470
+ cronExpression,
1471
+ timezone,
1472
+ nextRunAt,
1473
+ signingMode: patch.signingMode === undefined ? existing.signingMode : patch.signingMode,
1474
+ replayWindowSec: patch.replayWindowSec === undefined ? existing.replayWindowSec : patch.replayWindowSec,
1475
+ updatedByAgentId: actor.agentId ?? null,
1476
+ updatedByUserId: actor.userId ?? null,
1477
+ updatedAt: new Date(),
1478
+ })
1479
+ .where(eq(routineTriggers.id, id))
1480
+ .returning();
1481
+ if (!updated)
1482
+ return null;
1483
+ const routine = await txDb
1484
+ .select()
1485
+ .from(routines)
1486
+ .where(eq(routines.id, existing.routineId))
1487
+ .then((rows) => rows[0] ?? null);
1488
+ if (!routine)
1489
+ throw notFound("Routine not found");
1490
+ const appended = await appendRoutineRevision(txDb, routine, actor, {
1491
+ changeSummary: `Updated ${existing.kind} trigger`,
1492
+ });
1493
+ return { trigger: updated, revision: appended.revision };
1494
+ });
1495
+ },
1496
+ deleteTrigger: async (id, actor = {}) => {
1497
+ const existing = await getTriggerById(id);
1498
+ if (!existing)
1499
+ return { deleted: false, revision: null };
1500
+ return db.transaction(async (tx) => {
1501
+ const txDb = tx;
1502
+ await tx.execute(sql `select id from ${routines} where ${routines.id} = ${existing.routineId} for update`);
1503
+ await txDb.delete(routineTriggers).where(eq(routineTriggers.id, id));
1504
+ const routine = await txDb
1505
+ .select()
1506
+ .from(routines)
1507
+ .where(eq(routines.id, existing.routineId))
1508
+ .then((rows) => rows[0] ?? null);
1509
+ if (!routine)
1510
+ throw notFound("Routine not found");
1511
+ const appended = await appendRoutineRevision(txDb, routine, actor, {
1512
+ changeSummary: `Deleted ${existing.kind} trigger`,
1513
+ });
1514
+ return { deleted: true, revision: appended.revision };
1515
+ });
1516
+ },
1517
+ rotateTriggerSecret: async (id, actor) => {
1518
+ const existing = await getTriggerById(id);
1519
+ if (!existing)
1520
+ throw notFound("Routine trigger not found");
1521
+ if (existing.kind !== "webhook" || !existing.publicId || !existing.secretId) {
1522
+ throw unprocessable("Only webhook triggers can rotate secrets");
1523
+ }
1524
+ const secretValue = crypto.randomBytes(24).toString("hex");
1525
+ await secretsSvc.rotate(existing.secretId, { value: secretValue }, actor);
1526
+ const { trigger, revision } = await db.transaction(async (tx) => {
1527
+ const txDb = tx;
1528
+ await tx.execute(sql `select id from ${routines} where ${routines.id} = ${existing.routineId} for update`);
1529
+ const [updated] = await txDb
1530
+ .update(routineTriggers)
1531
+ .set({
1532
+ lastRotatedAt: new Date(),
1533
+ updatedByAgentId: actor.agentId ?? null,
1534
+ updatedByUserId: actor.userId ?? null,
1535
+ updatedAt: new Date(),
1536
+ })
1537
+ .where(eq(routineTriggers.id, id))
1538
+ .returning();
1539
+ const routine = await txDb
1540
+ .select()
1541
+ .from(routines)
1542
+ .where(eq(routines.id, existing.routineId))
1543
+ .then((rows) => rows[0] ?? null);
1544
+ if (!routine)
1545
+ throw notFound("Routine not found");
1546
+ const appended = await appendRoutineRevision(txDb, routine, actor, {
1547
+ changeSummary: "Rotated webhook trigger secret",
1548
+ });
1549
+ return { trigger: updated, revision: appended.revision };
1550
+ });
1551
+ return {
1552
+ trigger: trigger,
1553
+ secretMaterial: {
1554
+ webhookUrl: `${process.env.DEALDESK_API_URL}/api/routine-triggers/public/${existing.publicId}/fire`,
1555
+ webhookSecret: secretValue,
1556
+ },
1557
+ revision,
1558
+ };
1559
+ },
1560
+ listRevisions: async (routineId) => {
1561
+ const routine = await getRoutineById(routineId);
1562
+ if (!routine)
1563
+ throw notFound("Routine not found");
1564
+ const rows = await db
1565
+ .select()
1566
+ .from(routineRevisions)
1567
+ .where(and(eq(routineRevisions.companyId, routine.companyId), eq(routineRevisions.routineId, routine.id)))
1568
+ .orderBy(desc(routineRevisions.revisionNumber), desc(routineRevisions.createdAt))
1569
+ .limit(MAX_ROUTINE_REVISIONS);
1570
+ return rows.map(mapRoutineRevision);
1571
+ },
1572
+ restoreRevision: async (routineId, revisionId, actor) => {
1573
+ const existingRoutine = await getRoutineById(routineId);
1574
+ if (!existingRoutine)
1575
+ throw notFound("Routine not found");
1576
+ const targetRevision = await db
1577
+ .select()
1578
+ .from(routineRevisions)
1579
+ .where(and(eq(routineRevisions.companyId, existingRoutine.companyId), eq(routineRevisions.routineId, existingRoutine.id), eq(routineRevisions.id, revisionId)))
1580
+ .then((rows) => rows[0] ?? null);
1581
+ if (!targetRevision)
1582
+ throw notFound("Routine revision not found");
1583
+ const snapshot = targetRevision.snapshot;
1584
+ const routineSnapshot = snapshot.routine;
1585
+ await assertRestorableAssignee(existingRoutine.companyId, routineSnapshot.assigneeAgentId, actor);
1586
+ return db.transaction(async (tx) => {
1587
+ const txDb = tx;
1588
+ await tx.execute(sql `select id from ${routines} where ${routines.id} = ${existingRoutine.id} for update`);
1589
+ const locked = await txDb
1590
+ .select()
1591
+ .from(routines)
1592
+ .where(eq(routines.id, existingRoutine.id))
1593
+ .then((rows) => rows[0] ?? null);
1594
+ if (!locked)
1595
+ throw notFound("Routine not found");
1596
+ if (locked.latestRevisionId === targetRevision.id) {
1597
+ throw conflict("Selected revision is already the latest revision", {
1598
+ currentRevisionId: locked.latestRevisionId,
1599
+ });
1600
+ }
1601
+ const currentTriggers = await txDb
1602
+ .select({ id: routineTriggers.id })
1603
+ .from(routineTriggers)
1604
+ .where(and(eq(routineTriggers.companyId, locked.companyId), eq(routineTriggers.routineId, locked.id)));
1605
+ const currentTriggerIds = new Set(currentTriggers.map((trigger) => trigger.id));
1606
+ const missingWebhookTriggers = snapshot.triggers
1607
+ .filter((trigger) => trigger.kind === "webhook" && !currentTriggerIds.has(trigger.id));
1608
+ const recreatedWebhookSecrets = new Map();
1609
+ for (const trigger of missingWebhookTriggers) {
1610
+ const publicId = crypto.randomBytes(12).toString("hex");
1611
+ const created = await createWebhookSecret(locked.companyId, locked.id, actor, txDb);
1612
+ recreatedWebhookSecrets.set(trigger.id, {
1613
+ publicId,
1614
+ secretId: created.secret.id,
1615
+ secretMaterial: {
1616
+ triggerId: trigger.id,
1617
+ webhookUrl: `${process.env.DEALDESK_API_URL}/api/routine-triggers/public/${publicId}/fire`,
1618
+ webhookSecret: created.secretValue,
1619
+ },
1620
+ });
1621
+ }
1622
+ const now = new Date();
1623
+ const [restoredRoutine] = await txDb
1624
+ .update(routines)
1625
+ .set({
1626
+ projectId: routineSnapshot.projectId,
1627
+ goalId: routineSnapshot.goalId,
1628
+ parentIssueId: routineSnapshot.parentIssueId,
1629
+ title: routineSnapshot.title,
1630
+ description: routineSnapshot.description,
1631
+ assigneeAgentId: routineSnapshot.assigneeAgentId,
1632
+ priority: routineSnapshot.priority,
1633
+ status: routineSnapshot.status,
1634
+ concurrencyPolicy: routineSnapshot.concurrencyPolicy,
1635
+ catchUpPolicy: routineSnapshot.catchUpPolicy,
1636
+ variables: routineSnapshot.variables,
1637
+ updatedByAgentId: actor.agentId ?? null,
1638
+ updatedByUserId: actor.userId ?? null,
1639
+ updatedAt: now,
1640
+ })
1641
+ .where(eq(routines.id, locked.id))
1642
+ .returning();
1643
+ const snapshotTriggerIds = new Set(snapshot.triggers.map((trigger) => trigger.id));
1644
+ if (snapshotTriggerIds.size === 0) {
1645
+ await txDb
1646
+ .delete(routineTriggers)
1647
+ .where(and(eq(routineTriggers.companyId, locked.companyId), eq(routineTriggers.routineId, locked.id)));
1648
+ }
1649
+ else {
1650
+ await txDb
1651
+ .delete(routineTriggers)
1652
+ .where(and(eq(routineTriggers.companyId, locked.companyId), eq(routineTriggers.routineId, locked.id), not(inArray(routineTriggers.id, snapshot.triggers.map((trigger) => trigger.id)))));
1653
+ }
1654
+ for (const triggerSnapshot of snapshot.triggers) {
1655
+ const current = await txDb
1656
+ .select()
1657
+ .from(routineTriggers)
1658
+ .where(and(eq(routineTriggers.companyId, locked.companyId), eq(routineTriggers.id, triggerSnapshot.id)))
1659
+ .then((rows) => rows[0] ?? null);
1660
+ const webhookSecret = recreatedWebhookSecrets.get(triggerSnapshot.id);
1661
+ const restoredNextRunAt = triggerSnapshot.kind === "schedule" && triggerSnapshot.enabled
1662
+ && triggerSnapshot.cronExpression && triggerSnapshot.timezone
1663
+ ? nextCronTickInTimeZone(triggerSnapshot.cronExpression, triggerSnapshot.timezone, now)
1664
+ : null;
1665
+ const baseValues = {
1666
+ companyId: locked.companyId,
1667
+ routineId: locked.id,
1668
+ kind: triggerSnapshot.kind,
1669
+ label: triggerSnapshot.label,
1670
+ enabled: triggerSnapshot.enabled,
1671
+ cronExpression: triggerSnapshot.kind === "schedule" ? triggerSnapshot.cronExpression : null,
1672
+ timezone: triggerSnapshot.kind === "schedule" ? triggerSnapshot.timezone : null,
1673
+ publicId: triggerSnapshot.kind === "webhook" ? (current?.publicId ?? webhookSecret?.publicId ?? triggerSnapshot.publicId) : null,
1674
+ secretId: triggerSnapshot.kind === "webhook" ? (current?.secretId ?? webhookSecret?.secretId ?? null) : null,
1675
+ signingMode: triggerSnapshot.kind === "webhook" ? triggerSnapshot.signingMode : null,
1676
+ replayWindowSec: triggerSnapshot.kind === "webhook" ? triggerSnapshot.replayWindowSec : null,
1677
+ nextRunAt: restoredNextRunAt,
1678
+ updatedByAgentId: actor.agentId ?? null,
1679
+ updatedByUserId: actor.userId ?? null,
1680
+ updatedAt: now,
1681
+ };
1682
+ if (current) {
1683
+ await txDb.update(routineTriggers).set(baseValues).where(eq(routineTriggers.id, triggerSnapshot.id));
1684
+ }
1685
+ else {
1686
+ await txDb.insert(routineTriggers).values({
1687
+ id: triggerSnapshot.id,
1688
+ ...baseValues,
1689
+ createdByAgentId: actor.agentId ?? null,
1690
+ createdByUserId: actor.userId ?? null,
1691
+ createdAt: now,
1692
+ });
1693
+ }
1694
+ }
1695
+ const appended = await appendRoutineRevision(txDb, restoredRoutine ?? locked, actor, {
1696
+ changeSummary: `Restored from revision ${targetRevision.revisionNumber}`,
1697
+ restoredFromRevisionId: targetRevision.id,
1698
+ });
1699
+ return {
1700
+ routine: appended.routine,
1701
+ revision: appended.revision,
1702
+ restoredFromRevisionId: targetRevision.id,
1703
+ restoredFromRevisionNumber: targetRevision.revisionNumber,
1704
+ secretMaterials: [...recreatedWebhookSecrets.values()].map((entry) => entry.secretMaterial),
1705
+ };
1706
+ });
1707
+ },
1708
+ runRoutine: async (id, input, actor) => {
1709
+ const routine = await getRoutineById(id);
1710
+ if (!routine)
1711
+ throw notFound("Routine not found");
1712
+ if (routine.status === "archived")
1713
+ throw conflict("Routine is archived");
1714
+ await assertProject(routine.companyId, input.projectId ?? null);
1715
+ await assertAssignableAgent(routine.companyId, input.assigneeAgentId ?? null);
1716
+ const trigger = input.triggerId ? await getTriggerById(input.triggerId) : null;
1717
+ if (trigger && trigger.routineId !== routine.id)
1718
+ throw forbidden("Trigger does not belong to routine");
1719
+ if (trigger && !trigger.enabled)
1720
+ throw conflict("Routine trigger is not active");
1721
+ return dispatchRoutineRun({
1722
+ routine,
1723
+ trigger,
1724
+ source: input.source,
1725
+ payload: input.payload,
1726
+ variables: input.variables,
1727
+ projectId: input.projectId ?? null,
1728
+ assigneeAgentId: input.assigneeAgentId ?? null,
1729
+ idempotencyKey: input.idempotencyKey,
1730
+ executionWorkspaceId: input.executionWorkspaceId ?? null,
1731
+ executionWorkspacePreference: input.executionWorkspacePreference ?? null,
1732
+ executionWorkspaceSettings: input.executionWorkspaceSettings ?? null,
1733
+ actor,
1734
+ });
1735
+ },
1736
+ firePublicTrigger: async (publicId, input) => {
1737
+ const trigger = await db
1738
+ .select()
1739
+ .from(routineTriggers)
1740
+ .where(and(eq(routineTriggers.publicId, publicId), eq(routineTriggers.kind, "webhook")))
1741
+ .then((rows) => rows[0] ?? null);
1742
+ if (!trigger)
1743
+ throw notFound("Routine trigger not found");
1744
+ const routine = await getRoutineById(trigger.routineId);
1745
+ if (!routine)
1746
+ throw notFound("Routine not found");
1747
+ if (!trigger.enabled || routine.status !== "active")
1748
+ throw conflict("Routine trigger is not active");
1749
+ if (trigger.signingMode === "none") {
1750
+ // No authentication — the publicId in the URL acts as a shared secret.
1751
+ }
1752
+ else if (trigger.signingMode === "github_hmac") {
1753
+ const secretValue = await resolveTriggerSecret(trigger, routine.companyId);
1754
+ const rawBody = input.rawBody ?? Buffer.from(JSON.stringify(input.payload ?? {}));
1755
+ // Accept X-Hub-Signature-256 (GitHub/Sentry) or fall back to the
1756
+ // generic X-DealDesk-Signature header so operators can use github_hmac
1757
+ // mode with either header convention.
1758
+ const providedSignature = (input.hubSignatureHeader ?? input.signatureHeader)?.trim() ?? "";
1759
+ if (!providedSignature)
1760
+ throw unauthorized();
1761
+ const expectedHmac = crypto
1762
+ .createHmac("sha256", secretValue)
1763
+ .update(rawBody)
1764
+ .digest("hex");
1765
+ const normalizedSignature = providedSignature.replace(/^sha256=/, "");
1766
+ const normalizedBuf = Buffer.from(normalizedSignature);
1767
+ const expectedBuf = Buffer.from(expectedHmac);
1768
+ const valid = normalizedBuf.length === expectedBuf.length &&
1769
+ crypto.timingSafeEqual(normalizedBuf, expectedBuf);
1770
+ if (!valid)
1771
+ throw unauthorized();
1772
+ }
1773
+ else if (trigger.signingMode === "bearer") {
1774
+ const secretValue = await resolveTriggerSecret(trigger, routine.companyId);
1775
+ const expected = `Bearer ${secretValue}`;
1776
+ const provided = input.authorizationHeader?.trim() ?? "";
1777
+ const expectedBuf = Buffer.from(expected);
1778
+ const providedBuf = Buffer.alloc(expectedBuf.length);
1779
+ providedBuf.write(provided.slice(0, expectedBuf.length));
1780
+ const valid = provided.length === expected.length &&
1781
+ crypto.timingSafeEqual(providedBuf, expectedBuf);
1782
+ if (!valid) {
1783
+ throw unauthorized();
1784
+ }
1785
+ }
1786
+ else {
1787
+ const secretValue = await resolveTriggerSecret(trigger, routine.companyId);
1788
+ const rawBody = input.rawBody ?? Buffer.from(JSON.stringify(input.payload ?? {}));
1789
+ const providedSignature = input.signatureHeader?.trim() ?? "";
1790
+ const providedTimestamp = input.timestampHeader?.trim() ?? "";
1791
+ if (!providedSignature || !providedTimestamp)
1792
+ throw unauthorized();
1793
+ const tsMillis = normalizeWebhookTimestampMs(providedTimestamp);
1794
+ if (tsMillis == null)
1795
+ throw unauthorized();
1796
+ const replayWindowSec = trigger.replayWindowSec ?? 300;
1797
+ if (Math.abs(Date.now() - tsMillis) > replayWindowSec * 1000) {
1798
+ throw unauthorized();
1799
+ }
1800
+ const expectedHmac = crypto
1801
+ .createHmac("sha256", secretValue)
1802
+ .update(`${providedTimestamp}.`)
1803
+ .update(rawBody)
1804
+ .digest("hex");
1805
+ const normalizedSignature = providedSignature.replace(/^sha256=/, "");
1806
+ const valid = normalizedSignature.length === expectedHmac.length &&
1807
+ crypto.timingSafeEqual(Buffer.from(normalizedSignature), Buffer.from(expectedHmac));
1808
+ if (!valid)
1809
+ throw unauthorized();
1810
+ }
1811
+ return dispatchRoutineRun({
1812
+ routine,
1813
+ trigger,
1814
+ source: "webhook",
1815
+ payload: input.payload,
1816
+ variables: isPlainRecord(input.payload) && isPlainRecord(input.payload.variables)
1817
+ ? input.payload.variables
1818
+ : null,
1819
+ idempotencyKey: input.idempotencyKey,
1820
+ });
1821
+ },
1822
+ listRuns: async (routineId, limit = 50) => {
1823
+ const cappedLimit = Math.max(1, Math.min(limit, 200));
1824
+ const rows = await db
1825
+ .select({
1826
+ id: routineRuns.id,
1827
+ companyId: routineRuns.companyId,
1828
+ routineId: routineRuns.routineId,
1829
+ triggerId: routineRuns.triggerId,
1830
+ source: routineRuns.source,
1831
+ status: routineRuns.status,
1832
+ triggeredAt: routineRuns.triggeredAt,
1833
+ idempotencyKey: routineRuns.idempotencyKey,
1834
+ triggerPayload: routineRuns.triggerPayload,
1835
+ dispatchFingerprint: routineRuns.dispatchFingerprint,
1836
+ linkedIssueId: routineRuns.linkedIssueId,
1837
+ coalescedIntoRunId: routineRuns.coalescedIntoRunId,
1838
+ failureReason: routineRuns.failureReason,
1839
+ completedAt: routineRuns.completedAt,
1840
+ createdAt: routineRuns.createdAt,
1841
+ updatedAt: routineRuns.updatedAt,
1842
+ triggerKind: routineTriggers.kind,
1843
+ triggerLabel: routineTriggers.label,
1844
+ issueIdentifier: issues.identifier,
1845
+ issueTitle: issues.title,
1846
+ issueStatus: issues.status,
1847
+ issuePriority: issues.priority,
1848
+ issueUpdatedAt: issues.updatedAt,
1849
+ })
1850
+ .from(routineRuns)
1851
+ .leftJoin(routineTriggers, eq(routineRuns.triggerId, routineTriggers.id))
1852
+ .leftJoin(issues, eq(routineRuns.linkedIssueId, issues.id))
1853
+ .where(eq(routineRuns.routineId, routineId))
1854
+ .orderBy(desc(routineRuns.createdAt))
1855
+ .limit(cappedLimit);
1856
+ return rows.map((row) => ({
1857
+ id: row.id,
1858
+ companyId: row.companyId,
1859
+ routineId: row.routineId,
1860
+ triggerId: row.triggerId,
1861
+ source: row.source,
1862
+ status: row.status,
1863
+ triggeredAt: row.triggeredAt,
1864
+ idempotencyKey: row.idempotencyKey,
1865
+ triggerPayload: row.triggerPayload,
1866
+ dispatchFingerprint: row.dispatchFingerprint,
1867
+ linkedIssueId: row.linkedIssueId,
1868
+ coalescedIntoRunId: row.coalescedIntoRunId,
1869
+ failureReason: row.failureReason,
1870
+ completedAt: row.completedAt,
1871
+ createdAt: row.createdAt,
1872
+ updatedAt: row.updatedAt,
1873
+ linkedIssue: row.linkedIssueId
1874
+ ? {
1875
+ id: row.linkedIssueId,
1876
+ identifier: row.issueIdentifier,
1877
+ title: row.issueTitle ?? "Routine execution",
1878
+ status: row.issueStatus ?? "todo",
1879
+ priority: row.issuePriority ?? "medium",
1880
+ updatedAt: row.issueUpdatedAt ?? row.updatedAt,
1881
+ }
1882
+ : null,
1883
+ trigger: row.triggerId
1884
+ ? {
1885
+ id: row.triggerId,
1886
+ kind: row.triggerKind,
1887
+ label: row.triggerLabel,
1888
+ }
1889
+ : null,
1890
+ }));
1891
+ },
1892
+ tickScheduledTriggers: async (now = new Date()) => {
1893
+ const due = await db
1894
+ .select({
1895
+ trigger: routineTriggers,
1896
+ routine: routines,
1897
+ })
1898
+ .from(routineTriggers)
1899
+ .innerJoin(routines, eq(routineTriggers.routineId, routines.id))
1900
+ .where(and(eq(routineTriggers.kind, "schedule"), eq(routineTriggers.enabled, true), eq(routines.status, "active"), isNotNull(routineTriggers.nextRunAt), lte(routineTriggers.nextRunAt, now)))
1901
+ .orderBy(asc(routineTriggers.nextRunAt), asc(routineTriggers.createdAt));
1902
+ let triggered = 0;
1903
+ for (const row of due) {
1904
+ if (!row.trigger.nextRunAt || !row.trigger.cronExpression || !row.trigger.timezone)
1905
+ continue;
1906
+ let runCount = 1;
1907
+ let claimedNextRunAt = nextCronTickInTimeZone(row.trigger.cronExpression, row.trigger.timezone, now);
1908
+ if (row.routine.catchUpPolicy === "enqueue_missed_with_cap") {
1909
+ let cursor = row.trigger.nextRunAt;
1910
+ runCount = 0;
1911
+ while (cursor && cursor <= now && runCount < MAX_CATCH_UP_RUNS) {
1912
+ runCount += 1;
1913
+ claimedNextRunAt = nextCronTickInTimeZone(row.trigger.cronExpression, row.trigger.timezone, cursor);
1914
+ cursor = claimedNextRunAt;
1915
+ }
1916
+ }
1917
+ const claimed = await db
1918
+ .update(routineTriggers)
1919
+ .set({
1920
+ nextRunAt: claimedNextRunAt,
1921
+ updatedAt: new Date(),
1922
+ })
1923
+ .where(and(eq(routineTriggers.id, row.trigger.id), eq(routineTriggers.enabled, true), eq(routineTriggers.nextRunAt, row.trigger.nextRunAt)))
1924
+ .returning({ id: routineTriggers.id })
1925
+ .then((rows) => rows[0] ?? null);
1926
+ if (!claimed)
1927
+ continue;
1928
+ for (let i = 0; i < runCount; i += 1) {
1929
+ await dispatchRoutineRun({
1930
+ routine: row.routine,
1931
+ trigger: row.trigger,
1932
+ source: "schedule",
1933
+ });
1934
+ triggered += 1;
1935
+ }
1936
+ }
1937
+ return { triggered };
1938
+ },
1939
+ syncRunStatusForIssue: async (issueId) => {
1940
+ const issue = await db
1941
+ .select({
1942
+ id: issues.id,
1943
+ status: issues.status,
1944
+ originKind: issues.originKind,
1945
+ originRunId: issues.originRunId,
1946
+ })
1947
+ .from(issues)
1948
+ .where(eq(issues.id, issueId))
1949
+ .then((rows) => rows[0] ?? null);
1950
+ if (!issue || issue.originKind !== "routine_execution" || !issue.originRunId)
1951
+ return null;
1952
+ if (issue.status === "done") {
1953
+ return finalizeRun(issue.originRunId, {
1954
+ status: "completed",
1955
+ completedAt: new Date(),
1956
+ });
1957
+ }
1958
+ if (issue.status === "blocked" || issue.status === "cancelled") {
1959
+ return finalizeRun(issue.originRunId, {
1960
+ status: "failed",
1961
+ failureReason: `Execution issue moved to ${issue.status}`,
1962
+ completedAt: new Date(),
1963
+ });
1964
+ }
1965
+ return null;
1966
+ },
1967
+ };
1968
+ }
1969
+ //# sourceMappingURL=routines.js.map