@thebookingkit/server 0.1.1
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/.turbo/turbo-build.log +6 -0
- package/.turbo/turbo-test.log +20 -0
- package/CHANGELOG.md +9 -0
- package/dist/__tests__/api.test.d.ts +2 -0
- package/dist/__tests__/api.test.d.ts.map +1 -0
- package/dist/__tests__/api.test.js +280 -0
- package/dist/__tests__/api.test.js.map +1 -0
- package/dist/__tests__/auth.test.d.ts +2 -0
- package/dist/__tests__/auth.test.d.ts.map +1 -0
- package/dist/__tests__/auth.test.js +78 -0
- package/dist/__tests__/auth.test.js.map +1 -0
- package/dist/__tests__/concurrent-booking.test.d.ts +2 -0
- package/dist/__tests__/concurrent-booking.test.d.ts.map +1 -0
- package/dist/__tests__/concurrent-booking.test.js +111 -0
- package/dist/__tests__/concurrent-booking.test.js.map +1 -0
- package/dist/__tests__/multi-tenancy.test.d.ts +2 -0
- package/dist/__tests__/multi-tenancy.test.d.ts.map +1 -0
- package/dist/__tests__/multi-tenancy.test.js +196 -0
- package/dist/__tests__/multi-tenancy.test.js.map +1 -0
- package/dist/__tests__/serialization-retry.test.d.ts +2 -0
- package/dist/__tests__/serialization-retry.test.d.ts.map +1 -0
- package/dist/__tests__/serialization-retry.test.js +53 -0
- package/dist/__tests__/serialization-retry.test.js.map +1 -0
- package/dist/__tests__/webhooks.test.d.ts +2 -0
- package/dist/__tests__/webhooks.test.d.ts.map +1 -0
- package/dist/__tests__/webhooks.test.js +286 -0
- package/dist/__tests__/webhooks.test.js.map +1 -0
- package/dist/__tests__/workflows.test.d.ts +2 -0
- package/dist/__tests__/workflows.test.d.ts.map +1 -0
- package/dist/__tests__/workflows.test.js +299 -0
- package/dist/__tests__/workflows.test.js.map +1 -0
- package/dist/adapters/calendar-adapter.d.ts +47 -0
- package/dist/adapters/calendar-adapter.d.ts.map +1 -0
- package/dist/adapters/calendar-adapter.js +2 -0
- package/dist/adapters/calendar-adapter.js.map +1 -0
- package/dist/adapters/email-adapter.d.ts +65 -0
- package/dist/adapters/email-adapter.d.ts.map +1 -0
- package/dist/adapters/email-adapter.js +40 -0
- package/dist/adapters/email-adapter.js.map +1 -0
- package/dist/adapters/index.d.ts +9 -0
- package/dist/adapters/index.d.ts.map +1 -0
- package/dist/adapters/index.js +3 -0
- package/dist/adapters/index.js.map +1 -0
- package/dist/adapters/job-adapter.d.ts +26 -0
- package/dist/adapters/job-adapter.d.ts.map +1 -0
- package/dist/adapters/job-adapter.js +13 -0
- package/dist/adapters/job-adapter.js.map +1 -0
- package/dist/adapters/payment-adapter.d.ts +106 -0
- package/dist/adapters/payment-adapter.d.ts.map +1 -0
- package/dist/adapters/payment-adapter.js +8 -0
- package/dist/adapters/payment-adapter.js.map +1 -0
- package/dist/adapters/sms-adapter.d.ts +33 -0
- package/dist/adapters/sms-adapter.d.ts.map +1 -0
- package/dist/adapters/sms-adapter.js +8 -0
- package/dist/adapters/sms-adapter.js.map +1 -0
- package/dist/adapters/storage-adapter.d.ts +12 -0
- package/dist/adapters/storage-adapter.d.ts.map +1 -0
- package/dist/adapters/storage-adapter.js +2 -0
- package/dist/adapters/storage-adapter.js.map +1 -0
- package/dist/api.d.ts +223 -0
- package/dist/api.d.ts.map +1 -0
- package/dist/api.js +271 -0
- package/dist/api.js.map +1 -0
- package/dist/auth.d.ts +71 -0
- package/dist/auth.d.ts.map +1 -0
- package/dist/auth.js +81 -0
- package/dist/auth.js.map +1 -0
- package/dist/booking-tokens.d.ts +23 -0
- package/dist/booking-tokens.d.ts.map +1 -0
- package/dist/booking-tokens.js +52 -0
- package/dist/booking-tokens.js.map +1 -0
- package/dist/email-templates.d.ts +36 -0
- package/dist/email-templates.d.ts.map +1 -0
- package/dist/email-templates.js +112 -0
- package/dist/email-templates.js.map +1 -0
- package/dist/index.d.ts +13 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +22 -0
- package/dist/index.js.map +1 -0
- package/dist/multi-tenancy.d.ts +132 -0
- package/dist/multi-tenancy.d.ts.map +1 -0
- package/dist/multi-tenancy.js +188 -0
- package/dist/multi-tenancy.js.map +1 -0
- package/dist/notification-jobs.d.ts +143 -0
- package/dist/notification-jobs.d.ts.map +1 -0
- package/dist/notification-jobs.js +278 -0
- package/dist/notification-jobs.js.map +1 -0
- package/dist/serialization-retry.d.ts +28 -0
- package/dist/serialization-retry.d.ts.map +1 -0
- package/dist/serialization-retry.js +71 -0
- package/dist/serialization-retry.js.map +1 -0
- package/dist/webhooks.d.ts +164 -0
- package/dist/webhooks.d.ts.map +1 -0
- package/dist/webhooks.js +228 -0
- package/dist/webhooks.js.map +1 -0
- package/dist/workflows.d.ts +169 -0
- package/dist/workflows.d.ts.map +1 -0
- package/dist/workflows.js +251 -0
- package/dist/workflows.js.map +1 -0
- package/package.json +32 -0
- package/src/__tests__/api.test.ts +354 -0
- package/src/__tests__/auth.test.ts +111 -0
- package/src/__tests__/concurrent-booking.test.ts +170 -0
- package/src/__tests__/multi-tenancy.test.ts +267 -0
- package/src/__tests__/serialization-retry.test.ts +76 -0
- package/src/__tests__/webhooks.test.ts +412 -0
- package/src/__tests__/workflows.test.ts +422 -0
- package/src/adapters/calendar-adapter.ts +49 -0
- package/src/adapters/email-adapter.ts +108 -0
- package/src/adapters/index.ts +36 -0
- package/src/adapters/job-adapter.ts +26 -0
- package/src/adapters/payment-adapter.ts +118 -0
- package/src/adapters/sms-adapter.ts +35 -0
- package/src/adapters/storage-adapter.ts +11 -0
- package/src/api.ts +446 -0
- package/src/auth.ts +146 -0
- package/src/booking-tokens.ts +61 -0
- package/src/email-templates.ts +140 -0
- package/src/index.ts +192 -0
- package/src/multi-tenancy.ts +301 -0
- package/src/notification-jobs.ts +428 -0
- package/src/serialization-retry.ts +94 -0
- package/src/webhooks.ts +378 -0
- package/src/workflows.ts +441 -0
- package/tsconfig.json +9 -0
- package/vitest.config.ts +7 -0
|
@@ -0,0 +1,196 @@
|
|
|
1
|
+
import { describe, it, expect } from "vitest";
|
|
2
|
+
import { resolveEffectiveSettings, getRolePermissions, roleHasPermission, assertOrgPermission, assertTenantScope, buildOrgBookingUrl, parseOrgBookingPath, TenantAuthorizationError, GLOBAL_DEFAULTS, } from "../multi-tenancy.js";
|
|
3
|
+
// ---------------------------------------------------------------------------
|
|
4
|
+
// resolveEffectiveSettings
|
|
5
|
+
// ---------------------------------------------------------------------------
|
|
6
|
+
describe("resolveEffectiveSettings", () => {
|
|
7
|
+
it("uses global defaults when no settings provided", () => {
|
|
8
|
+
const resolved = resolveEffectiveSettings();
|
|
9
|
+
expect(resolved.timezone).toBe(GLOBAL_DEFAULTS.timezone);
|
|
10
|
+
expect(resolved.currency).toBe(GLOBAL_DEFAULTS.currency);
|
|
11
|
+
expect(resolved.bufferMinutes).toBe(GLOBAL_DEFAULTS.bufferMinutes);
|
|
12
|
+
expect(resolved.branding).toEqual({});
|
|
13
|
+
expect(resolved.bookingLimits).toEqual({});
|
|
14
|
+
});
|
|
15
|
+
it("applies org settings over global defaults", () => {
|
|
16
|
+
const org = {
|
|
17
|
+
defaultTimezone: "America/New_York",
|
|
18
|
+
defaultCurrency: "EUR",
|
|
19
|
+
branding: { primaryColor: "#6366f1" },
|
|
20
|
+
};
|
|
21
|
+
const resolved = resolveEffectiveSettings(org);
|
|
22
|
+
expect(resolved.timezone).toBe("America/New_York");
|
|
23
|
+
expect(resolved.currency).toBe("EUR");
|
|
24
|
+
expect(resolved.branding.primaryColor).toBe("#6366f1");
|
|
25
|
+
});
|
|
26
|
+
it("provider settings override org settings", () => {
|
|
27
|
+
const org = {
|
|
28
|
+
defaultTimezone: "America/New_York",
|
|
29
|
+
defaultCurrency: "USD",
|
|
30
|
+
};
|
|
31
|
+
const provider = {
|
|
32
|
+
timezone: "Europe/London",
|
|
33
|
+
currency: "GBP",
|
|
34
|
+
};
|
|
35
|
+
const resolved = resolveEffectiveSettings(org, provider);
|
|
36
|
+
expect(resolved.timezone).toBe("Europe/London");
|
|
37
|
+
expect(resolved.currency).toBe("GBP");
|
|
38
|
+
});
|
|
39
|
+
it("event type settings override provider settings", () => {
|
|
40
|
+
const org = { defaultTimezone: "UTC" };
|
|
41
|
+
const provider = { timezone: "America/New_York" };
|
|
42
|
+
const eventType = { timezone: "Asia/Tokyo" };
|
|
43
|
+
const resolved = resolveEffectiveSettings(org, provider, eventType);
|
|
44
|
+
expect(resolved.timezone).toBe("Asia/Tokyo");
|
|
45
|
+
});
|
|
46
|
+
it("merges branding settings additively", () => {
|
|
47
|
+
const org = {
|
|
48
|
+
branding: { primaryColor: "#000", logoUrl: "https://example.com/logo.png" },
|
|
49
|
+
};
|
|
50
|
+
const provider = {
|
|
51
|
+
branding: { primaryColor: "#fff" },
|
|
52
|
+
};
|
|
53
|
+
const resolved = resolveEffectiveSettings(org, provider);
|
|
54
|
+
expect(resolved.branding.primaryColor).toBe("#fff"); // provider overrides
|
|
55
|
+
expect(resolved.branding.logoUrl).toBe("https://example.com/logo.png"); // org preserved
|
|
56
|
+
});
|
|
57
|
+
it("merges booking limits additively", () => {
|
|
58
|
+
const org = {
|
|
59
|
+
defaultBookingLimits: { maxPerDay: 5 },
|
|
60
|
+
};
|
|
61
|
+
const provider = {
|
|
62
|
+
bookingLimits: { maxPerWeek: 20 },
|
|
63
|
+
};
|
|
64
|
+
const resolved = resolveEffectiveSettings(org, provider);
|
|
65
|
+
expect(resolved.bookingLimits).toMatchObject({ maxPerDay: 5, maxPerWeek: 20 });
|
|
66
|
+
});
|
|
67
|
+
it("handles null settings gracefully", () => {
|
|
68
|
+
expect(() => resolveEffectiveSettings(null, null, null)).not.toThrow();
|
|
69
|
+
});
|
|
70
|
+
});
|
|
71
|
+
// ---------------------------------------------------------------------------
|
|
72
|
+
// getRolePermissions
|
|
73
|
+
// ---------------------------------------------------------------------------
|
|
74
|
+
describe("getRolePermissions", () => {
|
|
75
|
+
it("owner has all permissions", () => {
|
|
76
|
+
const perms = getRolePermissions("owner");
|
|
77
|
+
expect(perms).toContain("manage:members");
|
|
78
|
+
expect(perms).toContain("manage:teams");
|
|
79
|
+
expect(perms).toContain("view:all-bookings");
|
|
80
|
+
expect(perms).toContain("manage:organization");
|
|
81
|
+
expect(perms).toContain("view:analytics");
|
|
82
|
+
});
|
|
83
|
+
it("admin has management permissions but not member management", () => {
|
|
84
|
+
const perms = getRolePermissions("admin");
|
|
85
|
+
expect(perms).toContain("manage:teams");
|
|
86
|
+
expect(perms).toContain("view:all-bookings");
|
|
87
|
+
expect(perms).not.toContain("manage:members");
|
|
88
|
+
expect(perms).not.toContain("manage:organization");
|
|
89
|
+
});
|
|
90
|
+
it("member has only own-resource permissions", () => {
|
|
91
|
+
const perms = getRolePermissions("member");
|
|
92
|
+
expect(perms).toContain("view:own-bookings");
|
|
93
|
+
expect(perms).toContain("manage:own-availability");
|
|
94
|
+
expect(perms).not.toContain("view:all-bookings");
|
|
95
|
+
expect(perms).not.toContain("manage:teams");
|
|
96
|
+
});
|
|
97
|
+
});
|
|
98
|
+
// ---------------------------------------------------------------------------
|
|
99
|
+
// roleHasPermission
|
|
100
|
+
// ---------------------------------------------------------------------------
|
|
101
|
+
describe("roleHasPermission", () => {
|
|
102
|
+
it("returns true for permitted action", () => {
|
|
103
|
+
expect(roleHasPermission("owner", "manage:members")).toBe(true);
|
|
104
|
+
});
|
|
105
|
+
it("returns false for unpermitted action", () => {
|
|
106
|
+
expect(roleHasPermission("member", "manage:members")).toBe(false);
|
|
107
|
+
});
|
|
108
|
+
it("admin can view all bookings", () => {
|
|
109
|
+
expect(roleHasPermission("admin", "view:all-bookings")).toBe(true);
|
|
110
|
+
});
|
|
111
|
+
});
|
|
112
|
+
// ---------------------------------------------------------------------------
|
|
113
|
+
// assertOrgPermission
|
|
114
|
+
// ---------------------------------------------------------------------------
|
|
115
|
+
describe("assertOrgPermission", () => {
|
|
116
|
+
const ownerMember = {
|
|
117
|
+
userId: "user-1",
|
|
118
|
+
organizationId: "org-1",
|
|
119
|
+
role: "owner",
|
|
120
|
+
};
|
|
121
|
+
const regularMember = {
|
|
122
|
+
userId: "user-2",
|
|
123
|
+
organizationId: "org-1",
|
|
124
|
+
role: "member",
|
|
125
|
+
};
|
|
126
|
+
it("does not throw for permitted action", () => {
|
|
127
|
+
expect(() => assertOrgPermission(ownerMember, "manage:members")).not.toThrow();
|
|
128
|
+
});
|
|
129
|
+
it("throws for unpermitted action", () => {
|
|
130
|
+
expect(() => assertOrgPermission(regularMember, "manage:members")).toThrow(TenantAuthorizationError);
|
|
131
|
+
});
|
|
132
|
+
it("throws with descriptive message", () => {
|
|
133
|
+
expect(() => assertOrgPermission(regularMember, "view:all-bookings")).toThrow('"member"');
|
|
134
|
+
});
|
|
135
|
+
});
|
|
136
|
+
// ---------------------------------------------------------------------------
|
|
137
|
+
// assertTenantScope
|
|
138
|
+
// ---------------------------------------------------------------------------
|
|
139
|
+
describe("assertTenantScope", () => {
|
|
140
|
+
it("does not throw when org IDs match", () => {
|
|
141
|
+
expect(() => assertTenantScope("org-1", "org-1")).not.toThrow();
|
|
142
|
+
});
|
|
143
|
+
it("does not throw when resource has no org ID", () => {
|
|
144
|
+
expect(() => assertTenantScope(null, "org-1")).not.toThrow();
|
|
145
|
+
expect(() => assertTenantScope(undefined, "org-1")).not.toThrow();
|
|
146
|
+
});
|
|
147
|
+
it("throws when org IDs don't match", () => {
|
|
148
|
+
expect(() => assertTenantScope("org-2", "org-1")).toThrow(TenantAuthorizationError);
|
|
149
|
+
expect(() => assertTenantScope("org-2", "org-1")).toThrow("does not belong to the current organization");
|
|
150
|
+
});
|
|
151
|
+
});
|
|
152
|
+
// ---------------------------------------------------------------------------
|
|
153
|
+
// buildOrgBookingUrl
|
|
154
|
+
// ---------------------------------------------------------------------------
|
|
155
|
+
describe("buildOrgBookingUrl", () => {
|
|
156
|
+
it("builds the correct URL", () => {
|
|
157
|
+
const url = buildOrgBookingUrl("acme-corp", "dr-smith", "consultation", "https://booking.example.com");
|
|
158
|
+
expect(url).toBe("https://booking.example.com/acme-corp/dr-smith/consultation");
|
|
159
|
+
});
|
|
160
|
+
});
|
|
161
|
+
// ---------------------------------------------------------------------------
|
|
162
|
+
// parseOrgBookingPath
|
|
163
|
+
// ---------------------------------------------------------------------------
|
|
164
|
+
describe("parseOrgBookingPath", () => {
|
|
165
|
+
it("parses valid path", () => {
|
|
166
|
+
const result = parseOrgBookingPath("/acme-corp/dr-smith/consultation");
|
|
167
|
+
expect(result).toEqual({
|
|
168
|
+
orgSlug: "acme-corp",
|
|
169
|
+
providerSlug: "dr-smith",
|
|
170
|
+
eventTypeSlug: "consultation",
|
|
171
|
+
});
|
|
172
|
+
});
|
|
173
|
+
it("returns null for invalid path", () => {
|
|
174
|
+
expect(parseOrgBookingPath("/only-two/segments")).toBeNull();
|
|
175
|
+
expect(parseOrgBookingPath("/too/many/path/segments")).toBeNull();
|
|
176
|
+
expect(parseOrgBookingPath("no-leading-slash")).toBeNull();
|
|
177
|
+
});
|
|
178
|
+
it("returns null for empty path", () => {
|
|
179
|
+
expect(parseOrgBookingPath("")).toBeNull();
|
|
180
|
+
});
|
|
181
|
+
});
|
|
182
|
+
// ---------------------------------------------------------------------------
|
|
183
|
+
// GLOBAL_DEFAULTS
|
|
184
|
+
// ---------------------------------------------------------------------------
|
|
185
|
+
describe("GLOBAL_DEFAULTS", () => {
|
|
186
|
+
it("uses UTC timezone", () => {
|
|
187
|
+
expect(GLOBAL_DEFAULTS.timezone).toBe("UTC");
|
|
188
|
+
});
|
|
189
|
+
it("uses USD currency", () => {
|
|
190
|
+
expect(GLOBAL_DEFAULTS.currency).toBe("USD");
|
|
191
|
+
});
|
|
192
|
+
it("uses 0 buffer minutes", () => {
|
|
193
|
+
expect(GLOBAL_DEFAULTS.bufferMinutes).toBe(0);
|
|
194
|
+
});
|
|
195
|
+
});
|
|
196
|
+
//# sourceMappingURL=multi-tenancy.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"multi-tenancy.test.js","sourceRoot":"","sources":["../../src/__tests__/multi-tenancy.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"serialization-retry.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/serialization-retry.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
import { describe, it, expect, vi } from "vitest";
|
|
2
|
+
import { withSerializableRetry, BookingConflictError, SerializationRetryExhaustedError, } from "../index.js";
|
|
3
|
+
function makePostgresError(code, message = "pg error") {
|
|
4
|
+
const err = new Error(message);
|
|
5
|
+
err.code = code;
|
|
6
|
+
return err;
|
|
7
|
+
}
|
|
8
|
+
describe("withSerializableRetry", () => {
|
|
9
|
+
it("returns the result on success", async () => {
|
|
10
|
+
const result = await withSerializableRetry(async () => ({ id: "123" }));
|
|
11
|
+
expect(result).toEqual({ id: "123" });
|
|
12
|
+
});
|
|
13
|
+
it("throws BookingConflictError on exclusion violation (23P01)", async () => {
|
|
14
|
+
const fn = vi.fn().mockRejectedValue(makePostgresError("23P01"));
|
|
15
|
+
await expect(withSerializableRetry(fn)).rejects.toThrow(BookingConflictError);
|
|
16
|
+
// Should NOT retry on exclusion violation
|
|
17
|
+
expect(fn).toHaveBeenCalledTimes(1);
|
|
18
|
+
});
|
|
19
|
+
it("retries on serialization failure (40001) and succeeds", async () => {
|
|
20
|
+
const fn = vi
|
|
21
|
+
.fn()
|
|
22
|
+
.mockRejectedValueOnce(makePostgresError("40001"))
|
|
23
|
+
.mockRejectedValueOnce(makePostgresError("40001"))
|
|
24
|
+
.mockResolvedValueOnce({ id: "456" });
|
|
25
|
+
const result = await withSerializableRetry(fn, {
|
|
26
|
+
maxRetries: 3,
|
|
27
|
+
baseDelayMs: 1, // fast for tests
|
|
28
|
+
});
|
|
29
|
+
expect(result).toEqual({ id: "456" });
|
|
30
|
+
expect(fn).toHaveBeenCalledTimes(3);
|
|
31
|
+
});
|
|
32
|
+
it("throws SerializationRetryExhaustedError when all retries fail", async () => {
|
|
33
|
+
const fn = vi.fn().mockRejectedValue(makePostgresError("40001"));
|
|
34
|
+
await expect(withSerializableRetry(fn, { maxRetries: 2, baseDelayMs: 1 })).rejects.toThrow(SerializationRetryExhaustedError);
|
|
35
|
+
// Initial attempt + 2 retries = 3 calls
|
|
36
|
+
expect(fn).toHaveBeenCalledTimes(3);
|
|
37
|
+
});
|
|
38
|
+
it("rethrows unknown errors without retrying", async () => {
|
|
39
|
+
const fn = vi.fn().mockRejectedValue(new Error("connection refused"));
|
|
40
|
+
await expect(withSerializableRetry(fn)).rejects.toThrow("connection refused");
|
|
41
|
+
expect(fn).toHaveBeenCalledTimes(1);
|
|
42
|
+
});
|
|
43
|
+
it("uses default options (3 retries, 50ms base delay)", async () => {
|
|
44
|
+
const fn = vi
|
|
45
|
+
.fn()
|
|
46
|
+
.mockRejectedValueOnce(makePostgresError("40001"))
|
|
47
|
+
.mockResolvedValueOnce("ok");
|
|
48
|
+
const result = await withSerializableRetry(fn);
|
|
49
|
+
expect(result).toBe("ok");
|
|
50
|
+
expect(fn).toHaveBeenCalledTimes(2);
|
|
51
|
+
});
|
|
52
|
+
});
|
|
53
|
+
//# sourceMappingURL=serialization-retry.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"serialization-retry.test.js","sourceRoot":"","sources":["../../src/__tests__/serialization-retry.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"webhooks.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/webhooks.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,286 @@
|
|
|
1
|
+
import { describe, it, expect, vi, beforeEach, afterEach } from "vitest";
|
|
2
|
+
import { signWebhookPayload, verifyWebhookSignature, createWebhookEnvelope, resolvePayloadTemplate, matchWebhookSubscriptions, getRetryDelay, isSuccessResponse, validateWebhookSubscription, WEBHOOK_TRIGGERS, } from "../webhooks.js";
|
|
3
|
+
// ---------------------------------------------------------------------------
|
|
4
|
+
// Fixtures
|
|
5
|
+
// ---------------------------------------------------------------------------
|
|
6
|
+
const testSecret = "whsec_test_secret_key_123";
|
|
7
|
+
const testPayload = '{"triggerEvent":"BOOKING_CREATED","payload":{}}';
|
|
8
|
+
const samplePayload = {
|
|
9
|
+
bookingId: "bk-1",
|
|
10
|
+
eventType: "consultation",
|
|
11
|
+
startTime: "2026-03-15T14:00:00.000Z",
|
|
12
|
+
endTime: "2026-03-15T14:30:00.000Z",
|
|
13
|
+
organizer: { name: "Dr. Smith", email: "smith@example.com" },
|
|
14
|
+
attendees: [{ email: "jane@example.com", name: "Jane Doe" }],
|
|
15
|
+
status: "confirmed",
|
|
16
|
+
};
|
|
17
|
+
function makeSub(overrides) {
|
|
18
|
+
return {
|
|
19
|
+
id: "wh-1",
|
|
20
|
+
subscriberUrl: "https://api.example.com/webhooks",
|
|
21
|
+
triggers: ["BOOKING_CREATED"],
|
|
22
|
+
isActive: true,
|
|
23
|
+
...overrides,
|
|
24
|
+
};
|
|
25
|
+
}
|
|
26
|
+
// ---------------------------------------------------------------------------
|
|
27
|
+
// signWebhookPayload
|
|
28
|
+
// ---------------------------------------------------------------------------
|
|
29
|
+
describe("signWebhookPayload", () => {
|
|
30
|
+
it("produces a hex-encoded HMAC-SHA256 signature", () => {
|
|
31
|
+
const sig = signWebhookPayload(testPayload, testSecret, 1710000000);
|
|
32
|
+
expect(sig).toMatch(/^[0-9a-f]{64}$/);
|
|
33
|
+
});
|
|
34
|
+
it("produces different signatures for different payloads", () => {
|
|
35
|
+
const sig1 = signWebhookPayload("body1", testSecret, 1710000000);
|
|
36
|
+
const sig2 = signWebhookPayload("body2", testSecret, 1710000000);
|
|
37
|
+
expect(sig1).not.toBe(sig2);
|
|
38
|
+
});
|
|
39
|
+
it("produces different signatures for different timestamps", () => {
|
|
40
|
+
const sig1 = signWebhookPayload(testPayload, testSecret, 1710000000);
|
|
41
|
+
const sig2 = signWebhookPayload(testPayload, testSecret, 1710000001);
|
|
42
|
+
expect(sig1).not.toBe(sig2);
|
|
43
|
+
});
|
|
44
|
+
it("produces different signatures for different secrets", () => {
|
|
45
|
+
const sig1 = signWebhookPayload(testPayload, "secret1", 1710000000);
|
|
46
|
+
const sig2 = signWebhookPayload(testPayload, "secret2", 1710000000);
|
|
47
|
+
expect(sig1).not.toBe(sig2);
|
|
48
|
+
});
|
|
49
|
+
});
|
|
50
|
+
// ---------------------------------------------------------------------------
|
|
51
|
+
// verifyWebhookSignature
|
|
52
|
+
// ---------------------------------------------------------------------------
|
|
53
|
+
describe("verifyWebhookSignature", () => {
|
|
54
|
+
beforeEach(() => {
|
|
55
|
+
vi.useFakeTimers();
|
|
56
|
+
});
|
|
57
|
+
afterEach(() => {
|
|
58
|
+
vi.useRealTimers();
|
|
59
|
+
});
|
|
60
|
+
it("returns valid for a correct signature within tolerance", () => {
|
|
61
|
+
const now = 1710000000;
|
|
62
|
+
vi.setSystemTime(now * 1000);
|
|
63
|
+
const sig = signWebhookPayload(testPayload, testSecret, now);
|
|
64
|
+
const result = verifyWebhookSignature(testPayload, sig, now, testSecret);
|
|
65
|
+
expect(result).toEqual({ valid: true });
|
|
66
|
+
});
|
|
67
|
+
it("returns signature_mismatch for tampered payload", () => {
|
|
68
|
+
const now = 1710000000;
|
|
69
|
+
vi.setSystemTime(now * 1000);
|
|
70
|
+
const sig = signWebhookPayload(testPayload, testSecret, now);
|
|
71
|
+
const result = verifyWebhookSignature("tampered_body", sig, now, testSecret);
|
|
72
|
+
expect(result).toEqual({ valid: false, reason: "signature_mismatch" });
|
|
73
|
+
});
|
|
74
|
+
it("returns signature_mismatch for wrong secret", () => {
|
|
75
|
+
const now = 1710000000;
|
|
76
|
+
vi.setSystemTime(now * 1000);
|
|
77
|
+
const sig = signWebhookPayload(testPayload, testSecret, now);
|
|
78
|
+
const result = verifyWebhookSignature(testPayload, sig, now, "wrong_secret");
|
|
79
|
+
expect(result).toEqual({ valid: false, reason: "signature_mismatch" });
|
|
80
|
+
});
|
|
81
|
+
it("returns timestamp_expired for old timestamp (>5 min)", () => {
|
|
82
|
+
const now = 1710000600; // 10 min later
|
|
83
|
+
vi.setSystemTime(now * 1000);
|
|
84
|
+
const oldTimestamp = now - 301; // 5 min 1 sec ago
|
|
85
|
+
const sig = signWebhookPayload(testPayload, testSecret, oldTimestamp);
|
|
86
|
+
const result = verifyWebhookSignature(testPayload, sig, oldTimestamp, testSecret);
|
|
87
|
+
expect(result).toEqual({ valid: false, reason: "timestamp_expired" });
|
|
88
|
+
});
|
|
89
|
+
it("accepts timestamp within tolerance", () => {
|
|
90
|
+
const now = 1710000000;
|
|
91
|
+
vi.setSystemTime(now * 1000);
|
|
92
|
+
const recentTimestamp = now - 299; // 4 min 59 sec ago
|
|
93
|
+
const sig = signWebhookPayload(testPayload, testSecret, recentTimestamp);
|
|
94
|
+
const result = verifyWebhookSignature(testPayload, sig, recentTimestamp, testSecret);
|
|
95
|
+
expect(result).toEqual({ valid: true });
|
|
96
|
+
});
|
|
97
|
+
it("supports custom tolerance", () => {
|
|
98
|
+
const now = 1710000000;
|
|
99
|
+
vi.setSystemTime(now * 1000);
|
|
100
|
+
const oldTimestamp = now - 61;
|
|
101
|
+
const sig = signWebhookPayload(testPayload, testSecret, oldTimestamp);
|
|
102
|
+
const result = verifyWebhookSignature(testPayload, sig, oldTimestamp, testSecret, { toleranceSeconds: 60 });
|
|
103
|
+
expect(result).toEqual({ valid: false, reason: "timestamp_expired" });
|
|
104
|
+
});
|
|
105
|
+
it("rejects future timestamps beyond tolerance", () => {
|
|
106
|
+
const now = 1710000000;
|
|
107
|
+
vi.setSystemTime(now * 1000);
|
|
108
|
+
const futureTimestamp = now + 400;
|
|
109
|
+
const sig = signWebhookPayload(testPayload, testSecret, futureTimestamp);
|
|
110
|
+
const result = verifyWebhookSignature(testPayload, sig, futureTimestamp, testSecret);
|
|
111
|
+
expect(result).toEqual({ valid: false, reason: "timestamp_expired" });
|
|
112
|
+
});
|
|
113
|
+
});
|
|
114
|
+
// ---------------------------------------------------------------------------
|
|
115
|
+
// createWebhookEnvelope
|
|
116
|
+
// ---------------------------------------------------------------------------
|
|
117
|
+
describe("createWebhookEnvelope", () => {
|
|
118
|
+
it("creates a valid envelope with ISO 8601 timestamp", () => {
|
|
119
|
+
const envelope = createWebhookEnvelope("BOOKING_CREATED", samplePayload);
|
|
120
|
+
expect(envelope.triggerEvent).toBe("BOOKING_CREATED");
|
|
121
|
+
expect(envelope.payload).toBe(samplePayload);
|
|
122
|
+
expect(new Date(envelope.createdAt).toISOString()).toBe(envelope.createdAt);
|
|
123
|
+
});
|
|
124
|
+
});
|
|
125
|
+
// ---------------------------------------------------------------------------
|
|
126
|
+
// resolvePayloadTemplate
|
|
127
|
+
// ---------------------------------------------------------------------------
|
|
128
|
+
describe("resolvePayloadTemplate", () => {
|
|
129
|
+
const envelope = createWebhookEnvelope("BOOKING_CREATED", samplePayload);
|
|
130
|
+
it("resolves all standard template variables", () => {
|
|
131
|
+
const template = '{"event":"{{triggerEvent}}","booking":"{{bookingId}}","type":"{{eventType}}","start":"{{startTime}}","end":"{{endTime}}","status":"{{status}}","org":"{{organizerName}}","email":"{{organizerEmail}}"}';
|
|
132
|
+
const result = resolvePayloadTemplate(template, envelope);
|
|
133
|
+
const parsed = JSON.parse(result);
|
|
134
|
+
expect(parsed.event).toBe("BOOKING_CREATED");
|
|
135
|
+
expect(parsed.booking).toBe("bk-1");
|
|
136
|
+
expect(parsed.type).toBe("consultation");
|
|
137
|
+
expect(parsed.start).toBe("2026-03-15T14:00:00.000Z");
|
|
138
|
+
expect(parsed.status).toBe("confirmed");
|
|
139
|
+
expect(parsed.org).toBe("Dr. Smith");
|
|
140
|
+
expect(parsed.email).toBe("smith@example.com");
|
|
141
|
+
});
|
|
142
|
+
it("handles template with no variables", () => {
|
|
143
|
+
const result = resolvePayloadTemplate('{"static":"value"}', envelope);
|
|
144
|
+
expect(result).toBe('{"static":"value"}');
|
|
145
|
+
});
|
|
146
|
+
});
|
|
147
|
+
// ---------------------------------------------------------------------------
|
|
148
|
+
// matchWebhookSubscriptions
|
|
149
|
+
// ---------------------------------------------------------------------------
|
|
150
|
+
describe("matchWebhookSubscriptions", () => {
|
|
151
|
+
const subs = [
|
|
152
|
+
makeSub({ id: "wh-1", triggers: ["BOOKING_CREATED", "BOOKING_CANCELLED"] }),
|
|
153
|
+
makeSub({ id: "wh-2", triggers: ["BOOKING_CREATED"], isActive: false }),
|
|
154
|
+
makeSub({
|
|
155
|
+
id: "wh-3",
|
|
156
|
+
triggers: ["BOOKING_CREATED"],
|
|
157
|
+
eventTypeId: "evt-1",
|
|
158
|
+
}),
|
|
159
|
+
makeSub({
|
|
160
|
+
id: "wh-4",
|
|
161
|
+
triggers: ["BOOKING_CREATED"],
|
|
162
|
+
teamId: "team-1",
|
|
163
|
+
}),
|
|
164
|
+
];
|
|
165
|
+
it("matches active webhooks with matching trigger", () => {
|
|
166
|
+
const matched = matchWebhookSubscriptions(subs, "BOOKING_CREATED");
|
|
167
|
+
expect(matched.map((s) => s.id)).toContain("wh-1");
|
|
168
|
+
});
|
|
169
|
+
it("excludes inactive webhooks", () => {
|
|
170
|
+
const matched = matchWebhookSubscriptions(subs, "BOOKING_CREATED");
|
|
171
|
+
expect(matched.find((s) => s.id === "wh-2")).toBeUndefined();
|
|
172
|
+
});
|
|
173
|
+
it("matches scoped webhooks when scope matches", () => {
|
|
174
|
+
const matched = matchWebhookSubscriptions(subs, "BOOKING_CREATED", {
|
|
175
|
+
eventTypeId: "evt-1",
|
|
176
|
+
});
|
|
177
|
+
expect(matched.find((s) => s.id === "wh-3")).toBeDefined();
|
|
178
|
+
});
|
|
179
|
+
it("excludes scoped webhooks when scope doesn't match", () => {
|
|
180
|
+
const matched = matchWebhookSubscriptions(subs, "BOOKING_CREATED", {
|
|
181
|
+
eventTypeId: "evt-99",
|
|
182
|
+
});
|
|
183
|
+
expect(matched.find((s) => s.id === "wh-3")).toBeUndefined();
|
|
184
|
+
});
|
|
185
|
+
it("matches team-scoped webhooks", () => {
|
|
186
|
+
const matched = matchWebhookSubscriptions(subs, "BOOKING_CREATED", {
|
|
187
|
+
teamId: "team-1",
|
|
188
|
+
});
|
|
189
|
+
expect(matched.find((s) => s.id === "wh-4")).toBeDefined();
|
|
190
|
+
});
|
|
191
|
+
it("returns empty for unsubscribed trigger", () => {
|
|
192
|
+
const matched = matchWebhookSubscriptions(subs, "OOO_CREATED");
|
|
193
|
+
expect(matched).toHaveLength(0);
|
|
194
|
+
});
|
|
195
|
+
});
|
|
196
|
+
// ---------------------------------------------------------------------------
|
|
197
|
+
// getRetryDelay
|
|
198
|
+
// ---------------------------------------------------------------------------
|
|
199
|
+
describe("getRetryDelay", () => {
|
|
200
|
+
it("returns correct delays for default config", () => {
|
|
201
|
+
expect(getRetryDelay(0)).toBe(10);
|
|
202
|
+
expect(getRetryDelay(1)).toBe(60);
|
|
203
|
+
expect(getRetryDelay(2)).toBe(300);
|
|
204
|
+
});
|
|
205
|
+
it("returns null when max retries exceeded", () => {
|
|
206
|
+
expect(getRetryDelay(3)).toBeNull();
|
|
207
|
+
expect(getRetryDelay(10)).toBeNull();
|
|
208
|
+
});
|
|
209
|
+
it("uses custom config", () => {
|
|
210
|
+
const config = { maxRetries: 2, backoffSeconds: [5, 30] };
|
|
211
|
+
expect(getRetryDelay(0, config)).toBe(5);
|
|
212
|
+
expect(getRetryDelay(1, config)).toBe(30);
|
|
213
|
+
expect(getRetryDelay(2, config)).toBeNull();
|
|
214
|
+
});
|
|
215
|
+
});
|
|
216
|
+
// ---------------------------------------------------------------------------
|
|
217
|
+
// isSuccessResponse
|
|
218
|
+
// ---------------------------------------------------------------------------
|
|
219
|
+
describe("isSuccessResponse", () => {
|
|
220
|
+
it("returns true for 200", () => {
|
|
221
|
+
expect(isSuccessResponse(200)).toBe(true);
|
|
222
|
+
});
|
|
223
|
+
it("returns true for 201", () => {
|
|
224
|
+
expect(isSuccessResponse(201)).toBe(true);
|
|
225
|
+
});
|
|
226
|
+
it("returns true for 204", () => {
|
|
227
|
+
expect(isSuccessResponse(204)).toBe(true);
|
|
228
|
+
});
|
|
229
|
+
it("returns false for 400", () => {
|
|
230
|
+
expect(isSuccessResponse(400)).toBe(false);
|
|
231
|
+
});
|
|
232
|
+
it("returns false for 500", () => {
|
|
233
|
+
expect(isSuccessResponse(500)).toBe(false);
|
|
234
|
+
});
|
|
235
|
+
it("returns false for 301 redirect", () => {
|
|
236
|
+
expect(isSuccessResponse(301)).toBe(false);
|
|
237
|
+
});
|
|
238
|
+
});
|
|
239
|
+
// ---------------------------------------------------------------------------
|
|
240
|
+
// validateWebhookSubscription
|
|
241
|
+
// ---------------------------------------------------------------------------
|
|
242
|
+
describe("validateWebhookSubscription", () => {
|
|
243
|
+
it("accepts valid subscription", () => {
|
|
244
|
+
expect(() => validateWebhookSubscription({
|
|
245
|
+
subscriberUrl: "https://example.com/webhook",
|
|
246
|
+
triggers: ["BOOKING_CREATED"],
|
|
247
|
+
isActive: true,
|
|
248
|
+
})).not.toThrow();
|
|
249
|
+
});
|
|
250
|
+
it("rejects missing URL", () => {
|
|
251
|
+
expect(() => validateWebhookSubscription({
|
|
252
|
+
subscriberUrl: "",
|
|
253
|
+
triggers: ["BOOKING_CREATED"],
|
|
254
|
+
isActive: true,
|
|
255
|
+
})).toThrow("Subscriber URL is required");
|
|
256
|
+
});
|
|
257
|
+
it("rejects invalid URL", () => {
|
|
258
|
+
expect(() => validateWebhookSubscription({
|
|
259
|
+
subscriberUrl: "not-a-url",
|
|
260
|
+
triggers: ["BOOKING_CREATED"],
|
|
261
|
+
isActive: true,
|
|
262
|
+
})).toThrow("Invalid subscriber URL");
|
|
263
|
+
});
|
|
264
|
+
it("rejects empty triggers array", () => {
|
|
265
|
+
expect(() => validateWebhookSubscription({
|
|
266
|
+
subscriberUrl: "https://example.com/webhook",
|
|
267
|
+
triggers: [],
|
|
268
|
+
isActive: true,
|
|
269
|
+
})).toThrow("At least one trigger");
|
|
270
|
+
});
|
|
271
|
+
it("rejects invalid trigger", () => {
|
|
272
|
+
expect(() => validateWebhookSubscription({
|
|
273
|
+
subscriberUrl: "https://example.com/webhook",
|
|
274
|
+
triggers: ["INVALID_TRIGGER"],
|
|
275
|
+
isActive: true,
|
|
276
|
+
})).toThrow('Invalid trigger: "INVALID_TRIGGER"');
|
|
277
|
+
});
|
|
278
|
+
it("accepts all valid triggers", () => {
|
|
279
|
+
expect(() => validateWebhookSubscription({
|
|
280
|
+
subscriberUrl: "https://example.com/webhook",
|
|
281
|
+
triggers: [...WEBHOOK_TRIGGERS],
|
|
282
|
+
isActive: true,
|
|
283
|
+
})).not.toThrow();
|
|
284
|
+
});
|
|
285
|
+
});
|
|
286
|
+
//# sourceMappingURL=webhooks.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"webhooks.test.js","sourceRoot":"","sources":["../../src/__tests__/webhooks.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"workflows.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/workflows.test.ts"],"names":[],"mappings":""}
|