Get the FREE Ultimate OpenClaw Setup Guide →

server-actions

npx machina-cli add skill davepoon/buildwithclaude/server-actions --openclaw
Files (1)
SKILL.md
7.5 KB

Next.js Server Actions

Overview

Server Actions are asynchronous functions that execute on the server. They can be called from Client and Server Components for data mutations, form submissions, and other server-side operations.

Defining Server Actions

In Server Components

Use the 'use server' directive inside an async function:

// app/page.tsx (Server Component)
export default function Page() {
  async function createPost(formData: FormData) {
    'use server'
    const title = formData.get('title') as string
    await db.post.create({ data: { title } })
  }

  return (
    <form action={createPost}>
      <input name="title" />
      <button type="submit">Create</button>
    </form>
  )
}

In Separate Files

Mark the entire file with 'use server':

// app/actions.ts
'use server'

export async function createPost(formData: FormData) {
  const title = formData.get('title') as string
  await db.post.create({ data: { title } })
}

export async function deletePost(id: string) {
  await db.post.delete({ where: { id } })
}

Form Handling

Basic Form

// app/actions.ts
'use server'

export async function submitContact(formData: FormData) {
  const name = formData.get('name') as string
  const email = formData.get('email') as string
  const message = formData.get('message') as string

  await db.contact.create({
    data: { name, email, message }
  })
}

// app/contact/page.tsx
import { submitContact } from '@/app/actions'

export default function ContactPage() {
  return (
    <form action={submitContact}>
      <input name="name" required />
      <input name="email" type="email" required />
      <textarea name="message" required />
      <button type="submit">Send</button>
    </form>
  )
}

With Validation (Zod)

// app/actions.ts
'use server'

import { z } from 'zod'

const schema = z.object({
  email: z.string().email(),
  password: z.string().min(8),
})

export async function signup(formData: FormData) {
  const parsed = schema.safeParse({
    email: formData.get('email'),
    password: formData.get('password'),
  })

  if (!parsed.success) {
    return { error: parsed.error.flatten() }
  }

  await createUser(parsed.data)
  return { success: true }
}

useFormState Hook

Handle form state and errors:

// app/signup/page.tsx
'use client'

import { useFormState } from 'react-dom'
import { signup } from '@/app/actions'

const initialState = {
  error: null,
  success: false,
}

export default function SignupPage() {
  const [state, formAction] = useFormState(signup, initialState)

  return (
    <form action={formAction}>
      <input name="email" type="email" />
      <input name="password" type="password" />
      {state.error && (
        <p className="text-red-500">{state.error}</p>
      )}
      <button type="submit">Sign Up</button>
    </form>
  )
}

// app/actions.ts
'use server'

export async function signup(prevState: any, formData: FormData) {
  const email = formData.get('email') as string

  if (!email.includes('@')) {
    return { error: 'Invalid email', success: false }
  }

  await createUser({ email })
  return { error: null, success: true }
}

useFormStatus Hook

Show loading states during submission:

// components/submit-button.tsx
'use client'

import { useFormStatus } from 'react-dom'

export function SubmitButton() {
  const { pending } = useFormStatus()

  return (
    <button type="submit" disabled={pending}>
      {pending ? 'Submitting...' : 'Submit'}
    </button>
  )
}

// Usage in form
import { SubmitButton } from '@/components/submit-button'

export default function Form() {
  return (
    <form action={submitAction}>
      <input name="title" />
      <SubmitButton />
    </form>
  )
}

Revalidation

revalidatePath

Revalidate a specific path:

'use server'

import { revalidatePath } from 'next/cache'

export async function createPost(formData: FormData) {
  await db.post.create({ data: { ... } })

  // Revalidate the posts list page
  revalidatePath('/posts')

  // Revalidate a dynamic route
  revalidatePath('/posts/[slug]', 'page')

  // Revalidate all paths under /posts
  revalidatePath('/posts', 'layout')
}

revalidateTag

Revalidate by cache tag:

// Fetching with tags
const posts = await fetch('https://api.example.com/posts', {
  next: { tags: ['posts'] }
})

// Server Action
'use server'

import { revalidateTag } from 'next/cache'

export async function createPost(formData: FormData) {
  await db.post.create({ data: { ... } })
  revalidateTag('posts')
}

Redirects After Actions

'use server'

import { redirect } from 'next/navigation'

export async function createPost(formData: FormData) {
  const post = await db.post.create({ data: { ... } })

  // Redirect to the new post
  redirect(`/posts/${post.slug}`)
}

Optimistic Updates

Update UI immediately while action completes:

'use client'

import { useOptimistic } from 'react'
import { addTodo } from '@/app/actions'

