Zachoo
Zachoo2mo ago

Auth token is not a valid JWT with ConvexProviderWithClerk (EXPO)

Issue I'm using Clerk with Convex in an Expo Router app, and everything loads fine, logs me in, and works as expected. However, I get the following error when using ConvexProviderWithClerk:
ERROR Auth token is not a valid JWT, cannot refetch the token
ERROR Auth token is not a valid JWT, cannot refetch the token
If I remove Convex and just use Clerk, the error disappears, and functionality remains the same. Code Setup ❌ Code that Causes the Error (Clerk + Convex)
import { ClerkProvider, ClerkLoaded, useAuth } from "@clerk/clerk-expo";
import { Slot } from "expo-router";
import { tokenCache } from "./cache";
import "../global.css";
import { BottomSheetModalProvider } from "@gorhom/bottom-sheet";
import { GestureHandlerRootView } from "react-native-gesture-handler";
import { ConvexProvider, ConvexReactClient } from "convex/react";
import { ConvexQueryCacheProvider } from "convex-helpers/react/cache";
import { ConvexProviderWithClerk } from "convex/react-clerk";

// Initialize Convex client
const convex = new ConvexReactClient(process.env.EXPO_PUBLIC_CONVEX_URL!);

export default function RootLayout() {
const publishableKey = process.env.EXPO_PUBLIC_CLERK_PUBLISHABLE_KEY!;

return (
<ClerkProvider tokenCache={tokenCache} publishableKey={publishableKey}>
<ClerkLoaded>
<ConvexProviderWithClerk useAuth={useAuth} client={convex}>
<ConvexQueryCacheProvider>
<GestureHandlerRootView style={{ flex: 1 }}>
<BottomSheetModalProvider>
<Slot screenOptions={{ headerShown: false }} />
</BottomSheetModalProvider>
</GestureHandlerRootView>
</ConvexQueryCacheProvider>
</ConvexProviderWithClerk>
</ClerkLoaded>
</ClerkProvider>
);
}
import { ClerkProvider, ClerkLoaded, useAuth } from "@clerk/clerk-expo";
import { Slot } from "expo-router";
import { tokenCache } from "./cache";
import "../global.css";
import { BottomSheetModalProvider } from "@gorhom/bottom-sheet";
import { GestureHandlerRootView } from "react-native-gesture-handler";
import { ConvexProvider, ConvexReactClient } from "convex/react";
import { ConvexQueryCacheProvider } from "convex-helpers/react/cache";
import { ConvexProviderWithClerk } from "convex/react-clerk";

// Initialize Convex client
const convex = new ConvexReactClient(process.env.EXPO_PUBLIC_CONVEX_URL!);

export default function RootLayout() {
const publishableKey = process.env.EXPO_PUBLIC_CLERK_PUBLISHABLE_KEY!;

return (
<ClerkProvider tokenCache={tokenCache} publishableKey={publishableKey}>
<ClerkLoaded>
<ConvexProviderWithClerk useAuth={useAuth} client={convex}>
<ConvexQueryCacheProvider>
<GestureHandlerRootView style={{ flex: 1 }}>
<BottomSheetModalProvider>
<Slot screenOptions={{ headerShown: false }} />
</BottomSheetModalProvider>
</GestureHandlerRootView>
</ConvexQueryCacheProvider>
</ConvexProviderWithClerk>
</ClerkLoaded>
</ClerkProvider>
);
}
53 Replies
Convex Bot
Convex Bot2mo ago
Thanks for posting in <#1088161997662724167>. Reminder: If you have a Convex Pro account, use the Convex Dashboard to file support tickets. - Provide context: What are you trying to achieve, what is the end-user interaction, what are you seeing? (full error message, command output, etc.) - Use search.convex.dev to search Docs, Stack, and Discord all at once. - Additionally, you can post your questions in the Convex Community's <#1228095053885476985> channel to receive a response from AI. - Avoid tagging staff unless specifically instructed. Thank you!
Zachoo
ZachooOP2mo ago
✅ Code That Works (Without Convex)
import { ClerkProvider, ClerkLoaded } from "@clerk/clerk-expo";
import { Slot } from "expo-router";
import { tokenCache } from "./cache";
import "../global.css";
import { BottomSheetModalProvider } from "@gorhom/bottom-sheet";
import { GestureHandlerRootView } from "react-native-gesture-handler";

