Zenuxs OAuth
June 17, 2026 ยท View on GitHub
Universal OAuth 2.0 + PKCE client for Zenuxs auth.
It works in browsers, Node.js, React Native, and hybrid apps with one small API:
const oauth = new ZenuxOAuth({ clientId: 'your-client-id' });
Everything else is optional.
Highlights
- Minimal setup: only
clientIdis required - Auto-detects
fetchin browser and modern Node runtimes - Auto-detects a usable storage adapter
- Frontend modes:
ui,popup,redirect,manual redirectis the default mode in browsersredirectis the default mode on the backend- Embedded UI supports
light,dark, andautothemes - Social providers can fall back from embedded mode to popup or full page
- Same-page callback handling works automatically
- Manual flow is still available when you want full control
- Built-in token refresh, user info, logout, and authenticated fetch
Installation
npm install zenuxs-oauth
Browser CDN:
<script src="https://unpkg.com/zenuxs-oauth@5.3.0/dist/zenux-oauth.min.js"></script>
Quick Start
Browser, same page, default UI mode
This is the new default flow.
If your login page is /login or /login.html, the SDK can:
- start OAuth from that page
- use the same page as the redirect URI
- detect
?code=...on return - finish the callback automatically
<button id="login">Continue with Zenuxs</button>
<script src="https://unpkg.com/zenuxs-oauth@5.3.0/dist/zenux-oauth.min.js"></script>
<script>
const oauth = new ZenuxOAuth({
clientId: 'your-client-id',
theme: 'auto'
});
oauth.on('login', (tokens) => {
console.log('Logged in', tokens);
});
oauth.on('error', (error) => {
console.error('OAuth error', error);
});
window.addEventListener('load', async () => {
await oauth.init();
});
document.getElementById('login').addEventListener('click', async () => {
await oauth.login();
});
</script>
Browser modes
const oauth = new ZenuxOAuth({ clientId: 'your-client-id' });
await oauth.login(); // default browser mode: ui
await oauth.login({ mode: 'ui' }); // embedded bottom-sheet auth
await oauth.login({ mode: 'popup' }); // popup window
await oauth.login({ mode: 'redirect' });// full-page redirect
const authData = await oauth.login({ mode: 'manual' });
console.log(authData.url);
Mode aliases:
inuiiframe
Both map to ui.
Theme and smoother embedded UI
const oauth = new ZenuxOAuth({
clientId: 'your-client-id',
theme: 'dark',
uiTitle: 'Continue with Zenuxs',
uiDescription: 'Secure sign in in a bottom sheet.',
uiFallbackMode: 'popup'
});
Theme options:
autolightdark
The embedded ui mode now:
- opens as a bottom sheet instead of a plain centered iframe box
- shows a loading screen while auth is starting and while the callback is finishing
- asks for confirmation before closing with the
Xbutton - can move to popup or full-page redirect when embedded provider login does not work
Social provider fallback
Some providers such as Google, Discord, and GitHub often do not behave well inside iframes.
The SDK now helps in two ways:
- If you directly request one of those providers,
uimode is promoted topopupautomatically. - If embedded auth returns to your page without usable callback params, the SDK can move the flow to popup or full page.
Example:
await oauth.login({
mode: 'ui',
provider: 'google',
uiFallbackMode: 'popup'
});
Node.js, same route
The same route can start login and finish the callback:
const express = require('express');
const session = require('express-session');
const ZenuxOAuth = require('zenuxs-oauth');
const app = express();
app.use(session({
secret: 'replace-me',
resave: false,
saveUninitialized: false
}));
const oauth = new ZenuxOAuth({
clientId: process.env.ZENUX_CLIENT_ID
});
app.get('/login', async (req, res, next) => {
try {
const tokens = await oauth.login({
request: req,
response: res
});
if (!tokens) {
return;
}
req.session.tokens = tokens;
res.redirect('/dashboard');
} catch (error) {
next(error);
}
});
Node.js, manual flow
const oauth = new ZenuxOAuth({
clientId: process.env.ZENUX_CLIENT_ID,
redirectUri: 'https://your-app.com/auth/callback'
});
app.get('/auth/login', async (req, res) => {
const authData = await oauth.login({ mode: 'manual' });
res.redirect(authData.url);
});
app.get('/auth/callback', async (req, res) => {
const tokens = await oauth.handleCallback({ request: req });
res.json(tokens);
});
Default Behavior
Frontend defaults
- default mode:
redirect - redirect URI: current page without
code,state, or OAuth error params - fetch: auto-detected
- storage: auto-detected
Backend defaults
- default mode:
redirect - fetch: auto-detected from
globalThis.fetch,node-fetch, orundici - storage: in-memory unless you pass your own adapter
Configuration
Only clientId is required.
const oauth = new ZenuxOAuth({
clientId: 'your-client-id',
redirectUri: 'https://your-app.com/login',
scopes: 'openid profile email',
mode: 'ui',
frontendMode: 'ui',
backendMode: 'redirect',
storage: 'sessionStorage',
storagePrefix: 'zenux_oauth_',
fetch: fetch,
debug: true,
theme: 'auto',
usePKCE: true,
validateState: true,
autoRefresh: true,
refreshThreshold: 60,
popupWidth: 540,
popupHeight: 720,
uiWidth: 460,
uiHeight: 720,
uiFallbackMode: 'popup',
uiAllowRedirectFallback: true,
uiCloseConfirm: true,
extraAuthParams: {
prompt: 'login'
},
extraTokenParams: {
client_secret: 'only-for-confidential-server-clients'
}
});
Important options
| Option | Required | Default | Notes |
|---|---|---|---|
clientId | yes | - | The only required option |
redirectUri | no | current page in browser | Useful for dedicated callback routes |
mode | no | environment default | Shortcut for default flow |
frontendMode | no | ui | Browser default |
backendMode | no | redirect | Server default |
theme | no | auto | Embedded UI theme: auto, light, or dark |
storage | no | auto | sessionStorage, localStorage, memory, Map, or custom adapter |
fetch / fetchFunction | no | auto | Override only when needed |
uiFallbackMode | no | popup | Preferred fallback when embedded provider auth cannot continue |
uiAllowRedirectFallback | no | true | Allows full-page fallback when popup is blocked or not desired |
uiCloseConfirm | no | true | Ask for confirmation before closing the embedded sheet |
debug | no | false | Logs SDK internals |
autoRefresh | no | false | Refreshes access tokens when possible |
Main Methods
login(options?)
Starts login, or completes it automatically if the current URL already contains OAuth callback params.
init(options?)
Best for browser page load.
It checks the current URL and finishes the callback automatically when needed.
handleCallback(callbackUrl?, options?)
Manual callback handling.
Use this when you want to process a specific URL or server request yourself.
getAuthorizationUrl(options?)
Returns the authorization request data without redirecting.
getTokens()
Returns stored tokens or null.
refreshTokens()
Uses the refresh token, if available.
getUserInfo()
Calls the user info endpoint using the current access token.
logout(options?)
Clears tokens and optionally revokes them.
getAuthenticatedFetch()
Returns a fetch wrapper that injects the bearer token automatically.
Events
oauth.on('loginRequest', (authData) => {
console.log(authData.url);
});
oauth.on('login', (tokens) => {
console.log('Logged in', tokens);
});
oauth.on('tokenRefresh', (tokens) => {
console.log('Refreshed', tokens);
});
oauth.on('tokenExpired', () => {
console.log('Refreshing soon');
});
oauth.on('logout', () => {
console.log('Logged out');
});
oauth.on('error', (error) => {
console.error(error);
});
Manual Flow Example
const oauth = new ZenuxOAuth({
clientId: 'your-client-id',
redirectUri: 'https://your-app.com/auth/callback'
});
const authData = await oauth.login({ mode: 'manual' });
console.log(authData.url);
console.log(authData.state);
console.log(authData.codeVerifier);
Later:
const tokens = await oauth.handleCallback('https://your-app.com/auth/callback?code=abc&state=xyz');
Authenticated Fetch
const authFetch = oauth.getAuthenticatedFetch();
const response = await authFetch('https://api.example.com/me');
const data = await response.json();
Supported Scopes
console.log(ZenuxOAuth.supportedScopes);
Default scopes:
openid profile email
Examples
- Browser same-page demo:
exmaples/browser/index.html - Browser dedicated callback demo:
exmaples/browser/callback.html - Express same-route demo:
exmaples/node/express-same-route.js testlogin.htmlfor a very small same-page setup
Serve the browser examples over http:// or https://, not file://.
Notes
- Keep secrets on the backend only
- Verify state unless you have a very specific reason not to
- Prefer the default same-page flow unless your app needs a dedicated callback route
popupanduiboth support automatic callback completion- For direct Google, Discord, or GitHub login, popup mode is usually the safest UX
Social API Wrappers
The SDK provides helper methods for interacting with users' connected social accounts (Twitter/X, LinkedIn, YouTube, Facebook, Instagram, etc.).
const oauth = new ZenuxOAuth({ clientId: 'your-client-id' });
await oauth.login();
// Access social providers
const x = oauth.x();
// Listen to events on this specific provider
x.on('post_created', (event) => {
console.log(`Successfully created a post on ${event.provider}!`, event.data);
});
// Get user data from provider
const userData = await x.getData();
// Get posts
const posts = await x.getPost();
// Create a post
await x.createPost({ message: 'Hello from Zenuxs API!' });
// Get analytics
const analytics = await x.analytics();
Supported Providers
oauth.x()oroauth.twitter()oauth.linkedin()oauth.youtube()oauth.facebook()oauth.instagram()
License
MIT