@blokjs/trigger-webhook 0.2.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.
package/CHANGELOG.md ADDED
@@ -0,0 +1,22 @@
1
+ # @blokjs/trigger-webhook
2
+
3
+ ## 0.2.0
4
+
5
+ ### Minor Changes
6
+
7
+ - Initial public release of Blok packages.
8
+
9
+ This release includes:
10
+
11
+ - Core packages: @blokjs/shared, @blokjs/helper, @blokjs/runner
12
+ - Node packages: @blokjs/api-call, @blokjs/if-else, @blokjs/react
13
+ - Trigger packages: pubsub, queue, webhook, websocket, worker, cron, grpc
14
+ - CLI tool: blokctl
15
+ - Editor support: @blokjs/lsp-server, @blokjs/syntax
16
+
17
+ ### Patch Changes
18
+
19
+ - Updated dependencies
20
+ - @blokjs/shared@0.2.0
21
+ - @blokjs/helper@0.2.0
22
+ - @blokjs/runner@0.2.0
@@ -0,0 +1,129 @@
1
+ /**
2
+ * WebhookTrigger - Handle webhook events from external services
3
+ *
4
+ * Extends TriggerBase to process webhook events from:
5
+ * - GitHub (push, pull_request, issues, etc.)
6
+ * - Stripe (payment_intent, checkout.session, etc.)
7
+ * - Shopify (orders, products, customers)
8
+ * - Custom webhooks
9
+ *
10
+ * Features:
11
+ * - Signature verification for security
12
+ * - Event type filtering
13
+ * - Retry support
14
+ * - Dead letter handling
15
+ */
16
+ import type { HelperResponse, WebhookTriggerOpts } from "@blok/helper";
17
+ import { DefaultLogger, type GlobalOptions, type BlokService, TriggerBase, type TriggerResponse } from "@blok/runner";
18
+ /**
19
+ * Webhook event structure
20
+ */
21
+ export interface WebhookEvent {
22
+ /** Unique event ID */
23
+ id: string;
24
+ /** Source service (github, stripe, shopify, custom) */
25
+ source: string;
26
+ /** Event type (e.g., push, payment_intent.succeeded) */
27
+ eventType: string;
28
+ /** Event payload */
29
+ payload: unknown;
30
+ /** Request headers */
31
+ headers: Record<string, string>;
32
+ /** Signature (if provided) */
33
+ signature?: string;
34
+ /** Timestamp */
35
+ timestamp: Date;
36
+ /** Raw request body */
37
+ rawBody: string;
38
+ }
39
+ /**
40
+ * Signature verification result
41
+ */
42
+ export interface VerificationResult {
43
+ valid: boolean;
44
+ error?: string;
45
+ }
46
+ /**
47
+ * Webhook source handlers
48
+ */
49
+ export interface WebhookSourceHandler {
50
+ /** Extract event type from request */
51
+ getEventType(headers: Record<string, string>, body: unknown): string;
52
+ /** Get signature from request */
53
+ getSignature(headers: Record<string, string>): string | undefined;
54
+ /** Verify signature */
55
+ verifySignature(rawBody: string, signature: string, secret: string): VerificationResult;
56
+ /** Get event ID */
57
+ getEventId(headers: Record<string, string>, body: unknown): string;
58
+ }
59
+ /**
60
+ * Workflow model with webhook trigger configuration
61
+ */
62
+ interface WebhookWorkflowModel {
63
+ path: string;
64
+ config: {
65
+ name: string;
66
+ version: string;
67
+ trigger?: {
68
+ webhook?: WebhookTriggerOpts;
69
+ [key: string]: unknown;
70
+ };
71
+ [key: string]: unknown;
72
+ };
73
+ }
74
+ /**
75
+ * Built-in source handlers
76
+ */
77
+ declare const sourceHandlers: Record<string, WebhookSourceHandler>;
78
+ /**
79
+ * WebhookTrigger - Handle webhook events
80
+ */
81
+ export declare abstract class WebhookTrigger extends TriggerBase {
82
+ protected nodeMap: GlobalOptions;
83
+ protected readonly tracer: import("@opentelemetry/api").Tracer;
84
+ protected readonly logger: DefaultLogger;
85
+ protected webhookWorkflows: WebhookWorkflowModel[];
86
+ protected abstract nodes: Record<string, BlokService<unknown>>;
87
+ protected abstract workflows: Record<string, HelperResponse>;
88
+ constructor();
89
+ /**
90
+ * Load nodes into the node map
91
+ */
92
+ loadNodes(): void;
93
+ /**
94
+ * Load workflows into the workflow map
95
+ */
96
+ loadWorkflows(): void;
97
+ /**
98
+ * Initialize webhook trigger (call after loading workflows)
99
+ */
100
+ listen(): Promise<number>;
101
+ /**
102
+ * Stop the webhook trigger
103
+ */
104
+ stop(): Promise<void>;
105
+ protected onHmrWorkflowChange(): Promise<void>;
106
+ /**
107
+ * Process an incoming webhook request
108
+ * Call this from your HTTP endpoint handler
109
+ */
110
+ handleWebhook(source: string, rawBody: string, headers: Record<string, string>): Promise<TriggerResponse | null>;
111
+ /**
112
+ * Get all workflows that have webhook triggers
113
+ */
114
+ protected getWebhookWorkflows(): WebhookWorkflowModel[];
115
+ /**
116
+ * Find workflow matching the webhook event
117
+ */
118
+ protected findMatchingWorkflow(event: WebhookEvent): WebhookWorkflowModel | null;
119
+ /**
120
+ * Execute a workflow for a webhook event
121
+ */
122
+ protected executeWorkflow(event: WebhookEvent, workflow: WebhookWorkflowModel, _config: WebhookTriggerOpts): Promise<TriggerResponse>;
123
+ /**
124
+ * Register a custom source handler
125
+ */
126
+ static registerSourceHandler(source: string, handler: WebhookSourceHandler): void;
127
+ }
128
+ export default WebhookTrigger;
129
+ export { sourceHandlers };
@@ -0,0 +1,355 @@
1
+ "use strict";
2
+ /**
3
+ * WebhookTrigger - Handle webhook events from external services
4
+ *
5
+ * Extends TriggerBase to process webhook events from:
6
+ * - GitHub (push, pull_request, issues, etc.)
7
+ * - Stripe (payment_intent, checkout.session, etc.)
8
+ * - Shopify (orders, products, customers)
9
+ * - Custom webhooks
10
+ *
11
+ * Features:
12
+ * - Signature verification for security
13
+ * - Event type filtering
14
+ * - Retry support
15
+ * - Dead letter handling
16
+ */
17
+ var __importDefault = (this && this.__importDefault) || function (mod) {
18
+ return (mod && mod.__esModule) ? mod : { "default": mod };
19
+ };
20
+ Object.defineProperty(exports, "__esModule", { value: true });
21
+ exports.sourceHandlers = exports.WebhookTrigger = void 0;
22
+ const node_crypto_1 = __importDefault(require("node:crypto"));
23
+ const runner_1 = require("@blok/runner");
24
+ const api_1 = require("@opentelemetry/api");
25
+ const uuid_1 = require("uuid");
26
+ /**
27
+ * Built-in source handlers
28
+ */
29
+ const sourceHandlers = {
30
+ github: {
31
+ getEventType: (headers) => headers["x-github-event"] || "unknown",
32
+ getSignature: (headers) => headers["x-hub-signature-256"] || headers["x-hub-signature"],
33
+ verifySignature: (rawBody, signature, secret) => {
34
+ const hmac = node_crypto_1.default.createHmac("sha256", secret);
35
+ const digest = `sha256=${hmac.update(rawBody).digest("hex")}`;
36
+ const sigBuffer = Buffer.from(signature);
37
+ const digestBuffer = Buffer.from(digest);
38
+ // Length check first to avoid timing attack on length
39
+ if (sigBuffer.length !== digestBuffer.length) {
40
+ return { valid: false, error: "Invalid GitHub signature" };
41
+ }
42
+ const valid = node_crypto_1.default.timingSafeEqual(sigBuffer, digestBuffer);
43
+ return { valid, error: valid ? undefined : "Invalid GitHub signature" };
44
+ },
45
+ getEventId: (headers) => headers["x-github-delivery"] || (0, uuid_1.v4)(),
46
+ },
47
+ stripe: {
48
+ getEventType: (_, body) => body?.type || "unknown",
49
+ getSignature: (headers) => headers["stripe-signature"],
50
+ verifySignature: (rawBody, signature, secret) => {
51
+ // Stripe signature format: t=timestamp,v1=signature
52
+ const parts = signature.split(",").reduce((acc, part) => {
53
+ const [key, value] = part.split("=");
54
+ acc[key] = value;
55
+ return acc;
56
+ }, {});
57
+ const timestamp = parts.t;
58
+ const expectedSig = parts.v1;
59
+ if (!timestamp || !expectedSig) {
60
+ return { valid: false, error: "Invalid Stripe signature format" };
61
+ }
62
+ const payload = `${timestamp}.${rawBody}`;
63
+ const hmac = node_crypto_1.default.createHmac("sha256", secret);
64
+ const computedSig = hmac.update(payload).digest("hex");
65
+ const sigBuffer = Buffer.from(expectedSig);
66
+ const computedBuffer = Buffer.from(computedSig);
67
+ if (sigBuffer.length !== computedBuffer.length) {
68
+ return { valid: false, error: "Invalid Stripe signature" };
69
+ }
70
+ const valid = node_crypto_1.default.timingSafeEqual(sigBuffer, computedBuffer);
71
+ return { valid, error: valid ? undefined : "Invalid Stripe signature" };
72
+ },
73
+ getEventId: (_, body) => body?.id || (0, uuid_1.v4)(),
74
+ },
75
+ shopify: {
76
+ getEventType: (headers) => headers["x-shopify-topic"] || "unknown",
77
+ getSignature: (headers) => headers["x-shopify-hmac-sha256"],
78
+ verifySignature: (rawBody, signature, secret) => {
79
+ const hmac = node_crypto_1.default.createHmac("sha256", secret);
80
+ const digest = hmac.update(rawBody, "utf8").digest("base64");
81
+ const sigBuffer = Buffer.from(signature, "base64");
82
+ const digestBuffer = Buffer.from(digest, "base64");
83
+ if (sigBuffer.length !== digestBuffer.length) {
84
+ return { valid: false, error: "Invalid Shopify signature" };
85
+ }
86
+ const valid = node_crypto_1.default.timingSafeEqual(sigBuffer, digestBuffer);
87
+ return { valid, error: valid ? undefined : "Invalid Shopify signature" };
88
+ },
89
+ getEventId: (headers) => headers["x-shopify-webhook-id"] || (0, uuid_1.v4)(),
90
+ },
91
+ custom: {
92
+ getEventType: (headers, body) => headers["x-event-type"] || body?.event || "custom",
93
+ getSignature: (headers) => headers["x-signature"] || headers["x-webhook-signature"],
94
+ verifySignature: (rawBody, signature, secret) => {
95
+ // Default: HMAC-SHA256
96
+ const hmac = node_crypto_1.default.createHmac("sha256", secret);
97
+ const digest = hmac.update(rawBody).digest("hex");
98
+ const valid = signature === digest || signature === `sha256=${digest}`;
99
+ return { valid, error: valid ? undefined : "Invalid signature" };
100
+ },
101
+ getEventId: (headers, body) => headers["x-event-id"] || body?.id || (0, uuid_1.v4)(),
102
+ },
103
+ };
104
+ exports.sourceHandlers = sourceHandlers;
105
+ /**
106
+ * WebhookTrigger - Handle webhook events
107
+ */
108
+ class WebhookTrigger extends runner_1.TriggerBase {
109
+ nodeMap = {};
110
+ tracer = api_1.trace.getTracer(process.env.PROJECT_NAME || "trigger-webhook-workflow", process.env.PROJECT_VERSION || "0.0.1");
111
+ logger = new runner_1.DefaultLogger();
112
+ webhookWorkflows = [];
113
+ constructor() {
114
+ super();
115
+ this.loadNodes();
116
+ this.loadWorkflows();
117
+ }
118
+ /**
119
+ * Load nodes into the node map
120
+ */
121
+ loadNodes() {
122
+ this.nodeMap.nodes = new runner_1.NodeMap();
123
+ const nodeKeys = Object.keys(this.nodes);
124
+ for (const key of nodeKeys) {
125
+ this.nodeMap.nodes.addNode(key, this.nodes[key]);
126
+ }
127
+ }
128
+ /**
129
+ * Load workflows into the workflow map
130
+ */
131
+ loadWorkflows() {
132
+ this.nodeMap.workflows = this.workflows;
133
+ }
134
+ /**
135
+ * Initialize webhook trigger (call after loading workflows)
136
+ */
137
+ async listen() {
138
+ const startTime = this.startCounter();
139
+ // Find all workflows with webhook triggers
140
+ this.webhookWorkflows = this.getWebhookWorkflows();
141
+ if (this.webhookWorkflows.length === 0) {
142
+ this.logger.log("No workflows with webhook triggers found");
143
+ }
144
+ else {
145
+ this.logger.log(`Webhook trigger initialized. ${this.webhookWorkflows.length} workflow(s) registered`);
146
+ }
147
+ // Enable HMR in development mode
148
+ if (process.env.BLOK_HMR === "true" || process.env.NODE_ENV === "development") {
149
+ await this.enableHotReload();
150
+ }
151
+ return this.endCounter(startTime);
152
+ }
153
+ /**
154
+ * Stop the webhook trigger
155
+ */
156
+ async stop() {
157
+ this.webhookWorkflows = [];
158
+ this.logger.log("Webhook trigger stopped");
159
+ }
160
+ async onHmrWorkflowChange() {
161
+ this.loadWorkflows();
162
+ this.webhookWorkflows = this.getWebhookWorkflows();
163
+ this.logger.log(`[HMR] Webhook workflows reloaded. ${this.webhookWorkflows.length} workflow(s) registered`);
164
+ }
165
+ /**
166
+ * Process an incoming webhook request
167
+ * Call this from your HTTP endpoint handler
168
+ */
169
+ async handleWebhook(source, rawBody, headers) {
170
+ const handler = sourceHandlers[source] || sourceHandlers.custom;
171
+ // Parse body
172
+ let body;
173
+ try {
174
+ body = JSON.parse(rawBody);
175
+ }
176
+ catch {
177
+ body = rawBody;
178
+ }
179
+ // Create webhook event
180
+ const event = {
181
+ id: handler.getEventId(headers, body),
182
+ source,
183
+ eventType: handler.getEventType(headers, body),
184
+ payload: body,
185
+ headers,
186
+ signature: handler.getSignature(headers),
187
+ timestamp: new Date(),
188
+ rawBody,
189
+ };
190
+ // Find matching workflow
191
+ const workflow = this.findMatchingWorkflow(event);
192
+ if (!workflow) {
193
+ this.logger.log(`No matching workflow for webhook: ${source}/${event.eventType}`);
194
+ return null;
195
+ }
196
+ const config = workflow.config.trigger?.webhook;
197
+ // Verify signature if secret is configured
198
+ if (config.secret && event.signature) {
199
+ const verification = handler.verifySignature(rawBody, event.signature, config.secret);
200
+ if (!verification.valid) {
201
+ this.logger.error(`Webhook signature verification failed: ${verification.error}`);
202
+ throw new Error(`Signature verification failed: ${verification.error}`);
203
+ }
204
+ }
205
+ else if (config.secret && !event.signature) {
206
+ this.logger.error("Webhook signature missing but secret is configured");
207
+ throw new Error("Signature missing");
208
+ }
209
+ return this.executeWorkflow(event, workflow, config);
210
+ }
211
+ /**
212
+ * Get all workflows that have webhook triggers
213
+ */
214
+ getWebhookWorkflows() {
215
+ const workflows = [];
216
+ for (const [path, workflow] of Object.entries(this.nodeMap.workflows || {})) {
217
+ const workflowConfig = workflow._config;
218
+ if (workflowConfig?.trigger) {
219
+ const triggerType = Object.keys(workflowConfig.trigger)[0];
220
+ if (triggerType === "webhook" && workflowConfig.trigger.webhook) {
221
+ workflows.push({
222
+ path,
223
+ config: workflowConfig,
224
+ });
225
+ }
226
+ }
227
+ }
228
+ return workflows;
229
+ }
230
+ /**
231
+ * Find workflow matching the webhook event
232
+ */
233
+ findMatchingWorkflow(event) {
234
+ for (const workflow of this.webhookWorkflows) {
235
+ const config = workflow.config.trigger?.webhook;
236
+ if (!config)
237
+ continue;
238
+ // Check source match
239
+ if (config.source !== event.source)
240
+ continue;
241
+ // Check event type match
242
+ if (config.events && config.events.length > 0) {
243
+ const matches = config.events.some((pattern) => {
244
+ // Support wildcards (e.g., "push", "pull_request.*")
245
+ if (pattern === "*")
246
+ return true;
247
+ if (pattern.endsWith(".*")) {
248
+ const prefix = pattern.slice(0, -2);
249
+ return event.eventType.startsWith(prefix);
250
+ }
251
+ return pattern === event.eventType;
252
+ });
253
+ if (!matches)
254
+ continue;
255
+ }
256
+ return workflow;
257
+ }
258
+ return null;
259
+ }
260
+ /**
261
+ * Execute a workflow for a webhook event
262
+ */
263
+ async executeWorkflow(event, workflow, _config) {
264
+ const executionId = (0, uuid_1.v4)();
265
+ const defaultMeter = api_1.metrics.getMeter("default");
266
+ const webhookExecutions = defaultMeter.createCounter("webhook_executions", {
267
+ description: "Webhook executions",
268
+ });
269
+ const webhookErrors = defaultMeter.createCounter("webhook_errors", {
270
+ description: "Webhook execution errors",
271
+ });
272
+ return new Promise((resolve) => {
273
+ this.tracer.startActiveSpan(`webhook:${event.source}/${event.eventType}`, async (span) => {
274
+ try {
275
+ const start = performance.now();
276
+ // Initialize configuration for this workflow
277
+ await this.configuration.init(workflow.path, this.nodeMap);
278
+ // Create context
279
+ const ctx = this.createContext(undefined, workflow.path, executionId);
280
+ // Populate request with webhook event
281
+ ctx.request = {
282
+ body: event.payload,
283
+ headers: event.headers,
284
+ query: {},
285
+ params: {
286
+ source: event.source,
287
+ eventType: event.eventType,
288
+ eventId: event.id,
289
+ },
290
+ };
291
+ // Store webhook context in vars
292
+ if (!ctx.vars)
293
+ ctx.vars = {};
294
+ ctx.vars._webhook_event = {
295
+ id: event.id,
296
+ source: event.source,
297
+ eventType: event.eventType,
298
+ timestamp: event.timestamp.toISOString(),
299
+ hasSignature: String(!!event.signature),
300
+ };
301
+ ctx.logger.log(`Processing webhook: ${event.source}/${event.eventType} (${event.id})`);
302
+ // Execute workflow
303
+ const response = await this.run(ctx);
304
+ const end = performance.now();
305
+ // Set span attributes
306
+ span.setAttribute("success", true);
307
+ span.setAttribute("event_id", event.id);
308
+ span.setAttribute("source", event.source);
309
+ span.setAttribute("event_type", event.eventType);
310
+ span.setAttribute("workflow_path", workflow.path);
311
+ span.setAttribute("elapsed_ms", end - start);
312
+ span.setStatus({ code: api_1.SpanStatusCode.OK });
313
+ // Record metrics
314
+ webhookExecutions.add(1, {
315
+ env: process.env.NODE_ENV,
316
+ source: event.source,
317
+ event_type: event.eventType,
318
+ workflow_name: this.configuration.name,
319
+ success: "true",
320
+ });
321
+ ctx.logger.log(`Webhook processed in ${(end - start).toFixed(2)}ms: ${event.id}`);
322
+ resolve(response);
323
+ }
324
+ catch (error) {
325
+ const errorMessage = error.message;
326
+ // Set span error
327
+ span.setAttribute("success", false);
328
+ span.recordException(error);
329
+ span.setStatus({ code: api_1.SpanStatusCode.ERROR, message: errorMessage });
330
+ // Record error metrics
331
+ webhookErrors.add(1, {
332
+ env: process.env.NODE_ENV,
333
+ source: event.source,
334
+ event_type: event.eventType,
335
+ workflow_name: this.configuration?.name || "unknown",
336
+ });
337
+ this.logger.error(`Webhook failed ${event.id}: ${errorMessage}`, error.stack);
338
+ throw error;
339
+ }
340
+ finally {
341
+ span.end();
342
+ }
343
+ });
344
+ });
345
+ }
346
+ /**
347
+ * Register a custom source handler
348
+ */
349
+ static registerSourceHandler(source, handler) {
350
+ sourceHandlers[source] = handler;
351
+ }
352
+ }
353
+ exports.WebhookTrigger = WebhookTrigger;
354
+ exports.default = WebhookTrigger;
355
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"WebhookTrigger.js","sourceRoot":"","sources":["../src/WebhookTrigger.ts"],"names":[],"mappings":";AAAA;;;;;;;;;;;;;;GAcG;;;;;;AAEH,8DAAiC;AAEjC,yCAOsB;AAEtB,4CAA+E;AAC/E,+BAAkC;AA8DlC;;GAEG;AACH,MAAM,cAAc,GAAyC;IAC5D,MAAM,EAAE;QACP,YAAY,EAAE,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,gBAAgB,CAAC,IAAI,SAAS;QACjE,YAAY,EAAE,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,qBAAqB,CAAC,IAAI,OAAO,CAAC,iBAAiB,CAAC;QACvF,eAAe,EAAE,CAAC,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,EAAE;YAC/C,MAAM,IAAI,GAAG,qBAAM,CAAC,UAAU,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;YACjD,MAAM,MAAM,GAAG,UAAU,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC;YAC9D,MAAM,SAAS,GAAG,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;YACzC,MAAM,YAAY,GAAG,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;YACzC,sDAAsD;YACtD,IAAI,SAAS,CAAC,MAAM,KAAK,YAAY,CAAC,MAAM,EAAE,CAAC;gBAC9C,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,0BAA0B,EAAE,CAAC;YAC5D,CAAC;YACD,MAAM,KAAK,GAAG,qBAAM,CAAC,eAAe,CAAC,SAAS,EAAE,YAAY,CAAC,CAAC;YAC9D,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,0BAA0B,EAAE,CAAC;QACzE,CAAC;QACD,UAAU,EAAE,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,mBAAmB,CAAC,IAAI,IAAA,SAAI,GAAE;KAC/D;IAED,MAAM,EAAE;QACP,YAAY,EAAE,CAAC,CAAC,EAAE,IAAI,EAAE,EAAE,CAAE,IAA0B,EAAE,IAAI,IAAI,SAAS;QACzE,YAAY,EAAE,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,kBAAkB,CAAC;QACtD,eAAe,EAAE,CAAC,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,EAAE;YAC/C,oDAAoD;YACpD,MAAM,KAAK,GAAG,SAAS,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,MAAM,CACxC,CAAC,GAAG,EAAE,IAAI,EAAE,EAAE;gBACb,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;gBACrC,GAAG,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC;gBACjB,OAAO,GAAG,CAAC;YACZ,CAAC,EACD,EAA4B,CAC5B,CAAC;YAEF,MAAM,SAAS,GAAG,KAAK,CAAC,CAAC,CAAC;YAC1B,MAAM,WAAW,GAAG,KAAK,CAAC,EAAE,CAAC;YAE7B,IAAI,CAAC,SAAS,IAAI,CAAC,WAAW,EAAE,CAAC;gBAChC,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,iCAAiC,EAAE,CAAC;YACnE,CAAC;YAED,MAAM,OAAO,GAAG,GAAG,SAAS,IAAI,OAAO,EAAE,CAAC;YAC1C,MAAM,IAAI,GAAG,qBAAM,CAAC,UAAU,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;YACjD,MAAM,WAAW,GAAG,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;YAEvD,MAAM,SAAS,GAAG,MAAM,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;YAC3C,MAAM,cAAc,GAAG,MAAM,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;YAChD,IAAI,SAAS,CAAC,MAAM,KAAK,cAAc,CAAC,MAAM,EAAE,CAAC;gBAChD,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,0BAA0B,EAAE,CAAC;YAC5D,CAAC;YACD,MAAM,KAAK,GAAG,qBAAM,CAAC,eAAe,CAAC,SAAS,EAAE,cAAc,CAAC,CAAC;YAChE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,0BAA0B,EAAE,CAAC;QACzE,CAAC;QACD,UAAU,EAAE,CAAC,CAAC,EAAE,IAAI,EAAE,EAAE,CAAE,IAAwB,EAAE,EAAE,IAAI,IAAA,SAAI,GAAE;KAChE;IAED,OAAO,EAAE;QACR,YAAY,EAAE,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,iBAAiB,CAAC,IAAI,SAAS;QAClE,YAAY,EAAE,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,uBAAuB,CAAC;QAC3D,eAAe,EAAE,CAAC,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,EAAE;YAC/C,MAAM,IAAI,GAAG,qBAAM,CAAC,UAAU,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;YACjD,MAAM,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;YAC7D,MAAM,SAAS,GAAG,MAAM,CAAC,IAAI,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAC;YACnD,MAAM,YAAY,GAAG,MAAM,CAAC,IAAI,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAC;YACnD,IAAI,SAAS,CAAC,MAAM,KAAK,YAAY,CAAC,MAAM,EAAE,CAAC;gBAC9C,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,2BAA2B,EAAE,CAAC;YAC7D,CAAC;YACD,MAAM,KAAK,GAAG,qBAAM,CAAC,eAAe,CAAC,SAAS,EAAE,YAAY,CAAC,CAAC;YAC9D,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,2BAA2B,EAAE,CAAC;QAC1E,CAAC;QACD,UAAU,EAAE,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,sBAAsB,CAAC,IAAI,IAAA,SAAI,GAAE;KAClE;IAED,MAAM,EAAE;QACP,YAAY,EAAE,CAAC,OAAO,EAAE,IAAI,EAAE,EAAE,CAAC,OAAO,CAAC,cAAc,CAAC,IAAK,IAA2B,EAAE,KAAK,IAAI,QAAQ;QAC3G,YAAY,EAAE,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,aAAa,CAAC,IAAI,OAAO,CAAC,qBAAqB,CAAC;QACnF,eAAe,EAAE,CAAC,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,EAAE;YAC/C,uBAAuB;YACvB,MAAM,IAAI,GAAG,qBAAM,CAAC,UAAU,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;YACjD,MAAM,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;YAClD,MAAM,KAAK,GAAG,SAAS,KAAK,MAAM,IAAI,SAAS,KAAK,UAAU,MAAM,EAAE,CAAC;YACvE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,mBAAmB,EAAE,CAAC;QAClE,CAAC;QACD,UAAU,EAAE,CAAC,OAAO,EAAE,IAAI,EAAE,EAAE,CAAC,OAAO,CAAC,YAAY,CAAC,IAAK,IAAwB,EAAE,EAAE,IAAI,IAAA,SAAI,GAAE;KAC/F;CACD,CAAC;AA8SO,wCAAc;AA5SvB;;GAEG;AACH,MAAsB,cAAe,SAAQ,oBAAW;IAC7C,OAAO,GAAkB,EAAmB,CAAC;IACpC,MAAM,GAAG,WAAK,CAAC,SAAS,CAC1C,OAAO,CAAC,GAAG,CAAC,YAAY,IAAI,0BAA0B,EACtD,OAAO,CAAC,GAAG,CAAC,eAAe,IAAI,OAAO,CACtC,CAAC;IACiB,MAAM,GAAG,IAAI,sBAAa,EAAE,CAAC;IACtC,gBAAgB,GAA2B,EAAE,CAAC;IAMxD;QACC,KAAK,EAAE,CAAC;QACR,IAAI,CAAC,SAAS,EAAE,CAAC;QACjB,IAAI,CAAC,aAAa,EAAE,CAAC;IACtB,CAAC;IAED;;OAEG;IACH,SAAS;QACR,IAAI,CAAC,OAAO,CAAC,KAAK,GAAG,IAAI,gBAAO,EAAE,CAAC;QACnC,MAAM,QAAQ,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACzC,KAAK,MAAM,GAAG,IAAI,QAAQ,EAAE,CAAC;YAC5B,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,GAAG,EAAE,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC;QAClD,CAAC;IACF,CAAC;IAED;;OAEG;IACH,aAAa;QACZ,IAAI,CAAC,OAAO,CAAC,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC;IACzC,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,MAAM;QACX,MAAM,SAAS,GAAG,IAAI,CAAC,YAAY,EAAE,CAAC;QAEtC,2CAA2C;QAC3C,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAEnD,IAAI,IAAI,CAAC,gBAAgB,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACxC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,0CAA0C,CAAC,CAAC;QAC7D,CAAC;aAAM,CAAC;YACP,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,gCAAgC,IAAI,CAAC,gBAAgB,CAAC,MAAM,yBAAyB,CAAC,CAAC;QACxG,CAAC;QAED,iCAAiC;QACjC,IAAI,OAAO,CAAC,GAAG,CAAC,QAAQ,KAAK,MAAM,IAAI,OAAO,CAAC,GAAG,CAAC,QAAQ,KAAK,aAAa,EAAE,CAAC;YAC/E,MAAM,IAAI,CAAC,eAAe,EAAE,CAAC;QAC9B,CAAC;QAED,OAAO,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC;IACnC,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,IAAI;QACT,IAAI,CAAC,gBAAgB,GAAG,EAAE,CAAC;QAC3B,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,yBAAyB,CAAC,CAAC;IAC5C,CAAC;IAEkB,KAAK,CAAC,mBAAmB;QAC3C,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,mBAAmB,EAAE,CAAC;QACnD,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,qCAAqC,IAAI,CAAC,gBAAgB,CAAC,MAAM,yBAAyB,CAAC,CAAC;IAC7G,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,aAAa,CAClB,MAAc,EACd,OAAe,EACf,OAA+B;QAE/B,MAAM,OAAO,GAAG,cAAc,CAAC,MAAM,CAAC,IAAI,cAAc,CAAC,MAAM,CAAC;QAEhE,aAAa;QACb,IAAI,IAAa,CAAC;QAClB,IAAI,CAAC;YACJ,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;QAC5B,CAAC;QAAC,MAAM,CAAC;YACR,IAAI,GAAG,OAAO,CAAC;QAChB,CAAC;QAED,uBAAuB;QACvB,MAAM,KAAK,GAAiB;YAC3B,EAAE,EAAE,OAAO,CAAC,UAAU,CAAC,OAAO,EAAE,IAAI,CAAC;YACrC,MAAM;YACN,SAAS,EAAE,OAAO,CAAC,YAAY,CAAC,OAAO,EAAE,IAAI,CAAC;YAC9C,OAAO,EAAE,IAAI;YACb,OAAO;YACP,SAAS,EAAE,OAAO,CAAC,YAAY,CAAC,OAAO,CAAC;YACxC,SAAS,EAAE,IAAI,IAAI,EAAE;YACrB,OAAO;SACP,CAAC;QAEF,yBAAyB;QACzB,MAAM,QAAQ,GAAG,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,CAAC;QAClD,IAAI,CAAC,QAAQ,EAAE,CAAC;YACf,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,qCAAqC,MAAM,IAAI,KAAK,CAAC,SAAS,EAAE,CAAC,CAAC;YAClF,OAAO,IAAI,CAAC;QACb,CAAC;QAED,MAAM,MAAM,GAAG,QAAQ,CAAC,MAAM,CAAC,OAAO,EAAE,OAA6B,CAAC;QAEtE,2CAA2C;QAC3C,IAAI,MAAM,CAAC,MAAM,IAAI,KAAK,CAAC,SAAS,EAAE,CAAC;YACtC,MAAM,YAAY,GAAG,OAAO,CAAC,eAAe,CAAC,OAAO,EAAE,KAAK,CAAC,SAAS,EAAE,MAAM,CAAC,MAAM,CAAC,CAAC;YACtF,IAAI,CAAC,YAAY,CAAC,KAAK,EAAE,CAAC;gBACzB,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,0CAA0C,YAAY,CAAC,KAAK,EAAE,CAAC,CAAC;gBAClF,MAAM,IAAI,KAAK,CAAC,kCAAkC,YAAY,CAAC,KAAK,EAAE,CAAC,CAAC;YACzE,CAAC;QACF,CAAC;aAAM,IAAI,MAAM,CAAC,MAAM,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC;YAC9C,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,oDAAoD,CAAC,CAAC;YACxE,MAAM,IAAI,KAAK,CAAC,mBAAmB,CAAC,CAAC;QACtC,CAAC;QAED,OAAO,IAAI,CAAC,eAAe,CAAC,KAAK,EAAE,QAAQ,EAAE,MAAM,CAAC,CAAC;IACtD,CAAC;IAED;;OAEG;IACO,mBAAmB;QAC5B,MAAM,SAAS,GAA2B,EAAE,CAAC;QAE7C,KAAK,MAAM,CAAC,IAAI,EAAE,QAAQ,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,IAAI,EAAE,CAAC,EAAE,CAAC;YAC7E,MAAM,cAAc,GAAI,QAAmE,CAAC,OAAO,CAAC;YAEpG,IAAI,cAAc,EAAE,OAAO,EAAE,CAAC;gBAC7B,MAAM,WAAW,GAAG,MAAM,CAAC,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC;gBAE3D,IAAI,WAAW,KAAK,SAAS,IAAI,cAAc,CAAC,OAAO,CAAC,OAAO,EAAE,CAAC;oBACjE,SAAS,CAAC,IAAI,CAAC;wBACd,IAAI;wBACJ,MAAM,EAAE,cAAc;qBACtB,CAAC,CAAC;gBACJ,CAAC;YACF,CAAC;QACF,CAAC;QAED,OAAO,SAAS,CAAC;IAClB,CAAC;IAED;;OAEG;IACO,oBAAoB,CAAC,KAAmB;QACjD,KAAK,MAAM,QAAQ,IAAI,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC9C,MAAM,MAAM,GAAG,QAAQ,CAAC,MAAM,CAAC,OAAO,EAAE,OAAO,CAAC;YAChD,IAAI,CAAC,MAAM;gBAAE,SAAS;YAEtB,qBAAqB;YACrB,IAAI,MAAM,CAAC,MAAM,KAAK,KAAK,CAAC,MAAM;gBAAE,SAAS;YAE7C,yBAAyB;YACzB,IAAI,MAAM,CAAC,MAAM,IAAI,MAAM,CAAC,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBAC/C,MAAM,OAAO,GAAG,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,OAAO,EAAE,EAAE;oBAC9C,qDAAqD;oBACrD,IAAI,OAAO,KAAK,GAAG;wBAAE,OAAO,IAAI,CAAC;oBACjC,IAAI,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;wBAC5B,MAAM,MAAM,GAAG,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;wBACpC,OAAO,KAAK,CAAC,SAAS,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC;oBAC3C,CAAC;oBACD,OAAO,OAAO,KAAK,KAAK,CAAC,SAAS,CAAC;gBACpC,CAAC,CAAC,CAAC;gBACH,IAAI,CAAC,OAAO;oBAAE,SAAS;YACxB,CAAC;YAED,OAAO,QAAQ,CAAC;QACjB,CAAC;QAED,OAAO,IAAI,CAAC;IACb,CAAC;IAED;;OAEG;IACO,KAAK,CAAC,eAAe,CAC9B,KAAmB,EACnB,QAA8B,EAC9B,OAA2B;QAE3B,MAAM,WAAW,GAAG,IAAA,SAAI,GAAE,CAAC;QAE3B,MAAM,YAAY,GAAG,aAAO,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC;QACjD,MAAM,iBAAiB,GAAG,YAAY,CAAC,aAAa,CAAC,oBAAoB,EAAE;YAC1E,WAAW,EAAE,oBAAoB;SACjC,CAAC,CAAC;QACH,MAAM,aAAa,GAAG,YAAY,CAAC,aAAa,CAAC,gBAAgB,EAAE;YAClE,WAAW,EAAE,0BAA0B;SACvC,CAAC,CAAC;QAEH,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;YAC9B,IAAI,CAAC,MAAM,CAAC,eAAe,CAAC,WAAW,KAAK,CAAC,MAAM,IAAI,KAAK,CAAC,SAAS,EAAE,EAAE,KAAK,EAAE,IAAU,EAAE,EAAE;gBAC9F,IAAI,CAAC;oBACJ,MAAM,KAAK,GAAG,WAAW,CAAC,GAAG,EAAE,CAAC;oBAEhC,6CAA6C;oBAC7C,MAAM,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;oBAE3D,iBAAiB;oBACjB,MAAM,GAAG,GAAY,IAAI,CAAC,aAAa,CAAC,SAAS,EAAE,QAAQ,CAAC,IAAI,EAAE,WAAW,CAAC,CAAC;oBAE/E,sCAAsC;oBACtC,GAAG,CAAC,OAAO,GAAG;wBACb,IAAI,EAAE,KAAK,CAAC,OAAO;wBACnB,OAAO,EAAE,KAAK,CAAC,OAAO;wBACtB,KAAK,EAAE,EAAE;wBACT,MAAM,EAAE;4BACP,MAAM,EAAE,KAAK,CAAC,MAAM;4BACpB,SAAS,EAAE,KAAK,CAAC,SAAS;4BAC1B,OAAO,EAAE,KAAK,CAAC,EAAE;yBACjB;qBAC4B,CAAC;oBAE/B,gCAAgC;oBAChC,IAAI,CAAC,GAAG,CAAC,IAAI;wBAAE,GAAG,CAAC,IAAI,GAAG,EAAE,CAAC;oBAC7B,GAAG,CAAC,IAAI,CAAC,cAAc,GAAG;wBACzB,EAAE,EAAE,KAAK,CAAC,EAAE;wBACZ,MAAM,EAAE,KAAK,CAAC,MAAM;wBACpB,SAAS,EAAE,KAAK,CAAC,SAAS;wBAC1B,SAAS,EAAE,KAAK,CAAC,SAAS,CAAC,WAAW,EAAE;wBACxC,YAAY,EAAE,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,SAAS,CAAC;qBACvC,CAAC;oBAEF,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,uBAAuB,KAAK,CAAC,MAAM,IAAI,KAAK,CAAC,SAAS,KAAK,KAAK,CAAC,EAAE,GAAG,CAAC,CAAC;oBAEvF,mBAAmB;oBACnB,MAAM,QAAQ,GAAoB,MAAM,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;oBACtD,MAAM,GAAG,GAAG,WAAW,CAAC,GAAG,EAAE,CAAC;oBAE9B,sBAAsB;oBACtB,IAAI,CAAC,YAAY,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;oBACnC,IAAI,CAAC,YAAY,CAAC,UAAU,EAAE,KAAK,CAAC,EAAE,CAAC,CAAC;oBACxC,IAAI,CAAC,YAAY,CAAC,QAAQ,EAAE,KAAK,CAAC,MAAM,CAAC,CAAC;oBAC1C,IAAI,CAAC,YAAY,CAAC,YAAY,EAAE,KAAK,CAAC,SAAS,CAAC,CAAC;oBACjD,IAAI,CAAC,YAAY,CAAC,eAAe,EAAE,QAAQ,CAAC,IAAI,CAAC,CAAC;oBAClD,IAAI,CAAC,YAAY,CAAC,YAAY,EAAE,GAAG,GAAG,KAAK,CAAC,CAAC;oBAC7C,IAAI,CAAC,SAAS,CAAC,EAAE,IAAI,EAAE,oBAAc,CAAC,EAAE,EAAE,CAAC,CAAC;oBAE5C,iBAAiB;oBACjB,iBAAiB,CAAC,GAAG,CAAC,CAAC,EAAE;wBACxB,GAAG,EAAE,OAAO,CAAC,GAAG,CAAC,QAAQ;wBACzB,MAAM,EAAE,KAAK,CAAC,MAAM;wBACpB,UAAU,EAAE,KAAK,CAAC,SAAS;wBAC3B,aAAa,EAAE,IAAI,CAAC,aAAa,CAAC,IAAI;wBACtC,OAAO,EAAE,MAAM;qBACf,CAAC,CAAC;oBAEH,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,wBAAwB,CAAC,GAAG,GAAG,KAAK,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,OAAO,KAAK,CAAC,EAAE,EAAE,CAAC,CAAC;oBAElF,OAAO,CAAC,QAAQ,CAAC,CAAC;gBACnB,CAAC;gBAAC,OAAO,KAAK,EAAE,CAAC;oBAChB,MAAM,YAAY,GAAI,KAAe,CAAC,OAAO,CAAC;oBAE9C,iBAAiB;oBACjB,IAAI,CAAC,YAAY,CAAC,SAAS,EAAE,KAAK,CAAC,CAAC;oBACpC,IAAI,CAAC,eAAe,CAAC,KAAc,CAAC,CAAC;oBACrC,IAAI,CAAC,SAAS,CAAC,EAAE,IAAI,EAAE,oBAAc,CAAC,KAAK,EAAE,OAAO,EAAE,YAAY,EAAE,CAAC,CAAC;oBAEtE,uBAAuB;oBACvB,aAAa,CAAC,GAAG,CAAC,CAAC,EAAE;wBACpB,GAAG,EAAE,OAAO,CAAC,GAAG,CAAC,QAAQ;wBACzB,MAAM,EAAE,KAAK,CAAC,MAAM;wBACpB,UAAU,EAAE,KAAK,CAAC,SAAS;wBAC3B,aAAa,EAAE,IAAI,CAAC,aAAa,EAAE,IAAI,IAAI,SAAS;qBACpD,CAAC,CAAC;oBAEH,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,kBAAkB,KAAK,CAAC,EAAE,KAAK,YAAY,EAAE,EAAG,KAAe,CAAC,KAAK,CAAC,CAAC;oBAEzF,MAAM,KAAK,CAAC;gBACb,CAAC;wBAAS,CAAC;oBACV,IAAI,CAAC,GAAG,EAAE,CAAC;gBACZ,CAAC;YACF,CAAC,CAAC,CAAC;QACJ,CAAC,CAAC,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,MAAM,CAAC,qBAAqB,CAAC,MAAc,EAAE,OAA6B;QACzE,cAAc,CAAC,MAAM,CAAC,GAAG,OAAO,CAAC;IAClC,CAAC;CACD;AAtSD,wCAsSC;AAED,kBAAe,cAAc,CAAC","sourcesContent":["/**\n * WebhookTrigger - Handle webhook events from external services\n *\n * Extends TriggerBase to process webhook events from:\n * - GitHub (push, pull_request, issues, etc.)\n * - Stripe (payment_intent, checkout.session, etc.)\n * - Shopify (orders, products, customers)\n * - Custom webhooks\n *\n * Features:\n * - Signature verification for security\n * - Event type filtering\n * - Retry support\n * - Dead letter handling\n */\n\nimport crypto from \"node:crypto\";\nimport type { HelperResponse, WebhookTriggerOpts } from \"@blok/helper\";\nimport {\n\tDefaultLogger,\n\ttype GlobalOptions,\n\ttype BlokService,\n\tNodeMap,\n\tTriggerBase,\n\ttype TriggerResponse,\n} from \"@blok/runner\";\nimport type { Context, RequestContext } from \"@blok/shared\";\nimport { type Span, SpanStatusCode, metrics, trace } from \"@opentelemetry/api\";\nimport { v4 as uuid } from \"uuid\";\n\n/**\n * Webhook event structure\n */\nexport interface WebhookEvent {\n\t/** Unique event ID */\n\tid: string;\n\t/** Source service (github, stripe, shopify, custom) */\n\tsource: string;\n\t/** Event type (e.g., push, payment_intent.succeeded) */\n\teventType: string;\n\t/** Event payload */\n\tpayload: unknown;\n\t/** Request headers */\n\theaders: Record<string, string>;\n\t/** Signature (if provided) */\n\tsignature?: string;\n\t/** Timestamp */\n\ttimestamp: Date;\n\t/** Raw request body */\n\trawBody: string;\n}\n\n/**\n * Signature verification result\n */\nexport interface VerificationResult {\n\tvalid: boolean;\n\terror?: string;\n}\n\n/**\n * Webhook source handlers\n */\nexport interface WebhookSourceHandler {\n\t/** Extract event type from request */\n\tgetEventType(headers: Record<string, string>, body: unknown): string;\n\t/** Get signature from request */\n\tgetSignature(headers: Record<string, string>): string | undefined;\n\t/** Verify signature */\n\tverifySignature(rawBody: string, signature: string, secret: string): VerificationResult;\n\t/** Get event ID */\n\tgetEventId(headers: Record<string, string>, body: unknown): string;\n}\n\n/**\n * Workflow model with webhook trigger configuration\n */\ninterface WebhookWorkflowModel {\n\tpath: string;\n\tconfig: {\n\t\tname: string;\n\t\tversion: string;\n\t\ttrigger?: {\n\t\t\twebhook?: WebhookTriggerOpts;\n\t\t\t[key: string]: unknown;\n\t\t};\n\t\t[key: string]: unknown;\n\t};\n}\n\n/**\n * Built-in source handlers\n */\nconst sourceHandlers: Record<string, WebhookSourceHandler> = {\n\tgithub: {\n\t\tgetEventType: (headers) => headers[\"x-github-event\"] || \"unknown\",\n\t\tgetSignature: (headers) => headers[\"x-hub-signature-256\"] || headers[\"x-hub-signature\"],\n\t\tverifySignature: (rawBody, signature, secret) => {\n\t\t\tconst hmac = crypto.createHmac(\"sha256\", secret);\n\t\t\tconst digest = `sha256=${hmac.update(rawBody).digest(\"hex\")}`;\n\t\t\tconst sigBuffer = Buffer.from(signature);\n\t\t\tconst digestBuffer = Buffer.from(digest);\n\t\t\t// Length check first to avoid timing attack on length\n\t\t\tif (sigBuffer.length !== digestBuffer.length) {\n\t\t\t\treturn { valid: false, error: \"Invalid GitHub signature\" };\n\t\t\t}\n\t\t\tconst valid = crypto.timingSafeEqual(sigBuffer, digestBuffer);\n\t\t\treturn { valid, error: valid ? undefined : \"Invalid GitHub signature\" };\n\t\t},\n\t\tgetEventId: (headers) => headers[\"x-github-delivery\"] || uuid(),\n\t},\n\n\tstripe: {\n\t\tgetEventType: (_, body) => (body as { type?: string })?.type || \"unknown\",\n\t\tgetSignature: (headers) => headers[\"stripe-signature\"],\n\t\tverifySignature: (rawBody, signature, secret) => {\n\t\t\t// Stripe signature format: t=timestamp,v1=signature\n\t\t\tconst parts = signature.split(\",\").reduce(\n\t\t\t\t(acc, part) => {\n\t\t\t\t\tconst [key, value] = part.split(\"=\");\n\t\t\t\t\tacc[key] = value;\n\t\t\t\t\treturn acc;\n\t\t\t\t},\n\t\t\t\t{} as Record<string, string>,\n\t\t\t);\n\n\t\t\tconst timestamp = parts.t;\n\t\t\tconst expectedSig = parts.v1;\n\n\t\t\tif (!timestamp || !expectedSig) {\n\t\t\t\treturn { valid: false, error: \"Invalid Stripe signature format\" };\n\t\t\t}\n\n\t\t\tconst payload = `${timestamp}.${rawBody}`;\n\t\t\tconst hmac = crypto.createHmac(\"sha256\", secret);\n\t\t\tconst computedSig = hmac.update(payload).digest(\"hex\");\n\n\t\t\tconst sigBuffer = Buffer.from(expectedSig);\n\t\t\tconst computedBuffer = Buffer.from(computedSig);\n\t\t\tif (sigBuffer.length !== computedBuffer.length) {\n\t\t\t\treturn { valid: false, error: \"Invalid Stripe signature\" };\n\t\t\t}\n\t\t\tconst valid = crypto.timingSafeEqual(sigBuffer, computedBuffer);\n\t\t\treturn { valid, error: valid ? undefined : \"Invalid Stripe signature\" };\n\t\t},\n\t\tgetEventId: (_, body) => (body as { id?: string })?.id || uuid(),\n\t},\n\n\tshopify: {\n\t\tgetEventType: (headers) => headers[\"x-shopify-topic\"] || \"unknown\",\n\t\tgetSignature: (headers) => headers[\"x-shopify-hmac-sha256\"],\n\t\tverifySignature: (rawBody, signature, secret) => {\n\t\t\tconst hmac = crypto.createHmac(\"sha256\", secret);\n\t\t\tconst digest = hmac.update(rawBody, \"utf8\").digest(\"base64\");\n\t\t\tconst sigBuffer = Buffer.from(signature, \"base64\");\n\t\t\tconst digestBuffer = Buffer.from(digest, \"base64\");\n\t\t\tif (sigBuffer.length !== digestBuffer.length) {\n\t\t\t\treturn { valid: false, error: \"Invalid Shopify signature\" };\n\t\t\t}\n\t\t\tconst valid = crypto.timingSafeEqual(sigBuffer, digestBuffer);\n\t\t\treturn { valid, error: valid ? undefined : \"Invalid Shopify signature\" };\n\t\t},\n\t\tgetEventId: (headers) => headers[\"x-shopify-webhook-id\"] || uuid(),\n\t},\n\n\tcustom: {\n\t\tgetEventType: (headers, body) => headers[\"x-event-type\"] || (body as { event?: string })?.event || \"custom\",\n\t\tgetSignature: (headers) => headers[\"x-signature\"] || headers[\"x-webhook-signature\"],\n\t\tverifySignature: (rawBody, signature, secret) => {\n\t\t\t// Default: HMAC-SHA256\n\t\t\tconst hmac = crypto.createHmac(\"sha256\", secret);\n\t\t\tconst digest = hmac.update(rawBody).digest(\"hex\");\n\t\t\tconst valid = signature === digest || signature === `sha256=${digest}`;\n\t\t\treturn { valid, error: valid ? undefined : \"Invalid signature\" };\n\t\t},\n\t\tgetEventId: (headers, body) => headers[\"x-event-id\"] || (body as { id?: string })?.id || uuid(),\n\t},\n};\n\n/**\n * WebhookTrigger - Handle webhook events\n */\nexport abstract class WebhookTrigger extends TriggerBase {\n\tprotected nodeMap: GlobalOptions = {} as GlobalOptions;\n\tprotected readonly tracer = trace.getTracer(\n\t\tprocess.env.PROJECT_NAME || \"trigger-webhook-workflow\",\n\t\tprocess.env.PROJECT_VERSION || \"0.0.1\",\n\t);\n\tprotected readonly logger = new DefaultLogger();\n\tprotected webhookWorkflows: WebhookWorkflowModel[] = [];\n\n\t// Subclasses provide these\n\tprotected abstract nodes: Record<string, BlokService<unknown>>;\n\tprotected abstract workflows: Record<string, HelperResponse>;\n\n\tconstructor() {\n\t\tsuper();\n\t\tthis.loadNodes();\n\t\tthis.loadWorkflows();\n\t}\n\n\t/**\n\t * Load nodes into the node map\n\t */\n\tloadNodes(): void {\n\t\tthis.nodeMap.nodes = new NodeMap();\n\t\tconst nodeKeys = Object.keys(this.nodes);\n\t\tfor (const key of nodeKeys) {\n\t\t\tthis.nodeMap.nodes.addNode(key, this.nodes[key]);\n\t\t}\n\t}\n\n\t/**\n\t * Load workflows into the workflow map\n\t */\n\tloadWorkflows(): void {\n\t\tthis.nodeMap.workflows = this.workflows;\n\t}\n\n\t/**\n\t * Initialize webhook trigger (call after loading workflows)\n\t */\n\tasync listen(): Promise<number> {\n\t\tconst startTime = this.startCounter();\n\n\t\t// Find all workflows with webhook triggers\n\t\tthis.webhookWorkflows = this.getWebhookWorkflows();\n\n\t\tif (this.webhookWorkflows.length === 0) {\n\t\t\tthis.logger.log(\"No workflows with webhook triggers found\");\n\t\t} else {\n\t\t\tthis.logger.log(`Webhook trigger initialized. ${this.webhookWorkflows.length} workflow(s) registered`);\n\t\t}\n\n\t\t// Enable HMR in development mode\n\t\tif (process.env.BLOK_HMR === \"true\" || process.env.NODE_ENV === \"development\") {\n\t\t\tawait this.enableHotReload();\n\t\t}\n\n\t\treturn this.endCounter(startTime);\n\t}\n\n\t/**\n\t * Stop the webhook trigger\n\t */\n\tasync stop(): Promise<void> {\n\t\tthis.webhookWorkflows = [];\n\t\tthis.logger.log(\"Webhook trigger stopped\");\n\t}\n\n\tprotected override async onHmrWorkflowChange(): Promise<void> {\n\t\tthis.loadWorkflows();\n\t\tthis.webhookWorkflows = this.getWebhookWorkflows();\n\t\tthis.logger.log(`[HMR] Webhook workflows reloaded. ${this.webhookWorkflows.length} workflow(s) registered`);\n\t}\n\n\t/**\n\t * Process an incoming webhook request\n\t * Call this from your HTTP endpoint handler\n\t */\n\tasync handleWebhook(\n\t\tsource: string,\n\t\trawBody: string,\n\t\theaders: Record<string, string>,\n\t): Promise<TriggerResponse | null> {\n\t\tconst handler = sourceHandlers[source] || sourceHandlers.custom;\n\n\t\t// Parse body\n\t\tlet body: unknown;\n\t\ttry {\n\t\t\tbody = JSON.parse(rawBody);\n\t\t} catch {\n\t\t\tbody = rawBody;\n\t\t}\n\n\t\t// Create webhook event\n\t\tconst event: WebhookEvent = {\n\t\t\tid: handler.getEventId(headers, body),\n\t\t\tsource,\n\t\t\teventType: handler.getEventType(headers, body),\n\t\t\tpayload: body,\n\t\t\theaders,\n\t\t\tsignature: handler.getSignature(headers),\n\t\t\ttimestamp: new Date(),\n\t\t\trawBody,\n\t\t};\n\n\t\t// Find matching workflow\n\t\tconst workflow = this.findMatchingWorkflow(event);\n\t\tif (!workflow) {\n\t\t\tthis.logger.log(`No matching workflow for webhook: ${source}/${event.eventType}`);\n\t\t\treturn null;\n\t\t}\n\n\t\tconst config = workflow.config.trigger?.webhook as WebhookTriggerOpts;\n\n\t\t// Verify signature if secret is configured\n\t\tif (config.secret && event.signature) {\n\t\t\tconst verification = handler.verifySignature(rawBody, event.signature, config.secret);\n\t\t\tif (!verification.valid) {\n\t\t\t\tthis.logger.error(`Webhook signature verification failed: ${verification.error}`);\n\t\t\t\tthrow new Error(`Signature verification failed: ${verification.error}`);\n\t\t\t}\n\t\t} else if (config.secret && !event.signature) {\n\t\t\tthis.logger.error(\"Webhook signature missing but secret is configured\");\n\t\t\tthrow new Error(\"Signature missing\");\n\t\t}\n\n\t\treturn this.executeWorkflow(event, workflow, config);\n\t}\n\n\t/**\n\t * Get all workflows that have webhook triggers\n\t */\n\tprotected getWebhookWorkflows(): WebhookWorkflowModel[] {\n\t\tconst workflows: WebhookWorkflowModel[] = [];\n\n\t\tfor (const [path, workflow] of Object.entries(this.nodeMap.workflows || {})) {\n\t\t\tconst workflowConfig = (workflow as unknown as { _config: WebhookWorkflowModel[\"config\"] })._config;\n\n\t\t\tif (workflowConfig?.trigger) {\n\t\t\t\tconst triggerType = Object.keys(workflowConfig.trigger)[0];\n\n\t\t\t\tif (triggerType === \"webhook\" && workflowConfig.trigger.webhook) {\n\t\t\t\t\tworkflows.push({\n\t\t\t\t\t\tpath,\n\t\t\t\t\t\tconfig: workflowConfig,\n\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn workflows;\n\t}\n\n\t/**\n\t * Find workflow matching the webhook event\n\t */\n\tprotected findMatchingWorkflow(event: WebhookEvent): WebhookWorkflowModel | null {\n\t\tfor (const workflow of this.webhookWorkflows) {\n\t\t\tconst config = workflow.config.trigger?.webhook;\n\t\t\tif (!config) continue;\n\n\t\t\t// Check source match\n\t\t\tif (config.source !== event.source) continue;\n\n\t\t\t// Check event type match\n\t\t\tif (config.events && config.events.length > 0) {\n\t\t\t\tconst matches = config.events.some((pattern) => {\n\t\t\t\t\t// Support wildcards (e.g., \"push\", \"pull_request.*\")\n\t\t\t\t\tif (pattern === \"*\") return true;\n\t\t\t\t\tif (pattern.endsWith(\".*\")) {\n\t\t\t\t\t\tconst prefix = pattern.slice(0, -2);\n\t\t\t\t\t\treturn event.eventType.startsWith(prefix);\n\t\t\t\t\t}\n\t\t\t\t\treturn pattern === event.eventType;\n\t\t\t\t});\n\t\t\t\tif (!matches) continue;\n\t\t\t}\n\n\t\t\treturn workflow;\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Execute a workflow for a webhook event\n\t */\n\tprotected async executeWorkflow(\n\t\tevent: WebhookEvent,\n\t\tworkflow: WebhookWorkflowModel,\n\t\t_config: WebhookTriggerOpts,\n\t): Promise<TriggerResponse> {\n\t\tconst executionId = uuid();\n\n\t\tconst defaultMeter = metrics.getMeter(\"default\");\n\t\tconst webhookExecutions = defaultMeter.createCounter(\"webhook_executions\", {\n\t\t\tdescription: \"Webhook executions\",\n\t\t});\n\t\tconst webhookErrors = defaultMeter.createCounter(\"webhook_errors\", {\n\t\t\tdescription: \"Webhook execution errors\",\n\t\t});\n\n\t\treturn new Promise((resolve) => {\n\t\t\tthis.tracer.startActiveSpan(`webhook:${event.source}/${event.eventType}`, async (span: Span) => {\n\t\t\t\ttry {\n\t\t\t\t\tconst start = performance.now();\n\n\t\t\t\t\t// Initialize configuration for this workflow\n\t\t\t\t\tawait this.configuration.init(workflow.path, this.nodeMap);\n\n\t\t\t\t\t// Create context\n\t\t\t\t\tconst ctx: Context = this.createContext(undefined, workflow.path, executionId);\n\n\t\t\t\t\t// Populate request with webhook event\n\t\t\t\t\tctx.request = {\n\t\t\t\t\t\tbody: event.payload,\n\t\t\t\t\t\theaders: event.headers,\n\t\t\t\t\t\tquery: {},\n\t\t\t\t\t\tparams: {\n\t\t\t\t\t\t\tsource: event.source,\n\t\t\t\t\t\t\teventType: event.eventType,\n\t\t\t\t\t\t\teventId: event.id,\n\t\t\t\t\t\t},\n\t\t\t\t\t} as unknown as RequestContext;\n\n\t\t\t\t\t// Store webhook context in vars\n\t\t\t\t\tif (!ctx.vars) ctx.vars = {};\n\t\t\t\t\tctx.vars._webhook_event = {\n\t\t\t\t\t\tid: event.id,\n\t\t\t\t\t\tsource: event.source,\n\t\t\t\t\t\teventType: event.eventType,\n\t\t\t\t\t\ttimestamp: event.timestamp.toISOString(),\n\t\t\t\t\t\thasSignature: String(!!event.signature),\n\t\t\t\t\t};\n\n\t\t\t\t\tctx.logger.log(`Processing webhook: ${event.source}/${event.eventType} (${event.id})`);\n\n\t\t\t\t\t// Execute workflow\n\t\t\t\t\tconst response: TriggerResponse = await this.run(ctx);\n\t\t\t\t\tconst end = performance.now();\n\n\t\t\t\t\t// Set span attributes\n\t\t\t\t\tspan.setAttribute(\"success\", true);\n\t\t\t\t\tspan.setAttribute(\"event_id\", event.id);\n\t\t\t\t\tspan.setAttribute(\"source\", event.source);\n\t\t\t\t\tspan.setAttribute(\"event_type\", event.eventType);\n\t\t\t\t\tspan.setAttribute(\"workflow_path\", workflow.path);\n\t\t\t\t\tspan.setAttribute(\"elapsed_ms\", end - start);\n\t\t\t\t\tspan.setStatus({ code: SpanStatusCode.OK });\n\n\t\t\t\t\t// Record metrics\n\t\t\t\t\twebhookExecutions.add(1, {\n\t\t\t\t\t\tenv: process.env.NODE_ENV,\n\t\t\t\t\t\tsource: event.source,\n\t\t\t\t\t\tevent_type: event.eventType,\n\t\t\t\t\t\tworkflow_name: this.configuration.name,\n\t\t\t\t\t\tsuccess: \"true\",\n\t\t\t\t\t});\n\n\t\t\t\t\tctx.logger.log(`Webhook processed in ${(end - start).toFixed(2)}ms: ${event.id}`);\n\n\t\t\t\t\tresolve(response);\n\t\t\t\t} catch (error) {\n\t\t\t\t\tconst errorMessage = (error as Error).message;\n\n\t\t\t\t\t// Set span error\n\t\t\t\t\tspan.setAttribute(\"success\", false);\n\t\t\t\t\tspan.recordException(error as Error);\n\t\t\t\t\tspan.setStatus({ code: SpanStatusCode.ERROR, message: errorMessage });\n\n\t\t\t\t\t// Record error metrics\n\t\t\t\t\twebhookErrors.add(1, {\n\t\t\t\t\t\tenv: process.env.NODE_ENV,\n\t\t\t\t\t\tsource: event.source,\n\t\t\t\t\t\tevent_type: event.eventType,\n\t\t\t\t\t\tworkflow_name: this.configuration?.name || \"unknown\",\n\t\t\t\t\t});\n\n\t\t\t\t\tthis.logger.error(`Webhook failed ${event.id}: ${errorMessage}`, (error as Error).stack);\n\n\t\t\t\t\tthrow error;\n\t\t\t\t} finally {\n\t\t\t\t\tspan.end();\n\t\t\t\t}\n\t\t\t});\n\t\t});\n\t}\n\n\t/**\n\t * Register a custom source handler\n\t */\n\tstatic registerSourceHandler(source: string, handler: WebhookSourceHandler): void {\n\t\tsourceHandlers[source] = handler;\n\t}\n}\n\nexport default WebhookTrigger;\nexport { sourceHandlers };\n"]}
@@ -0,0 +1,70 @@
1
+ /**
2
+ * @blok/trigger-webhook
3
+ *
4
+ * Webhook trigger for Blok workflows.
5
+ * Handle webhook events from external services.
6
+ *
7
+ * Supported Services:
8
+ * - GitHub (push, pull_request, issues, releases, etc.)
9
+ * - Stripe (payment_intent, checkout.session, customer, etc.)
10
+ * - Shopify (orders, products, customers, etc.)
11
+ * - Custom webhooks (any service with signature verification)
12
+ *
13
+ * Features:
14
+ * - Signature verification (HMAC-SHA256)
15
+ * - Event type filtering
16
+ * - Source-specific handlers
17
+ * - Custom source registration
18
+ *
19
+ * @example
20
+ * ```typescript
21
+ * import { WebhookTrigger } from "@blok/trigger-webhook";
22
+ *
23
+ * class MyWebhookTrigger extends WebhookTrigger {
24
+ * protected nodes = myNodes;
25
+ * protected workflows = myWorkflows;
26
+ * }
27
+ *
28
+ * const trigger = new MyWebhookTrigger();
29
+ * await trigger.listen();
30
+ *
31
+ * // In your HTTP endpoint handler:
32
+ * app.post("/webhooks/:source", async (req, res) => {
33
+ * const rawBody = JSON.stringify(req.body);
34
+ * const result = await trigger.handleWebhook(
35
+ * req.params.source,
36
+ * rawBody,
37
+ * req.headers as Record<string, string>
38
+ * );
39
+ * res.status(200).json({ received: true });
40
+ * });
41
+ * ```
42
+ *
43
+ * Workflow Definition:
44
+ * ```typescript
45
+ * Workflow({ name: "github-push", version: "1.0.0" })
46
+ * .addTrigger("webhook", {
47
+ * source: "github",
48
+ * events: ["push", "pull_request.*"],
49
+ * secret: process.env.GITHUB_WEBHOOK_SECRET,
50
+ * })
51
+ * .addStep({ ... });
52
+ * ```
53
+ *
54
+ * Custom Source Handler:
55
+ * ```typescript
56
+ * import { WebhookTrigger } from "@blok/trigger-webhook";
57
+ *
58
+ * WebhookTrigger.registerSourceHandler("my-service", {
59
+ * getEventType: (headers, body) => body.event_type,
60
+ * getSignature: (headers) => headers["x-my-signature"],
61
+ * verifySignature: (rawBody, signature, secret) => {
62
+ * // Your verification logic
63
+ * return { valid: true };
64
+ * },
65
+ * getEventId: (headers, body) => body.id,
66
+ * });
67
+ * ```
68
+ */
69
+ export { WebhookTrigger, sourceHandlers, type WebhookEvent, type VerificationResult, type WebhookSourceHandler, } from "./WebhookTrigger";
70
+ export type { WebhookTriggerOpts } from "@blok/helper";