export function TodoList({ todos }: { todos: Todo[] }) {
  const [optimisticTodos, addOptimisticTodo] = useOptimistic(
    todos,
    (state, newTodo: string) => [
      ...state,
      { id: 'temp', title: newTodo, completed: false }
    ]
  )

  async function handleSubmit(formData: FormData) {
    const title = formData.get('title') as string
    addOptimisticTodo(title) // Update UI immediately
    await addTodo(formData)  // Server action
  }

  return (
    <>
      <form action={handleSubmit}>
        <input name="title" />
        <button>Add</button>
      </form>
      <ul>
        {optimisticTodos.map(todo => (
          <li key={todo.id}>{todo.title}</li>
        ))}
      </ul>
    </>
  )
}

Non-Form Usage

Call Server Actions programmatically:

'use client'

import { deletePost } from '@/app/actions'

export function DeleteButton({ id }: { id: string }) {
  return (
    <button onClick={() => deletePost(id)}>
      Delete
    </button>
  )
}

Error Handling

'use server'

export async function createPost(formData: FormData) {
  try {
    await db.post.create({ data: { ... } })
    return { success: true }
  } catch (error) {
    if (error instanceof PrismaClientKnownRequestError) {
      if (error.code === 'P2002') {
        return { error: 'A post with this title already exists' }
      }
    }
    return { error: 'Failed to create post' }
  }
}

Security Considerations

  1. Always validate input - Never trust client data
  2. Check authentication - Verify user is authorized
  3. Use CSRF protection - Built-in with Server Actions
  4. Sanitize output - Prevent XSS attacks
'use server'

import { auth } from '@/lib/auth'

export async function deletePost(id: string) {
  const session = await auth()

  if (!session) {
    throw new Error('Unauthorized')
  }

  const post = await db.post.findUnique({ where: { id } })

  if (post.authorId !== session.user.id) {
    throw new Error('Forbidden')
  }

  await db.post.delete({ where: { id } })
}

Resources

For detailed patterns, see:

  • references/form-handling.md - Advanced form patterns
  • references/revalidation.md - Cache revalidation strategies
  • examples/mutation-patterns.md - Complete mutation examples

Source

git clone https://github.com/davepoon/buildwithclaude/blob/main/plugins/nextjs-expert/skills/server-actions/SKILL.mdView on GitHub

Overview

Server Actions are asynchronous server-side functions callable from Client and Server Components to perform data mutations and form submissions in the Next.js App Router. They rely on the 'use server' directive and can live inside server components or in separate files. This skill explains how to define, invoke, and validate actions, wire them to forms, and leverage form-state hooks like useFormState and useFormStatus for responsive UI and cache revalidation via revalidatePath and revalidateTag.

How This Skill Works

Define actions with 'use server' inside an async function in a Server Component, or mark the whole file with 'use server'. Connect a form's action attribute to the server action so the browser sends FormData to the server, where you perform DB mutations and return results (e.g., success or errors). Client-side hooks like useFormState and useFormStatus help manage UI state and loading indicators during submissions.

When to Use It

  • When building forms that submit data via the Next.js App Router using server actions
  • When performing create, update, or delete mutations on the server from a client action
  • When you need server-side validation (e.g., with Zod) before mutating data
  • When you want real-time UI feedback using useFormState and useFormStatus during submissions
  • When you need cache revalidation control using revalidatePath or revalidateTag after mutations

Quick Start

  1. Step 1: Create a server action using 'use server' in a Server Component or a dedicated actions.ts file
  2. Step 2: Attach the action to a form via the form's action attribute, e.g., <form action={submitAction}>
  3. Step 3: In a client component, use useFormState and useFormStatus to reflect submission state and handle results

Best Practices

  • Define actions either inside a Server Component with 'use server' or in a dedicated file marked with 'use server'
  • Validate inputs on the server (e.g., using Zod) and return structured results (success or error data)
  • Keep action return shapes simple (e.g., { success: true, error?: string }) for straightforward client handling
  • Coordinate with useFormState and useFormStatus to synchronize UI with server mutations
  • Use revalidatePath and revalidateTag to refresh stale data after mutations

Example Use Cases

  • Define createPost and deletePost in app/actions.ts with 'use server' and call them from a server component's form
  • Basic form submission pattern: submitContact action handling name, email, and message fields
  • Validation flow: signup action using Zod to validate email and password before mutation
  • useFormState integration: manage error/success state and pass a formAction to the form
  • useFormStatus integration: display loading state on a Submit button while a server action runs

Frequently Asked Questions

Add this skill to your agents
Sponsor this space

Reach thousands of developers