Remix Authentication
Easily add secure, edge- and SSR-friendly authentication to Remix with Clerk.
Clerk is the easiest way to add authentication and user management to your Remix application. After following this guide, you should have a working Remix app complete with:
- Fully fledged sign in and sign up flows.
- Google Social Login.
- Secure email/password authentication.
Install @clerk/remix
Once you have a Remix application ready, you need to install Clerk's Remix SDK. This gives you access to our prebuilt components and hooks for Remix applications.
1npm install @clerk/remix
Set Environment Keys
Below is an example of your .env file. To get the respective keys go to the API Keys page in the Clerk dashboard.
1CLERK_PUBLISHABLE_KEY=pk_test_••••••••••••••••••••••••••••••••••2CLERK_SECRET_KEY=sk_test_••••••••••••••••••••••••••••••••••
Disable ErrorBoundary
Currently we are unable to support ErrorBoundaryV2 but will support this when Remix V2 is in general release, make sure that it is set to false
in your remix.config.js
:
1/** @type {import('@remix-run/dev').AppConfig} */2module.exports = {3ignoredRouteFiles: ["**/.*"],4// appDirectory: "app",5// assetsBuildDirectory: "public/build",6// serverBuildPath: "build/index.js",7// publicPath: "/build/",8serverModuleFormat: "cjs",9future: {10v2_errorBoundary: false,11v2_meta: true,12v2_normalizeFormMethod: true,13v2_routeConvention: true,14},15};
Configure rootAuthLoader
To configure Clerk in your Remix application, you will need to update your root loader. This will enable us to have access to authentication state in any Remix routes.
1import type { MetaFunction,LoaderFunction } from "@remix-run/node";2import {3Links,4LiveReload,5Meta,6Outlet,7Scripts,8ScrollRestoration,9} from "@remix-run/react";1011import { rootAuthLoader } from "@clerk/remix/ssr.server";1213export const meta: MetaFunction = () => ({14charset: "utf-8",15title: "New Remix App",16viewport: "width=device-width,initial-scale=1",17});1819export const loader: LoaderFunction = (args) => rootAuthLoader(args);2021export default function App() {22return (23<html lang="en">24<head>25<Meta />26<Links />27</head>28<body>29<Outlet />30<ScrollRestoration />31<Scripts />32<LiveReload />33</body>34</html>35);36}
If you need to load in additonal data you can pass your loader directly to the rootAuthLoader
.
1//Imports234export const loader: LoaderFunction = args => {5return rootAuthLoader(args, ({ request }) => {6const { sessionId, userId, getToken } = request.auth;7// fetch data8return { yourData: 'here' };9});10};1112// Additonal application code.
Configure ClerkApp
Clerk provides a ClerkApp
wrapper to provide the authentication state to your React tree. This helper works with Remix SSR out-of-the-box and follows the "higher-order component" paradigm.
1import { ClerkApp } from "@clerk/remix";2import type { MetaFunction,LoaderFunction } from "@remix-run/node";34import {5Links,6LiveReload,7Meta,8Outlet,9Scripts,10ScrollRestoration,11} from "@remix-run/react";1213import { rootAuthLoader } from "@clerk/remix/ssr.server";14// Import ClerkApp15import { ClerkApp } from "@clerk/remix";1617export const meta: MetaFunction = () => ({18charset: "utf-8",19title: "New Remix App",20viewport: "width=device-width,initial-scale=1",21});2223export const loader: LoaderFunction = (args) => rootAuthLoader(args);2425function App() {26return (27<html lang="en">28<head>29<Meta />30<Links />31</head>32<body>33<Outlet />34<ScrollRestoration />35<Scripts />36<LiveReload />37</body>38</html>39);40}4142// Wrap your app in ClerkApp(app)43export default ClerkApp(App);44`
Set ClerkCatchBoundary
Clerk uses short lived tokens to keep your application secure, to refresh expired tokens Clerk uses Remix's catch boundary.
If you are seeing a 401 after a short period of time even though your user is logged in. Please verify you have implemented this section.
1import { ClerkApp } from "@clerk/remix";2import type { MetaFunction,LoaderFunction } from "@remix-run/node";34import {5Links,6LiveReload,7Meta,8Outlet,9Scripts,10ScrollRestoration,11} from "@remix-run/react";1213import { rootAuthLoader } from "@clerk/remix/ssr.server";14import { ClerkApp, ClerkCatchBoundary } from "@clerk/remix";1516export const meta: MetaFunction = () => ({17charset: "utf-8",18title: "New Remix App",19viewport: "width=device-width,initial-scale=1",20});2122export const loader: LoaderFunction = (args) => rootAuthLoader(args);23// add a Catch Boundary24export const CatchBoundary = ClerkCatchBoundary();2526function App() {27return (28<html lang="en">29<head>30<Meta />31<Links />32</head>33<body>34<Outlet />35<ScrollRestoration />36<Scripts />37<LiveReload />38</body>39</html>40);41}4243export default ClerkApp(App);
If you need to add a custom boundary to your application you can pass it as an argument to the ClerkCatchBoundary
.
1//imports2export const CatchBoundary = ClerkCatchBoundary(YourBoundary);3// Additional application code
Clerk offers a set of prebuilt components that you can use to embed sign in, sign up, and other user management functions into your Next.js application. We are going to use the <SignIn />
,<SignUp />
components by utlizing Remix routes.
The functionality of the components are controlled by the instance settings you specify in your Clerk Dashboard.
Build Your Sign Up
The page needs to be mounted on a catch all route for example:
app/routes/sign-up/$.tsx
1import { SignUp } from "@clerk/remix";23export default function SignUpPage() {4return (5<div>6<h1>Sign Up route</h1>7<SignUp routing={"path"} path={"/sign-up"} />8</div>9);10}
Build Your Sign In
The page needs to be mounted on a catch all route for example:
app/routes/sign-in/$.tsx
1import { SignIn } from "@clerk/remix";23export default function SignInPage() {4return (5<div>6<h1>Sign In route</h1>7<SignIn routing={"path"} path={"/sign-in"} />8</div>9);10}
Update your dashboard settings
To make sure you are using the embedded components you will need to update your instance settings in the Clerk Dashboard to use the embedded components.
- Select Paths
- Set paths to
/sign-in
and/sign-up
By default, the Clerk components inherit the font family.
Protecting Your Pages
Client Side
Clerk offers Control Components that allow you to protect your pages, below is a simple example using Clerk's hosted components and the <SignedIn/>
and <SignedOut/>
control components.
1import {2SignedIn,3SignedOut,4RedirectToSignIn,5UserButton,6} from "@clerk/remix";78export default function Index() {9return (10<div>11<SignedIn>12<h1>Index route</h1>13<p>You are signed in!</p>14<UserButton />15</SignedIn>16<SignedOut>17<RedirectToSignIn />18</SignedOut>19</div>20);21}
Server Side
To protect your routes you can use the the loader to check for the userId
singleton, and if it doesn't exists redirect your user back to the sign in page.
1import {2UserButton,3} from "@clerk/remix";4import { getAuth } from "@clerk/remix/ssr.server";5import { LoaderFunction, redirect } from "@remix-run/node";67export const loader: LoaderFunction = async (args) => {8const { userId } = await getAuth(args);9if(!userId){10return redirect("/sign-in");11}12return {};13}1415export default function Index() {16return (17<div>18<h1>Index route</h1>19<p>You are signed in!</p>20<UserButton afterSignOutUrl="/"/>21</div>22);23}
Read Session & User Data
Clerk provides a set of hooks and helpers that you can use to access the active session and user data in your Remix application. We have included examples of how to use these helpers to get you started.
Client Side
useAuth
The useAuth
hook is a convenient way to access the current auth state. This hook provides the minimal information needed for data-loading and helper methods to manage the current active session.
1import { useAuth } from "@clerk/remix";23export default function Example() {4const { isLoaded, userId, sessionId, getToken } = useAuth();56// In case the user signs out while on the page.7if (!isLoaded || !userId) {8return null;9}1011return (12<div>13Hello, {userId} your current active session is {sessionId}14</div>15);16}
useUser
The useUser
hook is a convenient way to access the current user data where you need it. This hook provides the user data and helper methods to manage the current active session.
1import { useUser } from "@clerk/remix";23export default function Example() {4const { isLoaded, isSignedIn, user } = useUser();56if (!isLoaded || !isSignedIn) {7return null;8}910return <div>Hello, {user.firstName} welcome to Clerk</div>;11}
Server Side
getAuth()
Using getAuth allows you to retrieve auth state and also ways to fetch data using the getToken
function.
12import {getAuth} from "@clerk/remix";3export const loader: LoaderFunction = async (args) => {4// Using getAuth to retrieve data as needed.5const { userId, sessionId, getToken } = await getAuth(args);67console.log("Use getAuth() to access the auth state:", userId, sessionId, getToken);89if (!userId) {10return redirect("/sign-in?redirect_url=" + args.request.url);11}12// fetch data where you need the userId.13const posts = await mockGetPosts(userId);14return { posts };15};
Retrieve Full User
1import {getAuth,clerkClient} from "@clerk/remix";23export const loader: LoaderFunction = async (args) => {4const { userId } = await getAuth(args);56if (!userId) {7return redirect("/sign-in?redirect_url=" + args.request.url);8}910const user = await createClerkClient({secretKey: process.env.CLERK_SECRET_KEY}).users.getUser(userId);11return { serialisedUser: JSON.stringify(user) };12};