Syncing Users Between Clerk and Your Database in Next.js icon

Syncing Users Between Clerk and Your Database in Next.js

A practical guide to implementing reliable user synchronization between Clerk authentication and your database using Drizzle ORM and Neon in Next.js applications.

13 min read
Cover image for Syncing Users Between Clerk and Your Database in Next.js

When building modern Next.js applications, many developers turn to third-party authentication providers like Clerk to handle user management. However, this introduces a critical architectural challenge:

how do you properly synchronize users from your auth provider to your own database?

After years of building production Next.js applications and advising teams on authentication architecture, I've developed reliable patterns for handling this synchronization. In this article, I'll share the most effective approach based on real-world experience.

#Understanding the Architecture Challenge

In a typical Next.js application with third-party authentication, you face an architectural decision: where should user data live?

While Clerk manages authentication and stores user identity information, it's not the right place for application-specific user data. Your own database needs to contain user records that can:

  1. Form relationships with other entities in your data model
  2. Store application-specific user attributes
  3. Enable high-performance queries without API rate limits
  4. Keep you in control of your user data

The critical challenge becomes: how do you ensure a user in Clerk always has a corresponding record in your database?

Two common approaches exist:

  1. Webhook-based synchronization: Create/update database records when Clerk sends webhook events
  2. First-access synchronization: Create/verify database records when users first access your application

After implementing both approaches in multiple production applications, I've found the first-access pattern to be significantly more reliable. Let me explain why.

ApproachHow It WorksProsCons
Webhook SyncClerk sends events to your server when users are created/updated/deletedAutomatic background sync. Works for API-only apps.Can fail due to network issues. Hard to debug. Race conditions possible.
First-Access SyncAfter login, users visit a special page that creates their database recordReliable and predictable. Easy to implement. Guarantees DB record exists.Adds an extra redirect. Only works for web applications.
Combined (Recommended)Use First-Access as primary method with webhooks as backupMaximum reliability. Handles all edge cases. Works in all scenarios.Slightly more complex to implement.

#Database Setup with Drizzle and Neon

Before diving into synchronization patterns, let's set up our database. We'll use Drizzle ORM with Neon, a serverless PostgreSQL database:

db/schema.ts
import {
	pgTable,
	text,
	timestamp,
	uuid
} from 'drizzle-orm/pg-core'

export const users = pgTable('users', {
	id: uuid('id').defaultRandom().primaryKey(),
	clerkId: text('clerk_id').unique().notNull(),
	email: text('email').unique().notNull(),
	name: text('name'),
	createdAt: timestamp('created_at').defaultNow().notNull(),
	updatedAt: timestamp('updated_at').defaultNow().notNull()
})

Notice we're being intentionally minimalist. We store:

  • A unique database-generated ID
  • The Clerk ID (for cross-referencing)
  • Email (for searching/display)
  • Name (for display)
  • Timestamps (for auditing)

We deliberately avoid storing information Clerk already manages well:

  • Authentication details
  • Profile pictures
  • OAuth connections
  • Email verification status

Next, let's create our database connection:

db/index.ts
import {neon} from '@neondatabase/serverless'
import {drizzle} from 'drizzle-orm/neon-http'
import * as schema from './schema'

// Create the connection
const sql = neon(process.env.DATABASE_URL!)
export const db = drizzle(sql, {schema})

#The First-Access Synchronization Pattern

Here's the approach I've found most reliable over years of developing Next.js applications: synchronize users when they first access your application after authentication.

