Server-side Firebase Admin SDK patterns for Next.js 14+ with secure initialization and token verification. PROACTIVELY activate for: (1) setting up Admin SDK with server-only protection, (2) implementing custom claims for RBAC, (3) verifying tokens in middleware. Triggers: "admin sdk", "server-only", "custom claims"
Install
npx skillscat add agentient/vibekit/firebase-admin-sdk-server-integration Install via the SkillsCat registry.
Firebase Admin SDK Server Integration
Overview
The Firebase Admin SDK provides privileged server-side access to Firebase services, bypassing security rules. It must be used exclusively in server environments (Server Components, API Routes, Server Actions).
Critical Security Pattern: server-only Package
MANDATORY: All Admin SDK files MUST import server-only to prevent client-side bundling.
// lib/firebase/admin.ts
import 'server-only'; // CRITICAL: First import, prevents client bundling
import { initializeApp, getApps, cert } from 'firebase-admin/app';Why: Without server-only, service account credentials could leak into the client bundle, exposing your entire Firebase project.
Secure Admin SDK Initialization
// lib/firebase/admin.ts
import 'server-only';
import { initializeApp, getApps, cert, type App } from 'firebase-admin/app';
import { getAuth, type Auth } from 'firebase-admin/auth';
import { getFirestore, type Firestore } from 'firebase-admin/firestore';
import { getStorage, type Storage } from 'firebase-admin/storage';
/**
* Initialize Firebase Admin SDK with service account credentials
* ONLY use in server-side code
*/
const adminConfig = {
projectId: process.env.FIREBASE_ADMIN_PROJECT_ID,
credential: cert({
projectId: process.env.FIREBASE_ADMIN_PROJECT_ID,
clientEmail: process.env.FIREBASE_ADMIN_CLIENT_EMAIL,
// Firebase private keys contain \n characters that need to be unescaped
privateKey: process.env.FIREBASE_ADMIN_PRIVATE_KEY?.replace(/\\n/g, '\n'),
}),
};
// Validate environment variables
if (
!process.env.FIREBASE_ADMIN_PROJECT_ID ||
!process.env.FIREBASE_ADMIN_CLIENT_EMAIL ||
!process.env.FIREBASE_ADMIN_PRIVATE_KEY
) {
throw new Error(
'Missing Firebase Admin SDK environment variables. Ensure .env.local is configured.'
);
}
// Singleton pattern (prevent multiple initializations)
let adminApp: App;
if (getApps().length === 0) {
adminApp = initializeApp(adminConfig, 'admin');
} else {
adminApp = getApps()[0];
}
/**
* Admin Authentication
* Use for: token verification, custom claims, user management
*/
export const adminAuth: Auth = getAuth(adminApp);
/**
* Admin Firestore
* Use for: privileged data access, bypassing security rules
*/
export const adminDb: Firestore = getFirestore(adminApp);
/**
* Admin Storage
* Use for: signed URLs, server-side file operations
*/
export const adminStorage: Storage = getStorage(adminApp);
export { adminApp };Environment Variables
.env.local (Server-only variables):
# Firebase Admin SDK Credentials (NEVER commit these!)
FIREBASE_ADMIN_PROJECT_ID=your-project-id
FIREBASE_ADMIN_CLIENT_EMAIL=firebase-adminsdk-xxxxx@your-project-id.iam.gserviceaccount.com
FIREBASE_ADMIN_PRIVATE_KEY="-----BEGIN PRIVATE KEY-----\nYourPrivateKeyHere\n-----END PRIVATE KEY-----\n"Get Credentials:
- Firebase Console -> Project Settings -> Service Accounts
- Click "Generate New Private Key"
- Download JSON file
- Extract values into
.env.local
Custom Claims (RBAC)
Set Custom Claims (Server-Side Only)
// app/api/admin/set-role/route.ts
import 'server-only';
import { adminAuth } from '@/lib/firebase/admin';
import { NextRequest, NextResponse } from 'next/server';
export async function POST(request: NextRequest) {
// Verify requester is admin
const token = request.cookies.get('firebase-token')?.value;
if (!token) {
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
}
const decodedToken = await adminAuth.verifyIdToken(token);
if (decodedToken.role !== 'admin') {
return NextResponse.json({ error: 'Forbidden' }, { status: 403 });
}
// Set custom claim
const { uid, role } = await request.json();
await adminAuth.setCustomUserClaims(uid, { role });
return NextResponse.json({ success: true });
}Read Custom Claims (Client-Side)
'use client';
import { auth } from '@/lib/firebase/client';
import { useEffect, useState } from 'react';
export function useUserRole() {
const [role, setRole] = useState<string | null>(null);
useEffect(() => {
const unsubscribe = auth.onAuthStateChanged(async (user) => {
if (user) {
const idTokenResult = await user.getIdTokenResult();
setRole(idTokenResult.claims.role as string || 'user');
} else {
setRole(null);
}
});
return unsubscribe;
}, []);
return role;
}IMPORTANT: After setting custom claims, client must force token refresh:
// Client-side after role is changed
await auth.currentUser?.getIdToken(true); // Force refreshAnti-Patterns
Importing Admin SDK in Client Component:
'use client';
import { adminDb } from '@/lib/firebase/admin'; // ERROR: Leaks credentials!Use Server Actions Instead:
// app/actions/getUser.ts
'use server';
import { adminDb } from '@/lib/firebase/admin';
export async function getUser(userId: string) {
const userDoc = await adminDb.collection('users').doc(userId).get();
return userDoc.data();
}
// Client component
'use client';
import { getUser } from '@/app/actions/getUser';Best Practices
Do:
- Always import
'server-only'first in admin files - Store credentials in server-side environment variables
- Validate environment variables on startup
- Use middleware for token verification
- Force token refresh after custom claims change
- Use Admin SDK for server-side data fetching
Don't:
- Import Admin SDK in client components
- Commit
.env.localto version control - Share service account keys
- Skip token verification
- Use Admin SDK for client-side operations
Related Skills: firebase-authentication-patterns, firebase-nextjs-integration-strategies