@remix-run/auth
v0.2.5
Published
Browser login, OAuth, and OIDC helpers for Remix
Downloads
9,134
Maintainers
Readme
auth
Composable browser authentication primitives for Remix. Use this package to verify credentials on your own server, start external OAuth or OIDC redirects, finish provider callbacks, and write an app-owned auth record into the session. Pair it with remix/middleware/auth when later requests need to resolve that session data into the current user and protect routes.
Features
- Small, composable primitives:
verifyCredentials(),startExternalAuth(),finishExternalAuth(),refreshExternalAuth(), andcompleteAuth() - Built-in provider support for Google, Microsoft, Okta, Auth0, GitHub, Facebook, X, and Atmosphere
- Module-scope provider configuration for boot-time validation and stable callback URLs
- App-owned session records so you decide what auth data to persist
- Shared session completion for credentials and external auth flows
- Designed to pair with
remix/middleware/authfor request-time auth resolution and route protection
Installation
npm i remixUsage
remix/auth exposes five primitives:
verifyCredentials(provider, context)parses submitted credentials and returns the authenticated result ornullstartExternalAuth(provider, context, options?)stores the in-progress OAuth transaction in the session and returns the provider redirect responsefinishExternalAuth(provider, context, options?)validates the callback, clears the stored transaction, and returns{ result, returnTo? }, including any provider tokens inresult.tokensrefreshExternalAuth(provider, tokens)exchanges a previously storedrefreshTokenfor a fresh provider token bundle when the provider runtime supports refreshcompleteAuth(context)rotates the current session id and returns the session for auth writes
The route owns redirects, flashes, and other app-specific behavior. remix/auth owns the protocol work.
Credentials Auth
Use createCredentialsAuthProvider() when your own server can verify submitted credentials directly, such as email/password logins.
import { auth, Auth, createSessionAuthScheme, requireAuth } from 'remix/middleware/auth'
import { completeAuth, createCredentialsAuthProvider, verifyCredentials } from 'remix/auth'
import { createCookie } from 'remix/cookie'
import { createRouter } from 'remix/router'
import { formData } from 'remix/middleware/form-data'
import { form, route } from 'remix/routes'
import type { GoodAuth } from 'remix/middleware/auth'
import { redirect } from 'remix/response/redirect'
import { Session } from 'remix/session'
import { session } from 'remix/middleware/session'
import { createCookieSessionStorage } from 'remix/session-storage/cookie'
let sessionCookie = createCookie('__session', {
secrets: [env.SESSION_SECRET],
httpOnly: true,
secure: true,
sameSite: 'lax',
path: '/',
})
let sessionStorage = createCookieSessionStorage()
let routes = route({
auth: {
session: {
login: form('/login'),
logout: { method: 'POST', pattern: '/logout' },
},
},
app: {
dashboard: '/dashboard',
},
})
let passwordProvider = createCredentialsAuthProvider({
parse(context) {
let formData = context.get(FormData)
if (formData == null) {
throw new Error('Expected formData() middleware before verifyCredentials()')
}
return {
email: String(formData.get('email') ?? ''),
password: String(formData.get('password') ?? ''),
}
},
async verify({ email, password }) {
return users.verifyPassword(email, password)
},
})
let router = createRouter({
middleware: [
session(sessionCookie, sessionStorage),
formData(),
auth({
schemes: [
createSessionAuthScheme({
read(session) {
return session.get('auth') as { userId: string } | null
},
verify(value) {
return users.getById(value.userId)
},
invalidate(session) {
session.unset('auth')
},
}),
],
}),
],
})
router.get(routes.auth.session.login.index, () => new Response('Login page'))
router.post(routes.auth.session.login.action, async (context) => {
let user = await verifyCredentials(passwordProvider, context)
if (user == null) {
return redirect(routes.auth.session.login.index.href())
}
let session = completeAuth(context)
session.set('auth', { userId: user.id })
return redirect(routes.app.dashboard.href())
})
router.post(routes.auth.session.logout, ({ get }) => {
let session = get(Session)
session.unset('auth')
session.regenerateId(true)
return redirect(routes.auth.session.login.index.href())
})
router.get(routes.app.dashboard, {
middleware: [requireAuth()],
handler(context) {
let auth = context.get(Auth) as GoodAuth<{ id: string; email: string }>
return Response.json({
id: auth.identity.id,
email: auth.identity.email,
method: auth.method,
})
},
})External Auth
Starting from the same session(), auth(), and createSessionAuthScheme() setup as the credentials example above, you can add a Google login flow like this. The provider is created once at module scope, and the routes compose startExternalAuth(), finishExternalAuth(), and completeAuth() directly.
import { auth, Auth, createSessionAuthScheme, requireAuth } from 'remix/middleware/auth'
import {
completeAuth,
createGoogleAuthProvider,
finishExternalAuth,
refreshExternalAuth,
startExternalAuth,
} from 'remix/auth'
import { createCookie } from 'remix/cookie'
import { createRouter } from 'remix/router'
import { route } from 'remix/routes'
import type { GoodAuth } from 'remix/middleware/auth'
import { redirect } from 'remix/response/redirect'
import { session } from 'remix/middleware/session'
import { createCookieSessionStorage } from 'remix/session-storage/cookie'
let sessionCookie = createCookie('__session', {
secrets: [env.SESSION_SECRET],
httpOnly: true,
secure: true,
sameSite: 'lax',
path: '/',
})
let sessionStorage = createCookieSessionStorage()
let routes = route({
auth: {
session: {
login: '/login',
},
google: {
login: '/login/google',
callback: '/auth/google/callback',
},
},
app: {
dashboard: '/dashboard',
},
})
let googleProvider = createGoogleAuthProvider({
clientId: env.GOOGLE_CLIENT_ID,
clientSecret: env.GOOGLE_CLIENT_SECRET,
redirectUri: new URL(routes.auth.google.callback.href(), env.APP_ORIGIN),
authorizationParams: {
access_type: 'offline',
prompt: 'consent',
},
})
let router = createRouter({
middleware: [
session(sessionCookie, sessionStorage),
auth({
schemes: [
createSessionAuthScheme({
read(session) {
return session.get('auth') as { userId: string } | null
},
verify(value) {
return users.getById(value.userId)
},
invalidate(session) {
session.unset('auth')
},
}),
],
}),
],
})
router.get(routes.auth.session.login, () => {
return new Response(`<a href="${routes.auth.google.login.href()}">Login with Google</a>`, {
headers: {
'Content-Type': 'text/html; charset=utf-8',
},
})
})
router.get(routes.auth.google.login, (context) =>
startExternalAuth(googleProvider, context, {
returnTo: context.url.searchParams.get('returnTo'),
}),
)
router.get(routes.auth.google.callback, async (context) => {
let { result, returnTo } = await finishExternalAuth(googleProvider, context)
let user = await users.upsertFromGoogle(result.profile)
await persistProviderTokens(user.id, result.tokens)
let session = completeAuth(context)
session.set('auth', { userId: user.id })
return redirect(returnTo ?? routes.app.dashboard.href())
})
async function getGoogleAccessToken(userId: string) {
let tokens = await readStoredProviderTokens(userId)
if (tokens == null) {
return null
}
if (tokens.expiresAt != null && tokens.expiresAt.getTime() <= Date.now()) {
tokens = (await refreshExternalAuth(googleProvider, tokens)).tokens
await persistProviderTokens(userId, tokens)
}
return tokens.accessToken
}
router.get(routes.app.dashboard, {
middleware: [requireAuth()],
handler(context) {
let auth = context.get(Auth) as GoodAuth<{ id: string; email: string | null }>
return Response.json({
id: auth.identity.id,
email: auth.identity.email,
method: auth.method,
})
},
})A typical external auth flow looks like this:
- Create the provider once at module scope. For Atmosphere, call
provider.prepare(handleOrDid)only when starting the login flow. - Call
startExternalAuth()from the login route. - Call
finishExternalAuth()from the callback route. - Persist any provider tokens you want to reuse later.
- Call
completeAuth(context)and write your auth record into the returned session. - On a later follow-up request, load the stored provider tokens, refresh them with
refreshExternalAuth()only if needed, then save the refreshed bundle back to storage. - Return your own redirect or other response.
Built-in External Auth Providers
When one of the built-in providers matches your auth provider, start there. Google, Microsoft, Okta, and Auth0 use the shared OIDC runtime. GitHub, Facebook, X, and Atmosphere use built-in custom OAuth flows.
import {
createAuth0AuthProvider,
createAtmosphereAuthProvider,
createFacebookAuthProvider,
createGitHubAuthProvider,
createGoogleAuthProvider,
createMicrosoftAuthProvider,
createOktaAuthProvider,
createXAuthProvider,
} from 'remix/auth'
let auth0Provider = createAuth0AuthProvider({
domain: env.AUTH0_DOMAIN,
clientId: env.AUTH0_CLIENT_ID,
clientSecret: env.AUTH0_CLIENT_SECRET,
redirectUri: new URL('/auth/auth0/callback', env.APP_ORIGIN),
})
let atmosphereProvider = createAtmosphereAuthProvider({
clientId: new URL('/oauth/client-metadata.json', env.APP_ORIGIN),
redirectUri: new URL('/auth/atmosphere/callback', env.APP_ORIGIN),
sessionSecret: env.SESSION_SECRET,
})
let facebookProvider = createFacebookAuthProvider({
clientId: env.FACEBOOK_CLIENT_ID,
clientSecret: env.FACEBOOK_CLIENT_SECRET,
redirectUri: new URL('/auth/facebook/callback', env.APP_ORIGIN),
})
let githubProvider = createGitHubAuthProvider({
clientId: env.GITHUB_CLIENT_ID,
clientSecret: env.GITHUB_CLIENT_SECRET,
redirectUri: new URL('/auth/github/callback', env.APP_ORIGIN),
})
let googleProvider = createGoogleAuthProvider({
clientId: env.GOOGLE_CLIENT_ID,
clientSecret: env.GOOGLE_CLIENT_SECRET,
redirectUri: new URL('/auth/google/callback', env.APP_ORIGIN),
})
let microsoftProvider = createMicrosoftAuthProvider({
tenant: 'organizations',
clientId: env.MICROSOFT_CLIENT_ID,
clientSecret: env.MICROSOFT_CLIENT_SECRET,
redirectUri: new URL('/auth/microsoft/callback', env.APP_ORIGIN),
})
let oktaProvider = createOktaAuthProvider({
issuer: env.OKTA_ISSUER,
clientId: env.OKTA_CLIENT_ID,
clientSecret: env.OKTA_CLIENT_SECRET,
redirectUri: new URL('/auth/okta/callback', env.APP_ORIGIN),
})
let xProvider = createXAuthProvider({
clientId: env.X_CLIENT_ID,
clientSecret: env.X_CLIENT_SECRET,
redirectUri: new URL('/auth/x/callback', env.APP_ORIGIN),
})Notes:
- OIDC providers use discovery by default at
/.well-known/openid-configuration - Pass
metadatawhen you want to skip discovery ordiscoveryUrlwhen the metadata document lives elsewhere - Default OIDC scopes are
openid profile email createGoogleAuthProvider()uses the same OIDC runtime with Google's published endpoints wired in directly, so it does not need a discovery requestcreateMicrosoftAuthProvider()adds thetenantoption and builds the issuer from itcreateOktaAuthProvider()expects the full Okta issuer URL, usually something likehttps://example.okta.com/oauth2/defaultcreateAuth0AuthProvider()expects your Auth0 domain and derives the issuer URL for youcreateAtmosphereAuthProvider()returns a module-scope provider withprepare(handleOrDid)for request-time atproto account discovery beforestartExternalAuth()- Atmosphere callback routes pass the module-scope provider directly to
finishExternalAuth(); the original handle or DID is stored in the sealed OAuth transaction state createAtmosphereAuthProvider()requiressessionSecretand seals the in-flight account, authorization server, nonce, and DPoP key state into the existing OAuth transaction stored in your app session, so you do not need a separate file or database store for the redirect stepcreateAtmosphereAuthProvider()returns DPoP-bound token material inresult.tokens, includingaccessToken,refreshToken, authorization server refresh details, anddpopJWK state for follow-up DPoP-signed requestsrefreshExternalAuth()supports built-in OIDC providers, X, and Atmosphere when the stored token bundle includes a refresh token- Providers only return refresh tokens when configured to request offline access, such as
authorizationParams: { access_type: 'offline' }for Google or addingoffline.accessto X scopes - Use
mapProfile()withcreateOIDCAuthProvider()when you wantresult.profileto have an app-specific type before it reaches your route code
Default scopes for OAuth providers that don't use OIDC discovery:
- GitHub:
read:user user:email - Facebook:
public_profile email - X:
tweet.read users.read
Pass scopes if you need a different set for a provider.
Custom Auth Providers
Use createOIDCAuthProvider() directly for custom external auth providers. This is the extension point for providers that support OpenID Connect discovery, authorization code flow, and a userinfo endpoint. Reach for a custom OAuth provider implementation only when the provider does not support OIDC.
import {
completeAuth,
createOIDCAuthProvider,
finishExternalAuth,
startExternalAuth,
} from 'remix/auth'
import { redirect } from 'remix/response/redirect'
let companyProvider = createOIDCAuthProvider({
name: 'company',
issuer: 'https://sso.acme.com',
clientId: 'acme-web',
clientSecret: 'acme-web-secret',
redirectUri: new URL('/auth/company/callback', 'https://app.acme.com'),
authorizationParams: {
prompt: 'login',
},
mapProfile({ claims }) {
return {
id: claims.sub,
email: claims.email ?? null,
name: claims.name ?? claims.preferred_username ?? 'Unknown user',
}
},
})
router.get('/login/company', (context) =>
startExternalAuth(companyProvider, context, {
returnTo: context.url.searchParams.get('returnTo'),
}),
)
router.get('/auth/company/callback', async (context) => {
let { result, returnTo } = await finishExternalAuth(companyProvider, context)
let user = await users.upsertFromCompanySSO(result.profile)
let session = completeAuth(context)
session.set('auth', { userId: user.id })
return redirect(returnTo ?? '/dashboard')
})Related Packages
auth-middleware- Request authentication and route protection helpersform-data-middleware- Form body parsing forcreateCredentialsAuthProvider()routessession-middleware- Request-scoped session loading and persistencesession- Session data model and storage backendsfetch-router- Router and middleware runtime
Related Work
License
See LICENSE