This solves several critical issues:

  1. It works even if webhooks fail (which happens more often than you'd expect)
  2. It ensures users always have database records before accessing protected routes
  3. It allows for a deterministic, testable synchronization flow
  4. It eliminates race conditions where a user might try to access data before their database record exists

Here's how to implement it:

app/new-user/page.tsx
import { redirect } from 'next/navigation'
import { currentUser } from '@clerk/nextjs/server'
import { db } from '@/db'
import { users } from '@/db/schema'
import { eq } from 'drizzle-orm'
import Link from 'next/link'

export default async function NewUserPage() {
  // Get the current user from Clerk
  const clerkUser = await currentUser()

  // If no user is signed in, redirect to sign-in
  if (!clerkUser) {
    redirect('/sign-in')
  }

  try {
    // Check if user already exists in database
    const [existingUser] = await db
      .select()
      .from(users)
      .where(eq(users.clerkId, clerkUser.id))

    // If user exists, proceed to dashboard
    if (existingUser) {
      redirect('/dashboard')
    }

    // Extract required information from Clerk user
    const email = clerkUser.emailAddresses[0]?.emailAddress || ''
    if (!email) {
      return (
        <div className="flex min-h-screen items-center justify-center">
          <div className="p-4 text-center">
            <h1 className="mb-2 text-lg font-medium">Missing email address</h1>
            <Link href="/" className="text-blue-600 hover:underline">
              Return home
            </Link>
          </div>
        </div>
      )
    }

    // Handle name extraction with fallback strategy
    const firstName = clerkUser.firstName || ''
    const lastName = clerkUser.lastName || ''
    let fullName = `${firstName} ${lastName}`.trim()

    if (!fullName) {
      fullName = email.split('@')[0]
    }

    // Insert user into database
    await db.insert(users).values({
      clerkId: clerkUser.id,
      email,
      name: fullName
    })

    console.log('User created with name:', fullName)
  } catch (error) {
    console.error('Error creating user:', error)
    return (
      <div className="flex min-h-screen items-center justify-center">
        <div className="p-4 text-center">
          <h1 className="mb-2 text-lg font-medium">Setup error</h1>
          <p className="text-sm text-gray-600 mb-4">
            We encountered a problem setting up your account.
          </p>
          <Link href="/dashboard" className="text-blue-600 hover:underline">
            Try again
          </Link>
        </div>
      </div>
    )
  }

  // Redirect to dashboard after successful creation
  redirect('/dashboard')
}

This implementation gives us:

  1. Reliability: The user record is created in a synchronous flow
  2. Error handling: Issues during user creation show a friendly error
  3. Fallback strategies: If name information is missing, we derive from email
  4. No dangling redirects: We redirect after ensuring the database operation completed

In your Clerk configuration, set the post-authentication redirect to /new-user. This ensures every user passes through this synchronization step before accessing protected content.

#Implementing Webhooks as a Safety Net

While the first-access pattern is highly reliable, implementing webhooks provides an additional safety net for specific scenarios:

  1. User deletion in Clerk (to cascade delete in your database)
  2. Email changes in Clerk (to update your database)
  3. Creating records for users who may access your API but not your frontend

Here's a robust webhook implementation:

app/api/clerk/route.ts
import {headers} from 'next/headers'
import {NextResponse} from 'next/server'
import {
	syncNewUser,
	handleUserDeletion,
	handleEmailUpdate
} from '@/lib/user'
import {Webhook} from 'svix'

// Define type for webhook events
interface WebhookEvent {
	data: {
		id: string
		first_name?: string | null
		last_name?: string | null
		email_addresses?: Array<{
			id: string
			email_address: string
		}>
		primary_email_address_id?: string
	}
	object: string
	type: string
}

export async function POST(req: Request) {
	const headersList = headers()
	const svix_id = headersList.get('svix-id')
	const svix_timestamp = headersList.get('svix-timestamp')
	const svix_signature = headersList.get('svix-signature')

	// Validate webhook headers
	if (!svix_id || !svix_timestamp || !svix_signature) {
		return new Response('Missing svix headers', {status: 400})
	}

	// Get the body
	const payload = await req.json()
	const body = JSON.stringify(payload)

	// Verify the webhook signature
	const wh = new Webhook(process.env.CLERK_WEBHOOK_SECRET || '')
	let evt: WebhookEvent

	try {
		evt = wh.verify(body, {
			'svix-id': svix_id,
			'svix-timestamp': svix_timestamp,
			'svix-signature': svix_signature
		}) as WebhookEvent
	} catch (error) {
		console.error('Error verifying webhook:', error)
		return new Response('Invalid signature', {status: 400})
	}

	// Process different event types
	const {type} = evt

	try {
		if (type === 'user.created') {
			await syncNewUser(evt.data)
			return NextResponse.json({
				success: true,
				event: 'user.created'
			})
		}

		if (type === 'user.deleted') {
			await handleUserDeletion(evt.data.id)
			return NextResponse.json({
				success: true,
				event: 'user.deleted'
			})
		}

		if (type === 'email.updated') {
			await handleEmailUpdate(evt.data)
			return NextResponse.json({
				success: true,
				event: 'email.updated'
			})
		}

		// Acknowledge other events but take no action
		return NextResponse.json({
			success: true,
			event: type,
			action: 'none'
		})
	} catch (error) {
		console.error(`Error handling webhook ${type}:`, error)
		// Still return 200 to acknowledge receipt (prevent retries)
		return NextResponse.json({
			success: false,
			error: 'Processing error'
		})
	}
}

The implementation functions might look like:

lib/user.ts
import {db} from '@/db'
import {users} from '@/db/schema'
import {eq} from 'drizzle-orm'

// Synchronize a new Clerk user to your database
export const syncNewUser = async (userData: any) => {
	try {
		// Check if user already exists
		const [existingUser] = await db
			.select()
			.from(users)
			.where(eq(users.clerkId, userData.id))

		// If user already exists, no need to create again
		if (existingUser) {
			return existingUser
		}

		// Extract email from Clerk user data
		const primaryEmailId = userData.primary_email_address_id
		let email = ''

		if (
			userData.email_addresses &&
			userData.email_addresses.length > 0
		) {
			const primaryEmail = userData.email_addresses.find(
				(e: any) => e.id === primaryEmailId
			)
			email =
				primaryEmail?.email_address ||
				userData.email_addresses[0]?.email_address ||
				''
		}

		// Extract name from Clerk user data
		const firstName = userData.first_name || ''
		const lastName = userData.last_name || ''
		let name = `${firstName} ${lastName}`.trim()

		// Use email username as fallback name
		if (!name && email) {
			name = email.split('@')[0]
		}

		// Create the user in your database
		const [newUser] = await db
			.insert(users)
			.values({
				clerkId: userData.id,
				email,
				name: name || null
			})
			.returning()

		return newUser
	} catch (error) {
		console.error('Error syncing user to database:', error)
		throw error
	}
}

// Handle user deletion
export const handleUserDeletion = async (clerkId: string) => {
	try {
		await db.delete(users).where(eq(users.clerkId, clerkId))
		return {success: true}
	} catch (error) {
		console.error('Error deleting user from database:', error)
		throw error
	}
}

// Handle email update
export const handleEmailUpdate = async (userData: any) => {
	try {
		const primaryEmailId = userData.primary_email_address_id
		let email = ''

		if (
			userData.email_addresses &&
			userData.email_addresses.length > 0
		) {
			const primaryEmail = userData.email_addresses.find(
				(e: any) => e.id === primaryEmailId
			)
			email =
				primaryEmail?.email_address ||
				userData.email_addresses[0]?.email_address ||
				''
		}

		if (email) {
			await db
				.update(users)
				.set({email, updatedAt: new Date()})
				.where(eq(users.clerkId, userData.id))
		}

		return {success: true}
	} catch (error) {
		console.error(
			'Error updating user email in database:',
			error
		)
		throw error
	}
}

#Middleware Configuration

With our synchronization patterns in place, we need to configure middleware to properly direct users through our application. This ensures users pass through the synchronization process and can't access protected routes without database records:

middleware.ts
import {
	clerkMiddleware,
	createRouteMatcher
} from '@clerk/nextjs/server'

const isPublicRoute = createRouteMatcher([
	'/', // Homepage
	'/about', // About page
	'/contact', // Contact page
	'/sign-in(.*)', // Sign-in and all sub-paths
	'/sign-up(.*)', // Sign-up and all sub-paths
	'/api/webhooks/clerk(.*)', // Webhooks endpoint
	'/new-user' // Critical: Our synchronization page
])

export default clerkMiddleware(async (auth, request) => {
	if (!isPublicRoute(request)) {
		await auth.protect()
	}
})

export const config = {
	matcher: [
		'/((?!.+\\.[\\w]+$|_next).*)',
		'/',
		'/(api|trpc)(.*)'
	]
}

This middleware configuration ensures:

  1. Public routes remain accessible to everyone
  2. The /new-user route is accessible after authentication but before database synchronization
  3. All other routes require authentication

#Helper Functions for Accessing User Data

To simplify accessing user data throughout your application, implement utility functions that abstract the synchronization logic:

lib/user.ts (additional functions)
import {currentUser} from '@clerk/nextjs/server'
import {db} from '@/db'
import {users} from '@/db/schema'
import {eq} from 'drizzle-orm'
import {cache} from 'react'

// Get current user from database with caching
// This uses React's cache for memoization within a single request
export const getCurrentUser = cache(async () => {
	const clerkUser = await currentUser()
	if (!clerkUser) return null

	// Get from database
	const [dbUser] = await db
		.select()
		.from(users)
		.where(eq(users.clerkId, clerkUser.id))

	// If for some reason we don't have a DB user, synchronize now
	// This might happen if webhooks failed and user somehow bypassed new-user page
	if (!dbUser) {
		await syncNewUser(clerkUser)

		// Now try again
		const [newUser] = await db
			.select()
			.from(users)
			.where(eq(users.clerkId, clerkUser.id))

		return newUser
	}

	return dbUser
})

// Get user by ID - useful for displaying other users' data
export const getUserById = async (userId: string) => {
	const [user] = await db
		.select()
		.from(users)
		.where(eq(users.id, userId))

	return user || null
}

// Get user by Clerk ID - useful for admin operations
export const getUserByClerkId = async (clerkId: string) => {
	const [user] = await db
		.select()
		.from(users)
		.where(eq(users.clerkId, clerkId))

	return user || null
}

These helper functions provide several advantages:

  1. Consistent access pattern: Every component uses the same function to get user data
  2. Caching: React's cache function prevents redundant database queries within a request
  3. Self-healing: If a user somehow has a Clerk account but no database record, it creates one
  4. Type safety: Returns properly typed user objects

#Advanced User Management

As your application grows, you'll likely need tools to manage users. Here's a server action to handle user deletion that removes records from both Clerk and your database:

app/admin/actions.ts
'use server'

import {currentUser} from '@clerk/nextjs/server'
import {clerkClient} from '@clerk/nextjs/server'
import {db} from '@/db'
import {users} from '@/db/schema'
import {eq} from 'drizzle-orm'
import {revalidatePath} from 'next/cache'

export async function deleteUser(formData: FormData) {
	// Get the current user for permission checking
	const clerkUser = await currentUser()

	// Check if the current user is an admin
	// Replace this with your actual admin check logic
	if (
		!clerkUser ||
		clerkUser.emailAddresses[0]?.emailAddress !==
			'admin@example.com'
	) {
		throw new Error('Unauthorized')
	}

	const userId = formData.get('userId') as string
	const clerkId = formData.get('clerkId') as string

	if (!userId || !clerkId) {
		throw new Error('Missing required user information')
	}

	try {
		// First delete from our database to maintain referential integrity
		// If you have foreign key constraints, you might need a transaction
		// or cascade delete setup
		await db.delete(users).where(eq(users.id, userId))

		// Then delete from Clerk
		try {
			await clerkClient.users.deleteUser(clerkId)
		} catch (clerkError) {
			console.error(
				'Error deleting user from Clerk:',
				clerkError
			)
			// If Clerk deletion fails, we continue since DB record is gone
			// This can be handled by re-syncing later if needed
		}

		// Revalidate the admin dashboard
		revalidatePath('/admin')

		return {success: true}
	} catch (error) {
		console.error('Error deleting user:', error)
		return {
			success: false,
			error:
				error instanceof Error ? error.message : 'Unknown error'
		}
	}
}

#Performance Considerations

When implementing user synchronization, consider these performance optimizations:

  1. Minimal database schema: Only store what you need in your database
  2. Efficient queries: Use indexed fields for lookups (like clerkId and email)
  3. Connection pooling: For high-traffic applications, implement connection pooling with your database
  4. Memoization: Use React's cache function to prevent duplicate queries

#Conclusion

After years of building authentication systems, I've found that the first-access synchronization pattern, combined with webhooks as a safety net, provides the most reliable approach to keeping users in sync with your database.

This approach gives you:

BenefitDescription
ReliabilityUsers always have database records before accessing protected content
SimplicityThe synchronization logic is straightforward and deterministic
PerformanceNo unnecessary database queries or API calls
ResilienceMultiple mechanisms for keeping data in sync

By following these patterns, you'll build a robust authentication system that gives you the best of both worlds: Clerk's excellent authentication capabilities with the full control and performance of your own database.

Remember that keeping your auth provider and database in sync is one of the most critical aspects of your application architecture. Taking the time to implement it correctly will save you countless hours of debugging strange authentication issues down the road.