docs/content/docs/plugins/oauth-proxy.mdx
A proxy plugin that allows you to proxy OAuth requests. Useful for development and preview deployments where the redirect URL can't be known in advance to add to the OAuth provider.
```ts title="auth.ts"
import { betterAuth } from "better-auth"
import { oAuthProxy } from "better-auth/plugins" // [!code highlight]
export const auth = betterAuth({
plugins: [
oAuthProxy({ // [!code highlight]
productionURL: "https://my-production-app.com", // [!code highlight]
secret: process.env.OAUTH_PROXY_SECRET, // [!code highlight]
}), // [!code highlight]
],
socialProviders: {
github: {
clientId: process.env.GITHUB_CLIENT_ID || "",
clientSecret: process.env.GITHUB_CLIENT_SECRET || "",
},
},
})
```
Set `OAUTH_PROXY_SECRET` to the same value on all environments (production, preview, localhost).
The plugin will automatically route OAuth requests through your production server.
In your OAuth provider's developer console (e.g. GitHub, Google), register the callback URL using your **production** domain. For example:
```
https://my-production-app.com/api/auth/callback/github
```
Only the production callback URL needs to be registered. The plugin handles routing OAuth requests from preview and development environments through production automatically.
Since preview and development servers redirect through production, you need to add them as `trustedOrigins` in your auth config:
```ts title="auth.ts"
export const auth = betterAuth({
// ...other config
trustedOrigins: [ // [!code highlight]
"http://localhost:3000", // [!code highlight]
"https://my-app-*-preview.example.com", // [!code highlight]
], // [!code highlight]
})
```
All environments (production, preview, localhost) must use the same encryption key to communicate. Configure a dedicated secret in the plugin options:
oAuthProxy({
productionURL: "https://my-production-app.com",
secret: process.env.OAUTH_PROXY_SECRET, // Same value on ALL environments // [!code highlight]
})
If you don't configure a shared secret, the plugin falls back to BETTER_AUTH_SECRET. Since production and preview typically have different main secrets (which is correct for security), the OAuth flow will fail with a state_mismatch error.
</Callout>
The plugin allows you to use a single OAuth client (registered with your production URL) across multiple environments like preview deployments or local development.
import { authClient } from "@/lib/auth-client"
await authClient.signIn.social({
provider: "github",
callbackURL: "/dashboard"
})
The encrypted profile data is passed via URL query parameters and can only be decrypted by servers sharing the same secret. This also allows preview deployments to use separate databases from production if needed.
<Callout type="info"> This plugin is intended for development and preview environments. If `baseURL` and `productionURL` are the same, the plugin will not proxy the request. </Callout>productionURL: The URL of your production server. If this value matches the baseURL in your auth config, requests will not be proxied. Defaults to the BETTER_AUTH_URL environment variable.
currentURL: The application's current URL is automatically determined by the plugin. It first checks the request URL, then vendor-specific environment variables from popular hosting providers, and finally falls back to the baseURL in your auth config. You only need to set this if the URL isn't being inferred correctly in your environment.
maxAge: Maximum age in seconds for encrypted profile payloads. Payloads older than this will be rejected to prevent replay attacks. Keep this value short (e.g., 30-60 seconds) to minimize the window for potential replay attacks while still allowing normal OAuth flows. Defaults to 60 seconds.
secret: A dedicated secret used for encrypting and decrypting data during the OAuth proxy flow. When set, this is used instead of the global BETTER_AUTH_SECRET, limiting the blast radius if the key is shared across environments — a leaked proxy secret cannot forge sessions or decrypt other data protected by the main secret. All environments participating in the proxy flow must share the same secret value.
state_mismatch or "State not persisted correctly" errorThis error typically occurs when production and preview environments have different secrets and no shared secret is configured in the plugin options.
What happens:
Solution: Configure a shared secret in the oAuthProxy options on all environments:
oAuthProxy({
productionURL: "https://my-production-app.com",
secret: process.env.OAUTH_PROXY_SECRET, // [!code highlight]
})
Make sure OAUTH_PROXY_SECRET has the same value on production, preview, and localhost.
Ensure all of the following:
https://production.com/api/auth/callback/github)