Build HTML Forms in Astro Pages
In SSR mode, Astro pages can both display and handle forms. In this recipe, you’ll use a standard HTML form to submit data to the server. Your frontmatter script will handle the data on the server, sending no JavaScript to the client.
Prerequisites
Section titled Prerequisites- A project with SSR (output: 'server') enabled
Recipe
Section titled Recipe- 
Create or identify a .astropage which will contain your form and your handling code. For example, you could add a registration page:src/pages/register.astro ------<h1>Register</h1>
- 
Add a <form>tag with some inputs to the page. Each input should have anameattribute that describes the value of that input.Be sure to include a <button>or<input type="submit">element to submit the form.src/pages/register.astro ------<h1>Register</h1><form><label>Username:<input type="text" name="username" /></label><label>Email:<input type="email" name="email" /></label><label>Password:<input type="password" name="password" /></label><button>Submit</button></form>
- 
Use validation attributes to provide basic client-side validation that works even if JavaScript is disabled. In this example, - requiredprevents form submission until the field is filled.
- minlengthsets a minimum required length for the input text.
- type="email"also introduces validation that will only accept a valid email format.
 src/pages/register.astro ------<h1>Register</h1><form><label>Username:<input type="text" name="username" required /></label><label>Email:<input type="email" name="email" required /></label><label>Password:<input type="password" name="password" required minlength="6" /></label><button>Submit</button></form>
- 
The form submission will cause the browser to request the page again. Change the form’s data transfer methodtoPOSTto send the form data as part of theRequestbody, rather than as URL parameters.src/pages/register.astro ------<h1>Register</h1><form method="POST"><label>Username:<input type="text" name="username" required /></label><label>Email:<input type="email" name="email" required /></label><label>Password:<input type="password" name="password" required minlength="6" /></label><button>Submit</button></form>
- 
Check for the POSTmethod in the frontmatter and access the form data usingAstro.request.formData(). Wrap this in atry ... catchblock to handle cases when thePOSTrequest wasn’t sent by a form and theformDatais invalid.src/pages/register.astro ---if (Astro.request.method === "POST") {try {const data = await Astro.request.formData();const name = data.get("username");const email = data.get("email");const password = data.get("password");// Do something with the data} catch (error) {if (error instanceof Error) {console.error(error.message);}}}---<h1>Register</h1><form method="POST"><label>Username:<input type="text" name="username" required /></label><label>Email:<input type="email" name="email" required /></label><label>Password:<input type="password" name="password" required minlength="6" /></label><button>Submit</button></form>
- 
Validate the form data on the server. This should include the same validation done on the client to prevent malicious submissions to your endpoint and to support the rare legacy browser that doesn’t have form validation. It can also include validation that can’t be done on the client. For example, this example checks if the email is already in the database. Error messages can be sent back to the client by storing them in an errorsobject and accessing it in the template.src/pages/register.astro ---import { isRegistered, registerUser } from "../../data/users"import { isValidEmail } from "../../utils/isValidEmail";const errors = { username: "", email: "", password: "" };if (Astro.request.method === "POST") {try {const data = await Astro.request.formData();const name = data.get("username");const email = data.get("email");const password = data.get("password");if (typeof name !== "string" || name.length < 1) {errors.username += "Please enter a username. ";}if (typeof email !== "string" || !isValidEmail(email)) {errors.email += "Email is not valid. ";} else if (await isRegistered(email)) {errors.email += "Email is already registered. ";}if (typeof password !== "string" || password.length < 6) {errors.password += "Password must be at least 6 characters. ";}const hasErrors = Object.values(errors).some(msg => msg)if (!hasErrors) {await registerUser({name, email, password});return Astro.redirect("/login");}} catch (error) {if (error instanceof Error) {console.error(error.message);}}}---<h1>Register</h1><form method="POST"><label>Username:<input type="text" name="username" /></label>{errors.username && <p>{errors.username}</p>}<label>Email:<input type="email" name="email" required /></label>{errors.email && <p>{errors.email}</p>}<label>Password:<input type="password" name="password" required minlength="6" /></label>{errors.password && <p>{errors.password}</p>}<button>Register</button></form>
More recipes
- 
	
	Share State Between IslandsLearn how to share state across framework components with Nano Stores. 
- 
	
	Add an RSS feedAdd an RSS feed to your Astro site to let users subscribe to your content. 
- 
	
	Installing a Vite or Rollup pluginLearn how you can import YAML data by adding a Rollup plugin to your project. 
- 
	
	Build a custom image componentLearn how to build a custom image component that supports media queries using the getImage function 
- 
	
	Build Forms With API RoutesLearn how to use JavaScript to send form submissions to an API Route 
- 
	
	Build HTML Forms in Astro PagesLearn how to build HTML forms and handle submissions in your frontmatter 
- 
	
	Use Bun with AstroLearn how to use Bun with your Astro site. 
- 
	
	Call endpoints from the serverLearn how to call endpoints from the server in Astro. 
- 
	
	Verify a CaptchaLearn how to create an API route and fetch it from the client. 
- 
	
	Build your Astro Site with DockerLearn how to build your Astro site using Docker. 
- 
	
	Dynamically Import ImagesLearn how to dynamically import images using Vite's import.meta.glob function 
- 
	
	Add icons to external linksLearn how to install a rehype plugin to add icons to external links in your Markdown files 
- 
	
	Add i18n featuresUse dynamic routing and content collections to add internationalization support to your Astro site. 
- 
	
	Add Last Modified TimeBuild a remark plugin to add the last modified time to your Markdown and MDX. 
- 
	
	Add Reading TimeBuild a remark plugin to add reading time to your Markdown or MDX files. 
- 
	
	Share State Between Astro ComponentsLearn how to share state across Astro components with Nano Stores. 
- 
	
	Using streaming to improve page performanceLearn how to use streaming to improve page performance. 
- 
	
	Style Rendered Markdown with Tailwind TypographyLearn how to use @tailwind/typography to style your rendered Markdown