The headless OAuth/OIDC Authorization Server in the Externalized Login & Consent pattern. A thin client of Authlete 3.0 on the inside, a standard OAuth/OIDC surface on the outside, with login and consent handed off to a separate UI (auth-ui).
Built on TypeScript · Hono · @authlete/typescript-sdk.
This is the AS half of the pair. The AS is headless — every screen a user sees during sign-in or consent is rendered by
auth-ui. The AS only owns protocol endpoints and the redirect back to the RP.
Intent. Decouple user authentication and consent from the OAuth/OIDC Authorization Server. The AS stays a thin, spec-compliant surface; a separate UI application (auth-ui) owns everything the user touches. The AS holds no per-transaction state.
| Component | Responsibility | What it sees |
|---|---|---|
| Relying Party (RP) | Initiates /authorize; receives code/tokens. |
Only the AS. |
| Authorization Server (AS) — this repo | OAuth/OIDC endpoints (/authorize, /token, /userinfo, /par, /introspect, /revoke, /jwks, .well-known/*). Delegates user-facing flow to auth-ui; owns the final redirect back to the RP. |
RP, Authlete, auth-ui — never external IdPs. |
| auth-ui | Authenticates the user with any combination of factors (password, MFA, passkeys, federation); collects consent; records the decision against the opaque interaction ticket. | Only the opaque ticket id — no codes, no tokens, no RP redirect_uris. |
| Authlete | OAuth/OIDC protocol engine. Owns per-transaction state via tickets. | Never reachable from the browser; only the AS calls it. |
- The Authlete ticket is the only handle for an in-flight authorization. Auth result, consent decision, and request context all hang off it.
- The AS holds no per-transaction state. The browser carries only the ticket id; the AS exchanges it back for context as needed. (One transitional exception is documented inline in
src/userstore.ts.) auth-uiholds the user session but not the OAuth transaction.
┌──────────┐ OAuth / OIDC ┌──────────┐ interaction protocol ┌──────────┐
│ RP │ ─────────────────→ │ AS │ ───(bearer-auth)──────→ │ auth-ui │
└──────────┘ │ (this) │ │ │
│ │ @authlete/sdk │ │
│ │ ─────────────────→ Authlete │
└──────────┘ └─────┬────┘
│
(future) federated IdPs · MFA · passkeys
- AS outward to RPs: standard OAuth/OIDC. One spec, no surprises.
- AS ↔ auth-ui: the Interaction Protocol — a bilateral signed-JWT contract for handing off the user-facing flow and exchanging state. Two channel modes (back-channel for production, front-channel for dev/test).
- AS ↔ Authlete: standard Authlete SDK over HTTPS.
- AS never federates outward. No social login, no upstream OIDC, no SAML. All of that lives in
auth-ui.
- Implementation-portable. A thin Authlete client with no user state can be this Node service, a sidecar, a reverse-proxy plugin, or live inside an API gateway / edge worker.
- Authentication grows in
auth-ui. MFA, passkeys, federation, step-up, risk-based prompts — none of which the AS ever sees. - Consent grows in
auth-ui. Granular per-claim choices, Rich Authorization Requests (RAR), persistent grant management — all UI work behind the same ticket interface. - Independent deploy and scale. Two services, one narrow protocol between them.
This separation matches the architecture Authlete is designed around: the engine owns the spec and per-transaction state; you own the user experience.
| Path | Spec | Purpose |
|---|---|---|
GET/POST /oauth/authorize |
OAuth 2.0, OIDC Core | Authorization endpoint — redirects to the interaction app for login + consent. |
POST /oauth/token |
RFC 6749 §3.2 | Token endpoint — authorization_code, refresh_token, client_credentials. |
GET/POST /oauth/userinfo |
OIDC Core §5.3 | UserInfo endpoint. |
POST /oauth/par |
RFC 9126 | Pushed Authorization Requests. |
POST /oauth/introspect |
RFC 7662 | Token introspection. |
POST /oauth/revoke |
RFC 7009 | Token revocation. |
GET /oauth/jwks |
RFC 7517 | Service JWK Set (merges Authlete-managed keys + the AS's own interaction-protocol signing key). |
GET /.well-known/openid-configuration |
OIDC Discovery | OIDC discovery metadata. |
GET /.well-known/oauth-authorization-server |
RFC 8414 | OAuth AS metadata. |
GET /api/authorizations/{id} |
Interaction Protocol | Interaction app fetches in-flight authorization state (JWT-bearer auth). |
POST /api/authorizations/{id}/decision |
Interaction Protocol | Interaction app submits the user's decision (JWT-bearer auth). |
GET /authorizations/{id}/resume |
Interaction Protocol | Browser returns here from the interaction app; the AS calls Authlete issue/fail and redirects the RP. |
The AS hands off all user-facing interactions (sign-in, consent, MFA, …) to a separate interaction app over the Interaction Protocol — a bilateral signed-JWT contract.
- Full spec:
INTERACTION_PROTOCOL.md— JWT envelope, verification rules, channel modes (back-channel for production, front-channel for dev/test), URL surface, per-operation claim shapes. - Endpoints this AS exposes for the protocol are listed in the Endpoints table above.
- Authentication is per-request signed JWT in
Authorization: Bearer. Each peer publishes a JWKS; each verifies the other's signatures against the published keyset. No OAuth-client registration is used by this protocol.
Copy .env.example to .env and fill in:
| Variable | Purpose |
|---|---|
AUTHLETE_BASE_URL |
Authlete cluster URL (default https://us.authlete.com). |
AUTHLETE_SERVICE_ID |
Numeric service id from the Authlete console. |
AUTHLETE_API_TOKEN |
Service access token from the Authlete console. |
AS_BASE_URL |
This server's public URL (RPs and auth-ui use it). |
AUTH_UI_URL |
Where auth-ui is reachable. |
PORT |
Listen port (default 3000). |
AS_CORS_ORIGINS |
Comma-separated allowlist of browser origins (e.g. Authlete OAuth Playground). Empty disables CORS. |
- Sign up at https://us.authlete.com and create a new Authlete 3.0 service.
- Register a test RP client in the service (Authorization Code + PKCE) for end-to-end testing.
- Generate an ES256 key pair for the AS's interaction-protocol signing. Register the public JWK with the Authlete service's JWKS (so it shows up in
/oauth/jwks); keep the private JWK in this AS's env asAS_SIGNING_JWKS. - Populate the AS's
.envfrom the Authlete console (service id + API token + URLs). - The interaction app needs its own ES256 key pair and JWKS publication — see its own setup docs.
pnpm install
pnpm devServer boots at http://localhost:3000. Health probe:
curl http://localhost:3000/healthDiscovery sanity:
curl http://localhost:3000/.well-known/openid-configuration | jq .End-to-end is exercised by auth-ui's smoke harness (auth-ui/scripts/smoke-e2e.mjs).
The AS surface grows with the OAuth/OIDC spec; authentication features grow in auth-ui.
- FAPI 2.0 — DPoP, JAR, JARM (PAR already shipped).
- mTLS client auth (
tls_client_auth). - CIBA (
urn:openid:params:grant-type:ciba). - Dynamic Client Registration (RFC 7591/7592).
- RP-Initiated Logout / Front- and Back-channel Logout.
- Grants Management API.
- Per-claim consent forwarding — plumb
consentedClaimsend-to-end at/auth/authorization/issueso/userinfohonors precisely what the user agreed to release (see theTODO(claims-leakage)block insrc/routes/userinfo.ts). - Front-channel transport implementation — JWT-via-browser-redirect carrier for dev/test deployments where the interaction app isn't directly reachable from the AS. Contract is already specified in
INTERACTION_PROTOCOL.md; only back-channel is shipped today.