From 8089fc6185c0c6bebab4a9d93830173af0366468 Mon Sep 17 00:00:00 2001 From: CahidArda Date: Mon, 21 Oct 2024 17:00:54 +0300 Subject: [PATCH 1/5] feat: add resend api --- src/client/api/email.test.ts | 23 +++++++++++++++++++++++ src/client/api/email.ts | 13 +++++++++++++ src/client/api/index.ts | 7 +++++++ src/client/client.ts | 23 +++++++++++++++++++++-- src/client/llm/utils.ts | 2 +- src/client/utils.ts | 32 ++++++++++++++++++++++++++------ 6 files changed, 91 insertions(+), 9 deletions(-) create mode 100644 src/client/api/email.test.ts create mode 100644 src/client/api/email.ts create mode 100644 src/client/api/index.ts diff --git a/src/client/api/email.test.ts b/src/client/api/email.test.ts new file mode 100644 index 0000000..09bc4d6 --- /dev/null +++ b/src/client/api/email.test.ts @@ -0,0 +1,23 @@ +/* eslint-disable @typescript-eslint/no-non-null-assertion */ +import { describe, test } from "bun:test"; +import { Client } from "../client"; +import { resend } from "./email"; + +describe("email", () => { + const client = new Client({ token: process.env.QSTASH_TOKEN! }); + + test("should use resend", async () => { + await client.publishJSON({ + api: { + name: "email", + provider: resend({ token: process.env.RESEND_TOKEN! }), + }, + body: { + from: "Acme ", + to: ["delivered@resend.dev"], + subject: "hello world", + html: "

it works!

", + }, + }); + }); +}); diff --git a/src/client/api/email.ts b/src/client/api/email.ts new file mode 100644 index 0000000..eb5a3a4 --- /dev/null +++ b/src/client/api/email.ts @@ -0,0 +1,13 @@ +export type EmailProviderReturnType = { + owner: "resend"; + baseUrl: "https://api.resend.com/emails"; + token: string; +}; + +export const resend = ({ token }: { token: string }): EmailProviderReturnType => { + return { + owner: "resend", + baseUrl: "https://api.resend.com/emails", + token, + }; +}; diff --git a/src/client/api/index.ts b/src/client/api/index.ts new file mode 100644 index 0000000..9d1d13c --- /dev/null +++ b/src/client/api/index.ts @@ -0,0 +1,7 @@ +import type { PublishRequest } from "../client"; + +export const appendAPIOptions = (request: PublishRequest, headers: Headers) => { + if (request.api?.name === "email") { + headers.set("Authorization", request.api.provider.token); + } +}; diff --git a/src/client/client.ts b/src/client/client.ts index a05785e..32a7555 100644 --- a/src/client/client.ts +++ b/src/client/client.ts @@ -1,3 +1,5 @@ +import { appendAPIOptions } from "./api"; +import type { EmailProviderReturnType } from "./api/email"; import { DLQ } from "./dlq"; import type { Duration } from "./duration"; import { HttpClient, type Requester, type RetryConfig } from "./http"; @@ -179,7 +181,7 @@ export type PublishRequest = { callback?: string; } | { - url?: string; + url?: never; urlGroup?: never; /** * The api endpoint the request should be sent to. @@ -196,15 +198,30 @@ export type PublishRequest = { * * @default undefined */ + topic?: never; callback: string; + } + | { + url?: never; + urlGroup?: never; + /** + * The api endpoint the request should be sent to. + */ + api: { + name: "email"; + provider: EmailProviderReturnType; + }; topic?: never; + callback?: string; } | { url?: never; urlGroup?: never; - api: never; + api?: never; /** * Deprecated. The topic the message should be sent to. Same as urlGroup + * + * @deprecated */ topic?: string; /** @@ -370,6 +387,8 @@ export class Client { ensureCallbackPresent(request); //If needed, this allows users to directly pass their requests to any open-ai compatible 3rd party llm directly from sdk. appendLLMOptionsIfNeeded(request, headers, this.http); + // append api options + appendAPIOptions(request, headers); // @ts-expect-error it's just internal const response = await this.publish({ diff --git a/src/client/llm/utils.ts b/src/client/llm/utils.ts index c0fe706..5a78f3f 100644 --- a/src/client/llm/utils.ts +++ b/src/client/llm/utils.ts @@ -8,7 +8,7 @@ export function appendLLMOptionsIfNeeded< // eslint-disable-next-line @typescript-eslint/no-unnecessary-type-parameters TRequest extends PublishRequest = PublishRequest, >(request: TRequest, headers: Headers, http: Requester) { - if (!request.api) return; + if (request.api?.name !== "llm") return; const provider = request.api.provider; const analytics = request.api.analytics; diff --git a/src/client/utils.ts b/src/client/utils.ts index b096df5..57689ab 100644 --- a/src/client/utils.ts +++ b/src/client/utils.ts @@ -1,4 +1,5 @@ import type { PublishRequest } from "./client"; +import { QstashError } from "./error"; const isIgnoredHeader = (header: string) => { const lowerCaseHeader = header.toLowerCase(); @@ -79,7 +80,18 @@ export function processHeaders(request: PublishRequest) { export function getRequestPath( request: Pick ): string { - return request.url ?? request.urlGroup ?? request.topic ?? `api/${request.api?.name}`; + // eslint-disable-next-line @typescript-eslint/no-deprecated + const nonApiPath = request.url ?? request.urlGroup ?? request.topic; + if (nonApiPath) return nonApiPath; + + // return llm api + if (request.api?.name === "llm") return `api/${request.api.name}`; + // return email api + if (request.api?.name === "email") { + return request.api.provider.baseUrl; + } + + throw new QstashError(`Failed to infer request path for ${JSON.stringify(request)}`); } const NANOID_CHARS = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789-_"; @@ -110,11 +122,19 @@ export function decodeBase64(base64: string) { return new TextDecoder().decode(intArray); } catch (error) { // this error should never happen essentially. It's only a failsafe - console.warn( - `Upstash Qstash: Failed while decoding base64 "${base64}".` + - ` Decoding with atob and returning it instead. ${error}` - ); - return atob(base64); + try { + const result = atob(base64); + console.warn( + `Upstash QStash: Failed while decoding base64 "${base64}".` + + ` Decoding with atob and returning it instead. ${error}` + ); + return result; + } catch (error) { + console.warn( + `Upstash QStash: Failed to decode base64 "${base64}" with atob. Returning it as it is. ${error}` + ); + return base64; + } } } From 156b5e256e9a53bb23e3d7219335da76b86cb89c Mon Sep 17 00:00:00 2001 From: CahidArda Date: Mon, 21 Oct 2024 17:44:03 +0300 Subject: [PATCH 2/5] fix: mock resend in email test --- src/client/api/email.test.ts | 48 +++++++++++++++++++++++++++--------- src/client/api/index.ts | 8 +----- src/client/api/utils.ts | 8 ++++++ src/client/client.ts | 2 +- src/index.ts | 1 + 5 files changed, 48 insertions(+), 19 deletions(-) create mode 100644 src/client/api/utils.ts diff --git a/src/client/api/email.test.ts b/src/client/api/email.test.ts index 09bc4d6..9bd17a6 100644 --- a/src/client/api/email.test.ts +++ b/src/client/api/email.test.ts @@ -1,22 +1,48 @@ -/* eslint-disable @typescript-eslint/no-non-null-assertion */ import { describe, test } from "bun:test"; import { Client } from "../client"; import { resend } from "./email"; +import { MOCK_QSTASH_SERVER_URL, mockQStashServer } from "../workflow/test-utils"; +import { nanoid } from "../utils"; describe("email", () => { - const client = new Client({ token: process.env.QSTASH_TOKEN! }); + const qstashToken = nanoid(); + const resendToken = nanoid(); + const client = new Client({ baseUrl: MOCK_QSTASH_SERVER_URL, token: qstashToken }); test("should use resend", async () => { - await client.publishJSON({ - api: { - name: "email", - provider: resend({ token: process.env.RESEND_TOKEN! }), + await mockQStashServer({ + execute: async () => { + await client.publishJSON({ + api: { + name: "email", + provider: resend({ token: resendToken }), + }, + body: { + from: "Acme ", + to: ["delivered@resend.dev"], + subject: "hello world", + html: "

it works!

", + }, + }); }, - body: { - from: "Acme ", - to: ["delivered@resend.dev"], - subject: "hello world", - html: "

it works!

", + responseFields: { + body: { messageId: "msgId" }, + status: 200, + }, + receivesRequest: { + method: "POST", + token: qstashToken, + url: "http://localhost:8080/v2/publish/https://api.resend.com/emails", + body: { + from: "Acme ", + to: ["delivered@resend.dev"], + subject: "hello world", + html: "

it works!

", + }, + headers: { + authorization: `Bearer ${qstashToken}`, + "upstash-forward-authorization": resendToken, + }, }, }); }); diff --git a/src/client/api/index.ts b/src/client/api/index.ts index 9d1d13c..0f81b62 100644 --- a/src/client/api/index.ts +++ b/src/client/api/index.ts @@ -1,7 +1 @@ -import type { PublishRequest } from "../client"; - -export const appendAPIOptions = (request: PublishRequest, headers: Headers) => { - if (request.api?.name === "email") { - headers.set("Authorization", request.api.provider.token); - } -}; +export { resend } from "./email"; diff --git a/src/client/api/utils.ts b/src/client/api/utils.ts new file mode 100644 index 0000000..ebaed9f --- /dev/null +++ b/src/client/api/utils.ts @@ -0,0 +1,8 @@ +import type { PublishRequest } from "../client"; + +export const appendAPIOptions = (request: PublishRequest, headers: Headers) => { + if (request.api?.name === "email") { + headers.set("Authorization", request.api.provider.token); + request.method = request.method ?? "POST"; + } +}; diff --git a/src/client/client.ts b/src/client/client.ts index 32a7555..74f2515 100644 --- a/src/client/client.ts +++ b/src/client/client.ts @@ -1,4 +1,4 @@ -import { appendAPIOptions } from "./api"; +import { appendAPIOptions } from "./api/utils"; import type { EmailProviderReturnType } from "./api/email"; import { DLQ } from "./dlq"; import type { Duration } from "./duration"; diff --git a/src/index.ts b/src/index.ts index cef6de9..15c00f9 100644 --- a/src/index.ts +++ b/src/index.ts @@ -10,3 +10,4 @@ export { decodeBase64 } from "./client/utils"; export * from "./client/llm/chat"; export * from "./client/llm/types"; export * from "./client/llm/providers"; +export * from "./client/api"; From 3c40574e5ddbe7e2e57c9a1218ce5045616732a8 Mon Sep 17 00:00:00 2001 From: CahidArda Date: Mon, 21 Oct 2024 18:15:09 +0300 Subject: [PATCH 3/5] fix: tests --- src/client/client.ts | 4 ++-- src/client/llm/utils.ts | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/src/client/client.ts b/src/client/client.ts index 74f2515..25d5599 100644 --- a/src/client/client.ts +++ b/src/client/client.ts @@ -181,7 +181,7 @@ export type PublishRequest = { callback?: string; } | { - url?: never; + url?: string; urlGroup?: never; /** * The api endpoint the request should be sent to. @@ -191,6 +191,7 @@ export type PublishRequest = { provider?: ProviderReturnType; analytics?: { name: "helicone"; token: string }; }; + topic?: never; /** * Use a callback url to forward the response of your destination server to your callback url. * @@ -198,7 +199,6 @@ export type PublishRequest = { * * @default undefined */ - topic?: never; callback: string; } | { diff --git a/src/client/llm/utils.ts b/src/client/llm/utils.ts index 5a78f3f..7406c9d 100644 --- a/src/client/llm/utils.ts +++ b/src/client/llm/utils.ts @@ -8,7 +8,7 @@ export function appendLLMOptionsIfNeeded< // eslint-disable-next-line @typescript-eslint/no-unnecessary-type-parameters TRequest extends PublishRequest = PublishRequest, >(request: TRequest, headers: Headers, http: Requester) { - if (request.api?.name !== "llm") return; + if (request.api?.name === "email" || !request.api) return; const provider = request.api.provider; const analytics = request.api.analytics; From a2fd543342ac91573fee4f8ad6e7dbc641b2836c Mon Sep 17 00:00:00 2001 From: CahidArda Date: Thu, 24 Oct 2024 13:48:12 +0300 Subject: [PATCH 4/5] fix: add appendAPIOptions to enqueue and batch --- src/client/client.ts | 5 +++++ src/client/queue.ts | 3 +++ 2 files changed, 8 insertions(+) diff --git a/src/client/client.ts b/src/client/client.ts index 25d5599..d926928 100644 --- a/src/client/client.ts +++ b/src/client/client.ts @@ -453,6 +453,11 @@ export class Client { // eslint-disable-next-line @typescript-eslint/ban-ts-comment //@ts-ignore this is required otherwise message header prevent ts to compile appendLLMOptionsIfNeeded(message, message.headers, this.http); + + // eslint-disable-next-line @typescript-eslint/ban-ts-comment + //@ts-ignore this is required otherwise message header prevent ts to compile + appendAPIOptions(message, message.headers); + (message.headers as Headers).set("Content-Type", "application/json"); } diff --git a/src/client/queue.ts b/src/client/queue.ts index 8a3c705..11f9e32 100644 --- a/src/client/queue.ts +++ b/src/client/queue.ts @@ -1,3 +1,4 @@ +import { appendAPIOptions } from "./api/utils"; import type { PublishRequest, PublishResponse } from "./client"; import type { Requester } from "./http"; import { appendLLMOptionsIfNeeded, ensureCallbackPresent } from "./llm/utils"; @@ -140,6 +141,8 @@ export class Queue { // If needed, this allows users to directly pass their requests to any open-ai compatible 3rd party llm directly from sdk. appendLLMOptionsIfNeeded(request, headers, this.http); + appendAPIOptions(request, headers); + const response = await this.enqueue({ ...request, body: JSON.stringify(request.body), From 1b2a7da160cfe1872fa0d1218659bcd38f6238e2 Mon Sep 17 00:00:00 2001 From: CahidArda Date: Mon, 28 Oct 2024 12:15:09 +0300 Subject: [PATCH 5/5] fix: add batch emails --- src/client/api/email.test.ts | 54 ++++++++++++++++++++++++++++++++++++ src/client/api/email.ts | 12 ++++++-- 2 files changed, 63 insertions(+), 3 deletions(-) diff --git a/src/client/api/email.test.ts b/src/client/api/email.test.ts index 9bd17a6..9e1b3a8 100644 --- a/src/client/api/email.test.ts +++ b/src/client/api/email.test.ts @@ -46,4 +46,58 @@ describe("email", () => { }, }); }); + + test("should use resend with batch", async () => { + await mockQStashServer({ + execute: async () => { + await client.publishJSON({ + api: { + name: "email", + provider: resend({ token: resendToken, batch: true }), + }, + body: [ + { + from: "Acme ", + to: ["foo@gmail.com"], + subject: "hello world", + html: "

it works!

", + }, + { + from: "Acme ", + to: ["bar@outlook.com"], + subject: "world hello", + html: "

it works!

", + }, + ], + }); + }, + responseFields: { + body: { messageId: "msgId" }, + status: 200, + }, + receivesRequest: { + method: "POST", + token: qstashToken, + url: "http://localhost:8080/v2/publish/https://api.resend.com/emails/batch", + body: [ + { + from: "Acme ", + to: ["foo@gmail.com"], + subject: "hello world", + html: "

it works!

", + }, + { + from: "Acme ", + to: ["bar@outlook.com"], + subject: "world hello", + html: "

it works!

", + }, + ], + headers: { + authorization: `Bearer ${qstashToken}`, + "upstash-forward-authorization": resendToken, + }, + }, + }); + }); }); diff --git a/src/client/api/email.ts b/src/client/api/email.ts index eb5a3a4..5de04d1 100644 --- a/src/client/api/email.ts +++ b/src/client/api/email.ts @@ -1,13 +1,19 @@ export type EmailProviderReturnType = { owner: "resend"; - baseUrl: "https://api.resend.com/emails"; + baseUrl: "https://api.resend.com/emails" | "https://api.resend.com/emails/batch"; token: string; }; -export const resend = ({ token }: { token: string }): EmailProviderReturnType => { +export const resend = ({ + token, + batch = false, +}: { + token: string; + batch?: boolean; +}): EmailProviderReturnType => { return { owner: "resend", - baseUrl: "https://api.resend.com/emails", + baseUrl: `https://api.resend.com/emails${batch ? "/batch" : ""}`, token, }; };