ZuploZuplo
LoginStart for Free
  • Documentation
  • API Reference
Introduction
Getting Started
    Develop on the web portal
      1 - Setup Your Gateway2 - Rate Limiting3 - API Key Auth4 - Deploy5 - Dynamic Rate LimitingDynamic MCP Server - Quickstart
    Develop locally with the CLI
      1 - Setup Your Gateway2 - Rate Limiting3 - API Key Auth4 - Deploy5 - Dynamic Rate LimitingDynamic MCP Server - Quickstart
Concepts
Development
Policies
Handlers
API Keys
Rate Limiting
MCP Server
MCP Gateway
AI Gateway
Developer Portal
Monetization
Deploying & Source Control
Analytics
Observability
Networking & Infrastructure
Account Management
Programming API
    Overview
    Request & Context
    Configuration
    Caching APIs
    Data Management
    Extensions & Hooks
    Error Handling
    Logging & Observability
    Types and Interfaces
    Web Standards
    Advanced Topics
      Node ModulesCode ReuseRoute Custom DataClone Request/ResponseRuntime Behaviorszp-body-removedZuplo Identity TokenJWT Service PluginOAuth Protected Resource Plugin
Build with AI
Zuplo CLI
Migration Guides
Platform LimitsSecuritySupportTrust & ComplianceChangelog
powered by Zudoku
Advanced Topics

JWT Service Plugin

Enterprise Feature

JWT Service Plugin is available as an add-on as part of an enterprise plan. If you would like to purchase this feature, please contact us at sales@zuplo.com or reach out to your account manager.

Most enterprise features can be used in a trial mode for a limited time. Feel free to use enterprise features for development and testing purposes.

The JWT Service Plugin allows you to create and issue short-lived JSON Web Tokens (JWTs) within your Zuplo API. This plugin is useful for scenarios where you need to issue tokens for authentication, authorization, or other purposes.

The plugin essentially turns your Zuplo API into its own identity provider that can issue JWTs. Your Zuplo API will also serve the standard /.well-known/openid-configuration endpoint and associated JWKS endpoint which can be used by clients to discover the public keys used to verify the JWTs issued by your API.

JWT Token

By default, this service issues JWTs using the EdDSA algorithm. This is the recommended algorithm for new applications due to its strong security properties and performance characteristics. However, not every library supports EdDSA, so you should ensure that your client library can handle this algorithm. If you need a different algorithm (for example, for compatibility with an existing key pair or client library), use the algorithm configuration option.

Use Cases

Some of the common use cases for the JWT service plugin include:

  • Securing downstream APIs by issuing JWTs that can be used to verify that the request is coming from your Zuplo API
  • Securing requests to other Zuplo API gateways (for example, when using the Federated Gateway capability on Zuplo managed dedicated deployments.)
  • Calling third-party APIs that can be configured with federated identity such as AWS, Azure, or Google Cloud.
  • Issuing short lived tokens for client side applications

Setup

To set up the JWT Service Plugin, you need to register it in your zuplo.runtime.ts file.

