UNPKG

svix

Version:

Svix webhooks API client and webhook verification library

422 lines (353 loc) 18.7 kB
import { BackgroundTaskType, MessageAttemptTriggerType, Svix } from "."; import { expect, test } from "@jest/globals"; import * as mockttp from "mockttp"; import { ApiException } from "./util"; import { Ordering } from "./models/ordering"; import { ValidationError, HttpErrorOut } from "./HttpErrors"; import { LIB_VERSION } from "./request"; const ApplicationOut = `{"uid":"unique-identifier","name":"My first application","rateLimit":0,"id":"app_1srOrx2ZWZBpBUvZwXKQmoEYga2","createdAt":"2019-08-24T14:15:22Z","updatedAt":"2019-08-24T14:15:22Z","metadata":{"property1":"string","property2":"string"}}`; const ListResponseMessageOut = `{"data":[{"eventId":"unique-identifier","eventType":"user.signup","payload":{"email":"test@example.com","type":"user.created","username":"test_user"},"channels":["project_123","group_2"],"id":"msg_1srOrx2ZWZBpBUvZwXKQmoEYga2","timestamp":"2019-08-24T14:15:22Z","tags":["project_1337"]}],"iterator":"iterator","prevIterator":"-iterator","done":true}`; const ListResponseApplicationOut = `{"data":[{"uid":"unique-identifier","name":"My first application","rateLimit":0,"id":"app_1srOrx2ZWZBpBUvZwXKQmoEYga2","createdAt":"2019-08-24T14:15:22Z","updatedAt":"2019-08-24T14:15:22Z","metadata":{"property1":"string","property2":"string"}}],"iterator":"iterator","prevIterator":"-iterator","done":true}`; const ListResponseMessageAttemptOut = `{"data":[{"url":"https://example.com/webhook/","response":"{}","responseStatusCode":200,"responseDurationMs":0,"status":0,"triggerType":0,"msgId":"msg_1srOrx2ZWZBpBUvZwXKQmoEYga2","endpointId":"ep_1srOrx2ZWZBpBUvZwXKQmoEYga2","id":"atmpt_1srOrx2ZWZBpBUvZwXKQmoEYga2","timestamp":"2025-02-16T21:38:21.977Z","msg":{"eventId":"unique-identifier","eventType":"user.signup","payload":{"email":"test@example.com","type":"user.created","username":"test_user"},"channels":["project_123","group_2"],"id":"msg_1srOrx2ZWZBpBUvZwXKQmoEYga2","timestamp":"2025-02-16T21:38:21.977Z","tags":["project_1337"]}}],"iterator":"iterator","prevIterator":"-iterator","done":true}`; const ListResponseOperationalWebhookEndpointOut = `{"data":[{"id":"ep_1srOrx2ZWZBpBUvZwXKQmoEYga2","description":"string","rateLimit":0,"uid":"unique-identifier","url":"https://example.com/webhook/","disabled":false,"filterTypes":["message.attempt.failing"],"createdAt":"2019-08-24T14:15:22Z","updatedAt":"2019-08-24T14:15:22Z","metadata":{"property1":"string","property2":"string"}}],"iterator":"iterator","prevIterator":"-iterator","done":true}`; const ListResponseBackgroundTaskOut = `{"data":[{"data":{},"id":"qtask_1srOrx2ZWZBpBUvZwXKQmoEYga2","status":"running","task":"endpoint.replay"}],"iterator":"iterator","prevIterator":"-iterator","done":true}`; const EventTypeImportOpenApiOut = `{"data":{"modified":["user.signup"],"to_modify":[{"name":"user.signup","description":"string","schemas":{},"deprecated":true,"featureFlag":"cool-new-feature","groupName":"user"}]}}`; const ReplayOut = `{"id":"qtask_1srOrx2ZWZBpBUvZwXKQmoEYga2","status":"running","task":"endpoint.replay"}`; const EndpointOut = `{"description":"string","rateLimit":0,"uid":"unique-identifier","url":"http://example.com","version":1,"disabled":true,"filterTypes":["user.signup"],"channels":["project_1337"],"secret":"whsec_C2FVsBQIhrscChlQIMV+b5sSYspob7oD","metadata":{"property1":"string","property2":"string"}}`; const ValidationErrorOut = `{"detail":[{"loc":["string"],"msg":"string","type":"string"}]}`; const IngestSourceOutCron = `{"type":"cron","config":{"schedule":"hello","payload":"world"},"id":"src_2yZwUhtgs5Ai8T9yRQJXA","uid":"unique-identifier","name":"string","ingestUrl":"http://example.com","createdAt":"2019-08-24T14:15:22Z","updatedAt":"2019-08-24T14:15:22Z"}`; const IngestSourceOutGeneric = `{"type":"generic-webhook","config":{},"id":"src_2yZwUhtgs5Ai8T9yRQJXA","uid":"unique-identifier","name":"string","ingestUrl":"http://example.com","createdAt":"2019-08-24T14:15:22Z","updatedAt":"2019-08-24T14:15:22Z"}`; const mockServer = mockttp.getLocal(); describe("mockttp tests", () => { beforeEach(async () => await mockServer.start(0)); afterEach(async () => await mockServer.stop()); test("test Date in query param", async () => { const endpointMock = await mockServer .forGet(/\/api\/v1\/app\/app1\/attempt\/endpoint\/ep1.*/) .thenReply(200, ListResponseMessageAttemptOut); const svx = new Svix("token", { serverUrl: mockServer.url }); await svx.messageAttempt.listByEndpoint("app1", "ep1", { before: new Date(1739741901977), }); const requests = await endpointMock.getSeenRequests(); expect(requests.length).toBe(1); expect(requests[0].url.endsWith("before=2025-02-16T21%3A38%3A21.977Z")).toBe(true); }); test("test Date request body", async () => { const endpointMock = await mockServer .forPost("/api/v1/app/app1/endpoint/ep1/replay-missing") .thenReply(200, ReplayOut); const svx = new Svix("token", { serverUrl: mockServer.url }); await svx.endpoint.replayMissing("app1", "ep1", { since: new Date(1739741901977) }); const requests = await endpointMock.getSeenRequests(); expect(requests.length).toBe(1); expect(await requests[0].body.getText()).toBe(`{"since":"2025-02-16T21:38:21.977Z"}`); }); test("test Date response body", async () => { const endpointMock = await mockServer .forGet("/api/v1/app/app1/attempt/endpoint/ep1") .thenReply(200, ListResponseMessageAttemptOut); const svx = new Svix("token", { serverUrl: mockServer.url }); const res = await svx.messageAttempt.listByEndpoint("app1", "ep1"); const requests = await endpointMock.getSeenRequests(); expect(requests.length).toBe(1); expect(res.data[0].timestamp?.getTime()).toBe(1739741901977); }); test("test listResponseOut deserializes correctly", async () => { const endpointMock = await mockServer .forGet("/api/v1/app/app1/attempt/endpoint/ep1") .thenReply(200, `{"data":[],"iterator":null,"prevIterator":null,"done":true}`); const svx = new Svix("token", { serverUrl: mockServer.url }); const res = await svx.messageAttempt.listByEndpoint("app1", "ep1"); const requests = await endpointMock.getSeenRequests(); expect(requests.length).toBe(1); expect(res.iterator).toBeNull(); expect(res.prevIterator).toBeNull(); }); test("test enum as query param", async () => { const endpointMock = await mockServer .forGet(/\/api\/v1\/app.*/) .thenReply(200, ListResponseApplicationOut); const svx = new Svix("token", { serverUrl: mockServer.url }); await svx.application.list({ order: Ordering.Ascending }); const requests = await endpointMock.getSeenRequests(); expect(requests.length).toBe(1); expect(requests[0].url.endsWith("order=ascending")).toBe(true); }); test("test list as query param", async () => { const endpointMock = await mockServer .forGet(/\/api\/v1\/app\/app1\/msg.*/) .thenReply(200, ListResponseMessageOut); const svx = new Svix("token", { serverUrl: mockServer.url }); await svx.message.list("app1", { eventTypes: ["val8", "val1", "val5"] }); const requests = await endpointMock.getSeenRequests(); expect(requests.length).toBe(1); expect( requests[0].url.endsWith("/api/v1/app/app1/msg?event_types=val8%2Cval1%2Cval5") ).toBe(true); }); test("test header param sent", async () => { const endpointMock = await mockServer .forPost(/\/api\/v1\/app.*/) .thenReply(200, ApplicationOut); const svx = new Svix("token", { serverUrl: mockServer.url }); await svx.application.create({ name: "test" }, { idempotencyKey: "random-key" }); const requests = await endpointMock.getSeenRequests(); expect(requests.length).toBe(1); expect(requests[0].headers["idempotency-key"]).toBe("random-key"); }); test("test retry for status => 500", async () => { const endpointMock = await mockServer .forGet("/api/v1/app") .thenReply(500, `{"code":"500","detail":"asd"}`); const svx = new Svix("token", { serverUrl: mockServer.url }); await expect(svx.application.list()).rejects.toThrow(ApiException); const requests = await endpointMock.getSeenRequests(); expect(requests.length).toBe(3); // same svix-req-id for each retry const req_id = requests[0].headers["svix-req-id"]; expect(req_id).toBeDefined(); expect(typeof req_id).toBe("string"); for (let i = 0; i < requests.length; i++) { expect(requests[i].headers["svix-req-id"]).toBe(req_id); if (i == 0) { // first request does not set svix-retry-count expect(requests[i].headers["svix-retry-count"]).toBeUndefined(); } else { expect(requests[i].headers["svix-retry-count"]).toBe(i.toString()); } } }); test("no body in response does not return anything", async () => { const endpointMock = await mockServer .forDelete("/api/v1/app/app1") .thenReply(204, ""); const svx = new Svix("token", { serverUrl: mockServer.url }); await svx.application.delete("app1"); const requests = await endpointMock.getSeenRequests(); expect(requests.length).toBe(1); }); test("422 returns validation error", async () => { const endpointMock = await mockServer .forGet("/api/v1/app") .thenReply(422, ValidationErrorOut); const svx = new Svix("token", { serverUrl: mockServer.url }); await expect(svx.application.list()).rejects.toThrow(ApiException<ValidationError>); try { await svx.application.list(); } catch (e) { expect(e).toHaveProperty("code", 422); expect(e).toHaveProperty("body", { detail: [{ loc: ["string"], msg: "string", type: "string" }], }); } const requests = await endpointMock.getSeenRequests(); expect(requests.length).toBe(2); }); test("400 returns ApiException", async () => { const endpointMock = await mockServer .forGet("/api/v1/app") .thenReply(400, `{"code":"400","detail":"text"}`); const svx = new Svix("token", { serverUrl: mockServer.url }); await expect(svx.application.list()).rejects.toThrow(ApiException<HttpErrorOut>); try { await svx.application.list(); } catch (e) { expect(e).toHaveProperty("code", 400); expect(e).toHaveProperty("body", { code: "400", detail: "text" }); } const requests = await endpointMock.getSeenRequests(); expect(requests.length).toBe(2); }); test("sub-resource works", async () => { const endpointMock = await mockServer .forGet("/api/v1/operational-webhook/endpoint") .thenReply(200, ListResponseOperationalWebhookEndpointOut); const svx = new Svix("token", { serverUrl: mockServer.url }); await svx.operationalWebhookEndpoint.list(); const requests = await endpointMock.getSeenRequests(); expect(requests.length).toBe(1); }); test("integer enum serialization", async () => { const endpointMock = await mockServer .forGet("/api/v1/app/app1/attempt/endpoint/endp1") .thenReply(200, ListResponseMessageAttemptOut); const svx = new Svix("token", { serverUrl: mockServer.url }); const res = await svx.messageAttempt.listByEndpoint("app1", "endp1"); expect(res.data[0].triggerType).toBe(MessageAttemptTriggerType.Scheduled); const requests = await endpointMock.getSeenRequests(); expect(requests.length).toBe(1); }); test("string enum de/serialization", async () => { const endpointMock = await mockServer .forGet("/api/v1/background-task") .thenReply(200, ListResponseBackgroundTaskOut); const svx = new Svix("token", { serverUrl: mockServer.url }); const res = await svx.backgroundTask.list({ task: BackgroundTaskType.EndpointReplay, }); expect(res.data[0].task).toBe(BackgroundTaskType.EndpointReplay); const requests = await endpointMock.getSeenRequests(); expect(requests.length).toBe(1); expect(requests[0].url.endsWith("api/v1/background-task?task=endpoint.replay")).toBe( true ); }); test("non-camelCase field name", async () => { const endpointMock = await mockServer .forPost("/api/v1/event-type/import/openapi") .thenReply(200, EventTypeImportOpenApiOut); const svx = new Svix("token", { serverUrl: mockServer.url }); const res = await svx.eventType.importOpenapi({ dryRun: true }); expect(res.data.toModify).toStrictEqual([ { name: "user.signup", description: "string", schemas: {}, deprecated: true, featureFlag: "cool-new-feature", groupName: "user", }, ]); const requests = await endpointMock.getSeenRequests(); expect(requests.length).toBe(1); }); test("patch request body", async () => { const endpointMock = await mockServer .forPatch("/api/v1/app/app1/endpoint/endp1") .thenReply(200, EndpointOut); const svx = new Svix("token", { serverUrl: mockServer.url }); await svx.endpoint.patch("app1", "endp1", { filterTypes: ["ty1"] }); await svx.endpoint.patch("app1", "endp1", { filterTypes: null }); await svx.endpoint.patch("app1", "endp1", { description: "text" }); const requests = await endpointMock.getSeenRequests(); expect(requests.length).toBe(3); // nullable field is sent expect(await requests[0].body.getText()).toBe(`{"filterTypes":["ty1"]}`); // nullable field is null expect(await requests[1].body.getText()).toBe(`{"filterTypes":null}`); // undefined field is omitted expect(await requests[2].body.getText()).toBe(`{"description":"text"}`); }); test("arbitrary json object body", async () => { const endpointMock = await mockServer .forPost("/api/v1/app/app1/msg") .thenReply(200, EndpointOut); const svx = new Svix("token", { serverUrl: mockServer.url }); await svx.message.create("app1", { eventType: "asd", payload: { key1: "val", list: ["val1"], obj: { key: "val2" } }, }); const requests = await endpointMock.getSeenRequests(); expect(requests.length).toBe(1); expect(await requests[0].body.getText()).toBe( `{"eventType":"asd","payload":{"key1":"val","list":["val1"],"obj":{"key":"val2"}}}` ); }); test("token/user-agent is sent", async () => { const endpointMock = await mockServer .forDelete("/api/v1/app/app1") .thenReply(200, EndpointOut); const svx = new Svix("token.eu", { serverUrl: mockServer.url }); await svx.application.delete("app1"); const requests = await endpointMock.getSeenRequests(); expect(requests.length).toBe(1); expect(requests[0].headers["authorization"]).toBe("Bearer token.eu"); expect(requests[0].headers["user-agent"]).toBe(`svix-libs/${LIB_VERSION}/javascript`); }); test("MessageAttemptOut without msg", async () => { const RES = `{ "data": [ { "url": "https://example.com/webhook/", "response": "{}", "responseStatusCode": 200, "responseDurationMs": 0, "status": 0, "triggerType": 0, "msgId": "msg_1srOrx2ZWZBpBUvZwXKQmoEYga2", "endpointId": "ep_1srOrx2ZWZBpBUvZwXKQmoEYga2", "id": "atmpt_1srOrx2ZWZBpBUvZwXKQmoEYga2", "timestamp": "2019-08-24T14:15:22Z" } ], "iterator": "iterator", "prevIterator": "-iterator", "done": true }`; const endpointMock = await mockServer .forGet("/api/v1/app/app/attempt/endpoint/edp") .thenReply(200, RES); const svx = new Svix("token.eu", { serverUrl: mockServer.url }); await svx.messageAttempt.listByEndpoint("app", "edp"); const requests = await endpointMock.getSeenRequests(); expect(requests.length).toBe(1); }); test("octothorpe in url query", async () => { const endpointMock = await mockServer .forGet("/api/v1/app/app1/msg") .thenReply(200, ListResponseMessageOut); const svx = new Svix("token.eu", { serverUrl: mockServer.url }); await svx.message.list("app1", { tag: "test#test" }); const requests = await endpointMock.getSeenRequests(); expect(requests.length).toBe(1); expect(requests[0].url.endsWith("api/v1/app/app1/msg?tag=test%23test")).toBe(true); }); test("content-type application/json is sent on request with body", async () => { const endpointMock = await mockServer .forPost(/\/api\/v1\/app.*/) .thenReply(200, ApplicationOut); const svx = new Svix("token", { serverUrl: mockServer.url }); await svx.application.create({ name: "test" }); const requests = await endpointMock.getSeenRequests(); expect(requests.length).toBe(1); expect(requests[0].headers["content-type"]).toBe("application/json"); }); test("content type not sent on request without body", async () => { const endpointMock = await mockServer .forGet("/api/v1/app") .thenReply(200, ListResponseApplicationOut); const svx = new Svix("token", { serverUrl: mockServer.url }); await svx.application.list(); const requests = await endpointMock.getSeenRequests(); expect(requests.length).toBe(1); expect(requests[0].headers["content-type"]).toBeUndefined(); }); test("struct enum with extra fields", async () => { const endpointMock = await mockServer .forPost("/ingest/api/v1/source") .thenReply(200, IngestSourceOutCron); const svx = new Svix("token", { serverUrl: mockServer.url }); const res = await svx.ingest.source.create({ name: "crontab -r", type: "cron", config: { schedule: "hello", payload: "world" }, }); expect(res.type).toBe("cron"); // this will smart cast res.config if (res.type === "cron") { expect(res.config.schedule).toBe("hello"); expect(res.config.payload).toBe("world"); } const requests = await endpointMock.getSeenRequests(); expect(await requests[0].body.getText()).toBe( '{"type":"cron","config":{"payload":"world","schedule":"hello"},"name":"crontab -r"}' ); }); test("struct enum with no extra fields", async () => { const endpointMock = await mockServer .forPost("/ingest/api/v1/source") .thenReply(200, IngestSourceOutGeneric); const svx = new Svix("token", { serverUrl: mockServer.url }); const res = await svx.ingest.source.create({ name: "generic over <T>", type: "generic-webhook", }); expect(res.type).toBe("generic-webhook"); const requests = await endpointMock.getSeenRequests(); // empty config object should be sent expect(await requests[0].body.getText()).toBe( '{"type":"generic-webhook","config":{},"name":"generic over <T>"}' ); }); });