@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.
- package/README.md +252 -0
- package/dist/__tests__/client.test.d.ts +1 -0
- package/dist/__tests__/client.test.js +79 -0
- package/dist/__tests__/webhooks.test.d.ts +1 -0
- package/dist/__tests__/webhooks.test.js +162 -0
- package/dist/client.d.ts +48 -0
- package/dist/client.js +229 -0
- package/dist/errors.d.ts +21 -0
- package/dist/errors.js +46 -0
- package/dist/index.d.ts +4 -0
- package/dist/index.js +29 -0
- package/dist/types.d.ts +102 -0
- package/dist/types.js +3 -0
- package/dist/webhooks.d.ts +38 -0
- package/dist/webhooks.js +138 -0
- package/package.json +42 -0
package/dist/webhooks.js
ADDED
|
@@ -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
|
+
}
|