modules/zuplo.runtime.ts
import { RuntimeExtensions, JwtServicePlugin } from "@zuplo/runtime"; export function runtimeInit(runtime: RuntimeExtensions) { // Default configuration (no options) const jwtService = new JwtServicePlugin(); runtime.addPlugin(jwtService); }

Configuration Options

The JWT Service Plugin accepts optional configuration to customize its behavior. You can pass a configuration object to the constructor:

modules/zuplo.runtime.ts
import { RuntimeExtensions, JwtServicePlugin, JwtServicePluginOptions, } from "@zuplo/runtime"; export function runtimeInit(runtime: RuntimeExtensions) { // Example 1: Custom configuration with both options const options: JwtServicePluginOptions = { // Custom base path for the issuer endpoint (default: "/__zuplo/issuer") basePath: "/custom", // Token expiration time (default: "1h") // Can be a number (seconds) or a time span string expiresIn: "5m", // or 300 for seconds }; const jwtService = new JwtServicePlugin(options); runtime.addPlugin(jwtService); }

Available Options

  • basePath (optional): The base path for the JWT issuer endpoint. Default is "/__zuplo/issuer". This affects the issuer URL and OIDC configuration endpoints.

  • algorithm (optional): The asymmetric signing algorithm used for issued JWTs. Default is "EdDSA". Supported values are EdDSA, RS256, RS384, RS512, PS256, PS384, PS512, ES256, ES384, and ES512. The algorithm must match the configured key pair — for example, an RSA key requires an RS* or PS* value and an Ed25519 key requires EdDSA. Symmetric algorithms (like HS256) aren't supported because the plugin publishes a JWKS endpoint.

  • expiresIn (optional): Sets the default expiration time for JWTs. Default is "1h". Can be either:

    • A number: Direct value in seconds (for example, 300 for 5 minutes)
    • A string: Time span format (for example, "5 minutes", "1 hour", "7 days")

    Valid time units include:

    • Seconds: "sec", "secs", "second", "seconds", "s"
    • Minutes: "minute", "minutes", "min", "mins", "m"
    • Hours: "hour", "hours", "hr", "hrs", "h"
    • Days: "day", "days", "d"
    • Weeks: "week", "weeks", "w"
    • Years: "year", "years", "yr", "yrs", "y" (365.25 days)

    Examples:

    Code
    expiresIn: 300; // 300 seconds expiresIn: "5 minutes"; // 5 minutes expiresIn: "2 hours"; // 2 hours expiresIn: "7 days"; // 7 days expiresIn: "30 mins"; // 30 minutes

    Note: Individual JWT creation can override this default by specifying expiresIn in the signJwt method.

Usage

Once the plugin is registered, you can use it to issue JWTs in custom handlers or policies.

modules/handlers/jwt-issue.ts
import { ZuploRequest, ZuploContext, JwtServicePlugin } from "@zuplo/runtime"; export async function getJwt(request: ZuploRequest, context: ZuploContext) { const jwt = await JwtServicePlugin.signJwt({ subject: "test-subject", }); return new Response(jwt, { headers: { "content-type": "text/plain" }, }); }

JWT Issuer and OIDC Configuration

When the JWT Service Plugin is enabled, your Zuplo API acts as an identity provider with the following endpoints:

  • Issuer URL: https://{deploymentName}.zuplo.app/__zuplo/issuer (or your custom domain if configured)
  • OIDC Configuration: https://{deploymentName}.zuplo.app/__zuplo/issuer/.well-known/openid-configuration
  • JWKS Endpoint: https://{deploymentName}.zuplo.app/__zuplo/issuer/.well-known/jwks.json

The OIDC configuration endpoint returns a standard OpenID Connect discovery document that includes the JWKS URI for retrieving the public keys used to verify JWTs.

Creating a JWT Authorization Policy

A common pattern is to create a custom policy that automatically adds JWT tokens to outbound requests. This is useful when calling downstream APIs that require authentication.

Here's an example of a custom policy that adds a JWT to the Authorization header of outbound requests:

modules/policies/jwt-auth-upstream.ts
import { ZuploRequest, ZuploContext, RequestHandlerPlugin, JwtServicePlugin, } from "@zuplo/runtime"; export default async function (request: ZuploRequest, context: ZuploContext) { // Generate a JWT with the configured options const jwt = await JwtServicePlugin.signJwt({ subject: request.user?.sub || "api-gateway", audience: request.url, }); // Add the JWT to the Authorization header of the request const headers = new Headers(request.headers); headers.set("Authorization", `Bearer ${jwt}`); return new ZuploRequest(request, { headers }); }

Validating JWTs in Upstream Services

Upstream services can validate the JWTs issued by your Zuplo API by verifying the signature and claims. The examples below use EdDSA, the plugin's default signing algorithm. If you configured a different algorithm using the algorithm option, use that value in the algorithms list instead.

Node.js/Express Example

This example uses the jose library because the popular jsonwebtoken library doesn't support the EdDSA algorithm.

validate-jwt.mjs
import { createRemoteJWKSet, jwtVerify } from "jose"; // Replace with your actual Zuplo deployment name or custom domain const ISSUER = "https://my-api.zuplo.app/__zuplo/issuer"; // Create a remote JWK Set that fetches and caches the public keys const JWKS = createRemoteJWKSet(new URL(`${ISSUER}/.well-known/jwks.json`)); // Middleware to validate JWT async function validateJwt(req, res, next) { const token = req.headers.authorization?.replace("Bearer ", ""); if (!token) { return res.status(401).json({ error: "No token provided" }); } try { const { payload } = await jwtVerify(token, JWKS, { issuer: ISSUER, algorithms: ["EdDSA"], }); req.user = payload; next(); } catch (err) { return res .status(401) .json({ error: "Invalid token", details: err.message }); } } // Example usage app.get("/protected", validateJwt, (req, res) => { res.json({ message: "Access granted", user: req.user, }); });

Python/FastAPI Example

EdDSA validation in PyJWT requires the cryptography package. Install PyJWT with the crypto extra: pip install pyjwt[crypto].

The keys in Zuplo's JWKS don't include a kid, so this example loads the key directly from the JWKS document rather than using PyJWKClient, which only matches keys by kid.

validate_jwt.py
from fastapi import FastAPI, Depends, HTTPException, Security from fastapi.security import HTTPBearer, HTTPAuthorizationCredentials import jwt from jwt import PyJWK import requests app = FastAPI() security = HTTPBearer() # Replace with your actual Zuplo deployment name or custom domain ISSUER = "https://my-api.zuplo.app/__zuplo/issuer" JWKS_URL = f"{ISSUER}/.well-known/jwks.json" def get_signing_key() -> PyJWK: # Zuplo publishes a single signing key. Consider caching this # response briefly to avoid fetching the JWKS on every request. jwks = requests.get(JWKS_URL, timeout=10).json() return PyJWK.from_dict(jwks["keys"][0]) async def validate_token(credentials: HTTPAuthorizationCredentials = Security(security)): token = credentials.credentials try: # Verify and decode the token payload = jwt.decode( token, get_signing_key(), algorithms=["EdDSA"], issuer=ISSUER, options={"verify_exp": True} ) return payload except jwt.ExpiredSignatureError: raise HTTPException(status_code=401, detail="Token has expired") except jwt.InvalidTokenError as e: raise HTTPException(status_code=401, detail=f"Invalid token: {str(e)}") @app.get("/protected") async def protected_route(token_data: dict = Depends(validate_token)): return { "message": "Access granted", "user": token_data }

Dynamic OIDC Discovery

For more flexible JWT validation, you can dynamically discover the OIDC configuration based on the issuer claim in the JWT. This example fetches the issuer's OIDC discovery document to find the JWKS endpoint, then verifies the token with the same jose library used in the Node.js example above.

Security Warning

This approach is particularly useful when you have multiple Zuplo APIs with different issuers or when the issuer URL might change (for example, between environments). It's CRITICAL that you validate the issuer claim in the JWT to ensure you are only allowing tokens from trusted issuers.

validate-jwt-dynamic.ts
import { createRemoteJWKSet, decodeJwt, jwtVerify } from "jose"; import type { JWTPayload } from "jose"; import type { NextFunction, Request, Response } from "express"; // Make the verified JWT payload available as req.user declare global { namespace Express { interface Request { user?: JWTPayload; } } } const ALLOWED_ISSUERS = [ "https://my-api.zuplo.app/__zuplo/issuer", "https://another-api.zuplo.app/__zuplo/issuer", // Add more allowed issuers as needed ]; // Cache the remote JWK Set for each issuer so discovery only runs once. // jose handles JWKS caching and key rotation automatically. const jwksCache = new Map<string, ReturnType<typeof createRemoteJWKSet>>(); async function getJwks(issuer: string) { let jwks = jwksCache.get(issuer); if (!jwks) { // Discover the OIDC configuration for the issuer const response = await fetch(`${issuer}/.well-known/openid-configuration`); if (!response.ok) { throw new Error(`OIDC discovery failed with status ${response.status}`); } const metadata = (await response.json()) as { issuer?: string; jwks_uri?: string; }; if (metadata.issuer !== issuer) { throw new Error("Discovery document issuer mismatch"); } if (!metadata.jwks_uri) { throw new Error("Issuer metadata is missing jwks_uri"); } jwks = createRemoteJWKSet(new URL(metadata.jwks_uri)); jwksCache.set(issuer, jwks); } return jwks; } async function validateJwtDynamic(token: string): Promise<JWTPayload> { // Read the issuer claim without verifying the signature yet const { iss: issuer } = decodeJwt(token); if (!issuer) { throw new Error("No issuer claim in token"); } // Validate the issuer against the allow list before fetching anything if (!ALLOWED_ISSUERS.includes(issuer)) { throw new Error(`Issuer ${issuer} isn't allowed`); } // Verify the signature and standard claims const { payload } = await jwtVerify(token, await getJwks(issuer), { issuer, algorithms: ["EdDSA"], }); return payload; } // Express middleware example function validateJwtMiddleware( req: Request, res: Response, next: NextFunction, ) { const token = req.headers.authorization?.replace("Bearer ", ""); if (!token) { res.status(401).json({ error: "No token provided" }); return; } validateJwtDynamic(token) .then((payload) => { req.user = payload; next(); }) .catch((error: Error) => { res .status(401) .json({ error: `JWT validation failed: ${error.message}` }); }); } // Usage app.get("/protected", validateJwtMiddleware, (req, res) => { res.json({ message: "Access granted", user: req.user, }); });

This approach is particularly useful when:

  • You need to validate JWTs from multiple Zuplo APIs with different issuers
  • The issuer URL might change (for example, between environments)
  • You want to leverage automatic OIDC discovery for configuration updates

Important Validation Steps

When validating JWTs from Zuplo:

  1. Verify the signature using the public keys from the JWKS endpoint
  2. Check the issuer matches your Zuplo API's issuer URL
  3. Validate expiration to ensure the token hasn't expired
  4. Verify audience if your tokens include audience claims
  5. Check any custom claims required by your application

The JWT Service Plugin handles key rotation automatically, so always fetch the current public keys from the JWKS endpoint rather than hard coding them.

Edit this page
Last modified on June 10, 2026
Zuplo Identity TokenOAuth Protected Resource Plugin
On this page
  • JWT Token
  • Use Cases
  • Setup
    • Configuration Options
  • Usage
    • JWT Issuer and OIDC Configuration
    • Creating a JWT Authorization Policy
  • Validating JWTs in Upstream Services
    • Node.js/Express Example
    • Python/FastAPI Example
    • Dynamic OIDC Discovery
    • Important Validation Steps
TypeScript
TypeScript
TypeScript
TypeScript
TypeScript
Javascript
TypeScript