Tutorials Using SSX with NextAuth Getting started authenticating using SSX and NextAuth
Overview
This guide provides an example of how to set up your dapp to authenticate your users using SSX and NextAuth. NextAuth allows you to authenticate with multiple identity providers, and SSX enables you to add Sign-in with Ethereum authentication to your Next.js dapp easily.
Prerequisites
This example follows along the ssx-test-nextauth
example implementation, built with npm init @rainbow-me/rainbowkit@0.1.9
. You can create from the rainbow template using the package manager of choice:
Copy npm init @rainbow-me/rainbowkit@0.1.9
# OR
yarn create @rainbow-me/rainbowkit
Install Dependencies
Install ssx
and next-auth
with your package manager of choice:
Copy npm install --save @spruceid/ssx-react @spruceid/ssx-server next-auth
# OR
yarn add @spruceid/ssx-react @spruceid/ssx-server next-auth
Add NextAuth API Routes
In your dapp, we will add an API route (pages/api/auth/[...nextauth].ts
) for NextAuth and configure it with SSX. SSX provides configured credentials
and authorize
functions to create a NextAuth provider. SSX also provides a session
function, but it is likely you will want to modify the contents of the function to provide specific session data from the server to the frontend client.
Copy import { NextApiRequest , NextApiResponse } from "next" ;
import NextAuth from "next-auth" ;
import CredentialsProvider from "next-auth/providers/credentials" ;
import { SSXNextAuth } from "@spruceid/ssx-react/next-auth/backend" ;
import { SSXServer } from "@spruceid/ssx-server" ;
export default async function auth (req : NextApiRequest , res : NextApiResponse ) {
const ssxConfig = {};
const ssx = new SSXServer (ssxConfig);
const { credentials , authorize } = SSXNextAuth (req , ssx);
const providers = [
CredentialsProvider ({
name : "Ethereum" ,
credentials ,
authorize ,
}) ,
];
return await NextAuth (req , res , {
providers ,
session : {
strategy : "jwt" ,
} ,
secret : process . env . NEXT_AUTH_SECRET ,
callbacks : {
session : (sessionData) => {
const { session , user , token } = sessionData;
if ( session .user) {
session . user .name = token .sub;
}
return session;
} ,
} ,
});
}
Add Providers to the Frontend
Next, you'll add the SSX provider and NextAuth Session provider to the front end. This is done in the pages/_app.tsx
file. Adding the providers here makes them available for any child component to access the session data.
Copy import '../styles/globals.css' ;
import '@rainbow-me/rainbowkit/styles.css' ;
import { RainbowKitProvider , getDefaultWallets } from '@rainbow-me/rainbowkit' ;
import { goerli , mainnet , configureChains , createClient , WagmiConfig } from 'wagmi' ;
import { alchemyProvider } from 'wagmi/providers/alchemy' ;
import { publicProvider } from 'wagmi/providers/public' ;
import { SSXProvider } from '@spruceid/ssx-react' ;
import { SSXNextAuthRouteConfig } from '@spruceid/ssx-react/next-auth/frontend' ;
import { SessionProvider } from "next-auth/react" ;
const { chains , provider , webSocketProvider } = configureChains (
[
goerli ,
mainnet
] ,
[
alchemyProvider ({
// This is Alchemy's default API key.
// You can get your own at https://dashboard.alchemyapi.io
apiKey : process . env . NEXT_PUBLIC_ALCHEMY_API_KEY ?? "" ,
}) ,
publicProvider () ,
]
);
const { connectors } = getDefaultWallets ({
appName : 'RainbowKit App' ,
chains ,
});
const wagmiClient = createClient ({
autoConnect : true ,
connectors ,
provider ,
webSocketProvider ,
});
const { server } = SSXNextAuthRouteConfig ({ signInOptions : { callbackUrl : '/protected' } });
const ssxConfig : any = {
siweConfig : {
domain : "localhost:3000" ,
} ,
providers : {
server ,
} ,
};
function MyApp ({ Component , pageProps } : any ) {
return (
< WagmiConfig client = {wagmiClient}>
< RainbowKitProvider chains = {chains}>
< SSXProvider ssxConfig = {ssxConfig}>
< SessionProvider session = { pageProps .session} refetchInterval = { 0 }>
< Component { ... pageProps} />
</ SessionProvider >
</ SSXProvider >
</ RainbowKitProvider >
</ WagmiConfig >
);
}
export default MyApp;
Using SSX + NextAuth Sessions in your dapp
Now that you've set up the SSX and NextAuth Providers, you can use their corresponding hooks to access user information to protect pages from unauthorized access. Let's create our first protected route in pages/protected.tsx
with the following code:
Copy import React from "react" ;
import styles from '../styles/Protected.module.css'
import { useSession , signOut as nextauthSignOut } from 'next-auth/react' ;
import { useSSX } from "@spruceid/ssx-react" ;
export default function Protected () {
const { data: session , status } = useSession ();
const { ssx , ssxLoaded } = useSSX ();
const signOut = async () => {
try {
await ssx ?.signOut ();
} catch (e) {
console .error (e);
}
nextauthSignOut ({ callbackUrl : '/' });
}
if (status === "loading" ) {
< div className = { styles .container}>
< h2 className = { styles .title}>Protected Page</ h2 >
< p className = { styles .description}>Loading</ p >
</ div >
}
if (status === "unauthenticated" ) {
return (< div className = { styles .container}>
< h2 className = { styles .title}>Protected Page</ h2 >
< p className = { styles .description}>
This page is only accessible to authenticated users.
</ p >
</ div >)
}
return (
< div className = { styles .container}>
< h2 className = { styles .title}>Protected Page</ h2 >
< p className = { styles .description}>
You are Authenticated as < br />
{ session ?. user ?.name}
</ p >
< button onClick = {signOut} disabled = { ! ssxLoaded}>Log Out</ button >
</ div >
)
}
And let's give it some style in styles/Protected.module.css
with the following:
Copy .container {
text-align : center ;
padding : 10 rem ;
}
.title {
font-size : 2 rem ;
}
.description {
font-size : 1 rem ;
line-height : 2 rem ;
}
Now update the pages/index.tsx
file with the following code:
Copy import { ConnectButton } from '@rainbow-me/rainbowkit' ;
import { useState } from 'react' ;
import Head from 'next/head' ;
import type { NextPage } from 'next' ;
import { useRouter } from 'next/router' ;
import { useSSX } from "@spruceid/ssx-react" ;
import styles from '../styles/Home.module.css' ;
const Home : NextPage = () => {
const { ssx , ssxLoaded } = useSSX ();
const router = useRouter ();
const [ address , setAddress ] = useState < string >();
const handleSignIn = async () => {
await ssx ?.signIn ();
router .push ( '/protected' );
};
return (
< div className = { styles .container}>
< Head >
< title >RainbowKit App</ title >
< meta
name = "description"
content = "Generated by @rainbow-me/create-rainbowkit"
/>
< link rel = "icon" href = "/favicon.ico" />
</ Head >
< div style = {{ display : "flex" , justifyContent : "end" }}>
< ConnectButton />
</ div >
< main className = { styles .main}>
< h1 className = { styles .title}>
Welcome to < a href = "" >RainbowKit</ a > + < a href = "" >wagmi</ a > +{ ' ' }
< a href = "https://docs.ssx.id/" >SSX</ a > + < a href = "https://nextjs.org" >Next.js</ a > + < a href = "https://next-auth.js.org/" >NextAuth.js!</ a >
</ h1 >
< p className = { styles .description}>
Sign-in with Ethereum powered by SSX
< br />
< button onClick = {handleSignIn} disabled = { ! ssxLoaded}>Sign Message</ button >
</ p >
{
address &&
< p className = { styles .description}>
Address: < code >{address}</ code >
</ p >
}
</ main >
< footer className = { styles .footer}>
< a href = "https://rainbow.me" target = "_blank" rel = "noopener noreferrer" >
Made with ❤️ by your frens at 🌈
</ a >
</ footer >
</ div >
);
};
export default Home;
Wrap up
Now that you have authentication in your dapp, take a deeper dive into some of the capabilities SSX gives you, like signing in on behalf of a multisig , resolving ENS/Lens names , or customizing the SIWE message.
Have any questions? Hop into the Spruce discord and ask!