From 170722c05d0e7b43724008367290e1483a4be1d3 Mon Sep 17 00:00:00 2001 From: Ben Holmes Date: Tue, 21 May 2024 10:55:23 -0400 Subject: [PATCH] Actions: support React 19 `useActionState()` with progressive enhancement (#11074) * feat(ex): Like with useActionState * feat: useActionState progressive enhancement! * feat: getActionState utility * chore: revert actions-blog fixture experimentation * fix: add back actions.ts export * feat(test): Like with use action state test * fix: stub form state client-side to avoid hydration error * fix: bad .safe chaining * fix: update actionState for client call * fix: correctly resume form state client side * refactor: unify and document reactServerActionResult * feat(test): useActionState assertions * feat(docs): explain my mess * refactor: add experimental_ prefix * refactor: move all react internals to integration * chore: remove unused getIslandProps * chore: remove unused imports * chore: undo format changes * refactor: get actionResult from middleware directly * refactor: remove bad result type * fix: like button disabled timeout * chore: changeset * refactor: remove request cloning --- .changeset/gentle-windows-enjoy.md | 46 ++++++++ packages/astro/e2e/actions-react-19.test.js | 29 ++++- .../actions-blog/src/actions/index.ts | 2 +- .../actions-react-19/src/actions/index.ts | 24 ++++- .../actions-react-19/src/components/Like.tsx | 17 +++ .../src/pages/blog/[...slug].astro | 7 +- packages/astro/src/@types/astro.ts | 2 + .../astro/src/actions/runtime/middleware.ts | 2 + packages/astro/src/actions/utils.ts | 2 +- packages/astro/src/core/render-context.ts | 9 +- packages/astro/templates/actions.mjs | 12 +++ packages/integrations/react/client.js | 11 ++ packages/integrations/react/package.json | 1 + packages/integrations/react/server.js | 55 ++++++++++ packages/integrations/react/src/actions.ts | 101 ++++++++++++++++++ 15 files changed, 312 insertions(+), 8 deletions(-) create mode 100644 .changeset/gentle-windows-enjoy.md create mode 100644 packages/integrations/react/src/actions.ts diff --git a/.changeset/gentle-windows-enjoy.md b/.changeset/gentle-windows-enjoy.md new file mode 100644 index 000000000000..ac9b1ccd92e9 --- /dev/null +++ b/.changeset/gentle-windows-enjoy.md @@ -0,0 +1,46 @@ +--- +"@astrojs/react": minor +"astro": minor +--- + +Add support for [the React 19 `useActionState()` hook](https://react.dev/reference/react/useActionState) when using Astro Actions. This introduces progressive enhancement when calling an Action with the `withState()` utility. + +This example calls a `like` action that accepts a `postId` and returns the number of likes. Pass this action to the `experimental_withState()` function to apply progressive enhancement info, and apply to `useActionState()` to track the result: + +```tsx +import { actions } from 'astro:actions'; +import { experimental_withState } from '@astrojs/react/actions'; + +export function Like({ postId }: { postId: string }) { + const [state, action, pending] = useActionState( + experimental_withState(actions.like), + 0, // initial likes + ); + + return ( +
+ + +
+ ); +} +``` + +You can also access the state stored by `useActionState()` from your action `handler`. Call `experimental_getActionState()` with the API context, and optionally apply a type to the result: + +```ts +import { defineAction, z } from 'astro:actions'; +import { experimental_getActionState } from '@astrojs/react/actions'; + +export const server = { + like: defineAction({ + input: z.object({ + postId: z.string(), + }), + handler: async ({ postId }, ctx) => { + const currentLikes = experimental_getActionState(ctx); + // write to database + return currentLikes + 1; + } + }) +} diff --git a/packages/astro/e2e/actions-react-19.test.js b/packages/astro/e2e/actions-react-19.test.js index bfb57d8f9096..5ce72a419acb 100644 --- a/packages/astro/e2e/actions-react-19.test.js +++ b/packages/astro/e2e/actions-react-19.test.js @@ -9,6 +9,11 @@ test.beforeAll(async ({ astro }) => { devServer = await astro.startDevServer(); }); +test.afterEach(async ({ astro }) => { + // Force database reset between tests + await astro.editFile('./db/seed.ts', (original) => original); +}); + test.afterAll(async () => { await devServer.stop(); }); @@ -21,6 +26,7 @@ test.describe('Astro Actions - React 19', () => { await expect(likeButton).toBeVisible(); await likeButton.click(); await expect(likeButton, 'like button should be disabled when pending').toBeDisabled(); + await expect(likeButton).not.toBeDisabled({ timeout: 5000 }); }); test('Like action - server progressive enhancement', async ({ page, astro }) => { @@ -30,7 +36,26 @@ test.describe('Astro Actions - React 19', () => { await expect(likeButton, 'like button starts with 10 likes').toContainText('10'); await likeButton.click(); - // May contain "12" after the client button test. - await expect(likeButton, 'like button increments').toContainText(/11|12/); + await expect(likeButton, 'like button increments').toContainText('11'); + }); + + test('Like action - client useActionState', async ({ page, astro }) => { + await page.goto(astro.resolveUrl('/blog/first-post/')); + + const likeButton = page.getByLabel('likes-action-client'); + await expect(likeButton).toBeVisible(); + await likeButton.click(); + + await expect(likeButton, 'like button increments').toContainText('11'); + }); + + test('Like action - server useActionState progressive enhancement', async ({ page, astro }) => { + await page.goto(astro.resolveUrl('/blog/first-post/')); + + const likeButton = page.getByLabel('likes-action-server'); + await expect(likeButton, 'like button starts with 10 likes').toContainText('10'); + await likeButton.click(); + + await expect(likeButton, 'like button increments').toContainText('11'); }); }); 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 036cc087fdca..0588f626c8a8 100644 --- a/packages/astro/e2e/fixtures/actions-blog/src/actions/index.ts +++ b/packages/astro/e2e/fixtures/actions-blog/src/actions/index.ts @@ -7,7 +7,7 @@ export const server = { like: defineAction({ input: z.object({ postId: z.string() }), handler: async ({ postId }) => { - await new Promise((r) => setTimeout(r, 200)); + await new Promise((r) => setTimeout(r, 1000)); const { likes } = await db .update(Likes) diff --git a/packages/astro/e2e/fixtures/actions-react-19/src/actions/index.ts b/packages/astro/e2e/fixtures/actions-react-19/src/actions/index.ts index 70ecd04aa2cf..9cb867603add 100644 --- a/packages/astro/e2e/fixtures/actions-react-19/src/actions/index.ts +++ b/packages/astro/e2e/fixtures/actions-react-19/src/actions/index.ts @@ -1,5 +1,6 @@ import { db, Likes, eq, sql } from 'astro:db'; -import { defineAction, z } from 'astro:actions'; +import { defineAction, getApiContext, z } from 'astro:actions'; +import { experimental_getActionState } from '@astrojs/react/actions'; export const server = { blog: { @@ -21,5 +22,26 @@ export const server = { return likes; }, }), + likeWithActionState: defineAction({ + accept: 'form', + input: z.object({ postId: z.string() }), + handler: async ({ postId }) => { + await new Promise((r) => setTimeout(r, 200)); + + const context = getApiContext(); + const state = await experimental_getActionState(context); + + const { likes } = await db + .update(Likes) + .set({ + likes: state + 1, + }) + .where(eq(Likes.postId, postId)) + .returning() + .get(); + + return likes; + }, + }), }, }; diff --git a/packages/astro/e2e/fixtures/actions-react-19/src/components/Like.tsx b/packages/astro/e2e/fixtures/actions-react-19/src/components/Like.tsx index 3928e9eda4a8..0d2dde00916e 100644 --- a/packages/astro/e2e/fixtures/actions-react-19/src/components/Like.tsx +++ b/packages/astro/e2e/fixtures/actions-react-19/src/components/Like.tsx @@ -1,5 +1,7 @@ import { actions } from 'astro:actions'; +import { useActionState } from 'react'; import { useFormStatus } from 'react-dom'; +import { experimental_withState } from '@astrojs/react/actions'; export function Like({ postId, label, likes }: { postId: string; label: string; likes: number }) { return ( @@ -10,6 +12,21 @@ export function Like({ postId, label, likes }: { postId: string; label: string; ); } + +export function LikeWithActionState({ postId, label, likes: initial }: { postId: string; label: string; likes: number }) { + const [likes, action] = useActionState( + experimental_withState(actions.blog.likeWithActionState), + 10, + ); + + return ( +
+ +