@objectstack/plugin-webhooks 7.4.1 → 7.6.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 (52) hide show
  1. package/.turbo/turbo-build.log +20 -32
  2. package/CHANGELOG.md +59 -0
  3. package/dist/chunk-HWFTXTTI.js +138 -0
  4. package/dist/chunk-HWFTXTTI.js.map +1 -0
  5. package/dist/chunk-KPKLAXNA.cjs +138 -0
  6. package/dist/chunk-KPKLAXNA.cjs.map +1 -0
  7. package/dist/index.cjs +62 -616
  8. package/dist/index.cjs.map +1 -1
  9. package/dist/index.d.cts +41 -325
  10. package/dist/index.d.ts +41 -325
  11. package/dist/index.js +52 -606
  12. package/dist/index.js.map +1 -1
  13. package/dist/schema.cjs +2 -6
  14. package/dist/schema.cjs.map +1 -1
  15. package/dist/schema.d.cts +5 -4764
  16. package/dist/schema.d.ts +5 -4764
  17. package/dist/schema.js +3 -7
  18. package/package.json +5 -12
  19. package/src/auto-enqueuer.test.ts +83 -116
  20. package/src/auto-enqueuer.ts +38 -27
  21. package/src/index.ts +13 -40
  22. package/src/schema.ts +11 -16
  23. package/src/webhook-outbox-plugin.ts +80 -296
  24. package/tsup.config.ts +1 -1
  25. package/dist/chunk-7HS5DLU2.js +0 -319
  26. package/dist/chunk-7HS5DLU2.js.map +0 -1
  27. package/dist/chunk-HF7CCDPB.cjs +0 -256
  28. package/dist/chunk-HF7CCDPB.cjs.map +0 -1
  29. package/dist/chunk-KNGLLSSP.js +0 -256
  30. package/dist/chunk-KNGLLSSP.js.map +0 -1
  31. package/dist/chunk-TDSI7UHY.cjs +0 -319
  32. package/dist/chunk-TDSI7UHY.cjs.map +0 -1
  33. package/dist/outbox-CIn7LSyB.d.cts +0 -155
  34. package/dist/outbox-CIn7LSyB.d.ts +0 -155
  35. package/dist/sql-outbox.cjs +0 -8
  36. package/dist/sql-outbox.cjs.map +0 -1
  37. package/dist/sql-outbox.d.cts +0 -55
  38. package/dist/sql-outbox.d.ts +0 -55
  39. package/dist/sql-outbox.js +0 -8
  40. package/dist/sql-outbox.js.map +0 -1
  41. package/src/dispatcher.test.ts +0 -324
  42. package/src/dispatcher.ts +0 -218
  43. package/src/http-sender.ts +0 -187
  44. package/src/memory-outbox.test.ts +0 -86
  45. package/src/memory-outbox.ts +0 -155
  46. package/src/outbox.ts +0 -175
  47. package/src/partition.ts +0 -19
  48. package/src/retention.test.ts +0 -116
  49. package/src/retention.ts +0 -144
  50. package/src/sql-outbox.test.ts +0 -490
  51. package/src/sql-outbox.ts +0 -343
  52. package/src/sys-webhook-delivery.object.ts +0 -224
