@auth/core
Version:
Authentication for the Web.
279 lines (252 loc) • 8.59 kB
text/typescript
/**
*
*
* This module contains functions and types
* to encode and decode {@link https://authjs.dev/concepts/session-strategies#jwt-session JWT}s
* issued and used by Auth.js.
*
* The JWT issued by Auth.js is _encrypted by default_, using the _A256CBC-HS512_ algorithm ({@link https://www.rfc-editor.org/rfc/rfc7518.html#section-5.2.5 JWE}).
* It uses the `AUTH_SECRET` environment variable or the passed `secret` propery to derive a suitable encryption key.
*
* :::info Note
* Auth.js JWTs are meant to be used by the same app that issued them.
* If you need JWT authentication for your third-party API, you should rely on your Identity Provider instead.
* :::
*
* ## Installation
*
* ```bash npm2yarn
* npm install @auth/core
* ```
*
* You can then import this submodule from `@auth/core/jwt`.
*
* ## Usage
*
* :::warning Warning
* This module *will* be refactored/changed. We do not recommend relying on it right now.
* :::
*
*
* ## Resources
*
* - [What is a JWT session strategy](https://authjs.dev/concepts/session-strategies#jwt-session)
* - [RFC7519 - JSON Web Token (JWT)](https://www.rfc-editor.org/rfc/rfc7519)
*
* @module jwt
*/
import { hkdf } from "@panva/hkdf"
import { EncryptJWT, base64url, calculateJwkThumbprint, jwtDecrypt } from "jose"
import { SessionStore } from "./lib/utils/cookie.js"
import { Awaitable } from "./types.js"
import type { LoggerInstance } from "./lib/utils/logger.js"
import { MissingSecret } from "./errors.js"
import { parse } from "cookie"
const DEFAULT_MAX_AGE = 30 * 24 * 60 * 60 // 30 days
const now = () => (Date.now() / 1000) | 0
const alg = "dir"
const enc = "A256CBC-HS512"
type Digest = Parameters<typeof calculateJwkThumbprint>[1]
/** Issues a JWT. By default, the JWT is encrypted using "A256CBC-HS512". */
export async function encode<Payload = JWT>(params: JWTEncodeParams<Payload>) {
const { token = {}, secret, maxAge = DEFAULT_MAX_AGE, salt } = params
const secrets = Array.isArray(secret) ? secret : [secret]
const encryptionSecret = await getDerivedEncryptionKey(enc, secrets[0], salt)
const thumbprint = await calculateJwkThumbprint(
{ kty: "oct", k: base64url.encode(encryptionSecret) },
`sha${encryptionSecret.byteLength << 3}` as Digest
)
// @ts-expect-error `jose` allows any object as payload.
return await new EncryptJWT(token)
.setProtectedHeader({ alg, enc, kid: thumbprint })
.setIssuedAt()
.setExpirationTime(now() + maxAge)
.setJti(crypto.randomUUID())
.encrypt(encryptionSecret)
}
/** Decodes a Auth.js issued JWT. */
export async function decode<Payload = JWT>(
params: JWTDecodeParams
): Promise<Payload | null> {
const { token, secret, salt } = params
const secrets = Array.isArray(secret) ? secret : [secret]
if (!token) return null
const { payload } = await jwtDecrypt(
token,
async ({ kid, enc }) => {
for (const secret of secrets) {
const encryptionSecret = await getDerivedEncryptionKey(
enc,
secret,
salt
)
if (kid === undefined) return encryptionSecret
const thumbprint = await calculateJwkThumbprint(
{ kty: "oct", k: base64url.encode(encryptionSecret) },
`sha${encryptionSecret.byteLength << 3}` as Digest
)
if (kid === thumbprint) return encryptionSecret
}
throw new Error("no matching decryption secret")
},
{
clockTolerance: 15,
keyManagementAlgorithms: [alg],
contentEncryptionAlgorithms: [enc, "A256GCM"],
}
)
return payload as Payload
}
export interface GetTokenParams<R extends boolean = false>
extends Pick<JWTDecodeParams, "salt" | "secret"> {
/** The request containing the JWT either in the cookies or in the `Authorization` header. */
req: Request | { headers: Headers | Record<string, string> }
/**
* Use secure prefix for cookie name, unless URL in `NEXTAUTH_URL` is http://
* or not set (e.g. development or test instance) case use unprefixed name
*/
secureCookie?: boolean
/** If the JWT is in the cookie, what name `getToken()` should look for. */
cookieName?: string
/**
* `getToken()` will return the raw JWT if this is set to `true`
*
* @default false
*/
raw?: R
decode?: JWTOptions["decode"]
logger?: LoggerInstance | Console
}
/**
* Takes an Auth.js request (`req`) and returns either the Auth.js issued JWT's payload,
* or the raw JWT string. We look for the JWT in the either the cookies, or the `Authorization` header.
*/
export async function getToken<R extends boolean = false>(
params: GetTokenParams<R>
): Promise<R extends true ? string : JWT | null>
export async function getToken(
params: GetTokenParams
): Promise<string | JWT | null> {
const {
secureCookie,
cookieName = secureCookie
? "__Secure-authjs.session-token"
: "authjs.session-token",
decode: _decode = decode,
salt = cookieName,
secret,
logger = console,
raw,
req,
} = params
if (!req) throw new Error("Must pass `req` to JWT getToken()")
if (!secret)
throw new MissingSecret("Must pass `secret` if not set to JWT getToken()")
const headers =
req.headers instanceof Headers ? req.headers : new Headers(req.headers)
const sessionStore = new SessionStore(
{ name: cookieName, options: { secure: secureCookie } },
parse(headers.get("cookie") ?? ""),
logger
)
let token = sessionStore.value
const authorizationHeader = headers.get("authorization")
if (!token && authorizationHeader?.split(" ")[0] === "Bearer") {
const urlEncodedToken = authorizationHeader.split(" ")[1]
token = decodeURIComponent(urlEncodedToken)
}
if (!token) return null
if (raw) return token
try {
return await _decode({ token, secret, salt })
} catch {
return null
}
}
async function getDerivedEncryptionKey(
enc: string,
keyMaterial: Parameters<typeof hkdf>[1],
salt: Parameters<typeof hkdf>[2]
) {
let length: number
switch (enc) {
case "A256CBC-HS512":
length = 64
break
case "A256GCM":
length = 32
break
default:
throw new Error("Unsupported JWT Content Encryption Algorithm")
}
return await hkdf(
"sha256",
keyMaterial,
salt,
`Auth.js Generated Encryption Key (${salt})`,
length
)
}
export interface DefaultJWT extends Record<string, unknown> {
name?: string | null
email?: string | null
picture?: string | null
sub?: string
iat?: number
exp?: number
jti?: string
}
/**
* Returned by the `jwt` callback when using JWT sessions
*
* [`jwt` callback](https://authjs.dev/reference/core/types#jwt)
*/
export interface JWT extends Record<string, unknown>, DefaultJWT {}
export interface JWTEncodeParams<Payload = JWT> {
/**
* The maximum age of the Auth.js issued JWT in seconds.
*
* @default 30 * 24 * 60 * 60 // 30 days
*/
maxAge?: number
/** Used in combination with `secret`, to derive the encryption secret for JWTs. */
salt: string
/** Used in combination with `salt`, to derive the encryption secret for JWTs. */
secret: string | string[]
/** The JWT payload. */
token?: Payload
}
export interface JWTDecodeParams {
/** Used in combination with `secret`, to derive the encryption secret for JWTs. */
salt: string
/**
* Used in combination with `salt`, to derive the encryption secret for JWTs.
*
* @note
* You can also pass an array of secrets, in which case the first secret that successfully
* decrypts the JWT will be used. This is useful for rotating secrets without invalidating existing sessions.
* The newer secret should be added to the start of the array, which will be used for all new sessions.
*/
secret: string | string[]
/** The Auth.js issued JWT to be decoded */
token?: string
}
export interface JWTOptions {
/**
* The secret used to encode/decode the Auth.js issued JWT.
* It can be an array of secrets, in which case the first secret that successfully
* decrypts the JWT will be used. This is useful for rotating secrets without invalidating existing sessions.
* @internal
*/
secret: string | string[]
/**
* The maximum age of the Auth.js issued JWT in seconds.
*
* @default 30 * 24 * 60 * 60 // 30 days
*/
maxAge: number
/** Override this method to control the Auth.js issued JWT encoding. */
encode: (params: JWTEncodeParams) => Awaitable<string>
/** Override this method to control the Auth.js issued JWT decoding. */
decode: (params: JWTDecodeParams) => Awaitable<JWT | null>
}