Agent skill
firebase-testing
Guide for testing Firebase Admin SDK with Vitest mocks. Use when writing tests that involve Firebase Auth, Firestore, or Firebase App.
Install this agent skill to your Project
npx add-skill https://github.com/majiayu000/claude-skill-registry/tree/main/skills/data/firebase-testing
SKILL.md
Firebase Testing Patterns
This skill provides patterns for mocking Firebase Admin SDK in Vitest tests.
Mock Helpers Location
Firebase mocks are located in app/tests/mocks/firebase.ts and provide:
createFirebaseAppMock()- Mock Firebase App instancecreateFirebaseAuthMock()- Mock Firebase Auth with verifyIdToken, getUsercreateFirestoreMock()- Mock Firestore with collection, doc, getresetFirebaseMocks()- Reset all mocks between tests
Firebase Plugin Test Template
import Fastify from "fastify";
import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
import {
createFirebaseAppMock,
createFirebaseAuthMock,
createFirestoreMock,
} from "../../mocks/firebase.js";
const mockApp = createFirebaseAppMock();
const mockAuth = createFirebaseAuthMock();
const mockFirestore = createFirestoreMock();
vi.mock("firebase-admin/app", () => ({
getApps: vi.fn(() => [mockApp]),
initializeApp: vi.fn(() => mockApp),
cert: vi.fn(),
}));
vi.mock("firebase-admin/auth", () => ({
getAuth: vi.fn(() => mockAuth),
}));
vi.mock("firebase-admin/firestore", () => ({
getFirestore: vi.fn(() => mockFirestore),
}));
describe("Firebase Plugin", () => {
beforeEach(() => {
vi.clearAllMocks();
});
afterEach(async () => {
// Clear module cache so dynamic imports get fresh mocked modules
vi.resetModules();
});
it("should register firebase decorator", async () => {
const { default: firebasePlugin } = await import(
"../../../src/plugins/firebase.js"
);
const fastify = Fastify();
await fastify.register(firebasePlugin);
await fastify.ready();
expect(fastify.firebase).toBeDefined();
await fastify.close();
});
});
Auth Testing Patterns
Valid Token Authentication
it("should authenticate with valid token", async () => {
const mockDecodedToken = {
uid: "test-user-123",
email: "test@example.com",
email_verified: true,
};
mockAuth.verifyIdToken.mockResolvedValue(mockDecodedToken);
const response = await fastify.inject({
method: "GET",
url: "/protected",
headers: { authorization: "Bearer valid-token" },
});
expect(response.statusCode).toBe(200);
expect(mockAuth.verifyIdToken).toHaveBeenCalledWith("valid-token", false);
});
Invalid Token Handling
it("should return 401 for invalid token", async () => {
mockAuth.verifyIdToken.mockRejectedValue(
new Error("Firebase ID token has invalid signature"),
);
const response = await fastify.inject({
method: "GET",
url: "/protected",
headers: { authorization: "Bearer invalid-token" },
});
expect(response.statusCode).toBe(401);
});
Token Revocation
it("should return 401 when token is revoked", async () => {
const revokedError = Object.assign(new Error("Token has been revoked"), {
code: "auth/id-token-revoked",
});
mockAuth.verifyIdToken.mockRejectedValue(revokedError);
const response = await fastify.inject({
method: "GET",
url: "/protected",
headers: { authorization: "Bearer revoked-token" },
});
expect(response.statusCode).toBe(401);
expect(response.json().message).toContain("Token has been revoked");
});
Missing Authorization Header
it("should return 401 when no authorization header", async () => {
const response = await fastify.inject({
method: "GET",
url: "/protected",
});
expect(response.statusCode).toBe(401);
});
Firestore Testing Patterns
Mocking Collection Queries
it("should query Firestore collection", async () => {
const mockDocs = [
{ id: "doc1", data: () => ({ name: "Test 1" }) },
{ id: "doc2", data: () => ({ name: "Test 2" }) },
];
mockFirestore.collection.mockReturnValue({
get: vi.fn().mockResolvedValue({ docs: mockDocs }),
limit: vi.fn().mockReturnThis(),
});
// Test code that queries Firestore
});
Mocking Document Operations
it("should get a document", async () => {
const mockDoc = {
exists: true,
id: "doc-id",
data: () => ({ field: "value" }),
};
mockFirestore.collection.mockReturnValue({
doc: vi.fn().mockReturnValue({
get: vi.fn().mockResolvedValue(mockDoc),
}),
});
// Test code that gets a document
});
Key Patterns
- Mock before import: Use
vi.mock()at module level before importing tested modules - Dynamic imports: Use
await import()after mocks are set up to ensure mocks are applied - Reset between tests:
vi.clearAllMocks()in beforeEach - clears mock call historyvi.resetModules()in afterEach - clears module cache so dynamic imports get fresh mocked modules
- Realistic data: Use realistic mock data that matches Firebase structures
Commands
cd app
npm run test # Run all tests
npm run test:coverage # Run tests with coverage
Boundaries
- Never use real Firebase credentials in tests
- Always mock Firebase Admin SDK modules
- Use the shared mock helpers from
tests/mocks/firebase.ts
Recommended Agent Skills
Expand your agent's capabilities with these related and highly-rated skills.
agent-ops-spec
Manage specification documents in .agent/specs/. Use when user provides requirements, acceptance criteria, or feature descriptions that need to be tracked and validated against implementation.
agent-ops-state
Maintain .agent state files. Use at session start, after meaningful steps, and before concluding: read/update constitution/memory/focus/issues/baseline consistently.
agent-ops-spec
Manage specification documents in .agent/specs/. Use when user provides requirements, acceptance criteria, or feature descriptions that need to be tracked and validated against implementation.
agent-ops-testing
Test strategy, execution, and coverage analysis. Use when designing tests, running test suites, or analyzing test results beyond baseline checks.
agent-ops-testing
Test strategy, execution, and coverage analysis. Use when designing tests, running test suites, or analyzing test results beyond baseline checks.
agent-ops-state
Maintain .agent state files. Use at session start, after meaningful steps, and before concluding: read/update constitution/memory/focus/issues/baseline consistently.
Didn't find tool you were looking for?