@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/README.md
ADDED
|
@@ -0,0 +1,252 @@
|
|
|
1
|
+
# @webhook-platform/node
|
|
2
|
+
|
|
3
|
+
Official Node.js SDK for [Webhook Platform](https://github.com/vadymkykalo/webhook-platform).
|
|
4
|
+
|
|
5
|
+
## Installation
|
|
6
|
+
|
|
7
|
+
```bash
|
|
8
|
+
npm install @webhook-platform/node
|
|
9
|
+
```
|
|
10
|
+
|
|
11
|
+
## Quick Start
|
|
12
|
+
|
|
13
|
+
```typescript
|
|
14
|
+
import { WebhookPlatform } from '@webhook-platform/node';
|
|
15
|
+
|
|
16
|
+
const client = new WebhookPlatform({
|
|
17
|
+
apiKey: 'wh_live_your_api_key',
|
|
18
|
+
baseUrl: 'http://localhost:8080', // optional, defaults to localhost
|
|
19
|
+
});
|
|
20
|
+
|
|
21
|
+
// Send an event
|
|
22
|
+
const event = await client.events.send({
|
|
23
|
+
type: 'order.completed',
|
|
24
|
+
data: {
|
|
25
|
+
orderId: 'ord_12345',
|
|
26
|
+
amount: 99.99,
|
|
27
|
+
currency: 'USD',
|
|
28
|
+
},
|
|
29
|
+
});
|
|
30
|
+
|
|
31
|
+
console.log(`Event created: ${event.eventId}`);
|
|
32
|
+
console.log(`Deliveries created: ${event.deliveriesCreated}`);
|
|
33
|
+
```
|
|
34
|
+
|
|
35
|
+
## API Reference
|
|
36
|
+
|
|
37
|
+
### Events
|
|
38
|
+
|
|
39
|
+
```typescript
|
|
40
|
+
// Send event with idempotency key
|
|
41
|
+
const event = await client.events.send(
|
|
42
|
+
{ type: 'order.completed', data: { orderId: '123' } },
|
|
43
|
+
'unique-idempotency-key'
|
|
44
|
+
);
|
|
45
|
+
```
|
|
46
|
+
|
|
47
|
+
### Endpoints
|
|
48
|
+
|
|
49
|
+
```typescript
|
|
50
|
+
// Create endpoint
|
|
51
|
+
const endpoint = await client.endpoints.create(projectId, {
|
|
52
|
+
url: 'https://api.example.com/webhooks',
|
|
53
|
+
description: 'Production webhooks',
|
|
54
|
+
enabled: true,
|
|
55
|
+
});
|
|
56
|
+
|
|
57
|
+
// List endpoints
|
|
58
|
+
const endpoints = await client.endpoints.list(projectId);
|
|
59
|
+
|
|
60
|
+
// Update endpoint
|
|
61
|
+
await client.endpoints.update(projectId, endpointId, {
|
|
62
|
+
enabled: false,
|
|
63
|
+
});
|
|
64
|
+
|
|
65
|
+
// Delete endpoint
|
|
66
|
+
await client.endpoints.delete(projectId, endpointId);
|
|
67
|
+
|
|
68
|
+
// Rotate secret
|
|
69
|
+
const updated = await client.endpoints.rotateSecret(projectId, endpointId);
|
|
70
|
+
console.log(`New secret: ${updated.secret}`);
|
|
71
|
+
|
|
72
|
+
// Test endpoint connectivity
|
|
73
|
+
const result = await client.endpoints.test(projectId, endpointId);
|
|
74
|
+
console.log(`Test ${result.success ? 'passed' : 'failed'}: ${result.latencyMs}ms`);
|
|
75
|
+
```
|
|
76
|
+
|
|
77
|
+
### Subscriptions
|
|
78
|
+
|
|
79
|
+
```typescript
|
|
80
|
+
// Subscribe endpoint to event types
|
|
81
|
+
const subscription = await client.subscriptions.create(projectId, {
|
|
82
|
+
endpointId: endpoint.id,
|
|
83
|
+
eventTypes: ['order.completed', 'order.cancelled'],
|
|
84
|
+
enabled: true,
|
|
85
|
+
});
|
|
86
|
+
|
|
87
|
+
// List subscriptions
|
|
88
|
+
const subscriptions = await client.subscriptions.list(projectId);
|
|
89
|
+
|
|
90
|
+
// Update subscription
|
|
91
|
+
await client.subscriptions.update(projectId, subscriptionId, {
|
|
92
|
+
eventTypes: ['order.*'],
|
|
93
|
+
});
|
|
94
|
+
|
|
95
|
+
// Delete subscription
|
|
96
|
+
await client.subscriptions.delete(projectId, subscriptionId);
|
|
97
|
+
```
|
|
98
|
+
|
|
99
|
+
### Deliveries
|
|
100
|
+
|
|
101
|
+
```typescript
|
|
102
|
+
// List deliveries with filters
|
|
103
|
+
const deliveries = await client.deliveries.list(projectId, {
|
|
104
|
+
status: 'FAILED',
|
|
105
|
+
page: 0,
|
|
106
|
+
size: 20,
|
|
107
|
+
});
|
|
108
|
+
|
|
109
|
+
console.log(`Total failed: ${deliveries.totalElements}`);
|
|
110
|
+
|
|
111
|
+
// Get delivery attempts
|
|
112
|
+
const attempts = await client.deliveries.getAttempts(deliveryId);
|
|
113
|
+
for (const attempt of attempts) {
|
|
114
|
+
console.log(`Attempt ${attempt.attemptNumber}: ${attempt.httpStatus} (${attempt.latencyMs}ms)`);
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
// Replay failed delivery
|
|
118
|
+
await client.deliveries.replay(deliveryId);
|
|
119
|
+
```
|
|
120
|
+
|
|
121
|
+
## Webhook Signature Verification
|
|
122
|
+
|
|
123
|
+
Verify incoming webhooks in your endpoint:
|
|
124
|
+
|
|
125
|
+
```typescript
|
|
126
|
+
import { verifySignature, constructEvent } from '@webhook-platform/node';
|
|
127
|
+
|
|
128
|
+
app.post('/webhooks', (req, res) => {
|
|
129
|
+
const payload = req.body; // raw body string
|
|
130
|
+
const signature = req.headers['x-signature'];
|
|
131
|
+
const secret = process.env.WEBHOOK_SECRET;
|
|
132
|
+
|
|
133
|
+
try {
|
|
134
|
+
// Option 1: Just verify
|
|
135
|
+
verifySignature(payload, signature, secret);
|
|
136
|
+
|
|
137
|
+
// Option 2: Verify and parse
|
|
138
|
+
const event = constructEvent(payload, req.headers, secret);
|
|
139
|
+
|
|
140
|
+
console.log(`Received ${event.type}:`, event.data);
|
|
141
|
+
|
|
142
|
+
// Handle the event
|
|
143
|
+
switch (event.type) {
|
|
144
|
+
case 'order.completed':
|
|
145
|
+
handleOrderCompleted(event.data);
|
|
146
|
+
break;
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
res.status(200).send('OK');
|
|
150
|
+
} catch (err) {
|
|
151
|
+
console.error('Webhook verification failed:', err.message);
|
|
152
|
+
res.status(400).send('Invalid signature');
|
|
153
|
+
}
|
|
154
|
+
});
|
|
155
|
+
```
|
|
156
|
+
|
|
157
|
+
### Express.js Example
|
|
158
|
+
|
|
159
|
+
```typescript
|
|
160
|
+
import express from 'express';
|
|
161
|
+
import { constructEvent } from '@webhook-platform/node';
|
|
162
|
+
|
|
163
|
+
const app = express();
|
|
164
|
+
|
|
165
|
+
// Important: Use raw body for signature verification
|
|
166
|
+
app.post('/webhooks', express.raw({ type: 'application/json' }), (req, res) => {
|
|
167
|
+
const event = constructEvent(
|
|
168
|
+
req.body.toString(),
|
|
169
|
+
req.headers,
|
|
170
|
+
process.env.WEBHOOK_SECRET
|
|
171
|
+
);
|
|
172
|
+
|
|
173
|
+
// Process event...
|
|
174
|
+
res.sendStatus(200);
|
|
175
|
+
});
|
|
176
|
+
```
|
|
177
|
+
|
|
178
|
+
## Error Handling
|
|
179
|
+
|
|
180
|
+
```typescript
|
|
181
|
+
import {
|
|
182
|
+
WebhookPlatformError,
|
|
183
|
+
RateLimitError,
|
|
184
|
+
AuthenticationError,
|
|
185
|
+
ValidationError
|
|
186
|
+
} from '@webhook-platform/node';
|
|
187
|
+
|
|
188
|
+
try {
|
|
189
|
+
await client.events.send({ type: 'test', data: {} });
|
|
190
|
+
} catch (err) {
|
|
191
|
+
if (err instanceof RateLimitError) {
|
|
192
|
+
// Wait and retry
|
|
193
|
+
console.log(`Rate limited. Retry after ${err.retryAfter}ms`);
|
|
194
|
+
await sleep(err.retryAfter);
|
|
195
|
+
} else if (err instanceof AuthenticationError) {
|
|
196
|
+
console.error('Invalid API key');
|
|
197
|
+
} else if (err instanceof ValidationError) {
|
|
198
|
+
console.error('Validation failed:', err.fieldErrors);
|
|
199
|
+
} else if (err instanceof WebhookPlatformError) {
|
|
200
|
+
console.error(`Error ${err.status}: ${err.message}`);
|
|
201
|
+
}
|
|
202
|
+
}
|
|
203
|
+
```
|
|
204
|
+
|
|
205
|
+
## Configuration
|
|
206
|
+
|
|
207
|
+
```typescript
|
|
208
|
+
const client = new WebhookPlatform({
|
|
209
|
+
apiKey: 'wh_live_xxx', // Required: Your API key
|
|
210
|
+
baseUrl: 'https://api.example.com', // Optional: API base URL
|
|
211
|
+
timeout: 30000, // Optional: Request timeout in ms (default: 30000)
|
|
212
|
+
});
|
|
213
|
+
```
|
|
214
|
+
|
|
215
|
+
## TypeScript Support
|
|
216
|
+
|
|
217
|
+
This SDK is written in TypeScript and includes full type definitions:
|
|
218
|
+
|
|
219
|
+
```typescript
|
|
220
|
+
import type {
|
|
221
|
+
Event,
|
|
222
|
+
EventResponse,
|
|
223
|
+
Endpoint,
|
|
224
|
+
Delivery,
|
|
225
|
+
DeliveryStatus
|
|
226
|
+
} from '@webhook-platform/node';
|
|
227
|
+
```
|
|
228
|
+
|
|
229
|
+
## Development
|
|
230
|
+
|
|
231
|
+
### Running Tests
|
|
232
|
+
|
|
233
|
+
**Local (requires Node.js 16+):**
|
|
234
|
+
```bash
|
|
235
|
+
npm install
|
|
236
|
+
npm test
|
|
237
|
+
```
|
|
238
|
+
|
|
239
|
+
**Docker:**
|
|
240
|
+
```bash
|
|
241
|
+
docker run --rm -v $(pwd):/app -w /app node:20-alpine sh -c "npm install && npm test"
|
|
242
|
+
```
|
|
243
|
+
|
|
244
|
+
### Building
|
|
245
|
+
|
|
246
|
+
```bash
|
|
247
|
+
npm run build
|
|
248
|
+
```
|
|
249
|
+
|
|
250
|
+
## License
|
|
251
|
+
|
|
252
|
+
MIT
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const client_1 = require("../client");
|
|
4
|
+
const errors_1 = require("../errors");
|
|
5
|
+
describe('WebhookPlatform Client', () => {
|
|
6
|
+
describe('constructor', () => {
|
|
7
|
+
it('should create client with API key', () => {
|
|
8
|
+
const client = new client_1.WebhookPlatform({ apiKey: 'test_api_key' });
|
|
9
|
+
expect(client).toBeInstanceOf(client_1.WebhookPlatform);
|
|
10
|
+
});
|
|
11
|
+
it('should throw error without API key', () => {
|
|
12
|
+
expect(() => new client_1.WebhookPlatform({ apiKey: '' })).toThrow('API key is required');
|
|
13
|
+
});
|
|
14
|
+
it('should use default base URL', () => {
|
|
15
|
+
const client = new client_1.WebhookPlatform({ apiKey: 'test_api_key' });
|
|
16
|
+
expect(client).toBeDefined();
|
|
17
|
+
});
|
|
18
|
+
it('should accept custom base URL', () => {
|
|
19
|
+
const client = new client_1.WebhookPlatform({
|
|
20
|
+
apiKey: 'test_api_key',
|
|
21
|
+
baseUrl: 'https://api.example.com',
|
|
22
|
+
});
|
|
23
|
+
expect(client).toBeDefined();
|
|
24
|
+
});
|
|
25
|
+
it('should accept custom timeout', () => {
|
|
26
|
+
const client = new client_1.WebhookPlatform({
|
|
27
|
+
apiKey: 'test_api_key',
|
|
28
|
+
timeout: 60000,
|
|
29
|
+
});
|
|
30
|
+
expect(client).toBeDefined();
|
|
31
|
+
});
|
|
32
|
+
it('should initialize all API modules', () => {
|
|
33
|
+
const client = new client_1.WebhookPlatform({ apiKey: 'test_api_key' });
|
|
34
|
+
expect(client.events).toBeDefined();
|
|
35
|
+
expect(client.endpoints).toBeDefined();
|
|
36
|
+
expect(client.subscriptions).toBeDefined();
|
|
37
|
+
expect(client.deliveries).toBeDefined();
|
|
38
|
+
});
|
|
39
|
+
});
|
|
40
|
+
});
|
|
41
|
+
describe('Error Classes', () => {
|
|
42
|
+
describe('WebhookPlatformError', () => {
|
|
43
|
+
it('should have correct properties', () => {
|
|
44
|
+
const error = new errors_1.WebhookPlatformError('Test error', 500, 'test_code');
|
|
45
|
+
expect(error.message).toBe('Test error');
|
|
46
|
+
expect(error.status).toBe(500);
|
|
47
|
+
expect(error.code).toBe('test_code');
|
|
48
|
+
expect(error.name).toBe('WebhookPlatformError');
|
|
49
|
+
});
|
|
50
|
+
});
|
|
51
|
+
describe('AuthenticationError', () => {
|
|
52
|
+
it('should have correct defaults', () => {
|
|
53
|
+
const error = new errors_1.AuthenticationError();
|
|
54
|
+
expect(error.message).toBe('Invalid API key');
|
|
55
|
+
expect(error.status).toBe(401);
|
|
56
|
+
expect(error.code).toBe('authentication_error');
|
|
57
|
+
expect(error.name).toBe('AuthenticationError');
|
|
58
|
+
});
|
|
59
|
+
it('should accept custom message', () => {
|
|
60
|
+
const error = new errors_1.AuthenticationError('Custom auth error');
|
|
61
|
+
expect(error.message).toBe('Custom auth error');
|
|
62
|
+
});
|
|
63
|
+
});
|
|
64
|
+
describe('ValidationError', () => {
|
|
65
|
+
it('should have field errors', () => {
|
|
66
|
+
const fieldErrors = { email: 'Invalid email', url: 'Invalid URL' };
|
|
67
|
+
const error = new errors_1.ValidationError('Validation failed', fieldErrors);
|
|
68
|
+
expect(error.message).toBe('Validation failed');
|
|
69
|
+
expect(error.status).toBe(400);
|
|
70
|
+
expect(error.code).toBe('validation_error');
|
|
71
|
+
expect(error.fieldErrors).toEqual(fieldErrors);
|
|
72
|
+
});
|
|
73
|
+
it('should default to empty field errors', () => {
|
|
74
|
+
const error = new errors_1.ValidationError('Validation failed');
|
|
75
|
+
expect(error.fieldErrors).toEqual({});
|
|
76
|
+
});
|
|
77
|
+
});
|
|
78
|
+
});
|
|
79
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"client.test.js","sourceRoot":"","sources":["../../src/__tests__/client.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { WebhookPlatform } from '../client';\nimport { WebhookPlatformError, AuthenticationError, ValidationError } from '../errors';\n\ndescribe('WebhookPlatform Client', () => {\n  describe('constructor', () => {\n    it('should create client with API key', () => {\n      const client = new WebhookPlatform({ apiKey: 'test_api_key' });\n      expect(client).toBeInstanceOf(WebhookPlatform);\n    });\n\n    it('should throw error without API key', () => {\n      expect(() => new WebhookPlatform({ apiKey: '' })).toThrow('API key is required');\n    });\n\n    it('should use default base URL', () => {\n      const client = new WebhookPlatform({ apiKey: 'test_api_key' });\n      expect(client).toBeDefined();\n    });\n\n    it('should accept custom base URL', () => {\n      const client = new WebhookPlatform({\n        apiKey: 'test_api_key',\n        baseUrl: 'https://api.example.com',\n      });\n      expect(client).toBeDefined();\n    });\n\n    it('should accept custom timeout', () => {\n      const client = new WebhookPlatform({\n        apiKey: 'test_api_key',\n        timeout: 60000,\n      });\n      expect(client).toBeDefined();\n    });\n\n    it('should initialize all API modules', () => {\n      const client = new WebhookPlatform({ apiKey: 'test_api_key' });\n      expect(client.events).toBeDefined();\n      expect(client.endpoints).toBeDefined();\n      expect(client.subscriptions).toBeDefined();\n      expect(client.deliveries).toBeDefined();\n    });\n  });\n});\n\ndescribe('Error Classes', () => {\n  describe('WebhookPlatformError', () => {\n    it('should have correct properties', () => {\n      const error = new WebhookPlatformError('Test error', 500, 'test_code');\n      expect(error.message).toBe('Test error');\n      expect(error.status).toBe(500);\n      expect(error.code).toBe('test_code');\n      expect(error.name).toBe('WebhookPlatformError');\n    });\n  });\n\n  describe('AuthenticationError', () => {\n    it('should have correct defaults', () => {\n      const error = new AuthenticationError();\n      expect(error.message).toBe('Invalid API key');\n      expect(error.status).toBe(401);\n      expect(error.code).toBe('authentication_error');\n      expect(error.name).toBe('AuthenticationError');\n    });\n\n    it('should accept custom message', () => {\n      const error = new AuthenticationError('Custom auth error');\n      expect(error.message).toBe('Custom auth error');\n    });\n  });\n\n  describe('ValidationError', () => {\n    it('should have field errors', () => {\n      const fieldErrors = { email: 'Invalid email', url: 'Invalid URL' };\n      const error = new ValidationError('Validation failed', fieldErrors);\n      expect(error.message).toBe('Validation failed');\n      expect(error.status).toBe(400);\n      expect(error.code).toBe('validation_error');\n      expect(error.fieldErrors).toEqual(fieldErrors);\n    });\n\n    it('should default to empty field errors', () => {\n      const error = new ValidationError('Validation failed');\n      expect(error.fieldErrors).toEqual({});\n    });\n  });\n});\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,162 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const webhooks_1 = require("../webhooks");
|
|
4
|
+
const errors_1 = require("../errors");
|
|
5
|
+
describe('Webhook Signature Verification', () => {
|
|
6
|
+
const secret = 'whsec_test_secret_key_123';
|
|
7
|
+
const payload = JSON.stringify({ type: 'order.completed', data: { orderId: '12345' } });
|
|
8
|
+
describe('generateSignature', () => {
|
|
9
|
+
it('should generate valid signature format', () => {
|
|
10
|
+
const signature = (0, webhooks_1.generateSignature)(payload, secret);
|
|
11
|
+
expect(signature).toMatch(/^t=\d+,v1=[a-f0-9]{64}$/);
|
|
12
|
+
});
|
|
13
|
+
it('should use provided timestamp', () => {
|
|
14
|
+
const timestamp = 1700000000000;
|
|
15
|
+
const signature = (0, webhooks_1.generateSignature)(payload, secret, timestamp);
|
|
16
|
+
expect(signature).toContain(`t=${timestamp}`);
|
|
17
|
+
});
|
|
18
|
+
it('should generate consistent signatures for same inputs', () => {
|
|
19
|
+
const timestamp = 1700000000000;
|
|
20
|
+
const sig1 = (0, webhooks_1.generateSignature)(payload, secret, timestamp);
|
|
21
|
+
const sig2 = (0, webhooks_1.generateSignature)(payload, secret, timestamp);
|
|
22
|
+
expect(sig1).toBe(sig2);
|
|
23
|
+
});
|
|
24
|
+
it('should generate different signatures for different payloads', () => {
|
|
25
|
+
const timestamp = 1700000000000;
|
|
26
|
+
const sig1 = (0, webhooks_1.generateSignature)(payload, secret, timestamp);
|
|
27
|
+
const sig2 = (0, webhooks_1.generateSignature)('different payload', secret, timestamp);
|
|
28
|
+
expect(sig1).not.toBe(sig2);
|
|
29
|
+
});
|
|
30
|
+
it('should generate different signatures for different secrets', () => {
|
|
31
|
+
const timestamp = 1700000000000;
|
|
32
|
+
const sig1 = (0, webhooks_1.generateSignature)(payload, secret, timestamp);
|
|
33
|
+
const sig2 = (0, webhooks_1.generateSignature)(payload, 'different_secret', timestamp);
|
|
34
|
+
expect(sig1).not.toBe(sig2);
|
|
35
|
+
});
|
|
36
|
+
});
|
|
37
|
+
describe('verifySignature', () => {
|
|
38
|
+
it('should verify valid signature', () => {
|
|
39
|
+
const timestamp = Date.now();
|
|
40
|
+
const signature = (0, webhooks_1.generateSignature)(payload, secret, timestamp);
|
|
41
|
+
expect((0, webhooks_1.verifySignature)(payload, signature, secret)).toBe(true);
|
|
42
|
+
});
|
|
43
|
+
it('should throw on missing signature', () => {
|
|
44
|
+
expect(() => (0, webhooks_1.verifySignature)(payload, '', secret)).toThrow(errors_1.WebhookPlatformError);
|
|
45
|
+
expect(() => (0, webhooks_1.verifySignature)(payload, '', secret)).toThrow('Missing signature header');
|
|
46
|
+
});
|
|
47
|
+
it('should throw on invalid signature format', () => {
|
|
48
|
+
expect(() => (0, webhooks_1.verifySignature)(payload, 'invalid', secret)).toThrow('Invalid signature format');
|
|
49
|
+
});
|
|
50
|
+
it('should throw on missing timestamp', () => {
|
|
51
|
+
expect(() => (0, webhooks_1.verifySignature)(payload, 'v1=abc123', secret)).toThrow('Invalid signature format');
|
|
52
|
+
});
|
|
53
|
+
it('should throw on missing v1 signature', () => {
|
|
54
|
+
expect(() => (0, webhooks_1.verifySignature)(payload, 't=1700000000000', secret)).toThrow('Invalid signature format');
|
|
55
|
+
});
|
|
56
|
+
it('should throw on expired timestamp', () => {
|
|
57
|
+
const oldTimestamp = Date.now() - 600000; // 10 minutes ago
|
|
58
|
+
const signature = (0, webhooks_1.generateSignature)(payload, secret, oldTimestamp);
|
|
59
|
+
expect(() => (0, webhooks_1.verifySignature)(payload, signature, secret)).toThrow('outside tolerance window');
|
|
60
|
+
});
|
|
61
|
+
it('should throw on future timestamp outside tolerance', () => {
|
|
62
|
+
const futureTimestamp = Date.now() + 600000; // 10 minutes in future
|
|
63
|
+
const signature = (0, webhooks_1.generateSignature)(payload, secret, futureTimestamp);
|
|
64
|
+
expect(() => (0, webhooks_1.verifySignature)(payload, signature, secret)).toThrow('outside tolerance window');
|
|
65
|
+
});
|
|
66
|
+
it('should accept timestamp within tolerance', () => {
|
|
67
|
+
const recentTimestamp = Date.now() - 60000; // 1 minute ago
|
|
68
|
+
const signature = (0, webhooks_1.generateSignature)(payload, secret, recentTimestamp);
|
|
69
|
+
expect((0, webhooks_1.verifySignature)(payload, signature, secret)).toBe(true);
|
|
70
|
+
});
|
|
71
|
+
it('should throw on invalid signature value', () => {
|
|
72
|
+
const timestamp = Date.now();
|
|
73
|
+
const signature = `t=${timestamp},v1=invalid_signature_that_will_not_match`;
|
|
74
|
+
expect(() => (0, webhooks_1.verifySignature)(payload, signature, secret)).toThrow('Invalid signature');
|
|
75
|
+
});
|
|
76
|
+
it('should throw on tampered payload', () => {
|
|
77
|
+
const timestamp = Date.now();
|
|
78
|
+
const signature = (0, webhooks_1.generateSignature)(payload, secret, timestamp);
|
|
79
|
+
const tamperedPayload = JSON.stringify({ type: 'order.cancelled', data: {} });
|
|
80
|
+
expect(() => (0, webhooks_1.verifySignature)(tamperedPayload, signature, secret)).toThrow('Invalid signature');
|
|
81
|
+
});
|
|
82
|
+
it('should respect custom tolerance', () => {
|
|
83
|
+
const oldTimestamp = Date.now() - 60000; // 1 minute ago
|
|
84
|
+
const signature = (0, webhooks_1.generateSignature)(payload, secret, oldTimestamp);
|
|
85
|
+
// Should fail with 30s tolerance
|
|
86
|
+
expect(() => (0, webhooks_1.verifySignature)(payload, signature, secret, { tolerance: 30000 }))
|
|
87
|
+
.toThrow('outside tolerance window');
|
|
88
|
+
// Should pass with 2min tolerance
|
|
89
|
+
expect((0, webhooks_1.verifySignature)(payload, signature, secret, { tolerance: 120000 })).toBe(true);
|
|
90
|
+
});
|
|
91
|
+
});
|
|
92
|
+
describe('constructEvent', () => {
|
|
93
|
+
it('should construct event from valid request', () => {
|
|
94
|
+
const timestamp = Date.now();
|
|
95
|
+
const signature = (0, webhooks_1.generateSignature)(payload, secret, timestamp);
|
|
96
|
+
const headers = {
|
|
97
|
+
'x-signature': signature,
|
|
98
|
+
'x-timestamp': timestamp.toString(),
|
|
99
|
+
'x-event-id': 'evt_123',
|
|
100
|
+
'x-delivery-id': 'dlv_456',
|
|
101
|
+
};
|
|
102
|
+
const event = (0, webhooks_1.constructEvent)(payload, headers, secret);
|
|
103
|
+
expect(event.eventId).toBe('evt_123');
|
|
104
|
+
expect(event.deliveryId).toBe('dlv_456');
|
|
105
|
+
expect(event.timestamp).toBe(timestamp);
|
|
106
|
+
expect(event.type).toBe('order.completed');
|
|
107
|
+
expect(event.data).toEqual({ orderId: '12345' });
|
|
108
|
+
});
|
|
109
|
+
it('should handle uppercase headers', () => {
|
|
110
|
+
const timestamp = Date.now();
|
|
111
|
+
const signature = (0, webhooks_1.generateSignature)(payload, secret, timestamp);
|
|
112
|
+
const headers = {
|
|
113
|
+
'X-Signature': signature,
|
|
114
|
+
'X-Timestamp': timestamp.toString(),
|
|
115
|
+
'X-Event-Id': 'evt_123',
|
|
116
|
+
'X-Delivery-Id': 'dlv_456',
|
|
117
|
+
};
|
|
118
|
+
const event = (0, webhooks_1.constructEvent)(payload, headers, secret);
|
|
119
|
+
expect(event.eventId).toBe('evt_123');
|
|
120
|
+
});
|
|
121
|
+
it('should throw on missing signature header', () => {
|
|
122
|
+
const headers = {
|
|
123
|
+
'x-timestamp': Date.now().toString(),
|
|
124
|
+
};
|
|
125
|
+
expect(() => (0, webhooks_1.constructEvent)(payload, headers, secret))
|
|
126
|
+
.toThrow('Missing X-Signature header');
|
|
127
|
+
});
|
|
128
|
+
it('should throw on invalid JSON payload', () => {
|
|
129
|
+
const timestamp = Date.now();
|
|
130
|
+
const invalidPayload = 'not valid json';
|
|
131
|
+
const signature = (0, webhooks_1.generateSignature)(invalidPayload, secret, timestamp);
|
|
132
|
+
const headers = {
|
|
133
|
+
'x-signature': signature,
|
|
134
|
+
'x-timestamp': timestamp.toString(),
|
|
135
|
+
};
|
|
136
|
+
expect(() => (0, webhooks_1.constructEvent)(invalidPayload, headers, secret))
|
|
137
|
+
.toThrow('Invalid JSON payload');
|
|
138
|
+
});
|
|
139
|
+
it('should handle payload without nested data field', () => {
|
|
140
|
+
const flatPayload = JSON.stringify({ type: 'test.event', value: 123 });
|
|
141
|
+
const timestamp = Date.now();
|
|
142
|
+
const signature = (0, webhooks_1.generateSignature)(flatPayload, secret, timestamp);
|
|
143
|
+
const headers = {
|
|
144
|
+
'x-signature': signature,
|
|
145
|
+
'x-timestamp': timestamp.toString(),
|
|
146
|
+
};
|
|
147
|
+
const event = (0, webhooks_1.constructEvent)(flatPayload, headers, secret);
|
|
148
|
+
expect(event.type).toBe('test.event');
|
|
149
|
+
expect(event.data).toEqual({ type: 'test.event', value: 123 });
|
|
150
|
+
});
|
|
151
|
+
it('should use current timestamp if not provided in headers', () => {
|
|
152
|
+
const timestamp = Date.now();
|
|
153
|
+
const signature = (0, webhooks_1.generateSignature)(payload, secret, timestamp);
|
|
154
|
+
const headers = {
|
|
155
|
+
'x-signature': signature,
|
|
156
|
+
};
|
|
157
|
+
const event = (0, webhooks_1.constructEvent)(payload, headers, secret);
|
|
158
|
+
expect(event.timestamp).toBeGreaterThan(0);
|
|
159
|
+
});
|
|
160
|
+
});
|
|
161
|
+
});
|
|
162
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"webhooks.test.js","sourceRoot":"","sources":["../../src/__tests__/webhooks.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { verifySignature, constructEvent, generateSignature } from '../webhooks';\nimport { WebhookPlatformError } from '../errors';\n\ndescribe('Webhook Signature Verification', () => {\n  const secret = 'whsec_test_secret_key_123';\n  const payload = JSON.stringify({ type: 'order.completed', data: { orderId: '12345' } });\n\n  describe('generateSignature', () => {\n    it('should generate valid signature format', () => {\n      const signature = generateSignature(payload, secret);\n      expect(signature).toMatch(/^t=\\d+,v1=[a-f0-9]{64}$/);\n    });\n\n    it('should use provided timestamp', () => {\n      const timestamp = 1700000000000;\n      const signature = generateSignature(payload, secret, timestamp);\n      expect(signature).toContain(`t=${timestamp}`);\n    });\n\n    it('should generate consistent signatures for same inputs', () => {\n      const timestamp = 1700000000000;\n      const sig1 = generateSignature(payload, secret, timestamp);\n      const sig2 = generateSignature(payload, secret, timestamp);\n      expect(sig1).toBe(sig2);\n    });\n\n    it('should generate different signatures for different payloads', () => {\n      const timestamp = 1700000000000;\n      const sig1 = generateSignature(payload, secret, timestamp);\n      const sig2 = generateSignature('different payload', secret, timestamp);\n      expect(sig1).not.toBe(sig2);\n    });\n\n    it('should generate different signatures for different secrets', () => {\n      const timestamp = 1700000000000;\n      const sig1 = generateSignature(payload, secret, timestamp);\n      const sig2 = generateSignature(payload, 'different_secret', timestamp);\n      expect(sig1).not.toBe(sig2);\n    });\n  });\n\n  describe('verifySignature', () => {\n    it('should verify valid signature', () => {\n      const timestamp = Date.now();\n      const signature = generateSignature(payload, secret, timestamp);\n      expect(verifySignature(payload, signature, secret)).toBe(true);\n    });\n\n    it('should throw on missing signature', () => {\n      expect(() => verifySignature(payload, '', secret)).toThrow(WebhookPlatformError);\n      expect(() => verifySignature(payload, '', secret)).toThrow('Missing signature header');\n    });\n\n    it('should throw on invalid signature format', () => {\n      expect(() => verifySignature(payload, 'invalid', secret)).toThrow('Invalid signature format');\n    });\n\n    it('should throw on missing timestamp', () => {\n      expect(() => verifySignature(payload, 'v1=abc123', secret)).toThrow('Invalid signature format');\n    });\n\n    it('should throw on missing v1 signature', () => {\n      expect(() => verifySignature(payload, 't=1700000000000', secret)).toThrow('Invalid signature format');\n    });\n\n    it('should throw on expired timestamp', () => {\n      const oldTimestamp = Date.now() - 600000; // 10 minutes ago\n      const signature = generateSignature(payload, secret, oldTimestamp);\n      expect(() => verifySignature(payload, signature, secret)).toThrow('outside tolerance window');\n    });\n\n    it('should throw on future timestamp outside tolerance', () => {\n      const futureTimestamp = Date.now() + 600000; // 10 minutes in future\n      const signature = generateSignature(payload, secret, futureTimestamp);\n      expect(() => verifySignature(payload, signature, secret)).toThrow('outside tolerance window');\n    });\n\n    it('should accept timestamp within tolerance', () => {\n      const recentTimestamp = Date.now() - 60000; // 1 minute ago\n      const signature = generateSignature(payload, secret, recentTimestamp);\n      expect(verifySignature(payload, signature, secret)).toBe(true);\n    });\n\n    it('should throw on invalid signature value', () => {\n      const timestamp = Date.now();\n      const signature = `t=${timestamp},v1=invalid_signature_that_will_not_match`;\n      expect(() => verifySignature(payload, signature, secret)).toThrow('Invalid signature');\n    });\n\n    it('should throw on tampered payload', () => {\n      const timestamp = Date.now();\n      const signature = generateSignature(payload, secret, timestamp);\n      const tamperedPayload = JSON.stringify({ type: 'order.cancelled', data: {} });\n      expect(() => verifySignature(tamperedPayload, signature, secret)).toThrow('Invalid signature');\n    });\n\n    it('should respect custom tolerance', () => {\n      const oldTimestamp = Date.now() - 60000; // 1 minute ago\n      const signature = generateSignature(payload, secret, oldTimestamp);\n      \n      // Should fail with 30s tolerance\n      expect(() => verifySignature(payload, signature, secret, { tolerance: 30000 }))\n        .toThrow('outside tolerance window');\n      \n      // Should pass with 2min tolerance\n      expect(verifySignature(payload, signature, secret, { tolerance: 120000 })).toBe(true);\n    });\n  });\n\n  describe('constructEvent', () => {\n    it('should construct event from valid request', () => {\n      const timestamp = Date.now();\n      const signature = generateSignature(payload, secret, timestamp);\n      \n      const headers = {\n        'x-signature': signature,\n        'x-timestamp': timestamp.toString(),\n        'x-event-id': 'evt_123',\n        'x-delivery-id': 'dlv_456',\n      };\n\n      const event = constructEvent(payload, headers, secret);\n      \n      expect(event.eventId).toBe('evt_123');\n      expect(event.deliveryId).toBe('dlv_456');\n      expect(event.timestamp).toBe(timestamp);\n      expect(event.type).toBe('order.completed');\n      expect(event.data).toEqual({ orderId: '12345' });\n    });\n\n    it('should handle uppercase headers', () => {\n      const timestamp = Date.now();\n      const signature = generateSignature(payload, secret, timestamp);\n      \n      const headers = {\n        'X-Signature': signature,\n        'X-Timestamp': timestamp.toString(),\n        'X-Event-Id': 'evt_123',\n        'X-Delivery-Id': 'dlv_456',\n      };\n\n      const event = constructEvent(payload, headers, secret);\n      expect(event.eventId).toBe('evt_123');\n    });\n\n    it('should throw on missing signature header', () => {\n      const headers = {\n        'x-timestamp': Date.now().toString(),\n      };\n\n      expect(() => constructEvent(payload, headers, secret))\n        .toThrow('Missing X-Signature header');\n    });\n\n    it('should throw on invalid JSON payload', () => {\n      const timestamp = Date.now();\n      const invalidPayload = 'not valid json';\n      const signature = generateSignature(invalidPayload, secret, timestamp);\n      \n      const headers = {\n        'x-signature': signature,\n        'x-timestamp': timestamp.toString(),\n      };\n\n      expect(() => constructEvent(invalidPayload, headers, secret))\n        .toThrow('Invalid JSON payload');\n    });\n\n    it('should handle payload without nested data field', () => {\n      const flatPayload = JSON.stringify({ type: 'test.event', value: 123 });\n      const timestamp = Date.now();\n      const signature = generateSignature(flatPayload, secret, timestamp);\n      \n      const headers = {\n        'x-signature': signature,\n        'x-timestamp': timestamp.toString(),\n      };\n\n      const event = constructEvent(flatPayload, headers, secret);\n      expect(event.type).toBe('test.event');\n      expect(event.data).toEqual({ type: 'test.event', value: 123 });\n    });\n\n    it('should use current timestamp if not provided in headers', () => {\n      const timestamp = Date.now();\n      const signature = generateSignature(payload, secret, timestamp);\n      \n      const headers = {\n        'x-signature': signature,\n      };\n\n      const event = constructEvent(payload, headers, secret);\n      expect(event.timestamp).toBeGreaterThan(0);\n    });\n  });\n});\n"]}
|
package/dist/client.d.ts
ADDED
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
import { WebhookPlatformConfig, Event, EventResponse, Endpoint, EndpointCreateParams, EndpointUpdateParams, Subscription, SubscriptionCreateParams, Delivery, DeliveryAttempt, DeliveryListParams, PaginatedResponse, EndpointTestResult } from './types';
|
|
2
|
+
export declare class WebhookPlatform {
|
|
3
|
+
private readonly apiKey;
|
|
4
|
+
private readonly baseUrl;
|
|
5
|
+
private readonly timeout;
|
|
6
|
+
readonly events: Events;
|
|
7
|
+
readonly endpoints: Endpoints;
|
|
8
|
+
readonly subscriptions: Subscriptions;
|
|
9
|
+
readonly deliveries: Deliveries;
|
|
10
|
+
constructor(config: WebhookPlatformConfig);
|
|
11
|
+
request<T>(method: string, path: string, body?: unknown, idempotencyKey?: string): Promise<T>;
|
|
12
|
+
private extractRateLimitInfo;
|
|
13
|
+
private handleError;
|
|
14
|
+
}
|
|
15
|
+
declare class Events {
|
|
16
|
+
private client;
|
|
17
|
+
constructor(client: WebhookPlatform);
|
|
18
|
+
send(event: Event, idempotencyKey?: string): Promise<EventResponse>;
|
|
19
|
+
}
|
|
20
|
+
declare class Endpoints {
|
|
21
|
+
private client;
|
|
22
|
+
constructor(client: WebhookPlatform);
|
|
23
|
+
create(projectId: string, params: EndpointCreateParams): Promise<Endpoint>;
|
|
24
|
+
get(projectId: string, endpointId: string): Promise<Endpoint>;
|
|
25
|
+
list(projectId: string): Promise<Endpoint[]>;
|
|
26
|
+
update(projectId: string, endpointId: string, params: EndpointUpdateParams): Promise<Endpoint>;
|
|
27
|
+
delete(projectId: string, endpointId: string): Promise<void>;
|
|
28
|
+
rotateSecret(projectId: string, endpointId: string): Promise<Endpoint>;
|
|
29
|
+
test(projectId: string, endpointId: string): Promise<EndpointTestResult>;
|
|
30
|
+
}
|
|
31
|
+
declare class Subscriptions {
|
|
32
|
+
private client;
|
|
33
|
+
constructor(client: WebhookPlatform);
|
|
34
|
+
create(projectId: string, params: SubscriptionCreateParams): Promise<Subscription>;
|
|
35
|
+
get(projectId: string, subscriptionId: string): Promise<Subscription>;
|
|
36
|
+
list(projectId: string): Promise<Subscription[]>;
|
|
37
|
+
update(projectId: string, subscriptionId: string, params: Partial<SubscriptionCreateParams>): Promise<Subscription>;
|
|
38
|
+
delete(projectId: string, subscriptionId: string): Promise<void>;
|
|
39
|
+
}
|
|
40
|
+
declare class Deliveries {
|
|
41
|
+
private client;
|
|
42
|
+
constructor(client: WebhookPlatform);
|
|
43
|
+
get(deliveryId: string): Promise<Delivery>;
|
|
44
|
+
list(projectId: string, params?: DeliveryListParams): Promise<PaginatedResponse<Delivery>>;
|
|
45
|
+
getAttempts(deliveryId: string): Promise<DeliveryAttempt[]>;
|
|
46
|
+
replay(deliveryId: string): Promise<void>;
|
|
47
|
+
}
|
|
48
|
+
export {};
|