Module @forgerock/oidc-client

OIDC Client

A generic OpenID Connect (OIDC) client library for JavaScript and TypeScript, designed to work with PingOne platforms.

The oidc module follows the OIDC specification and provides a simple and easy-to-use API to interact with the OIDC server. It allows you to authenticate, retrieve the access token, revoke the token, and sign out from the OIDC server.

pnpm add @forgerock/oidc-client
# or
npm install @forgerock/oidc-client
# or
yarn add @forgerock/oidc-client
import { oidc } from '@forgerock/oidc-client';
import { OidcConfig, OidcClient } from '@forgerock/oidc-client/types';

const config: OidcConfig = {
serverConfig: { wellknown: 'https://example.com/.well-known/openid-configuration' },
clientId: 'example-client-id',
redirectUri: 'https://example-app/redirect-uri',
scope: 'openid profile email',
};

const oidcClient: OidcClient = await oidc({ config });

The oidc() initialization function accepts the following configuration:

  • serverConfig (required)
    • wellknown (required) - URL to the OIDC provider's well-known configuration endpoint
  • clientId (required) - Your application's client ID registered with the OIDC provider
  • redirectUri (required) - The URI where the OIDC provider will redirect after authentication
  • scope (required) - Space-separated list of requested scopes (e.g., 'openid profile email')
  • storage (optional) - Storage configuration for tokens (defaults to localStorage)
  • timeout (optional) - Request timeout in milliseconds
  • additionalParameters (optional) - Additional parameters to include in authorization requests

Here's a minimal example to get started:

import { oidc } from '@forgerock/oidc-client';

// Initialize the client
const oidcClient = await oidc({ config });

// Start authorization in the background
const authResponse = await oidcClient.authorize.background();

// Get tokens
const tokens = await oidcClient.token.exchange(authResponse.code, authResponse.state);

// Get user information
const user = await oidcClient.user.info();

// Clean up: logout and revoke tokens
await oidcClient.user.logout();

Methods for creating and handling authorization flows.

Creates an authorization URL with the provided options or defaults from the configuration.

  • Parameters: GetAuthorizationUrlOptions (optional)
  • Returns: Promise<string | GenericError> - The authorization URL or an error
const authUrl = await oidcClient.authorize.url();

Initiates the authorization process in the background, returning the authorization code and state or an error. This method handles the authorization flow without requiring user interaction.

  • Parameters: GetAuthorizationUrlOptions (optional)
  • Returns: Promise<AuthorizationSuccess | AuthorizationError> - An object containing code and state on success, or error details on failure
const authResponse = await oidcClient.authorize.background();

Methods for managing OAuth tokens.

Exchanges an authorization code for tokens using the token endpoint from the wellknown configuration. The tokens are automatically stored in the configured storage.

  • Parameters:
    • code (string) - The authorization code received from the authorization server
    • state (string) - The state parameter from the authorization URL creation
    • options (Partial<StorageConfig>, optional) - Storage configuration for persisting tokens
  • Returns: Promise<OauthTokens | TokenExchangeErrorResponse | GenericError> - The new tokens or an error
const tokens = await oidcClient.token.exchange(authCode, authState);

Retrieves the current OAuth tokens from storage. Optionally auto-renews tokens if they are expired or if backgroundRenew is enabled.

  • Parameters: GetTokensOptions (optional)
    • forceRenew - Force token renewal even if not expired
    • backgroundRenew - Automatically renew expired tokens
    • authorizeOptions - Options for authorization during renewal
    • storageOptions - Storage configuration options
  • Returns: Promise<OauthTokens | TokenExchangeErrorResponse | AuthorizationError | GenericError> - The tokens or an error
const tokens = await oidcClient.token.get();

Revokes the access token using the revocation endpoint from the wellknown configuration. Requires an access token stored in the configured storage.

  • Parameters: None
  • Returns: Promise<GenericError | RevokeSuccessResult | RevokeErrorResult> - Confirmation of revocation or an error
const response = await oidcClient.token.revoke();

Methods for user information and session management.

Retrieves user information using the userinfo endpoint from the wellknown configuration. Requires an access token stored in the configured storage.

  • Parameters: None
  • Returns: Promise<GenericError | UserInfoResponse> - User information object or an error
const user = await oidcClient.user.info();

Logs out the user by revoking tokens and clearing the storage. Uses the end session endpoint from the wellknown configuration.

  • Parameters: None
  • Returns: Promise<GenericError | LogoutSuccessResult | LogoutErrorResult> - Confirmation of logout or an error
const logoutResponse = await oidcClient.user.logout();

Checks whether the user has an active session at the AS without prompting for login (prompt=none).

  • Parameters: SessionCheckOptions (optional)
  • Returns: Promise<SessionCheckSuccess | GenericError>
