Forms and Mutations

Forms enable you to create and update data in web applications. Next.js provides a powerful way to handle form submissions and data mutations using Server Actions.

How Server Actions Work

With Server Actions, you don’t need to manually create API endpoints. Instead, you define asynchronous server functions that can be called directly from your components.

🎥 Watch: Learn more about forms and mutations with the App Router → YouTube (10 minutes) ↗.

Server Actions can be defined in Server Components or called from Client Components. Defining the action in a Server Component allows the form to function without JavaScript, providing progressive enhancement.

Enable Server Actions in your next.config.js file:

next.config.js

module.exports = {
  experimental: {
    serverActions: true,
  },
}

Good to know:

  • Forms calling Server Actions from Server Components can function without JavaScript.
  • Forms calling Server Actions from Client Components will queue submissions if JavaScript isn’t loaded yet, prioritizing client hydration.
  • Server Actions inherit the runtime from the page or layout they are used on.
  • Server Actions work with fully static routes (including revalidating data with ISR).

Revalidating Cached Data

Server Actions integrate deeply with the Next.js caching and revalidation architecture. When a form is submitted, the Server Action can update cached data and revalidate any cache keys that should change.

Rather than being limited to a single form per route like traditional applications, Server Actions enable having multiple actions per route. Further, the browser does not need to refresh on form submission. In a single network roundtrip, Next.js can return both the updated UI and the refreshed data.

View the examples below for revalidating data from Server Actions.

Examples

Server-only Forms

To create a server-only form, define the Server Action in a Server Component. The action can either be defined inline with the "use server" directive at the top of the function, or in a separate file with the directive at the top of the file.

app/page.tsx

export default function Page() {
  async function create(formData: FormData) {
    'use server'
 
    // mutate data
    // revalidate cache
  }
 
  return <form action={create}>...</form>
}

Good to know: <form action={create}> takes the FormData ↗ data type. In the example above, the FormData submitted via the HTML form is accessible in the server action create.

Revalidating Data

Server Actions allow you to invalidate the Next.js Cache on demand. You can invalidate an entire route segment with revalidatePath:

app/actions.ts

'use server'
 
import { revalidatePath } from 'next/cache'
 
export default async function submit() {
  await submitForm()
  revalidatePath('/')
}

Or invalidate a specific data fetch with a cache tag using revalidateTag:

app/actions.ts

'use server'
 
import { revalidateTag } from 'next/cache'
 
export default async function submit() {
  await addPost()
  revalidateTag('posts')
}

Redirecting

If you would like to redirect the user to a different route after the completion of a Server Action, you can use redirect and any absolute or relative URL:

app/actions.ts

'use server'
 
import { redirect } from 'next/navigation'
import { revalidateTag } from 'next/cache'
 
export default async function submit() {
  const id = await addPost()
  revalidateTag('posts') // Update cached posts
  redirect(`/post/${id}`) // Navigate to new route
}

Form Validation

We recommend using HTML validation like required and type="email" for basic form validation.

For more advanced server-side validation, use a schema validation library like zod ↗ to validate the structure of the parsed form data:

app/actions.ts

import { z } from 'zod'
 
const schema = z.object({
  // ...
})
 
export default async function submit(formData: FormData) {
  const parsed = schema.parse({
    id: formData.get('id'),
  })
  // ...
}

Displaying Loading State

Use the useFormStatus hook to show a loading state when a form is submitting on the server. The useFormStatus hook can only be used as a child of a form element using a Server Action.

For example, the following submit button:

app/submit-button.tsx

'use client'
 
import { experimental_useFormStatus as useFormStatus } from 'react-dom'
 
export function SubmitButton() {
  const { pending } = useFormStatus()
 
  return (
    <button type="submit" aria-disabled={pending}>
      Add
    </button>
  )
}

<SubmitButton /> can then be used in a form with a Server Action:

app/page.tsx

import { SubmitButton } from '@/app/submit-button'
 
export default async function Home() {
  return (
    <form action={...}>
      <input type="text" name="field-name" />
      <SubmitButton />
    </form>
  )
}

Error Handling

Server Actions can also return serializable objects ↗. For example, your Server Action might handle errors from creating a new item:

app/actions.ts

'use server'
 
export async function createTodo(prevState: any, formData: FormData) {
  try {
    await createItem(formData.get('todo'))
    return revalidatePath('/')
  } catch (e) {
    return { message: 'Failed to create' }
  }
}

Then, from a Client Component, you can read this value and display an error message.

app/add-form.tsx

'use client'
 
import { experimental_useFormState as useFormState } from 'react-dom'
import { experimental_useFormStatus as useFormStatus } from 'react-dom'
import { createTodo } from '@/app/actions'
 
const initialState = {
  message: null,
}
 
function SubmitButton() {
  const { pending } = useFormStatus()
 
  return (
    <button type="submit" aria-disabled={pending}>
      Add
    </button>
  )
}
 
export function AddForm() {
  const [state, formAction] = useFormState(createTodo, initialState)
 
  return (
    <form action={formAction}>
      <label htmlFor="todo">Enter Task</label>
      <input type="text" id="todo" name="todo" required />
      <SubmitButton />
      <p aria-live="polite" className="sr-only">
        {state?.message}
      </p>
    </form>
  )
}

Optimistic Updates

Use useOptimistic to optimistically update the UI before the Server Action finishes, rather than waiting for the response:

app/page.tsx

'use client'
 
import { experimental_useOptimistic as useOptimistic } from 'react'
import { send } from './actions'
 
type Message = {
  message: string
}
 
export function Thread({ messages }: { messages: Message[] }) {
  const [optimisticMessages, addOptimisticMessage] = useOptimistic<Message[]>(
    messages,
    (state: Message[], newMessage: string) => [
      ...state,
      { message: newMessage },
    ]
  )
 
  return (
    <div>
      {optimisticMessages.map((m, k) => (
        <div key={k}>{m.message}</div>
      ))}
      <form
        action={async (formData: FormData) => {
          const message = formData.get('message')
          addOptimisticMessage(message)
          await send(message)
        }}
      >
        <input type="text" name="message" />
        <button type="submit">Send</button>
      </form>
    </div>
  )
}

Setting Cookies

You can set cookies inside a Server Action using the cookies function:

app/actions.ts

'use server'
 
import { cookies } from 'next/headers'
 
export async function create() {
  const cart = await createCart()
  cookies().set('cartId', cart.id)
}

Reading Cookies

You can read cookies inside a Server Action using the cookies function:

app/actions.ts

'use server'
 
import { cookies } from 'next/headers'
 
export async function read() {
  const auth = cookies().get('authorization')?.value
  // ...
}

Deleting Cookies

You can delete cookies inside a Server Action using the cookies function:

app/actions.ts

'use server'
 
import { cookies } from 'next/headers'
 
export async function delete() {
  cookies().delete('name')
  // ...
}

See additional examples for deleting cookies from Server Actions.

Last updated on