Server Actions — Complete Guide
Server Actions — Complete Guide: free step-by-step lesson with examples, common mistakes, and interview tips — part of Next.js Tutorial on Toolliyo Academy.
On this page
Next.js Tutorial (LearnHub) · Lesson 17 of 100
Server Actions
Beginner → Intermediate → Advanced → Professional
Beginner · 1 — Foundations · ~12 min read · Module 2: Layouts & Styling
Introduction
This lesson is part of the beginner section. We explain Server Actions slowly, with examples you can copy and run. If something is unclear, read it twice — that is how everyone learns. Server Actions are async functions that run on the server, triggered by forms or startTransition. Mark with "use server" — they can mutate data and revalidate cache without a separate API route. Enrolling in a course or marking a lesson complete is a POST. Server Actions let LearnHub forms stay simple — no manual fetch URL and CSRF boilerplate for every button.
Server Actions appears in almost every LearnHub page you will build. Once it clicks, data fetching and auth become much easier.
When will you use this?
You use layouts and styling in every page you build from your first screen to production.
- Course catalogs, lesson sidebars, and instructor dashboards all use layouts and shared navigation.
- When a student opens a lesson page, nested layouts keep the header and progress bar consistent.
Real-world: Naukri-style job board
The Recruitment team building Naukri-style job board uses Server Actions to enroll a student or mark a lesson complete with a form post — no separate API route. job seekers and recruiters never see the TypeScript files — they just get a fast, reliable job search, apply forms, and employer dashboards.
Production-style code
'use server';
import { revalidatePath } from 'next/cache';
export async function enrollInCourse(formData: FormData) {
const courseId = formData.get('courseId') as string;
await fetch(`${process.env.API_URL}/enroll`, {
method: 'POST',
body: JSON.stringify({ courseId }),
headers: { 'Content-Type': 'application/json' }
});
revalidatePath('/dashboard');
}
What happens in production: In Naukri-style job board, getting Server Actions right means job seekers and recruiters trust the job search, apply forms, and employer dashboards every day.
Lesson example (start here)
Copy this smaller example first. Once it works, compare it with the real-world code above.
'use server';
import { revalidatePath } from 'next/cache';
export async function enrollInCourse(formData: FormData) {
const courseId = formData.get('courseId');
if (!courseId || typeof courseId !== 'string') {
return { error: 'Invalid course' };
}
await fetch(`${process.env.API_URL}/enroll`, {
method: 'POST',
body: JSON.stringify({ courseId }),
headers: { 'Content-Type': 'application/json' }
});
revalidatePath('/dashboard');
return { success: true };
}
Line-by-line walkthrough
| Code | What it means |
|---|---|
'use server'; | Server Actions directive — function runs only on the server. |
import { revalidatePath } from 'next/cache'; | Imports a module so you can use its exports in this file. |
export async function enrollInCourse(formData: FormData) { | Exported async function — often a Server Action or API handler. |
const courseId = formData.get('courseId'); | Part of the Server Actions example — read it together with the lines before and after. |
if (!courseId || typeof courseId !== 'string') { | Part of the Server Actions example — read it together with the lines before and after. |
return { error: 'Invalid course' }; | Part of the Server Actions example — read it together with the lines before and after. |
} | Closes a block started by { above. |
await fetch(`${process.env.API_URL}/enroll`, { | Fetches data over HTTP — runs on the server in Server Components or in Server Actions. |
method: 'POST', | Part of the Server Actions example — read it together with the lines before and after. |
body: JSON.stringify({ courseId }), | Part of the Server Actions example — read it together with the lines before and after. |
headers: { 'Content-Type': 'application/json' } | Part of the Server Actions example — read it together with the lines before and after. |
}); | Closes a block started by { above. |
revalidatePath('/dashboard'); | Tells Next.js to refresh cached data after a mutation. |
return { success: true }; | Part of the Server Actions example — read it together with the lines before and after. |
How it works (big picture)
- Form action={enrollInCourse} posts FormData to the server function.
- revalidatePath refreshes dashboard cache.
- Return values can update UI via useFormState in Client Components.
Do this on your computer
- Create actions/enroll.ts with "use server"
- Wire a form with action={enrollInCourse} and hidden courseId
- Submit and confirm dashboard updates
- Add basic error return for missing courseId
- Read the real-world section and name which part of LearnHub uses this topic.
- Run the example locally with npm run dev and confirm the same behavior.
- Change one value in the example (route, text, or course id) and predict what will happen before you save.
Experiments — try changing this
- Change a string or route in the example and save — watch the browser update.
- Break the code on purpose (remove a bracket), read the error overlay, then fix it.
- Change the API URL or course id and see how the page data changes.
Remember
"use server" = server-only mutation. Works with
Common questions
Server Actions vs API routes?
Actions for form mutations from your UI; API routes for external clients and webhooks.
How long should I spend on Server Actions?
Until you can explain it in your own words and run the example without looking at the answer. Beginners often need 30–60 minutes per new concept; setup lessons may take one afternoon.
What if I get stuck on Server Actions?
Re-read the line-by-line walkthrough, check the terminal and browser overlay for errors, and compare your code character-by-character with the example. Search the exact error text — someone else had it too.
Where is Server Actions used in real jobs?
See the real-world section above — the same pattern appears in LMS, e-commerce, SaaS, and dashboards. Interviewers ask you to explain it using one concrete example.