export default function RootLayout() {
const publishableKey = process.env.EXPO_PUBLIC_CLERK_PUBLISHABLE_KEY!;

return (
<ClerkProvider tokenCache={tokenCache} publishableKey={publishableKey}>
<ClerkLoaded>
<GestureHandlerRootView style={{ flex: 1 }}>
<BottomSheetModalProvider>
<Slot screenOptions={{ headerShown: false }} />
</BottomSheetModalProvider>
</GestureHandlerRootView>
</ClerkLoaded>
</ClerkProvider>
);
}
import { ClerkProvider, ClerkLoaded } from "@clerk/clerk-expo";
import { Slot } from "expo-router";
import { tokenCache } from "./cache";
import "../global.css";
import { BottomSheetModalProvider } from "@gorhom/bottom-sheet";
import { GestureHandlerRootView } from "react-native-gesture-handler";

export default function RootLayout() {
const publishableKey = process.env.EXPO_PUBLIC_CLERK_PUBLISHABLE_KEY!;

return (
<ClerkProvider tokenCache={tokenCache} publishableKey={publishableKey}>
<ClerkLoaded>
<GestureHandlerRootView style={{ flex: 1 }}>
<BottomSheetModalProvider>
<Slot screenOptions={{ headerShown: false }} />
</BottomSheetModalProvider>
</GestureHandlerRootView>
</ClerkLoaded>
</ClerkProvider>
);
}
What I've Tried: Confirmed that Clerk auth works ✅ Checked that useAuth() provides a valid JWT ✅ Removed Convex, and the error disappears ✅ Double-checked Convex URL and Clerk keys ✅ Question Why is Convex failing to validate the Clerk token? Is there an additional step needed to correctly pass the JWT to Convex? Any ideas on resolving this error? Thanks in advance! 🙏
erquhart
erquhart5w ago
Can you enable verbose logging and share the logs you get? add { verbose: true } as second arg to new ConvexReactClient() What you're seeing isn't supposed to be possible 😅 : https://github.com/get-convex/convex-js/blob/60ce9add55f12cc5f68c813beafccb3867378685/src/browser/sync/authentication_manager.ts#L308-L313 But I'm hoping the inner decode function is throwing and we'll get some more info here: https://github.com/get-convex/convex-js/blob/60ce9add55f12cc5f68c813beafccb3867378685/src/browser/sync/authentication_manager.ts#L412-L414
Zachoo
ZachooOP5w ago
@erquhart // Initialize ConvexReactClient const convex = new ConvexReactClient(process.env.EXPO_PUBLIC_CONVEX_URL!, { verbose: true, }); Still returns same logs the weird thing is i am logged in and everything works when i see error ] } DEBUG 2025-03-04T20:39:40.666Z received ws message with type Transition DEBUG 2025-03-04T20:39:40.666Z server confirmed new auth token is valid [v2] DEBUG 2025-03-04T20:39:40.666Z Error decoding token: 0, import_jwt_decode.default is not a function (it is Object) [v2] ERROR Auth token is not a valid JWT, cannot refetch the token You are correct appears to be an issue with decoder
erquhart
erquhart5w ago
So you're seeing the error message, but not seeing any impacts?
Zachoo
ZachooOP5w ago
yeah no impacts just error message will screen record to show you quickly
erquhart
erquhart5w ago
If verbose isn't adding any logs, the decoder isn't throwing, just returning something falsy
Zachoo
ZachooOP5w ago
i am geetting verbose erro " DEBUG 2025-03-04T20:39:40.666Z received ws message with type Transition DEBUG 2025-03-04T20:39:40.666Z server confirmed new auth token is valid [v2] DEBUG 2025-03-04T20:39:40.666Z Error decoding token: 0, import_jwt_decode.default is not a function (it is Object) [v2] ERROR Auth token is not a valid JWT, cannot refetch the token
Zachoo
ZachooOP5w ago
Error decoding token: 0, import_jwt_decode.default is not a function (it is Object) [v2] which is same log line shown in https://github.com/get-convex/convex-js/blob/60ce9add55f12cc5f68c813beafccb3867378685/src/browser/sync/authentication_manager.ts#L412-L414
GitHub
convex-js/src/browser/sync/authentication_manager.ts at 60ce9add55f...
TypeScript/JavaScript client library for Convex. Contribute to get-convex/convex-js development by creating an account on GitHub.
Zachoo
ZachooOP5w ago
Look like convex get token fine but then fails on decode
erquhart
erquhart5w ago
ah the second one is the one we're looking for Error decoding token: 0, import_jwt_decode.default is not a function (it is Object) [v2] What version of convex are you running
Zachoo
ZachooOP5w ago
"convex": "^1.18.0", "convex-helpers": "^0.1.56",
erquhart
erquhart5w ago
Looks like an esm issue What's in your root tsconfig under compilerOptions for lib and moduleResolution
Zachoo
ZachooOP5w ago
{ "extends": "expo/tsconfig.base", "compilerOptions": { "strict": true, "paths": { "@repo/ui": ["../../packages/ui/src"], "@repo/ui/": ["../../packages/ui/src/"] } } }
erquhart
erquhart5w ago
Try these under compilerOptions (I use them with my expo/convex project):
"lib": ["ES2023"],
"moduleResolution": "Bundler",
"lib": ["ES2023"],
"moduleResolution": "Bundler",
very puzzling that this is the only place you're getting errors, though, if this is a general module resolution issue. And that you have no actual bugs occurring as a result.
Zachoo
ZachooOP5w ago
hmm yeah also weird that it is an error buyt i can authenticate on convex fine had a bug here where convex cant authenticate but if i refresh its fine ... after making those changes i still get error " DEBUG 2025-03-04T20:57:05.709Z received ws message with type Transition DEBUG 2025-03-04T20:57:05.709Z server confirmed new auth token is valid [v2] DEBUG 2025-03-04T20:57:05.710Z Error decoding token: 0, import_jwt_decode.default is not a function (it is Object) [v2] ERROR Auth token is not a valid JWT, cannot refetch the token" do i need to install a jwt decode package ?
erquhart
erquhart5w ago
No it's a direct dependency of convex-js
Zachoo
ZachooOP5w ago
thought so is my convex version fine
erquhart
erquhart5w ago
Actually latest is 1.19, try installing latest
Zachoo
ZachooOP5w ago
i am also using monorepo using turborepo
erquhart
erquhart5w ago
hmm actually this looks like a legit bug, jwt-decode exports jwtDecode as a named export, not default
Zachoo
ZachooOP5w ago
is there a command to greacefully update convex version ? or just do pnpm install and read notes for breaking changes have others reported ?
erquhart
erquhart5w ago
Nope, and I don't see it in my own logs either. And while the jwt-decode docs say named export, I do see a default export also present. Maybe changed in a new version though using pnpm I'm guessin need to determine what version of jwt-decode you're using, I'm suspecting you're running 4.x If you're using npm , npm ls jwt-decode Oh I missed this, just update to latest, shouldn't hurt anything pnpm update convex@latest I think pnpm list jwt-decode to get that version
Zachoo
ZachooOP5w ago
this prints nothing @erquhart
erquhart
erquhart5w ago
Try pnpm list jwt-decode --depth 10
Zachoo
ZachooOP5w ago
dependencies: @convex-dev/aggregate 0.1.20 └─┬ convex 1.19.5 peer └── jwt-decode 3.1.2 @convex-dev/auth 0.0.80 ├─┬ convex 1.19.5 peer │ └── jwt-decode 3.1.2 └── jwt-decode 4.0.0 convex 1.19.5 └── jwt-decode 3.1.2
erquhart
erquhart5w ago
looks like convex auth is pulling in 4.0
Zachoo
ZachooOP5w ago
So should remove convex auth the issue might be use to use convex auth now i switched to clerk
erquhart
erquhart5w ago
Oh if you're not using convex auth, yeah that's the easy fix. otherwise this is down to either pnpm or turborepo messing up the deps. No reason convex should be using convex auth's copy of that dependency but yeah that should resolve it
Zachoo
ZachooOP5w ago
would you have any indication why to if i leave my app for a while in background and then return to the app and open a page to say fetch mail which requiresthe user to be authenticated i get a white screen appear and and error in convex user not authenticated but if i clear app from background and reopeni am logged in fine 🙂 updating convex and removing convex auth fixed jwt error thanks for the help man !
erquhart
erquhart5w ago
You're seeing this with Clerk?
Zachoo
ZachooOP5w ago
Yes with clerk
erquhart
erquhart5w ago
Using nextjs by chance?
Zachoo
ZachooOP5w ago
using expo native app
erquhart
erquhart5w ago
oh right, native yeah, there are fixes coming very soon for that
Zachoo
ZachooOP5w ago
if i send app to background wait say 2 hours and reopen and fetch data from convex app just shows a white screen and does nothing until i clear app from background and reopen then i am logged in fine and its all good
erquhart
erquhart5w ago
android or ios?
Zachoo
ZachooOP5w ago
ios i will get a screen recording for your reference aswell as the logs if it is a know bug
erquhart
erquhart5w ago
when you say "reopen and fetch data", you just mean you go to the app and it's blank, or you go to the app, and then you interact with it somehow, and then it goes blank or you go to it, it fetches automatically, and goes blank after a second or so
Zachoo
ZachooOP5w ago
I open the app and it is authough it has the cached view so i load homeroute fine then if i switch to another route the screen goes white
erquhart
erquhart5w ago
Do you capture error logs anywhere, like sentry
Zachoo
ZachooOP5w ago
same white screen if i leave app in background reopen it will show me the splash and then splash disapears i load home route which fetches data from convex re current user and i get error in convex and white screen in app but if i clear from background and reopen everything loads fine Im adding sentry currently , short answer no
erquhart
erquhart5w ago
what's the error you get in convex when this happens and yeah add sentry and capture client errors for sure, really helpful
Zachoo
ZachooOP5w ago
just a user is unauthenticated error
erquhart
erquhart5w ago
especially for native
Zachoo
ZachooOP5w ago
very true cant just open console haha
Zachoo
ZachooOP5w ago
No description
Zachoo
ZachooOP5w ago
these functions call when i hit dashboard and fail i think because they fire before clerk has resynced user session with convex after inactivity
erquhart
erquhart5w ago
Yeah hmm
Zachoo
ZachooOP5w ago
because if i clear app from background/app draw and reopen everything loads fine user does not have to login again import { ClerkProvider, ClerkLoaded, useAuth, useUser, } from "@clerk/clerk-expo"; import { Slot, SplashScreen } from "expo-router"; import { tokenCache } from "./cache"; import "../global.css"; import { BottomSheetModalProvider } from "@gorhom/bottom-sheet"; import { GestureHandlerRootView } from "react-native-gesture-handler"; import { ConvexProvider, ConvexReactClient } from "convex/react"; import { ConvexQueryCacheProvider } from "convex-helpers/react/cache"; import { ConvexProviderWithClerk } from "convex/react-clerk"; import { useEffect, useState } from "react"; // Initialize ConvexReactClient const convex = new ConvexReactClient(process.env.EXPO_PUBLIC_CONVEX_URL!, { verbose: true, }); function AuthenticatedLayout() { const { isLoaded: isUserLoaded, isSignedIn } = useUser(); useEffect(() => { if (isUserLoaded) { SplashScreen.hideAsync(); } }, [isUserLoaded]); return ( <ConvexProviderWithClerk useAuth={useAuth} client={convex}> <ConvexQueryCacheProvider> <BottomSheetModalProvider> <Slot screenOptions={{ headerShown: false }} /> </BottomSheetModalProvider> </ConvexQueryCacheProvider> </ConvexProviderWithClerk> ); } export default function RootLayout() { const publishableKey = process.env.EXPO_PUBLIC_CLERK_PUBLISHABLE_KEY!; useEffect(() => { SplashScreen.preventAutoHideAsync(); }, []); return ( <GestureHandlerRootView style={{ flex: 1 }}> <ClerkProvider tokenCache={tokenCache} publishableKey={publishableKey}> <ClerkLoaded> <AuthenticatedLayout /> </ClerkLoaded> </ClerkProvider> </GestureHandlerRootView> ); } have even tried to split out to ensure clerk is loaded first that file is extract from root _layout.tsx
erquhart
erquhart5w ago
There are a few race conditions that can occur when ios apps come from background, I've been troubleshooting it with Convex Auth specifically so not sure if Clerk brings it's own issues or not. Incoming fixes should help with this.
Zachoo
ZachooOP5w ago
yes I was having those issue with clerk auth hence setup with clerk to see if they fix haha Great apreciate the work you are doing i love the convex platform so all good ! I will share sentery logs with you aswell once theyare in and a screen recording incase is helpful just need to leave app in background for several hours before i get the issue so annoying
erquhart
erquhart3w ago
let me know if you're up for trying the aforementioned fixes before they're released Still seeing this with the latest auth related fixes in the Convex client. I believe the issue is the browser delaying execution of tasks, so a token will be fetched, but then not authenticated for potentially hours, at which point the token is expired. It does this until runs out of retries and then it clears auth state. Currently trying a change that avoids attempting to authenticate a known expired token

Did you find this page helpful?