Troubleshooting
Common problems
Section titled “Common problems”Multiple decisions for the same request
Section titled “Multiple decisions for the same request”If you are seeing multiple decisions for the same request, it is likely that Arcjet is running in multiple places in your application. This can happen if you are using a middleware and also calling the Arcjet SDK in a route handler.
You can fix this by ensuring that Arcjet is only called once per request. You can combine multiple rules into a single client configuration and/or you can exclude routes from your middleware.
Decision errors from platform liveness or health checks when using Arcjet as a middleware
Section titled “Decision errors from platform liveness or health checks when using Arcjet as a middleware”Platforms like Kubernetes have support for automatic liveness and health
check probes. These requests do not come from a browser and do not have all
the same metadata that browsers do (and will not include important headers
like X-Real-Ip or X-Forwarded-For). This will cause Arcjet to reject these
probes, which will make the platform think your app is either not alive or
unhealthy. This can manifest in CrashLoopBackoff errors in Kubernetes or your
platform of choice restarting your app constantly.
In order to fix this, omit your healthcheck route from your Arcjet middleware.
Next.js
Section titled “Next.js”Assuming you have your Arcjet middleware at middleware.ts, adjust the
config.matcher regex to include an omission for the /healthz route:
export const config = {  // matcher tells Next.js which routes to run the middleware on.  // This runs the middleware on all routes except for static assets.  matcher: ["/((?!_next/static|_next/image|favicon.ico).*)"],  matcher: ["/((?!_next/static|_next/image|favicon.ico|healthz).*)"],};Then create an API route at /app/healthz/route.ts:
import { NextRequest, NextResponse } from "next/server";
export const GET = async (req: NextRequest) => {  if (req.headers.get("user-agent") !== "healthz") {    return NextResponse.json({ error: "Not found" }, { status: 404 });  }
  return new Response("OK", {    status: 200,    headers: {      "Content-Type": "text/plain",    },  });};Then create an API route at /pages/healthz.ts:
import type { NextApiRequest, NextApiResponse } from "next";
export default function handler(req: NextApiRequest, res: NextApiResponse) {  if (req.headers["user-agent"] !== "healthz") {    return res.status(404).send("Not found");  }
  res.status(200).send("OK");}Then create an API route at /app/healthz/route.js:
import { NextResponse } from "next/server";
export const GET = async (req) => {  if (req.headers.get("user-agent") !== "healthz") {    return NextResponse.json({ error: "Not found" }, { status: 404 });  }
  return new Response("OK", {    status: 200,    headers: {      "Content-Type": "text/plain",    },  });};Then create an API route at /pages/healthz.js:
export default function handler(req, res) {  if (req.headers["user-agent"] !== "healthz") {    return res.status(404).send("Not found");  }
  res.status(200).send("OK");}When you deploy your app next, Arcjet should not trigger on this route.
Override development IP
Section titled “Override development IP”Arcjet’s IP geolocation and VPN & proxy
detection features require a real IP address
to work. In local development, Arcjet defaults to 127.0.0.1 so the IP analysis
metadata will be empty.
You can set the IP address to a real public IP by overriding the IP address
field (ip) in the request object that is passed to the Arcjet protect
function.
// Override the IP address in the request object if `ip` is provided. You could// set this using an environment variable or some other method to ensure it is// only set in development.//const ip = "8.8.8.8"; // Google's public DNS serverconst decision = await aj.protect(ip ? { ...req, ip } : req);Common errors
Section titled “Common errors”[unauthenticated] invalid key
Section titled “[unauthenticated] invalid key”✦Aj Encountered problem getting remote decision: [unauthenticated] invalid keyThis error means that the Arcjet key you are using is invalid. Each site has a unique key to identify it within Arcjet. You can find your key in the Arcjet dashboard by clicking your site name and then going to the SDK installation tab.
Arcjet keys are always prefixed with ajkey_
You provide the Arcjet SDK with the key when instantiating the client. For
example in this case the key is defined in an environment variable called
ARCJET_KEY:
import arcjet from "arcjet";
const aj = arcjet({  // Get your site key from https://app.arcjet.com  // and set it as an environment variable rather than hard coding.  // See: https://www.npmjs.com/package/dotenv  key: process.env.ARCJET_KEY,  rules: [    // Protect against common attacks with Arcjet Shield    shield({      mode: "LIVE", // will block requests. Use "DRY_RUN" to log only    }),  ],});[deadline_exceeded] the operation timed out
Section titled “[deadline_exceeded] the operation timed out”✦Aj Encountered problem getting remote decision: [deadline_exceeded] the operation timed outConnectError: [deadline_exceeded] the operation timed outThis error means that the Arcjet SDK was unable to connect to the Cloud API within the short timeout period. This is usually caused by a network issue. The SDK is designed to fail open so that in the event of a network problem or outage, requests will be allowed.
If you are seeing this error frequently, please contact us so we can investigate.
Arcjet will use 127.0.0.1 when missing public IP address in development mode
Section titled “Arcjet will use 127.0.0.1 when missing public IP address in development mode”Arcjet expects a valid public IP address to enable features like geolocation and
VPN/proxy detection. In production, if the IP is invalid (e.g. 127.0.0.1), the
request is rejected to prevent spoofing.
In development, private/internal addresses are common. To avoid errors, set the
environment variable ARCJET_ENV to development. This will disable the IP
address validation and allow you to test your application without errors.
This log means Arcjet is falling back to 127.0.0.1 because it’s running in
development mode. If this appears in production, check that ARCJET_ENV is not
incorrectly set to development.
generateFingerprint: ip is empty / [failed_precondition] client IP not provided
Section titled “generateFingerprint: ip is empty / [failed_precondition] client IP not provided”Arcjet normally detects when it is running in a local environment and uses a
default IP address of 127.0.0.1. However, if it can’t determine whether it is
running locally or not then it will be safe and assume it is running in
production. In production environments we disallow private/internal addresses,
but they are allowed in development to allow you to test locally.
You can force the SDK to run in development mode by setting the environment
variable ARCJET_ENV to development in your .env.local file:
ARCJET_ENV=developmentArcjet uses the client IP address as part of a fingerprint to identify where each request is coming from. If the IP address is not provided, or it is running in production and a private/internal address is provided, then Arcjet will show this error.
[failed_precondition] request details failed validation
Section titled “[failed_precondition] request details failed validation”Arcjet expects certain fields to exist on the object used to analyze each request. This is normally handled by the SDK for the framework you are using, but if you are doing something custom then you will need to provide the correct fields:
The required fields are:
- ip- The client IP address. Internally our SDKs use our- @arcjet/ippackage to detect the correct IP which you can also use directly.
- method- The HTTP method of the request.
- host- The host of the request.
- pathor- url- The path of the request. In the Node.js SDK this field is called- url. If you are not using an adapter, this is called- path.
- headers- The headers of the request.
Property ‘[INTERNALS]’ is missing in type
Section titled “Property ‘[INTERNALS]’ is missing in type”When using the Arcjet Next.js SDK you may see a warning in your editor like this:
Argument of type x is not assignable to parameter of type 'NextMiddleware'.  Types of parameters 'req' and 'request' are incompatible.    Property '[INTERNALS]' is missing in type 'import(".../next/dist/server/web/spec-extension/request").NextRequest' but required in type 'import(".../next/dist/server/web/spec-extension/request").NextRequest'.ts(2345)request.d.ts(7, 5): '[INTERNALS]' is declared here.This is usually caused by a mismatch between the version of Next.js you are using and the version of the Arcjet SDK. You have several options for fixing this:
- 
Update the Arcjet SDK to the latest version and update your version of Next.js to the latest version. We try to keep the SDK up to date with the latest Next.js releases so they should generally match. 
- 
Add a resolution override to your package.jsonto force the SDK to use the same version of Next.js. For example, if you are using Next.js 14.1.4 you can add this override to yourpackage.jsonto force the Arcjet SDK to use the same version:{"overrides": {"next": "14.1.4"}}
- 
You can ignore the warning. The warning is just a TypeScript error and does not affect the runtime behavior of your application. You can add a comment on the line that is causing the error to ignore it: // @ts-ignore
Calling protect() with no rules is deprecated
Section titled “Calling protect() with no rules is deprecated”When using the Arcjet SDK you may see a warning like this:
Calling `protect()` with no rules is deprecated. Did you mean to configure the Shield rule?Prior to the JS SDK release v1.0.0-alpha.12 we allowed empty rules and
automatically configured a Shield rule for you. From v1.0.0-alpha.12 onwards
you must explicitly configure the Shield rule. This allows you to choose if you
want to block suspicious requests or not. We are maintaining the old behavior
for a while to give you time to update your code, which is why this is warning
you. Shield is still configured automatically but you should now migrate to
explicitly configuring it.
Check out the Shield documentation to learn how to configure Shield rules. To restore the old behavior you can add a simple Shield rule:
import arcjet, { shield } from "@arcjet/next";
const aj = arcjet({  // Get your site key from https://app.arcjet.com  // and set it as an environment variable rather than hard coding.  // See: https://nextjs.org/docs/pages/building-your-application/configuring/environment-variables  key: process.env.ARCJET_KEY,  rules: [    // Protect against common attacks with Arcjet Shield    shield({      mode: "LIVE", // will block requests. Use "DRY_RUN" to log only    }),  ],});The current file is a CommonJS module whose imports will produce ‘require’ calls
Section titled “The current file is a CommonJS module whose imports will produce ‘require’ calls”When using the Arcjet SDK you may see a warning like this:
The current file is a CommonJS module whose imports will produce 'require'calls; however, the referenced file is an ECMAScript module and cannot beimported with 'require'.This warning is caused by a mismatch between the module system used by the Arcjet SDK and the module system used by your application. The Arcjet SDK uses ECMAScript modules (ESM) but your application is using CommonJS modules (CJS).
The Arcjet SDK does not support CJS. You can fix this by changing your
application to use ESM. This is usually done by adding "type": "module" to
your package.json.
Alternatively, you can rename the files where you are using the Arcjet SDK to use
the .mjs or .mts extension. This will tell Node.js to treat the file as an
ESM file.
Error: Log is required / Error: Client is required
Section titled “Error: Log is required / Error: Client is required”You may see an error like this:
Error: Log is required    at arcjet (/node_modules/arcjet/index.js:520:15)...Error: Client is required    at arcjet (/node_modules/arcjet/index.js:526:15)This error usually means that you are trying to use the base arcjet package
rather than an adapter. You probably meant to use the @arcjet/node package
instead.
The arcjet package is a low-level package that is used to build adapters for
different frameworks and runtimes we support. Unless you are
doing something custom, we recommend using one of the adapters.
Error: Body already read or Body is unusable
Section titled “Error: Body already read or Body is unusable”You may see this error if you try to access the request body after it has already been read by something else.
For example, if Arcjet processes the body to analyze it for sensitive information and then you try and access the body later in your handler.
Another example is if you access a field in the body to perform email validation and then later try to access the body again in another part of your request handler.
To fix this with sensitive information detection, see the Accessing the body section of the reference.
To fix this with other cases, clone the request before accessing the body:
// Clone the request and extract the emailconst { email } = await req.clone().json();
// Call Arcjet with the cloned requestconst decision = await aj.protect(req, { email });
// ...// You can now access the body again laterError: Cannot read properties of undefined (reading ‘flags’)
Section titled “Error: Cannot read properties of undefined (reading ‘flags’)”If you see this error then you are probably running an older version of Next.js. Arcjet supports the active and maintenance LTS versions of Next.js as indicated in the SDK reference. Upgrade your Next.js version to the latest version to fix this error.
http2.connect is not implemented yet!
Section titled “http2.connect is not implemented yet!”You may see this error if you are using an old version of Deno, either directly or through a platform such as Supabase Edge Functions:
http2.connect is not implemented yet!Arcjet supports the following versions of Deno:
- Deno 1.46+ if only using the Deno.serveAPI
OR
- Deno 2.0+ if you are using Deno’s servecommand
Follow the update instructions to upgrade Deno to the latest version.
If you are self-hosting Supabase, you can follow the instructions in the Supabase docs. Otherwise, see this discussion for other Supabase instances.
Get help
Section titled “Get help”We provide technical support for all Arcjet users so if you need help, email us or join our Discord.
Discussion