Awesome-claude-code-toolkit testing-strategies
Testing strategies including contract testing, snapshot testing, mutation testing, property-based testing, and test organization
install
source · Clone the upstream repo
git clone https://github.com/rohitg00/awesome-claude-code-toolkit
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/rohitg00/awesome-claude-code-toolkit "$T" && mkdir -p ~/.claude/skills && cp -r "$T/skills/testing-strategies" ~/.claude/skills/rohitg00-awesome-claude-code-toolkit-testing-strategies && rm -rf "$T"
manifest:
skills/testing-strategies/SKILL.mdsource content
Testing Strategies
Test Structure (Arrange-Act-Assert)
describe("OrderService", () => { describe("createOrder", () => { it("creates an order with valid items and returns order ID", async () => { const repo = new InMemoryOrderRepository(); const service = new OrderService(repo); const input = { customerId: "c1", items: [{ productId: "p1", quantity: 2 }] }; const result = await service.createOrder(input); expect(result.id).toBeDefined(); expect(result.status).toBe("pending"); expect(result.items).toHaveLength(1); const saved = await repo.findById(result.id); expect(saved).toEqual(result); }); it("rejects order with empty items", async () => { const service = new OrderService(new InMemoryOrderRepository()); await expect( service.createOrder({ customerId: "c1", items: [] }) ).rejects.toThrow("Order must have at least one item"); }); }); });
Name tests by behavior, not method name. Each test should be independent and self-contained.
Contract Testing (Pact)
import { PactV4 } from "@pact-foundation/pact"; const provider = new PactV4({ consumer: "OrderService", provider: "UserService", }); describe("UserService contract", () => { it("returns user by ID", async () => { await provider .addInteraction() .given("user with id user-1 exists") .uponReceiving("a request for user user-1") .withRequest("GET", "/api/users/user-1") .willRespondWith(200, (builder) => { builder.jsonBody({ id: "user-1", name: "Alice", email: "alice@example.com", }); }) .executeTest(async (mockServer) => { const client = new UserClient(mockServer.url); const user = await client.getUser("user-1"); expect(user.name).toBe("Alice"); }); }); });
Contract tests verify that consumer expectations match provider capabilities without requiring both services to be running.
Snapshot Testing
import { render } from "@testing-library/react"; it("renders the user profile card", () => { const { container } = render( <UserCard user={{ name: "Alice", email: "alice@example.com", role: "admin" }} /> ); expect(container).toMatchSnapshot(); }); it("renders the order summary with inline snapshot", () => { const summary = formatOrderSummary(mockOrder); expect(summary).toMatchInlineSnapshot(` "Order #123 Items: 3 Total: $45.99 Status: Pending" `); });
Use inline snapshots for small outputs. Review snapshot diffs carefully during code review.
Property-Based Testing
import fc from "fast-check"; describe("sortUsers", () => { it("always returns the same number of elements", () => { fc.assert( fc.property( fc.array(fc.record({ name: fc.string(), age: fc.nat(120) })), (users) => { const sorted = sortUsers(users, "name"); return sorted.length === users.length; } ) ); }); it("produces a sorted result for any input", () => { fc.assert( fc.property( fc.array(fc.record({ name: fc.string(), age: fc.nat(120) })), (users) => { const sorted = sortUsers(users, "age"); for (let i = 1; i < sorted.length; i++) { if (sorted[i].age < sorted[i - 1].age) return false; } return true; } ) ); }); });
Integration Test with Test Containers
import { PostgreSqlContainer } from "@testcontainers/postgresql"; let container: any; let db: Database; beforeAll(async () => { container = await new PostgreSqlContainer("postgres:16").start(); db = await createDatabase(container.getConnectionUri()); await db.migrate(); }, 60000); afterAll(async () => { await db.close(); await container.stop(); }); it("creates and retrieves a user", async () => { const user = await db.user.create({ name: "Alice", email: "alice@test.com" }); const found = await db.user.findById(user.id); expect(found).toEqual(user); });
Test Doubles
function createMockEmailService(): EmailService { const sent: Array<{ to: string; subject: string }> = []; return { send: async (to, subject, body) => { sent.push({ to, subject }); }, getSent: () => sent, }; } const emailService = createMockEmailService(); const service = new NotificationService(emailService); await service.notifyUser("user-1", "Welcome"); expect(emailService.getSent()).toHaveLength(1); expect(emailService.getSent()[0].subject).toBe("Welcome");
Anti-Patterns
- Testing implementation details instead of behavior
- Sharing mutable state between tests (no
reset)beforeEach - Writing tests that depend on execution order
- Mocking everything instead of using real dependencies for integration tests
- Ignoring flaky tests instead of fixing the root cause
- Testing trivial getters/setters while missing edge cases
Checklist
- Tests organized by behavior, not by method or file
- Each test follows Arrange-Act-Assert structure
- Contract tests verify inter-service API compatibility
- Snapshot tests reviewed during code review (not blindly updated)
- Property-based tests cover invariants for algorithmic code
- Integration tests use test containers for real dependencies
- Test doubles are minimal and behavior-focused
- CI fails on flaky test detection