Fastify Authentication
Easily add authentication focused on security, speed, and DX to your Fastify server.
Overview
Clerk is the easiest way to add authentication and user management to your Fastify server. This guide will walk you through the necessary steps to install and use Clerk in a basic Fastify-powered application.
After following this guide, you should have a working Fastify app with public and private routes, authenticated using the `clerkPlugin` and `getAuth` helpers.
Looking for a quickstart? We created clerk-fastify-starter to show you how to add Clerk to your project.
Before you start
You need to create a Clerk Application in your Clerk Dashboard before you can set up Clerk React. For more information, check out our Setup your application guide.
Creating a new Fastify app
Start by creating a new Fastify application - for more details, follow the official Fastify "Getting-Started" guide.
npm i fastify
Installing Clerk
Install the `@clerk/fastify` package. This package can be used with ESM and CJS imports.
1npm install @clerk/fastify
Setting Environment Keys
Two environment variables for the Clerk SDK can be set. Below is an example of your .env
file. To get the respective keys, go to the API Keys page in the Clerk dashboard.
.env.local1
You can also pass these keys into the Clerk plugin directly.
Most applications rely on environment variables, so your app's infrastructure for reading them should already be set up. For demonstration purposes, in this example, we will use `dontenv`:
1npm install dotenv2npm install -D types/dotenv
Adding `clerkPlugin`
Adding Clerk is pretty straightforward. You only need to import and register the `clerkPlugin` helper, as shown below.
1import * as dotenv from "dotenv";23dotenv.config();45import Fastify from "fastify";6import { clerkClient, clerkPlugin, getAuth } from "@clerk/fastify";78const fastify = Fastify({ logger: true });910/**11* Register the Clerk plugin globally.12* By default, Clerk will initialise using the API keys from the environment if found.13*14* If you prefer to pass the keys to the plugin explicitly, see `src/using-runtime-keys.ts`15* If you prefer to register the plugin for specific routes only, see `src/authenticating-specific-routes.ts`16*/17fastify.register(clerkPlugin);1819const start = async () => {20try {21await fastify.listen({ port: 3000 });22} catch (err) {23fastify.log.error(err);24process.exit(1);25}26};2728start();29
Accessing the auth state using `getAuth`
Next, add a new route and call the `getAuth`helper as shown below.
1import * as dotenv from "dotenv";23dotenv.config();45import Fastify from "fastify";6import { clerkClient, clerkPlugin, getAuth } from "@clerk/fastify";78const fastify = Fastify({ logger: true });910fastify.register(clerkPlugin);1112fastify.get("/", async (req, reply) => {13/**14* Access the auth state for this request.15* In this example, we use the userId to load the whole User object16* from the Clerk servers17*/18const { userId } = getAuth(req);19const user = userId ? await clerkClient.users.getUser(userId) : null;20return { user };21});2223const start = async () => {24try {25await fastify.listen({ port: 3000 });26} catch (err) {27fastify.log.error(err);28process.exit(1);29}30};3132start();33
Requiring authentication
The easiest way to require authentication before accessing a protected route is to check wether a session exists within the route handler, as shown below.
1fastify.get("/protected", async (request, reply) => {2const { userId } = getAuth(request);3if (!userId) {4return reply.code(403).send();5}67const user = userId ? await clerkClient.users.getUser(userId) : null;8return { user };9});
Using Clerk for specific routes only
Fastify provides an easy way to register plugins for specific scopes only using the `register` helper. In the following example, we split our routes into two different fastify plugins - but we only register Clerk in one of them.
1import * as dotenv from "dotenv";23dotenv.config();456import Fastify, { FastifyPluginCallback } from "fastify";7import { clerkClient, clerkPlugin, getAuth } from "@clerk/fastify";89const fastify = Fastify({ logger: true });1011/**12* Register Clerk only for a subset of your routes13*/14const protectedRoutes: FastifyPluginCallback = (instance, opts, done) => {15instance.register(clerkPlugin);16instance.get("/protected", async (request, reply) => {17const { userId } = getAuth(request);18if (!userId) {19return reply.code(403).send();20}2122const user = userId ? await clerkClient.users.getUser(userId) : null;23return { user };24});25done();26};2728const publicRoutes: FastifyPluginCallback = (instance, opts, done) => {29instance.get("/", async (request, reply) => {30return { message: "This is a public endpoint. Request /protected to test the Clerk auth middleware" };31});32done();33};3435/**36* Register your routes as you normally would37*/38fastify.register(protectedRoutes);39fastify.register(publicRoutes);4041const start = async () => {42try {43await fastify.listen({ port: 3000 });44} catch (err) {45fastify.log.error(err);46process.exit(1);47}48};4950start();51