From b0902922c79a655fa723cd4d17e0af0d7b02b899 Mon Sep 17 00:00:00 2001 From: Wes Souza Date: Tue, 26 Nov 2024 21:05:24 -0500 Subject: [PATCH] feat(actions): add support for accept: 'search' for GET requests --- .../actions-blog/src/actions/index.ts | 2 +- .../src/actions/runtime/virtual/server.ts | 41 ++++++++++++++++--- packages/astro/templates/actions.mjs | 15 +++++-- packages/astro/test/actions.test.js | 14 +++++++ .../fixtures/actions/src/actions/index.ts | 8 ++++ .../astro/test/types/define-action-accept.ts | 18 ++++++++ 6 files changed, 87 insertions(+), 11 deletions(-) diff --git a/packages/astro/e2e/fixtures/actions-blog/src/actions/index.ts b/packages/astro/e2e/fixtures/actions-blog/src/actions/index.ts index a5437ad49a3c3..c644bc41652cd 100644 --- a/packages/astro/e2e/fixtures/actions-blog/src/actions/index.ts +++ b/packages/astro/e2e/fixtures/actions-blog/src/actions/index.ts @@ -57,7 +57,7 @@ export const server = { }), }, sum: defineAction({ - accept: "form", + accept: "search", input: z.object({ a: z.number(), b: z.number(), diff --git a/packages/astro/src/actions/runtime/virtual/server.ts b/packages/astro/src/actions/runtime/virtual/server.ts index 01e8fbd86bfee..84c8b063b036e 100644 --- a/packages/astro/src/actions/runtime/virtual/server.ts +++ b/packages/astro/src/actions/runtime/virtual/server.ts @@ -11,7 +11,7 @@ import { ActionError, ActionInputError, type SafeResult, callSafely } from './sh export * from './shared.js'; -export type ActionAccept = 'form' | 'json'; +export type ActionAccept = 'form' | 'json' | 'search'; export type ActionHandler = TInputSchema extends z.ZodType ? (input: z.infer, context: ActionAPIContext) => MaybePromise @@ -62,7 +62,9 @@ export function defineAction< const serverHandler = accept === 'form' ? getFormServerHandler(handler, inputSchema) - : getJsonServerHandler(handler, inputSchema); + : accept === 'search' + ? getSearchServerHandler(handler, inputSchema) + : getJsonServerHandler(handler, inputSchema); async function safeServerHandler(this: ActionAPIContext, unparsedInput: unknown) { // The ActionAPIContext should always contain the `params` property @@ -132,9 +134,36 @@ function getJsonServerHandler( }; } +function getSearchServerHandler( + handler: ActionHandler, + inputSchema?: TInputSchema, +) { + return async (unparsedInput: unknown, context: ActionAPIContext): Promise> => { + if (!(unparsedInput instanceof URLSearchParams)) { + throw new ActionError({ + code: 'UNSUPPORTED_MEDIA_TYPE', + message: 'This action only accepts URLSearchParams.', + }); + } + + if (!inputSchema) return await handler(unparsedInput, context); + + const baseSchema = unwrapBaseObjectSchema(inputSchema, unparsedInput); + const parsed = await inputSchema.safeParseAsync( + baseSchema instanceof z.ZodObject + ? formDataToObject(unparsedInput, baseSchema) + : unparsedInput, + ); + if (!parsed.success) { + throw new ActionInputError(parsed.error.issues); + } + return await handler(parsed.data, context); + }; +} + /** Transform form data to an object based on a Zod schema. */ export function formDataToObject( - formData: FormData, + formData: FormData | URLSearchParams, schema: T, ): Record { const obj: Record = @@ -171,7 +200,7 @@ export function formDataToObject( function handleFormDataGetAll( key: string, - formData: FormData, + formData: FormData | URLSearchParams, validator: z.ZodArray, ) { const entries = Array.from(formData.getAll(key)); @@ -186,7 +215,7 @@ function handleFormDataGetAll( function handleFormDataGet( key: string, - formData: FormData, + formData: FormData | URLSearchParams, validator: unknown, baseValidator: unknown, ) { @@ -197,7 +226,7 @@ function handleFormDataGet( return validator instanceof z.ZodNumber ? Number(value) : value; } -function unwrapBaseObjectSchema(schema: z.ZodType, unparsedInput: FormData) { +function unwrapBaseObjectSchema(schema: z.ZodType, unparsedInput: FormData | URLSearchParams) { while (schema instanceof z.ZodEffects || schema instanceof z.ZodPipeline) { if (schema instanceof z.ZodEffects) { schema = schema._def.schema; diff --git a/packages/astro/templates/actions.mjs b/packages/astro/templates/actions.mjs index dad29aa374cfa..252942b3f986c 100644 --- a/packages/astro/templates/actions.mjs +++ b/packages/astro/templates/actions.mjs @@ -76,8 +76,15 @@ async function handleAction(param, path, context) { // When running client-side, make a fetch request to the action path. const headers = new Headers(); headers.set('Accept', 'application/json'); - let body = param; - if (!(body instanceof FormData)) { + let method = 'POST' + let search = '' + let body + if (param instanceof URLSearchParams) { + method = 'GET' + search = `?${param.toString()}` + } else if (param instanceof FormData) { + body = param + } else { try { body = JSON.stringify(param); } catch (e) { @@ -92,8 +99,8 @@ async function handleAction(param, path, context) { headers.set('Content-Length', '0'); } } - const rawResult = await fetch(`${import.meta.env.BASE_URL.replace(/\/$/, '')}/_actions/${path}`, { - method: 'POST', + const rawResult = await fetch(`${import.meta.env.BASE_URL.replace(/\/$/, '')}/_actions/${path}${search}`, { + method, body, headers, }); diff --git a/packages/astro/test/actions.test.js b/packages/astro/test/actions.test.js index eed4d87344154..24162a29542a2 100644 --- a/packages/astro/test/actions.test.js +++ b/packages/astro/test/actions.test.js @@ -235,6 +235,20 @@ describe('Astro Actions', () => { assert.equal($('#user').text(), 'Houston'); }); + it('Supports URLSearchParams', async () => { + const req = new Request('http://example.com/_actions/getUserProperty?property=name', { + method: 'GET', + }); + + const res = await app.render(req); + + assert.equal(res.ok, true); + assert.equal(res.headers.get('Content-Type'), 'application/json+devalue'); + + const data = devalue.parse(await res.text()); + assert.deepEqual(data, 'Houston'); + }); + it('Respects custom errors', async () => { const req = new Request('http://example.com/user-or-throw?_astroAction=getUserOrThrow', { method: 'POST', diff --git a/packages/astro/test/fixtures/actions/src/actions/index.ts b/packages/astro/test/fixtures/actions/src/actions/index.ts index 78cc39620baa6..1191f9d6dbfc3 100644 --- a/packages/astro/test/fixtures/actions/src/actions/index.ts +++ b/packages/astro/test/fixtures/actions/src/actions/index.ts @@ -75,6 +75,14 @@ export const server = { return locals.user; }, }), + getUserProperty: defineAction({ + accept: 'search', + input: z + .object({ property: z.string() }), + handler: async (input, { locals }) => { + return locals.user[input.property]; + }, + }), validatePassword: defineAction({ accept: 'form', input: z diff --git a/packages/astro/test/types/define-action-accept.ts b/packages/astro/test/types/define-action-accept.ts index c9a9ca315a34a..7f81511827b9a 100644 --- a/packages/astro/test/types/define-action-accept.ts +++ b/packages/astro/test/types/define-action-accept.ts @@ -18,6 +18,7 @@ describe('defineAction accept', () => { expectTypeOf(jsonAction).parameter(0).toBeAny(); expectTypeOf(jsonAction).parameter(0).not.toEqualTypeOf(); }); + it('accepts type `FormData` when input is omitted with accept form', async () => { const action = defineAction({ accept: 'form', @@ -26,6 +27,14 @@ describe('defineAction accept', () => { expectTypeOf(action).parameter(0).toEqualTypeOf(); }); + it('accepts type `URLSearchParams` when input is omitted with accept form', async () => { + const action = defineAction({ + accept: 'search', + handler: () => {}, + }); + expectTypeOf(action).parameter(0).toEqualTypeOf(); + }); + it('accept type safe values for input with accept json', async () => { const action = defineAction({ input: z.object({ name: z.string() }), @@ -42,4 +51,13 @@ describe('defineAction accept', () => { }); expectTypeOf(action).parameter(0).toEqualTypeOf(); }); + + it('accepts type `URLSearchParams` for all inputs with accept form', async () => { + const action = defineAction({ + accept: 'form', + input: z.object({ name: z.string() }), + handler: () => {}, + }); + expectTypeOf(action).parameter(0).toEqualTypeOf(); + }); });