const session = await oidcClient.user.session();
if ('error' in session) {
// No active session
}

Use when you only need to know if a session is alive — no claims are returned.

Requires a stored id_token to send as id_token_hint. Fails immediately with no_id_token_hint if storage is empty.

How the check runs depends on redirectUri:

  • With redirectUri (iframe): a hidden iframe loads the authorization URL. The AS redirects to redirectUri on success or appends error params on failure. The redirectUri must be same-origin — the browser blocks cross-origin iframe content access.
  • Without redirectUri (fetch): a plain GET to the authorization endpoint — no iframe. The AS returns 204 on success or 400 on failure. Use this when no same-origin callback page is available.

Returns { mode: 'none' } on success.

Use when you need the user's claims back from the session. The AS issues a fresh id_token whose decoded claims are returned on success.

Always uses the iframe path — redirectUri is required. A stored id_token is sent as id_token_hint if available but is not required.

state and nonce are validated before claims are returned. If subject is provided, the sub claim must also match — otherwise any active session's claims are returned.

const session = await oidcClient.user.session({
responseType: 'id_token',
subject: knownUserId, // optional — omit to get claims for whoever is logged in
});
if (!('error' in session)) {
console.log(session.claims); // JWTPayload
}

Returns { mode: 'id_token', claims: JWTPayload } on success.

PingOne SaaS does not support response_type=none. Its authorization server only accepts code, token, and id_token as response types — none is not listed as an option in the PingOne application configuration, the PingOne response types reference, the PingOne Platform API reference, or the PingOne support for response_mode options.

Use responseType: 'id_token' when connecting to PingOne.

Here's a practical example of implementing a redirect-based authentication flow. The user is redirected to the OIDC provider's login page:

import { oidc } from '@forgerock/oidc-client';

// 1. Initialize the client
const oidcClient = await oidc({ config });

// 2. Generate authorization URL and redirect user to OIDC provider
const authUrl = await oidcClient.authorize.url();
if (typeof authUrl !== 'string' && 'error' in authUrl) {
console.error('Failed to generate authorization URL:', authUrl.error);
} else {
// Redirect to OIDC provider's login page
window.location.assign(authUrl);
}

// After user logs in and is redirected back to your app with authorization code
// 3. Exchange authorization code for tokens
const urlParams = new URLSearchParams(window.location.search);
const code = urlParams.get('code');
const state = urlParams.get('state');

const tokens = await oidcClient.token.exchange(code, state);
if ('error' in tokens) {
console.error('Failed to exchange code for tokens:', tokens.error);
}

// 4. Retrieve user information
const userInfo = await oidcClient.user.info();
if ('error' in userInfo) {
console.error('Failed to fetch user info:', userInfo.error);
}

// 5. Later, when user wants to logout
const logoutResult = await oidcClient.user.logout();
if ('error' in logoutResult) {
console.error('Logout failed:', logoutResult.error);
}

Here's an example of initiating the authorization process in the background without user interaction. This method returns the authorization code and state directly:

import { oidc } from '@forgerock/oidc-client';

// 1. Initialize the client
const oidcClient = await oidc({ config });

// 2. Start authorization in the background
const authResponse = await oidcClient.authorize.background();
if ('error' in authResponse) {
console.error('Background authorization failed:', authResponse.error);
} else {
// 3. Exchange the authorization code for tokens
const tokens = await oidcClient.token.exchange(authResponse.code, authResponse.state);
if ('error' in tokens) {
console.error('Failed to exchange code for tokens:', tokens.error);
}

// 4. Retrieve user information
const userInfo = await oidcClient.user.info();
if ('error' in userInfo) {
console.error('Failed to fetch user info:', userInfo.error);
}

// 5. Later, when user wants to logout
const logoutResult = await oidcClient.user.logout();
if ('error' in logoutResult) {
console.error('Logout failed:', logoutResult.error);
}
}

Use automatic token renewal to keep the user's session valid. With the backgroundRenew option, this will either return valid tokens from storage if they exist or fetch new tokens if they are expired.

// Get tokens with automatic renewal if expired
const tokens = await oidcClient.token.get({
backgroundRenew: true,
});

if ('error' in tokens) {
console.error('Failed to retrieve tokens:', tokens.error);
} else {
console.log('Access token:', tokens.access_token);
}

The library uses a consistent error handling pattern. All methods return either a success response or an error object. Check if the response contains an error property:

// Pattern for handling responses
const result = await oidcClient.user.info();
if ('error' in result) {
// Handle error case
console.error('Error:', result.error);
console.error('Error description:', result.error_description);
} else {
// Handle success case
console.log('User:', result);
}

Modules

index
types