Sronds
Sronds
CCConvex Community
Created by Riki on 1/18/2025 in #support-community
Convex Auth with Expo and Native Apple and Google Sign In
Instead of this
14 replies
CCConvex Community
Created by Riki on 1/18/2025 in #support-community
Convex Auth with Expo and Native Apple and Google Sign In
14 replies
CCConvex Community
Created by Riki on 1/18/2025 in #support-community
Convex Auth with Expo and Native Apple and Google Sign In
managed to get a working version. this is what it looks like
14 replies
CCConvex Community
Created by Riki on 1/18/2025 in #support-community
Convex Auth with Expo and Native Apple and Google Sign In
@Tom
const credentials = await AppleAuthentication.signInAsync({
requestedScopes: [
AppleAuthentication.AppleAuthenticationScope.FULL_NAME,
AppleAuthentication.AppleAuthenticationScope.EMAIL,
],
})
await signIn('native-apple', credentials)
const credentials = await AppleAuthentication.signInAsync({
requestedScopes: [
AppleAuthentication.AppleAuthenticationScope.FULL_NAME,
AppleAuthentication.AppleAuthenticationScope.EMAIL,
],
})
await signIn('native-apple', credentials)
ConvexCredentials({
id: 'native-apple',
authorize: async (credentials, ctx) => {
// Extract the identity token from the credentials
const { identityToken, user: clientAppleUserId } = credentials

if (!identityToken || !clientAppleUserId) {
throw new Error('Invalid Apple credentials')
}

const validatedTokenData = await ctx.runAction(internal.node.verifyToken, {
identityToken: identityToken as string,
})

// Verify that the user ID from the token matches the one from the client
if (validatedTokenData.appleUserId !== clientAppleUserId) {
throw new Error('User ID mismatch')
}

try {
// Check if user already exists
const existingAccount = await retrieveAccount(ctx, {
provider: 'native-apple',
account: { id: validatedTokenData.email },
})

if (existingAccount) {
// User exists, return their ID
console.log('User exists, returning their ID')
return { userId: existingAccount.user._id }
}
} catch (error) {
console.log('User does not exist, creating user')
}

const createdAccount = await createAccount(ctx, {
provider: 'native-apple',
account: { id: validatedTokenData.email },
profile: {
email: validatedTokenData.email,
name: 'User',
emailVerificationTime: Date.now(),
},
shouldLinkViaEmail: true,
})

return { userId: createdAccount.user._id }
},
}),
],
ConvexCredentials({
id: 'native-apple',
authorize: async (credentials, ctx) => {
// Extract the identity token from the credentials
const { identityToken, user: clientAppleUserId } = credentials

if (!identityToken || !clientAppleUserId) {
throw new Error('Invalid Apple credentials')
}

const validatedTokenData = await ctx.runAction(internal.node.verifyToken, {
identityToken: identityToken as string,
})

// Verify that the user ID from the token matches the one from the client
if (validatedTokenData.appleUserId !== clientAppleUserId) {
throw new Error('User ID mismatch')
}

try {
// Check if user already exists
const existingAccount = await retrieveAccount(ctx, {
provider: 'native-apple',
account: { id: validatedTokenData.email },
})

if (existingAccount) {
// User exists, return their ID
console.log('User exists, returning their ID')
return { userId: existingAccount.user._id }
}
} catch (error) {
console.log('User does not exist, creating user')
}

const createdAccount = await createAccount(ctx, {
provider: 'native-apple',
account: { id: validatedTokenData.email },
profile: {
email: validatedTokenData.email,
name: 'User',
emailVerificationTime: Date.now(),
},
shouldLinkViaEmail: true,
})

return { userId: createdAccount.user._id }
},
}),
],
14 replies
CCConvex Community
Created by Riki on 1/18/2025 in #support-community
Convex Auth with Expo and Native Apple and Google Sign In
btw as a reference for a potential implementation, this is how Supabase does this https://supabase.com/docs/guides/auth/social-login/auth-apple?queryGroups=platform&platform=react-native
14 replies
CCConvex Community
Created by Riki on 1/18/2025 in #support-community
Convex Auth with Expo and Native Apple and Google Sign In
I've started looking into this today, will see if i can find a solution
14 replies
CCConvex Community
Created by Riki on 1/18/2025 in #support-community
Convex Auth with Expo and Native Apple and Google Sign In
Mainly pricing. Plus i've already implemented Convex Auth in production with hundreds of users so i don't think i can switch easily even if i wanted to. But mainly not wanting to pay a monthly fee for just storing users in a database.
14 replies
CCConvex Community
Created by Riki on 1/18/2025 in #support-community
Convex Auth with Expo and Native Apple and Google Sign In
@Tom can you point me in the general right direction to attempt to get this setup? I would like to try and make this work but I'm not even sure where to start to implement this. Switching to Clerk is not a realistic option for my project so i'm looking for a Convex Auth solution here
14 replies
CCConvex Community
Created by Riki on 1/18/2025 in #support-community
Convex Auth with Expo and Native Apple and Google Sign In
+1
14 replies
CCConvex Community
Created by gioru on 8/27/2024 in #support-community
Need help with handling Convex Auth signIn errors in prod
currently running into a similar issue and would love to know if there is a clean way to show the relevant error on the client
40 replies
CCConvex Community
Created by gioru on 8/27/2024 in #support-community
Need help with handling Convex Auth signIn errors in prod
are there any updates on this? @Tom
40 replies
CCConvex Community
Created by Clever Tagline on 12/23/2024 in #general
clevertagline's Thread
+1
3 replies
CCConvex Community
Created by THEO on 12/16/2024 in #support-community
Clerk integration call functions infinitely - need help
thanks team, good work! ❤️❤️
12 replies
CCConvex Community
Created by THEO on 12/16/2024 in #support-community
Clerk integration call functions infinitely - need help
just ran into this bug myself! thanks
12 replies
CCConvex Community
Created by Eva on 9/1/2024 in #support-community
Deleting users with Convex Auth
if it helps anyone else, i wrote this function which works great.
export const deleteUser = userMutation({
args: {},
handler: async ({ ctx, userId }) => {
// delete all instances of the user from the database
console.log('Deleting user with id:', userId)

// authAccounts
const authAccount = await ctx.db
.query('authAccounts')
.withIndex('userIdAndProvider', (q) => q.eq('userId', userId))
.collect()

await Promise.all(
authAccount.map(async (account) => {
await ctx.db.delete(account._id)
})
)

const authSessions = await ctx.db
.query('authSessions')
.withIndex('userId', (q) => q.eq('userId', userId))
.collect()

// loop through all sessions and delete the associated refresh tokens in the authRefreshTokens table
await Promise.all(
authSessions.map(async (session) => {
const authRefreshTokens = await ctx.db
.query('authRefreshTokens')
.withIndex('sessionId', (q) => q.eq('sessionId', session._id))
.collect()

await Promise.all(
authRefreshTokens.map(async (token) => {
await ctx.db.delete(token._id)
})
)
})
)

// delete all sessions
await Promise.all(
authSessions.map(async (session) => {
await ctx.db.delete(session._id)
})
)

// MAKE SURE TO DELETE ANY OTHER USER DATA YOU'VE CREATED YOURSELF IN OTHER TABLES (messages, posts, etc...)

// delete user from the users table
return await ctx.db.delete(userId)
},
})
export const deleteUser = userMutation({
args: {},
handler: async ({ ctx, userId }) => {
// delete all instances of the user from the database
console.log('Deleting user with id:', userId)

// authAccounts
const authAccount = await ctx.db
.query('authAccounts')
.withIndex('userIdAndProvider', (q) => q.eq('userId', userId))
.collect()

await Promise.all(
authAccount.map(async (account) => {
await ctx.db.delete(account._id)
})
)

const authSessions = await ctx.db
.query('authSessions')
.withIndex('userId', (q) => q.eq('userId', userId))
.collect()

// loop through all sessions and delete the associated refresh tokens in the authRefreshTokens table
await Promise.all(
authSessions.map(async (session) => {
const authRefreshTokens = await ctx.db
.query('authRefreshTokens')
.withIndex('sessionId', (q) => q.eq('sessionId', session._id))
.collect()

await Promise.all(
authRefreshTokens.map(async (token) => {
await ctx.db.delete(token._id)
})
)
})
)

// delete all sessions
await Promise.all(
authSessions.map(async (session) => {
await ctx.db.delete(session._id)
})
)

// MAKE SURE TO DELETE ANY OTHER USER DATA YOU'VE CREATED YOURSELF IN OTHER TABLES (messages, posts, etc...)

// delete user from the users table
return await ctx.db.delete(userId)
},
})
if anyone is curious, this is my userMutation:
export const userMutation = customMutation(
mutation,
customCtx(async (ctx) => {
const userId = await getUserIdentity(ctx)
return { userId, ctx }
})
)
export const userMutation = customMutation(
mutation,
customCtx(async (ctx) => {
const userId = await getUserIdentity(ctx)
return { userId, ctx }
})
)
18 replies
CCConvex Community
Created by Web Dev Cody on 9/3/2024 in #support-community
Convex Auth Question
Thanks @Indy, appreciate you ❤️
9 replies
CCConvex Community
Created by Web Dev Cody on 9/3/2024 in #support-community
Convex Auth Question
which is then used here
const result = await openAuthSessionAsync(redirect!.toString(), redirectTo)
const result = await openAuthSessionAsync(redirect!.toString(), redirectTo)
9 replies
CCConvex Community
Created by Web Dev Cody on 9/3/2024 in #support-community
Convex Auth Question
looking at the code in signIn.js (@convex-dev/auth/dist/server/implmenetation/signIn.js) y'all are using
const redirect = new URL(requireEnv("CONVEX_SITE_URL") + `/api/auth/signin/${provider.id}`);
const redirect = new URL(requireEnv("CONVEX_SITE_URL") + `/api/auth/signin/${provider.id}`);
9 replies
CCConvex Community
Created by Web Dev Cody on 9/3/2024 in #support-community
Convex Auth Question
@Web Dev Cody @sshader i've tried to setup a custom domain but unfortunately google auth is not picking it up and only using original convex one. Any updates on this?
9 replies
CCConvex Community
Created by alixi on 11/25/2024 in #show-and-tell
Expo Router Convex Auth template
your timing is immaculate
5 replies