@webhook-platform/node 1.0.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.
@@ -0,0 +1,138 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.verifySignature = verifySignature;
37
+ exports.constructEvent = constructEvent;
38
+ exports.generateSignature = generateSignature;
39
+ const crypto = __importStar(require("crypto"));
40
+ const errors_1 = require("./errors");
41
+ const SIGNATURE_HEADER = 'x-signature';
42
+ const TIMESTAMP_HEADER = 'x-timestamp';
43
+ const EVENT_ID_HEADER = 'x-event-id';
44
+ const DELIVERY_ID_HEADER = 'x-delivery-id';
45
+ const DEFAULT_TOLERANCE = 300000; // 5 minutes in milliseconds
46
+ /**
47
+ * Verifies the webhook signature using HMAC-SHA256
48
+ * @param payload - Raw request body as string
49
+ * @param signature - X-Signature header value (format: t=timestamp,v1=signature)
50
+ * @param secret - Endpoint webhook secret
51
+ * @param options - Verification options
52
+ * @returns true if signature is valid
53
+ * @throws WebhookPlatformError if signature is invalid
54
+ */
55
+ function verifySignature(payload, signature, secret, options = {}) {
56
+ const tolerance = options.tolerance ?? DEFAULT_TOLERANCE;
57
+ if (!signature) {
58
+ throw new errors_1.WebhookPlatformError('Missing signature header', 400, 'invalid_signature');
59
+ }
60
+ const parts = signature.split(',');
61
+ let timestamp;
62
+ let sig;
63
+ for (const part of parts) {
64
+ const [key, value] = part.split('=');
65
+ if (key === 't')
66
+ timestamp = value;
67
+ if (key === 'v1')
68
+ sig = value;
69
+ }
70
+ if (!timestamp || !sig) {
71
+ throw new errors_1.WebhookPlatformError('Invalid signature format. Expected: t=timestamp,v1=signature', 400, 'invalid_signature');
72
+ }
73
+ const timestampMs = parseInt(timestamp, 10);
74
+ const now = Date.now();
75
+ if (Math.abs(now - timestampMs) > tolerance) {
76
+ throw new errors_1.WebhookPlatformError('Webhook timestamp is outside tolerance window', 400, 'timestamp_expired');
77
+ }
78
+ const signedPayload = `${timestamp}.${payload}`;
79
+ const expectedSignature = crypto
80
+ .createHmac('sha256', secret)
81
+ .update(signedPayload)
82
+ .digest('hex');
83
+ const sigBuffer = Buffer.from(sig);
84
+ const expectedBuffer = Buffer.from(expectedSignature);
85
+ if (sigBuffer.length !== expectedBuffer.length || !crypto.timingSafeEqual(sigBuffer, expectedBuffer)) {
86
+ throw new errors_1.WebhookPlatformError('Invalid signature', 400, 'invalid_signature');
87
+ }
88
+ return true;
89
+ }
90
+ /**
91
+ * Constructs a webhook event from the request
92
+ * @param payload - Raw request body as string
93
+ * @param headers - Request headers
94
+ * @param secret - Endpoint webhook secret
95
+ * @param options - Verification options
96
+ * @returns Parsed and verified webhook event
97
+ */
98
+ function constructEvent(payload, headers, secret, options = {}) {
99
+ const signature = headers[SIGNATURE_HEADER] || headers['X-Signature'];
100
+ const timestamp = headers[TIMESTAMP_HEADER] || headers['X-Timestamp'];
101
+ const eventId = headers[EVENT_ID_HEADER] || headers['X-Event-Id'];
102
+ const deliveryId = headers[DELIVERY_ID_HEADER] || headers['X-Delivery-Id'];
103
+ if (!signature) {
104
+ throw new errors_1.WebhookPlatformError('Missing X-Signature header', 400, 'missing_header');
105
+ }
106
+ verifySignature(payload, signature, secret, options);
107
+ let data;
108
+ try {
109
+ data = JSON.parse(payload);
110
+ }
111
+ catch {
112
+ throw new errors_1.WebhookPlatformError('Invalid JSON payload', 400, 'invalid_payload');
113
+ }
114
+ return {
115
+ eventId: eventId || '',
116
+ deliveryId: deliveryId || '',
117
+ timestamp: timestamp ? parseInt(timestamp, 10) : Date.now(),
118
+ type: data.type || '',
119
+ data: data.data || data,
120
+ };
121
+ }
122
+ /**
123
+ * Generates a signature for testing purposes
124
+ * @param payload - Request body as string
125
+ * @param secret - Webhook secret
126
+ * @param timestamp - Optional timestamp (defaults to now)
127
+ * @returns Signature string in format t=timestamp,v1=signature
128
+ */
129
+ function generateSignature(payload, secret, timestamp) {
130
+ const ts = timestamp ?? Date.now();
131
+ const signedPayload = `${ts}.${payload}`;
132
+ const signature = crypto
133
+ .createHmac('sha256', secret)
134
+ .update(signedPayload)
135
+ .digest('hex');
136
+ return `t=${ts},v1=${signature}`;
137
+ }
138
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"webhooks.js","sourceRoot":"","sources":["../src/webhooks.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'crypto';\nimport { WebhookEvent } from './types';\nimport { WebhookPlatformError } from './errors';\n\nconst SIGNATURE_HEADER = 'x-signature';\nconst TIMESTAMP_HEADER = 'x-timestamp';\nconst EVENT_ID_HEADER = 'x-event-id';\nconst DELIVERY_ID_HEADER = 'x-delivery-id';\n\nconst DEFAULT_TOLERANCE = 300000; // 5 minutes in milliseconds\n\nexport interface WebhookHeaders {\n  'x-signature'?: string;\n  'x-timestamp'?: string;\n  'x-event-id'?: string;\n  'x-delivery-id'?: string;\n  [key: string]: string | undefined;\n}\n\nexport interface VerifyOptions {\n  tolerance?: number;\n}\n\n/**\n * Verifies the webhook signature using HMAC-SHA256\n * @param payload - Raw request body as string\n * @param signature - X-Signature header value (format: t=timestamp,v1=signature)\n * @param secret - Endpoint webhook secret\n * @param options - Verification options\n * @returns true if signature is valid\n * @throws WebhookPlatformError if signature is invalid\n */\nexport function verifySignature(\n  payload: string,\n  signature: string,\n  secret: string,\n  options: VerifyOptions = {}\n): boolean {\n  const tolerance = options.tolerance ?? DEFAULT_TOLERANCE;\n\n  if (!signature) {\n    throw new WebhookPlatformError('Missing signature header', 400, 'invalid_signature');\n  }\n\n  const parts = signature.split(',');\n  let timestamp: string | undefined;\n  let sig: string | undefined;\n\n  for (const part of parts) {\n    const [key, value] = part.split('=');\n    if (key === 't') timestamp = value;\n    if (key === 'v1') sig = value;\n  }\n\n  if (!timestamp || !sig) {\n    throw new WebhookPlatformError(\n      'Invalid signature format. Expected: t=timestamp,v1=signature',\n      400,\n      'invalid_signature'\n    );\n  }\n\n  const timestampMs = parseInt(timestamp, 10);\n  const now = Date.now();\n\n  if (Math.abs(now - timestampMs) > tolerance) {\n    throw new WebhookPlatformError(\n      'Webhook timestamp is outside tolerance window',\n      400,\n      'timestamp_expired'\n    );\n  }\n\n  const signedPayload = `${timestamp}.${payload}`;\n  const expectedSignature = crypto\n    .createHmac('sha256', secret)\n    .update(signedPayload)\n    .digest('hex');\n\n  const sigBuffer = Buffer.from(sig);\n  const expectedBuffer = Buffer.from(expectedSignature);\n\n  if (sigBuffer.length !== expectedBuffer.length || !crypto.timingSafeEqual(sigBuffer, expectedBuffer)) {\n    throw new WebhookPlatformError('Invalid signature', 400, 'invalid_signature');\n  }\n\n  return true;\n}\n\n/**\n * Constructs a webhook event from the request\n * @param payload - Raw request body as string\n * @param headers - Request headers\n * @param secret - Endpoint webhook secret\n * @param options - Verification options\n * @returns Parsed and verified webhook event\n */\nexport function constructEvent(\n  payload: string,\n  headers: WebhookHeaders,\n  secret: string,\n  options: VerifyOptions = {}\n): WebhookEvent {\n  const signature = headers[SIGNATURE_HEADER] || headers['X-Signature'];\n  const timestamp = headers[TIMESTAMP_HEADER] || headers['X-Timestamp'];\n  const eventId = headers[EVENT_ID_HEADER] || headers['X-Event-Id'];\n  const deliveryId = headers[DELIVERY_ID_HEADER] || headers['X-Delivery-Id'];\n\n  if (!signature) {\n    throw new WebhookPlatformError('Missing X-Signature header', 400, 'missing_header');\n  }\n\n  verifySignature(payload, signature, secret, options);\n\n  let data: Record<string, unknown>;\n  try {\n    data = JSON.parse(payload);\n  } catch {\n    throw new WebhookPlatformError('Invalid JSON payload', 400, 'invalid_payload');\n  }\n\n  return {\n    eventId: eventId || '',\n    deliveryId: deliveryId || '',\n    timestamp: timestamp ? parseInt(timestamp, 10) : Date.now(),\n    type: (data.type as string) || '',\n    data: (data.data as Record<string, unknown>) || data,\n  };\n}\n\n/**\n * Generates a signature for testing purposes\n * @param payload - Request body as string\n * @param secret - Webhook secret\n * @param timestamp - Optional timestamp (defaults to now)\n * @returns Signature string in format t=timestamp,v1=signature\n */\nexport function generateSignature(\n  payload: string,\n  secret: string,\n  timestamp?: number\n): string {\n  const ts = timestamp ?? Date.now();\n  const signedPayload = `${ts}.${payload}`;\n  const signature = crypto\n    .createHmac('sha256', secret)\n    .update(signedPayload)\n    .digest('hex');\n\n  return `t=${ts},v1=${signature}`;\n}\n"]}
package/package.json ADDED
@@ -0,0 +1,42 @@
1
+ {
2
+ "name": "@webhook-platform/node",
3
+ "version": "1.0.0",
4
+ "description": "Official Node.js SDK for Webhook Platform",
5
+ "main": "dist/index.js",
6
+ "types": "dist/index.d.ts",
7
+ "files": [
8
+ "dist"
9
+ ],
10
+ "scripts": {
11
+ "build": "tsc",
12
+ "test": "jest",
13
+ "lint": "eslint src --ext .ts",
14
+ "prepublishOnly": "npm run build"
15
+ },
16
+ "keywords": [
17
+ "webhook",
18
+ "webhooks",
19
+ "api",
20
+ "events",
21
+ "delivery"
22
+ ],
23
+ "author": "Webhook Platform",
24
+ "license": "MIT",
25
+ "repository": {
26
+ "type": "git",
27
+ "url": "https://github.com/vadymkykalo/webhook-platform.git",
28
+ "directory": "sdks/node"
29
+ },
30
+ "engines": {
31
+ "node": ">=16.0.0"
32
+ },
33
+ "dependencies": {},
34
+ "devDependencies": {
35
+ "@types/jest": "^29.5.0",
36
+ "@types/node": "^20.10.0",
37
+ "jest": "^29.7.0",
38
+ "ts-jest": "^29.1.0",
39
+ "typescript": "^5.3.0"
40
+ },
41
+ "peerDependencies": {}
42
+ }