May 07, 2022
Colin Sidoti
Next.js SSR authentication is easy with Clerk – just a few lines of code to get started.
Updated: 25/08/2022
React was originally built to run on the client. Once React started, hooks would run to load data, and eventually a full page would be generated.
But since React is just a Javascript library, there was no reason it couldn't run on a server, too. Server-side rendering (SSR) runs React on the server to generate a page's initial HTML (a.k.a the first render), then runs React again on the client to provide reactivity.
Server-side rendering is particularly helpful for pages that must be indexed in search engines, since search engines cannot index pages that are rendered client-side.
The bigger benefit, though, is that server-side render can reduce the complexity of an application and lead to fewer lines of code. This is especially true in the modern era, where frameworks like Next.js provide helpers that drastically reduce the setup time for SSR.
Server-side rendering can be used by including export async function getServerSideProps()
on any pages you need SSR. Below is an example of how a page would look when using SSR:
function Page({ data }) {// use the data on our page<div>{data.content}</div>}export async function getServerSideProps() {// Fetch data from an APIconst res = await fetch(`https://api.example.com/data`)const data = await res.json()// Pass data to the page via propsreturn { props: { data } }}export default Page
This example retrieves some data from api.example.com. Then, we pass the data to the UI using the return statement. Finally, in the UI, we present the data to the user.
You can read more about Server-side rendering in the Next.js SSR documentation
Now that you have a basic understanding of how SSR works, let us investigate how Clerk and SSR can work together.
Clerk allows you to verify a user is authenticated and to retrieve user profile data when using SSR. Below, we will cover how to implement both.
With Clerk and Next.js SSR, we can check if the user is authenticated and, if not, redirect them to the sign-in page without rendering anything on the client. Clerk provides a helper called withServerSideAuth
, which allows you to access authentication information:
import { withServerSideAuth } from "@clerk/nextjs/ssr";export const getServerSideProps = withServerSideAuth(({ req, resolvedUrl }) => {const { sessionId } = req.auth;if (!sessionId) {return { redirect: { destination: "/sign-in?redirect_url=" + resolvedUrl } };}return { props: {} };});
The snippet above checks to see if there is a session. If not, it will redirect the user to the sign-in page. Once the user has successfully authenticated, Clerk will redirect the user back to the page they were initially trying to access. Here is a quick GIF showing the process in action.
If you want to see it in action, check out this fully functional example
Server-side rendering gives you the opportunity to retrieve data from 3rd party APIs and your own database. When using Clerk, you have the option of retrieving the user ID, or if you are using a Clerk integration such as Hasura you can also retrieve the JWT ready for use.
Below is an example of retrieving a Hasura JWT token ready to retrieve data before sending the page to the user.
import { withServerSideAuth } from "@clerk/nextjs/ssr";export const getServerSideProps = withServerSideAuth(async ({ req }) => {const {sessionId, getToken } = req.auth;if (!sessionId) {return { redirect: { destination: "/sign-in?redirect_url=" + resolvedUrl } };}// use a token for your Clerk integrationsconst hasuraToken = await getToken({ template: 'hasura' });// retrieve data from your Hasura integrationreturn { props: {} };});
If you aren’t using a JWT template or a Clerk integration you can just retrieve the userId and use that against your own database.
import { withServerSideAuth } from "@clerk/nextjs/ssr";export const getServerSideProps = withServerSideAuth(async ({ req }) => {const {sessionId,userId } = req.auth;if (!sessionId) {return { redirect: { destination: "/sign-in?redirect_url=" + resolvedUrl } };}// use the userId to retrieve your own datareturn { props: {} };});
In some cases, you may need the full User object available to you. For example, if you need to retrieve the user’s primary email address
Clerk can use an additional network request to retrieve the full User object. To enable it, you will need to add { loadUser: true }
to your SSR request. Then, the complete User object will be available:
import { withServerSideAuth } from "@clerk/nextjs/ssr";export const getServerSideProps = withServerSideAuth(async ({ req, resolvedUrl }) => {const { sessionId, getToken, userId } = req.auth;// retrieve the user objectconst { user } = req;// return the users primary email address.const email = user.emailAddresses.find(email => {return (email.id === user.primaryEmailAddressId);});// retrieve data using the email address.const data = getDataFromEmail(email)return { props: { data } };},{ loadUser: true });
Be sure to visit our Next.js authentication with Clerk page
Have more technical questions? Be sure to join our Discord
Start completely free for up to 5,000 monthly active users and up to 10 monthly active orgs. No credit card required.
Learn more about our transparent per-user costs to estimate how much your company could save by implementing Clerk.
The latest news and updates from Clerk, sent to your inbox.