Using timingSafeEqual
Protect against timing attacks by safely comparing values using timingSafeEqual.
If you want to get started quickly, click on the button below.
This creates a repository in your GitHub account and deploys the application to Cloudflare Workers.
The crypto.subtle.timingSafeEqual function compares two values using a constant-time algorithm. The time taken is independent of the contents of the values.
When strings are compared using the equality operator (== or ===), the comparison will end at the first mismatched character. By using timingSafeEqual, an attacker would not be able to use timing to find where at which point in the two strings there is a difference.
The timingSafeEqual function takes two ArrayBuffer or TypedArray values to compare. These buffers must be of equal length, otherwise an exception is thrown.
Note that this function is not constant time with respect to the length of the parameters and also does not guarantee constant time for the surrounding code.
Handling of secrets should be taken with care to not introduce timing side channels.
In order to compare two strings, you must use the TextEncoder API.
interface Environment {  MY_SECRET_VALUE?: string;}
export default {  async fetch(req: Request, env: Environment) {    if (!env.MY_SECRET_VALUE) {      return new Response("Missing secret binding", { status: 500 });    }
    const authToken = req.headers.get("Authorization") || "";
    if (authToken.length !== env.MY_SECRET_VALUE.length) {      return new Response("Unauthorized", { status: 401 });    }
    const encoder = new TextEncoder();
    const a = encoder.encode(authToken);    const b = encoder.encode(env.MY_SECRET_VALUE);
    if (a.byteLength !== b.byteLength) {      return new Response("Unauthorized", { status: 401 });    }
    if (!crypto.subtle.timingSafeEqual(a, b)) {      return new Response("Unauthorized", { status: 401 });    }
    return new Response("Welcome!");  },};from workers import WorkerEntrypoint, Responsefrom js import TextEncoder, crypto
class Default(WorkerEntrypoint):    async def fetch(self, request):        auth_token = request.headers["Authorization"] or ""        secret = self.env.MY_SECRET_VALUE
        if secret is None:            return Response("Missing secret binding", status=500)
        if len(auth_token) != len(secret):            return Response("Unauthorized", status=401)
        encoder = TextEncoder.new()        a = encoder.encode(auth_token)        b = encoder.encode(secret)
        if a.byteLength != b.byteLength:            return Response("Unauthorized", status=401)
        if not crypto.subtle.timingSafeEqual(a, b):            return Response("Unauthorized", status=401)
        return Response("Welcome!")import { Hono } from 'hono';
interface Environment {  Bindings: {    MY_SECRET_VALUE?: string;  }}
const app = new Hono<Environment>();
// Middleware to handle authentication with timing-safe comparisonapp.use('*', async (c, next) => {  const secret = c.env.MY_SECRET_VALUE;
  if (!secret) {    return c.text("Missing secret binding", 500);  }
  const authToken = c.req.header("Authorization") || "";
  // Early length check to avoid unnecessary processing  if (authToken.length !== secret.length) {    return c.text("Unauthorized", 401);  }
  const encoder = new TextEncoder();
  const a = encoder.encode(authToken);  const b = encoder.encode(secret);
  if (a.byteLength !== b.byteLength) {    return c.text("Unauthorized", 401);  }
  // Perform timing-safe comparison  if (!crypto.subtle.timingSafeEqual(a, b)) {    return c.text("Unauthorized", 401);  }
  // If we got here, the auth token is valid  await next();});
// Protected routeapp.get('*', (c) => {  return c.text("Welcome!");});
export default app;Was this helpful?
- Resources
- API
- New to Cloudflare?
- Directory
- Sponsorships
- Open Source
- Support
- Help Center
- System Status
- Compliance
- GDPR
- Company
- cloudflare.com
- Our team
- Careers
- © 2025 Cloudflare, Inc.
- Privacy Policy
- Terms of Use
- Report Security Issues
- Trademark