@@ -1,319 +0,0 @@
1
- // src/sys-webhook.object.ts
2
- import { ObjectSchema, Field } from "@objectstack/spec/data";
3
- var SysWebhook = ObjectSchema.create({
4
- name: "sys_webhook",
5
- label: "Webhook",
6
- pluralLabel: "Webhooks",
7
- icon: "webhook",
8
- isSystem: true,
9
- managedBy: "config",
10
- // Authoring a webhook from the UI requires a structured form for the
11
- // headers / auth / retry / payload blocks — the generic JSON textarea
12
- // is acceptable as a v1 until a dedicated builder lands. Re-enable
13
- // create/edit/delete so admins can at least toggle `active` and edit
14
- // simple URL/method fields without round-tripping through code.
15
- userActions: { create: true, edit: true, delete: true, import: false },
16
- description: "Outbound HTTP webhook subscription. Authored via defineWebhook() in code or the Studio editor; executed by the HTTP connector plugin.",
17
- displayNameField: "name",
18
- titleFormat: "{label}",
19
- compactLayout: ["name", "object_name", "url", "active", "updated_at"],
20
- listViews: {
21
- active: {
22
- type: "grid",
23
- name: "active",
24
- label: "Active",
25
- data: { provider: "object", object: "sys_webhook" },
26
- columns: ["label", "object_name", "url", "method", "active", "updated_at"],
27
- filter: [{ field: "active", operator: "equals", value: true }],
28
- sort: [{ field: "label", order: "asc" }],
29
- pagination: { pageSize: 50 }
30
- },
31
- inactive: {
32
- type: "grid",
33
- name: "inactive",
34
- label: "Inactive",
35
- data: { provider: "object", object: "sys_webhook" },
36
- columns: ["label", "object_name", "url", "method", "active", "updated_at"],
37
- filter: [{ field: "active", operator: "equals", value: false }],
38
- sort: [{ field: "label", order: "asc" }],
39
- pagination: { pageSize: 50 }
40
- },
41
- by_object: {
42
- type: "grid",
43
- name: "by_object",
44
- label: "By Object",
45
- data: { provider: "object", object: "sys_webhook" },
46
- columns: ["object_name", "label", "url", "active", "updated_at"],
47
- sort: [{ field: "object_name", order: "asc" }, { field: "label", order: "asc" }],
48
- grouping: { fields: [{ field: "object_name", order: "asc", collapsed: false }] },
49
- pagination: { pageSize: 100 }
50
- },
51
- all_webhooks: {
52
- type: "grid",
53
- name: "all_webhooks",
54
- label: "All",
55
- data: { provider: "object", object: "sys_webhook" },
56
- columns: ["label", "object_name", "url", "method", "active", "updated_at"],
57
- sort: [{ field: "label", order: "asc" }],
58
- pagination: { pageSize: 50 }
59
- }
60
- },
61
- fields: {
62
- id: Field.text({ label: "Webhook ID", required: true, readonly: true, group: "System" }),
63
- name: Field.text({
64
- label: "Name",
65
- required: true,
66
- maxLength: 100,
67
- description: "Unique snake_case name \u2014 referenced in logs and audit",
68
- group: "Definition"
69
- }),
70
- label: Field.text({
71
- label: "Display Label",
72
- required: false,
73
- maxLength: 200,
74
- group: "Definition"
75
- }),
76
- object_name: Field.text({
77
- label: "Object",
78
- required: false,
79
- maxLength: 100,
80
- description: "Short object name whose events fire this webhook (blank = manual / API-triggered)",
81
- group: "Definition"
82
- }),
83
- triggers: Field.text({
84
- label: "Triggers",
85
- required: false,
86
- maxLength: 200,
87
- description: "Comma-separated event list: create,update,delete,undelete,api",
88
- group: "Definition"
89
- }),
90
- url: Field.text({
91
- label: "Target URL",
92
- required: true,
93
- maxLength: 2048,
94
- description: "External endpoint that receives the POST",
95
- group: "Definition"
96
- }),
97
- method: Field.text({
98
- label: "HTTP Method",
99
- required: true,
100
- defaultValue: "POST",
101
- maxLength: 10,
102
- description: "GET / POST / PUT / PATCH / DELETE",
103
- group: "Definition"
104
- }),
105
- description: Field.textarea({ label: "Description", required: false, group: "Definition" }),
106
- active: Field.boolean({
107
- label: "Active",
108
- required: true,
109
- defaultValue: true,
110
- description: "Inactive webhooks are skipped by the dispatcher",
111
- group: "Definition"
112
- }),
113
- definition_json: Field.textarea({
114
- label: "Definition",
115
- required: true,
116
- description: "Serialised Webhook JSON (see @objectstack/spec/automation/webhook) \u2014 full headers/auth/retry/payload config",
117
- group: "Definition"
118
- }),
119
- created_at: Field.datetime({
120
- label: "Created At",
121
- required: true,
122
- defaultValue: "NOW()",
123
- readonly: true,
124
- group: "System"
125
- }),
126
- updated_at: Field.datetime({ label: "Updated At", required: false, group: "System" })
127
- },
128
- indexes: [
129
- { fields: ["name"], unique: true },
130
- { fields: ["object_name"] },
131
- { fields: ["active", "object_name"] }
132
- ]
133
- });
134
-
135
- // src/sys-webhook-delivery.object.ts
136
- import { Field as Field2, ObjectSchema as ObjectSchema2 } from "@objectstack/spec/data";
137
- var SysWebhookDelivery = ObjectSchema2.create({
138
- name: "sys_webhook_delivery",
139
- label: "Webhook Delivery",
140
- pluralLabel: "Webhook Deliveries",
141
- icon: "package",
142
- isSystem: true,
143
- managedBy: "config",
144
- userActions: { create: false, edit: false, delete: false, import: false },
145
- description: "Durable outbox row for one webhook attempt. Managed by @objectstack/plugin-webhooks; do not write directly.",
146
- displayNameField: "id",
147
- titleFormat: "{event_type} \u2192 {url}",
148
- compactLayout: ["event_type", "url", "status", "attempts", "next_retry_at"],
149
- actions: [
150
- {
151
- name: "redeliver",
152
- label: "Redeliver",
153
- icon: "refresh-cw",
154
- variant: "secondary",
155
- locations: ["list_item", "record_header"],
156
- type: "api",
157
- target: "/api/v1/webhooks/redeliver",
158
- method: "POST",
159
- recordIdParam: "deliveryId",
160
- confirmText: "Replay this delivery? The receiver will get the original payload again \u2014 they must be idempotent on the X-Objectstack-Delivery header.",
161
- successMessage: "Queued for redelivery",
162
- refreshAfter: true,
163
- // Only terminal rows are safe to replay. Pending / in_flight rows
164
- // are either already queued or actively being sent — replaying
165
- // would double-deliver.
166
- disabled: "!(status in ['success', 'failed', 'dead'])"
167
- }
168
- ],
169
- listViews: {
170
- recent: {
171
- type: "grid",
172
- name: "recent",
173
- label: "Recent",
174
- data: { provider: "object", object: "sys_webhook_delivery" },
175
- columns: ["event_type", "url", "status", "attempts", "response_code", "updated_at"],
176
- sort: [{ field: "updated_at", order: "desc" }],
177
- pagination: { pageSize: 50 }
178
- },
179
- failures: {
180
- type: "grid",
181
- name: "failures",
182
- label: "Failures",
183
- data: { provider: "object", object: "sys_webhook_delivery" },
184
- columns: ["event_type", "url", "status", "attempts", "response_code", "error", "updated_at"],
185
- filter: [{ field: "status", operator: "in", value: ["failed", "dead"] }],
186
- sort: [{ field: "updated_at", order: "desc" }],
187
- pagination: { pageSize: 50 }
188
- },
189
- in_flight: {
190
- type: "grid",
191
- name: "in_flight",
192
- label: "In Flight",
193
- data: { provider: "object", object: "sys_webhook_delivery" },
194
- columns: ["event_type", "url", "attempts", "claimed_by", "claimed_at"],
195
- filter: [{ field: "status", operator: "equals", value: "in_flight" }],
196
- sort: [{ field: "claimed_at", order: "desc" }],
197
- pagination: { pageSize: 50 }
198
- },
199
- pending: {
200
- type: "grid",
201
- name: "pending",
202
- label: "Pending",
203
- data: { provider: "object", object: "sys_webhook_delivery" },
204
- columns: ["event_type", "url", "attempts", "next_retry_at", "updated_at"],
205
- filter: [{ field: "status", operator: "equals", value: "pending" }],
206
- sort: [{ field: "next_retry_at", order: "asc" }],
207
- pagination: { pageSize: 50 }
208
- },
209
- by_status: {
210
- type: "grid",
211
- name: "by_status",
212
- label: "By Status",
213
- data: { provider: "object", object: "sys_webhook_delivery" },
214
- columns: ["status", "event_type", "url", "attempts", "updated_at"],
215
- sort: [{ field: "status", order: "asc" }, { field: "updated_at", order: "desc" }],
216
- grouping: { fields: [{ field: "status", order: "asc", collapsed: false }] },
217
- pagination: { pageSize: 100 }
218
- },
219
- by_webhook: {
220
- type: "grid",
221
- name: "by_webhook",
222
- label: "By Webhook",
223
- data: { provider: "object", object: "sys_webhook_delivery" },
224
- columns: ["webhook_id", "event_type", "status", "attempts", "updated_at"],
225
- sort: [{ field: "webhook_id", order: "asc" }, { field: "updated_at", order: "desc" }],
226
- grouping: { fields: [{ field: "webhook_id", order: "asc", collapsed: true }] },
227
- pagination: { pageSize: 100 }
228
- },
229
- all_deliveries: {
230
- type: "grid",
231
- name: "all_deliveries",
232
- label: "All",
233
- data: { provider: "object", object: "sys_webhook_delivery" },
234
- columns: ["event_type", "url", "status", "attempts", "response_code", "updated_at"],
235
- sort: [{ field: "updated_at", order: "desc" }],
236
- pagination: { pageSize: 100 }
237
- }
238
- },
239
- fields: {
240
- id: Field2.text({
241
- label: "Delivery ID",
242
- required: true,
243
- maxLength: 64,
244
- description: "UUID \u2014 also doubles as the receiver-side idempotency key"
245
- }),
246
- webhook_id: Field2.text({
247
- label: "Webhook ID",
248
- required: true,
249
- maxLength: 64,
250
- description: "FK to sys_webhook.id (loosely coupled \u2014 denormalised URL/secret on row)"
251
- }),
252
- event_id: Field2.text({
253
- label: "Event ID",
254
- required: true,
255
- maxLength: 128,
256
- description: "Source event id; UNIQUE(event_id, webhook_id) for dedup"
257
- }),
258
- event_type: Field2.text({
259
- label: "Event Type",
260
- required: true,
261
- maxLength: 128,
262
- description: "e.g. data.record.created"
263
- }),
264
- url: Field2.text({
265
- label: "Target URL",
266
- required: true,
267
- maxLength: 2048,
268
- description: "Snapshotted at enqueue so config edits do not rewrite live rows"
269
- }),
270
- method: Field2.text({ label: "Method", required: false, maxLength: 10 }),
271
- headers_json: Field2.textarea({ label: "Headers JSON", required: false }),
272
- secret: Field2.text({ label: "HMAC Secret", required: false, maxLength: 256 }),
273
- timeout_ms: Field2.number({ label: "Timeout (ms)", required: false }),
274
- payload_json: Field2.textarea({ label: "Payload JSON", required: true }),
275
- partition_key: Field2.number({
276
- label: "Partition",
277
- required: true,
278
- description: "hash(webhook_id) mod partitionCount \u2014 precomputed for cheap WHERE"
279
- }),
280
- status: Field2.text({
281
- label: "Status",
282
- required: true,
283
- defaultValue: "pending",
284
- maxLength: 16,
285
- description: "pending | in_flight | success | failed | dead"
286
- }),
287
- attempts: Field2.number({
288
- label: "Attempts",
289
- required: true,
290
- defaultValue: 0,
291
- description: "Number of POST attempts made so far"
292
- }),
293
- claimed_by: Field2.text({ label: "Claimed By", required: false, maxLength: 128 }),
294
- claimed_at: Field2.number({ label: "Claimed At (ms)", required: false }),
295
- next_retry_at: Field2.number({ label: "Next Retry At (ms)", required: false }),
296
- last_attempted_at: Field2.number({ label: "Last Attempted At (ms)", required: false }),
297
- response_code: Field2.number({ label: "HTTP Status", required: false }),
298
- response_body: Field2.textarea({ label: "Response Body (capped)", required: false }),
299
- error: Field2.textarea({ label: "Error", required: false }),
300
- created_at: Field2.number({ label: "Created At (ms)", required: true }),
301
- updated_at: Field2.number({ label: "Updated At (ms)", required: true })
302
- },
303
- indexes: [
304
- { fields: ["event_id", "webhook_id"], unique: true },
305
- // Hot path: claim query
306
- { fields: ["status", "partition_key", "next_retry_at"] },
307
- // Reaper: scan stale in_flight rows by claimed_at
308
- { fields: ["status", "claimed_at"] },
309
- { fields: ["webhook_id"] }
310
- ]
311
- });
312
- var SYS_WEBHOOK_DELIVERY = "sys_webhook_delivery";
313
-
314
- export {
315
- SysWebhook,
316
- SysWebhookDelivery,
317
- SYS_WEBHOOK_DELIVERY
318
- };
319
- //# sourceMappingURL=chunk-7HS5DLU2.js.map
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/sys-webhook.object.ts","../src/sys-webhook-delivery.object.ts"],"sourcesContent":["// Copyright (c) 2025 ObjectStack. Licensed under the Apache-2.0 license.\n\nimport { ObjectSchema, Field } from '@objectstack/spec/data';\n\n/**\n * sys_webhook — Outbound HTTP integration configuration (runtime).\n *\n * Persists a single {@link Webhook} envelope per row so administrators\n * can author, enable/disable, and edit webhook subscriptions from the\n * Studio UI without code changes. The canonical Zod schema for the\n * `definition_json` envelope lives at `@objectstack/spec/automation/webhook`.\n *\n * One row per `name`. The automation runtime\n * (`@objectstack/service-automation`, built-in `http_request` node) loads\n * active rows on boot + on `sys_webhook:changed` events, registers\n * `afterInsert` / `afterUpdate` / `afterDelete` listeners for the\n * targeted object, and dispatches outbound HTTP calls when matching\n * record events fire.\n *\n * Ownership (ADR-0029 K2.a): this object is **owned by\n * `@objectstack/plugin-webhooks`** — the plugin that consumes these rows —\n * alongside its sibling `sys_webhook_delivery`. It used to live in the\n * `@objectstack/platform-objects` monolith and be imported here; the\n * definition now lives with its owner so the plugin ships both data and\n * behavior as one unit.\n *\n * Platform-wide on purpose: every project (standalone, single-tenant,\n * cloud) can integrate with external systems (Slack, Stripe, internal\n * services) the same way.\n *\n * @namespace sys\n */\nexport const SysWebhook = ObjectSchema.create({\n name: 'sys_webhook',\n label: 'Webhook',\n pluralLabel: 'Webhooks',\n icon: 'webhook',\n isSystem: true,\n managedBy: 'config',\n // Authoring a webhook from the UI requires a structured form for the\n // headers / auth / retry / payload blocks — the generic JSON textarea\n // is acceptable as a v1 until a dedicated builder lands. Re-enable\n // create/edit/delete so admins can at least toggle `active` and edit\n // simple URL/method fields without round-tripping through code.\n userActions: { create: true, edit: true, delete: true, import: false },\n description: 'Outbound HTTP webhook subscription. Authored via defineWebhook() in code or the Studio editor; executed by the HTTP connector plugin.',\n displayNameField: 'name',\n titleFormat: '{label}',\n compactLayout: ['name', 'object_name', 'url', 'active', 'updated_at'],\n\n listViews: {\n active: {\n type: 'grid',\n name: 'active',\n label: 'Active',\n data: { provider: 'object', object: 'sys_webhook' },\n columns: ['label', 'object_name', 'url', 'method', 'active', 'updated_at'],\n filter: [{ field: 'active', operator: 'equals', value: true }],\n sort: [{ field: 'label', order: 'asc' }],\n pagination: { pageSize: 50 },\n },\n inactive: {\n type: 'grid',\n name: 'inactive',\n label: 'Inactive',\n data: { provider: 'object', object: 'sys_webhook' },\n columns: ['label', 'object_name', 'url', 'method', 'active', 'updated_at'],\n filter: [{ field: 'active', operator: 'equals', value: false }],\n sort: [{ field: 'label', order: 'asc' }],\n pagination: { pageSize: 50 },\n },\n by_object: {\n type: 'grid',\n name: 'by_object',\n label: 'By Object',\n data: { provider: 'object', object: 'sys_webhook' },\n columns: ['object_name', 'label', 'url', 'active', 'updated_at'],\n sort: [{ field: 'object_name', order: 'asc' }, { field: 'label', order: 'asc' }],\n grouping: { fields: [{ field: 'object_name', order: 'asc', collapsed: false }] },\n pagination: { pageSize: 100 },\n },\n all_webhooks: {\n type: 'grid',\n name: 'all_webhooks',\n label: 'All',\n data: { provider: 'object', object: 'sys_webhook' },\n columns: ['label', 'object_name', 'url', 'method', 'active', 'updated_at'],\n sort: [{ field: 'label', order: 'asc' }],\n pagination: { pageSize: 50 },\n },\n },\n\n fields: {\n id: Field.text({ label: 'Webhook ID', required: true, readonly: true, group: 'System' }),\n\n name: Field.text({\n label: 'Name',\n required: true,\n maxLength: 100,\n description: 'Unique snake_case name — referenced in logs and audit',\n group: 'Definition',\n }),\n\n label: Field.text({\n label: 'Display Label',\n required: false,\n maxLength: 200,\n group: 'Definition',\n }),\n\n object_name: Field.text({\n label: 'Object',\n required: false,\n maxLength: 100,\n description: 'Short object name whose events fire this webhook (blank = manual / API-triggered)',\n group: 'Definition',\n }),\n\n triggers: Field.text({\n label: 'Triggers',\n required: false,\n maxLength: 200,\n description: 'Comma-separated event list: create,update,delete,undelete,api',\n group: 'Definition',\n }),\n\n url: Field.text({\n label: 'Target URL',\n required: true,\n maxLength: 2048,\n description: 'External endpoint that receives the POST',\n group: 'Definition',\n }),\n\n method: Field.text({\n label: 'HTTP Method',\n required: true,\n defaultValue: 'POST',\n maxLength: 10,\n description: 'GET / POST / PUT / PATCH / DELETE',\n group: 'Definition',\n }),\n\n description: Field.textarea({ label: 'Description', required: false, group: 'Definition' }),\n\n active: Field.boolean({\n label: 'Active',\n required: true,\n defaultValue: true,\n description: 'Inactive webhooks are skipped by the dispatcher',\n group: 'Definition',\n }),\n\n definition_json: Field.textarea({\n label: 'Definition',\n required: true,\n description: 'Serialised Webhook JSON (see @objectstack/spec/automation/webhook) — full headers/auth/retry/payload config',\n group: 'Definition',\n }),\n\n created_at: Field.datetime({\n label: 'Created At',\n required: true,\n defaultValue: 'NOW()',\n readonly: true,\n group: 'System',\n }),\n\n updated_at: Field.datetime({ label: 'Updated At', required: false, group: 'System' }),\n },\n\n indexes: [\n { fields: ['name'], unique: true },\n { fields: ['object_name'] },\n { fields: ['active', 'object_name'] },\n ],\n});\n","// Copyright (c) 2026 ObjectStack. Licensed under the Apache-2.0 license.\n\nimport { Field, ObjectSchema } from '@objectstack/spec/data';\n\n/**\n * sys_webhook_delivery — Durable outbox row for one HTTP attempt.\n *\n * Schema is owned by `@objectstack/plugin-webhooks`. Add it to your stack\n * via:\n *\n * import { SysWebhookDelivery } from '@objectstack/plugin-webhooks/schema';\n * defineStack({ objects: [SysWebhookDelivery, ...], plugins: [...] });\n *\n * Designed for the SqlWebhookOutbox claim algorithm:\n *\n * 1. Producers INSERT pending rows (dedup'd by (event_id, webhook_id)).\n * 2. The dispatcher's per-partition lock-holder runs:\n * SELECT id WHERE status='pending' AND partition_key=? AND (next_retry_at <= now OR null)\n * UPDATE SET status='in_flight' WHERE id IN (...) AND status='pending' ← atomic claim\n * POST to target URL\n * UPDATE SET status=success/pending/dead, attempts=attempts+1, ...\n *\n * `partition_key` is precomputed on enqueue (hash(webhook_id) mod N) so the\n * dispatcher can filter cheaply without DB-side hash functions.\n *\n * Indexes are tuned for the hot path: `(status, partition_key, next_retry_at)`\n * is the claim query; `(event_id, webhook_id)` is the dedup uniqueness.\n *\n * @namespace sys\n */\nexport const SysWebhookDelivery = ObjectSchema.create({\n name: 'sys_webhook_delivery',\n label: 'Webhook Delivery',\n pluralLabel: 'Webhook Deliveries',\n icon: 'package',\n isSystem: true,\n managedBy: 'config',\n userActions: { create: false, edit: false, delete: false, import: false },\n description:\n 'Durable outbox row for one webhook attempt. Managed by @objectstack/plugin-webhooks; do not write directly.',\n displayNameField: 'id',\n titleFormat: '{event_type} → {url}',\n compactLayout: ['event_type', 'url', 'status', 'attempts', 'next_retry_at'],\n\n actions: [\n {\n name: 'redeliver',\n label: 'Redeliver',\n icon: 'refresh-cw',\n variant: 'secondary',\n locations: ['list_item', 'record_header'],\n type: 'api',\n target: '/api/v1/webhooks/redeliver',\n method: 'POST',\n recordIdParam: 'deliveryId',\n confirmText:\n 'Replay this delivery? The receiver will get the original payload again — they must be idempotent on the X-Objectstack-Delivery header.',\n successMessage: 'Queued for redelivery',\n refreshAfter: true,\n // Only terminal rows are safe to replay. Pending / in_flight rows\n // are either already queued or actively being sent — replaying\n // would double-deliver.\n disabled: \"!(status in ['success', 'failed', 'dead'])\",\n },\n ],\n\n listViews: {\n recent: {\n type: 'grid',\n name: 'recent',\n label: 'Recent',\n data: { provider: 'object', object: 'sys_webhook_delivery' },\n columns: ['event_type', 'url', 'status', 'attempts', 'response_code', 'updated_at'],\n sort: [{ field: 'updated_at', order: 'desc' }],\n pagination: { pageSize: 50 },\n },\n failures: {\n type: 'grid',\n name: 'failures',\n label: 'Failures',\n data: { provider: 'object', object: 'sys_webhook_delivery' },\n columns: ['event_type', 'url', 'status', 'attempts', 'response_code', 'error', 'updated_at'],\n filter: [{ field: 'status', operator: 'in', value: ['failed', 'dead'] }],\n sort: [{ field: 'updated_at', order: 'desc' }],\n pagination: { pageSize: 50 },\n },\n in_flight: {\n type: 'grid',\n name: 'in_flight',\n label: 'In Flight',\n data: { provider: 'object', object: 'sys_webhook_delivery' },\n columns: ['event_type', 'url', 'attempts', 'claimed_by', 'claimed_at'],\n filter: [{ field: 'status', operator: 'equals', value: 'in_flight' }],\n sort: [{ field: 'claimed_at', order: 'desc' }],\n pagination: { pageSize: 50 },\n },\n pending: {\n type: 'grid',\n name: 'pending',\n label: 'Pending',\n data: { provider: 'object', object: 'sys_webhook_delivery' },\n columns: ['event_type', 'url', 'attempts', 'next_retry_at', 'updated_at'],\n filter: [{ field: 'status', operator: 'equals', value: 'pending' }],\n sort: [{ field: 'next_retry_at', order: 'asc' }],\n pagination: { pageSize: 50 },\n },\n by_status: {\n type: 'grid',\n name: 'by_status',\n label: 'By Status',\n data: { provider: 'object', object: 'sys_webhook_delivery' },\n columns: ['status', 'event_type', 'url', 'attempts', 'updated_at'],\n sort: [{ field: 'status', order: 'asc' }, { field: 'updated_at', order: 'desc' }],\n grouping: { fields: [{ field: 'status', order: 'asc', collapsed: false }] },\n pagination: { pageSize: 100 },\n },\n by_webhook: {\n type: 'grid',\n name: 'by_webhook',\n label: 'By Webhook',\n data: { provider: 'object', object: 'sys_webhook_delivery' },\n columns: ['webhook_id', 'event_type', 'status', 'attempts', 'updated_at'],\n sort: [{ field: 'webhook_id', order: 'asc' }, { field: 'updated_at', order: 'desc' }],\n grouping: { fields: [{ field: 'webhook_id', order: 'asc', collapsed: true }] },\n pagination: { pageSize: 100 },\n },\n all_deliveries: {\n type: 'grid',\n name: 'all_deliveries',\n label: 'All',\n data: { provider: 'object', object: 'sys_webhook_delivery' },\n columns: ['event_type', 'url', 'status', 'attempts', 'response_code', 'updated_at'],\n sort: [{ field: 'updated_at', order: 'desc' }],\n pagination: { pageSize: 100 },\n },\n },\n\n fields: {\n id: Field.text({\n label: 'Delivery ID',\n required: true,\n maxLength: 64,\n description: 'UUID — also doubles as the receiver-side idempotency key',\n }),\n\n webhook_id: Field.text({\n label: 'Webhook ID',\n required: true,\n maxLength: 64,\n description: 'FK to sys_webhook.id (loosely coupled — denormalised URL/secret on row)',\n }),\n\n event_id: Field.text({\n label: 'Event ID',\n required: true,\n maxLength: 128,\n description: 'Source event id; UNIQUE(event_id, webhook_id) for dedup',\n }),\n\n event_type: Field.text({\n label: 'Event Type',\n required: true,\n maxLength: 128,\n description: 'e.g. data.record.created',\n }),\n\n url: Field.text({\n label: 'Target URL',\n required: true,\n maxLength: 2048,\n description: 'Snapshotted at enqueue so config edits do not rewrite live rows',\n }),\n\n method: Field.text({ label: 'Method', required: false, maxLength: 10 }),\n headers_json: Field.textarea({ label: 'Headers JSON', required: false }),\n secret: Field.text({ label: 'HMAC Secret', required: false, maxLength: 256 }),\n timeout_ms: Field.number({ label: 'Timeout (ms)', required: false }),\n payload_json: Field.textarea({ label: 'Payload JSON', required: true }),\n\n partition_key: Field.number({\n label: 'Partition',\n required: true,\n description: 'hash(webhook_id) mod partitionCount — precomputed for cheap WHERE',\n }),\n\n status: Field.text({\n label: 'Status',\n required: true,\n defaultValue: 'pending',\n maxLength: 16,\n description: 'pending | in_flight | success | failed | dead',\n }),\n\n attempts: Field.number({\n label: 'Attempts',\n required: true,\n defaultValue: 0,\n description: 'Number of POST attempts made so far',\n }),\n\n claimed_by: Field.text({ label: 'Claimed By', required: false, maxLength: 128 }),\n claimed_at: Field.number({ label: 'Claimed At (ms)', required: false }),\n next_retry_at: Field.number({ label: 'Next Retry At (ms)', required: false }),\n last_attempted_at: Field.number({ label: 'Last Attempted At (ms)', required: false }),\n response_code: Field.number({ label: 'HTTP Status', required: false }),\n response_body: Field.textarea({ label: 'Response Body (capped)', required: false }),\n error: Field.textarea({ label: 'Error', required: false }),\n\n created_at: Field.number({ label: 'Created At (ms)', required: true }),\n updated_at: Field.number({ label: 'Updated At (ms)', required: true }),\n },\n\n indexes: [\n { fields: ['event_id', 'webhook_id'], unique: true },\n // Hot path: claim query\n { fields: ['status', 'partition_key', 'next_retry_at'] },\n // Reaper: scan stale in_flight rows by claimed_at\n { fields: ['status', 'claimed_at'] },\n { fields: ['webhook_id'] },\n ],\n});\n\n/** Canonical object name — exported so SqlWebhookOutbox callers can override if needed. */\nexport const SYS_WEBHOOK_DELIVERY = 'sys_webhook_delivery' as const;\n"],"mappings":"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","names":["Field","ObjectSchema"]}
@@ -1,256 +0,0 @@
1
- "use strict";Object.defineProperty(exports, "__esModule", {value: true}); function _nullishCoalesce(lhs, rhsFn) { if (lhs != null) { return lhs; } else { return rhsFn(); } } function _optionalChain(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; }
2
-
3
- var _chunkTDSI7UHYcjs = require('./chunk-TDSI7UHY.cjs');
4
-
5
- // src/sql-outbox.ts
6
- var _crypto = require('crypto');
7
-
8
- // src/outbox.ts
9
- var RedeliverError = class extends Error {
10
- constructor(message, code) {
11
- super(message);
12
- this.code = code;
13
- this.name = "RedeliverError";
14
- }
15
- };
16
-
17
- // src/partition.ts
18
- function hashPartition(key, count) {
19
- if (count <= 0) throw new Error("partition count must be > 0");
20
- let h = 2166136261;
21
- for (let i = 0; i < key.length; i++) {
22
- h ^= key.charCodeAt(i);
23
- h = Math.imul(h, 16777619);
24
- }
25
- return Math.abs(h | 0) % count;
26
- }
27
-
28
- // src/sql-outbox.ts
29
- var SqlWebhookOutbox = class {
30
- constructor(engine, opts) {
31
- this.engine = engine;
32
- if (opts.partitionCount <= 0) {
33
- throw new Error("SqlWebhookOutbox: partitionCount must be > 0");
34
- }
35
- this.objectName = _nullishCoalesce(opts.objectName, () => ( _chunkTDSI7UHYcjs.SYS_WEBHOOK_DELIVERY));
36
- this.partitionCount = opts.partitionCount;
37
- }
38
- async enqueue(input) {
39
- const existing = await this.engine.findOne(this.objectName, {
40
- where: { event_id: input.eventId, webhook_id: input.webhookId },
41
- fields: ["id"]
42
- });
43
- if (_optionalChain([existing, 'optionalAccess', _ => _.id])) return existing.id;
44
- const id = _crypto.randomUUID.call(void 0, );
45
- const now = Date.now();
46
- const row = {
47
- id,
48
- webhook_id: input.webhookId,
49
- event_id: input.eventId,
50
- event_type: input.eventType,
51
- url: input.url,
52
- method: _nullishCoalesce(input.method, () => ( "POST")),
53
- headers_json: input.headers ? JSON.stringify(input.headers) : void 0,
54
- secret: input.secret,
55
- timeout_ms: input.timeoutMs,
56
- payload_json: JSON.stringify(_nullishCoalesce(input.payload, () => ( null))),
57
- partition_key: hashPartition(input.webhookId, this.partitionCount),
58
- status: "pending",
59
- attempts: 0,
60
- created_at: now,
61
- updated_at: now
62
- };
63
- try {
64
- await this.engine.insert(this.objectName, row);
65
- return id;
66
- } catch (err) {
67
- const winner = await this.engine.findOne(this.objectName, {
68
- where: { event_id: input.eventId, webhook_id: input.webhookId },
69
- fields: ["id"]
70
- });
71
- if (_optionalChain([winner, 'optionalAccess', _2 => _2.id])) return winner.id;
72
- throw err;
73
- }
74
- }
75
- async claim(opts) {
76
- const now = _nullishCoalesce(opts.now, () => ( Date.now()));
77
- await this.engine.update(
78
- this.objectName,
79
- { status: "pending", claimed_by: null, claimed_at: null, updated_at: now },
80
- {
81
- where: {
82
- status: "in_flight",
83
- claimed_at: { $lt: now - opts.claimTtlMs }
84
- },
85
- multi: true
86
- }
87
- );
88
- const partitionFilter = opts.partition ? { partition_key: opts.partition.index } : {};
89
- const candidates = await this.engine.find(this.objectName, {
90
- where: {
91
- status: "pending",
92
- ...partitionFilter,
93
- // next_retry_at <= now OR null
94
- $or: [
95
- { next_retry_at: null },
96
- { next_retry_at: { $lte: now } }
97
- ]
98
- },
99
- fields: ["id"],
100
- // No orderBy for portability — drivers handle the natural insert order.
101
- limit: opts.limit
102
- });
103
- if (candidates.length === 0) return [];
104
- const ids = candidates.map((c) => c.id);
105
- await this.engine.update(
106
- this.objectName,
107
- {
108
- status: "in_flight",
109
- claimed_by: opts.nodeId,
110
- claimed_at: now,
111
- updated_at: now
112
- },
113
- {
114
- where: { id: { $in: ids }, status: "pending" },
115
- multi: true
116
- }
117
- );
118
- const claimed = await this.engine.find(this.objectName, {
119
- where: {
120
- id: { $in: ids },
121
- claimed_by: opts.nodeId,
122
- claimed_at: now,
123
- status: "in_flight"
124
- }
125
- });
126
- return claimed.map((r) => this.toDelivery(r));
127
- }
128
- async ack(id, result) {
129
- const current = await this.engine.findOne(this.objectName, {
130
- where: { id },
131
- fields: ["attempts"]
132
- });
133
- if (!current) return;
134
- const now = Date.now();
135
- let status;
136
- let nextRetryAt;
137
- let error;
138
- if (result.success) {
139
- status = "success";
140
- nextRetryAt = null;
141
- error = null;
142
- } else if (result.dead) {
143
- status = "dead";
144
- nextRetryAt = null;
145
- error = _nullishCoalesce(result.error, () => ( null));
146
- } else {
147
- status = "pending";
148
- nextRetryAt = _nullishCoalesce(result.nextRetryAt, () => ( null));
149
- error = _nullishCoalesce(result.error, () => ( null));
150
- }
151
- await this.engine.update(
152
- this.objectName,
153
- {
154
- status,
155
- attempts: (_nullishCoalesce(current.attempts, () => ( 0))) + 1,
156
- last_attempted_at: now,
157
- claimed_by: null,
158
- claimed_at: null,
159
- response_code: _nullishCoalesce(result.httpStatus, () => ( null)),
160
- response_body: _nullishCoalesce(result.responseBody, () => ( null)),
161
- next_retry_at: nextRetryAt,
162
- error,
163
- updated_at: now
164
- },
165
- { where: { id }, multi: false }
166
- );
167
- }
168
- async list(filter) {
169
- const rows = await this.engine.find(this.objectName, {
170
- where: _optionalChain([filter, 'optionalAccess', _3 => _3.status]) ? { status: filter.status } : {}
171
- });
172
- return rows.map((r) => this.toDelivery(r));
173
- }
174
- async redeliver(id) {
175
- const current = await this.engine.findOne(this.objectName, {
176
- where: { id }
177
- });
178
- if (!current) {
179
- throw new RedeliverError(
180
- `Delivery row '${id}' not found`,
181
- "not_found"
182
- );
183
- }
184
- if (current.status !== "success" && current.status !== "failed" && current.status !== "dead") {
185
- throw new RedeliverError(
186
- `Delivery row '${id}' is '${current.status}', expected one of: success, failed, dead`,
187
- "not_eligible"
188
- );
189
- }
190
- const now = Date.now();
191
- await this.engine.update(
192
- this.objectName,
193
- {
194
- status: "pending",
195
- attempts: 0,
196
- claimed_by: null,
197
- claimed_at: null,
198
- next_retry_at: null,
199
- last_attempted_at: null,
200
- response_code: null,
201
- response_body: null,
202
- error: null,
203
- updated_at: now
204
- },
205
- {
206
- where: {
207
- id,
208
- status: { $in: ["success", "failed", "dead"] }
209
- },
210
- multi: false
211
- }
212
- );
213
- const after = await this.engine.findOne(this.objectName, {
214
- where: { id }
215
- });
216
- if (!after || after.status !== "pending") {
217
- throw new RedeliverError(
218
- `Delivery row '${id}' state changed during redeliver`,
219
- "not_eligible"
220
- );
221
- }
222
- return this.toDelivery(after);
223
- }
224
- toDelivery(r) {
225
- return {
226
- id: r.id,
227
- webhookId: r.webhook_id,
228
- eventId: r.event_id,
229
- eventType: r.event_type,
230
- url: r.url,
231
- method: _nullishCoalesce(r.method, () => ( void 0)),
232
- headers: r.headers_json ? JSON.parse(r.headers_json) : void 0,
233
- secret: _nullishCoalesce(r.secret, () => ( void 0)),
234
- timeoutMs: _nullishCoalesce(r.timeout_ms, () => ( void 0)),
235
- payload: JSON.parse(r.payload_json),
236
- status: r.status,
237
- attempts: r.attempts,
238
- claimedBy: _nullishCoalesce(r.claimed_by, () => ( void 0)),
239
- claimedAt: _nullishCoalesce(r.claimed_at, () => ( void 0)),
240
- nextRetryAt: _nullishCoalesce(r.next_retry_at, () => ( void 0)),
241
- lastAttemptedAt: _nullishCoalesce(r.last_attempted_at, () => ( void 0)),
242
- responseCode: _nullishCoalesce(r.response_code, () => ( void 0)),
243
- responseBody: _nullishCoalesce(r.response_body, () => ( void 0)),
244
- error: _nullishCoalesce(r.error, () => ( void 0)),
245
- createdAt: r.created_at,
246
- updatedAt: r.updated_at
247
- };
248
- }
249
- };
250
-
251
-
252
-
253
-
254
-
255
- exports.RedeliverError = RedeliverError; exports.hashPartition = hashPartition; exports.SqlWebhookOutbox = SqlWebhookOutbox;
256
- //# sourceMappingURL=chunk-HF7CCDPB.cjs.map