@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 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"]}
@@ -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 {};