contract-testing-builder
Contract Testing Builder
Ensure API contracts don't break consumers.
Contract Testing Concepts
Consumer → Defines expected contract → Provider must satisfy
Benefits:
- Catch breaking changes early
- Independent development
- Fast feedback (no integration env needed)
- Documentation as code
Pact Setup (Consumer Side)
// consumer/tests/pacts/user-api.pact.test.ts
import { PactV3 } from "@pact-foundation/pact";
import { userApi } from "../api/userApi";
const provider = new PactV3({
consumer: "UserWebApp",
provider: "UserAPI",
dir: path.resolve(__dirname, "../../pacts"),
});
describe("User API Contract", () => {
it("should get user by ID", async () => {
// Define expected interaction
await provider
.given("user 123 exists")
.uponReceiving("a request for user 123")
.withRequest({
method: "GET",
path: "/api/users/123",
headers: {
Authorization: "Bearer token123",
},
})
.willRespondWith({
status: 200,
headers: {
"Content-Type": "application/json",
},
body: {
id: "123",
email: "john@example.com",
name: "John Doe",
role: "USER",
createdAt: like("2024-01-01T00:00:00Z"),
},
})
.executeTest(async (mockServer) => {
// Make actual API call against mock server
const user = await userApi.getUser("123", mockServer.url);
// Verify consumer can handle response
expect(user.id).toBe("123");
expect(user.email).toBe("john@example.com");
});
});
it("should return 404 when user not found", async () => {
await provider
.given("user 999 does not exist")
.uponReceiving("a request for non-existent user")
.withRequest({
method: "GET",
path: "/api/users/999",
})
.willRespondWith({
status: 404,
headers: {
"Content-Type": "application/json",
},
body: {
error: "User not found",
},
})
.executeTest(async (mockServer) => {
await expect(userApi.getUser("999", mockServer.url)).rejects.toThrow(
"User not found"
);
});
});
});
Pact Verification (Provider Side)
// provider/tests/pacts/verify.test.ts
import { Verifier } from "@pact-foundation/pact";
import { app } from "../src/app";
describe("Pact Verification", () => {
let server: Server;
beforeAll(async () => {
server = app.listen(3000);
});
afterAll(() => {
server.close();
});
it("should validate consumer contracts", async () => {
const verifier = new Verifier({
provider: "UserAPI",
providerBaseUrl: "http://localhost:3000",
// Fetch pacts from broker or local files
pactUrls: [
path.resolve(__dirname, "../../pacts/UserWebApp-UserAPI.json"),
],
// Provider states setup
stateHandlers: {
"user 123 exists": async () => {
// Seed database with user 123
await db.user.create({
id: "123",
email: "john@example.com",
name: "John Doe",
role: "USER",
});
},
"user 999 does not exist": async () => {
// Ensure user 999 doesn't exist
await db.user.deleteMany({ where: { id: "999" } });
},
},
// Teardown after each test
afterEach: async () => {
await db.$executeRaw`TRUNCATE TABLE users CASCADE`;
},
});
await verifier.verifyProvider();
});
});
OpenAPI Contract Testing
# contracts/user-api.yaml
openapi: 3.0.0
info:
title: User API
version: 1.0.0
paths:
/api/users/{id}:
get:
parameters:
- name: id
in: path
required: true
schema:
type: string
responses:
"200":
description: User found
content:
application/json:
schema:
$ref: "#/components/schemas/User"
"404":
description: User not found
content:
application/json:
schema:
$ref: "#/components/schemas/Error"
components:
schemas:
User:
type: object
required:
- id
- email
- name
- role
properties:
id:
type: string
email:
type: string
format: email
name:
type: string
role:
type: string
enum: [USER, ADMIN]
createdAt:
type: string
format: date-time
Contract Validation (OpenAPI)
// tests/contract-validation.test.ts
import * as OpenAPIValidator from "express-openapi-validator";
import * as fs from "fs";
import * as yaml from "js-yaml";
describe("API Contract Validation", () => {
it("should match OpenAPI spec", async () => {
const spec = yaml.load(
fs.readFileSync("./contracts/user-api.yaml", "utf8")
);
app.use(
OpenAPIValidator.middleware({
apiSpec: spec,
validateRequests: true,
validateResponses: true,
})
);
// Valid request - should pass
await request(app)
.get("/api/users/123")
.expect(200)
.expect((res) => {
expect(res.body).toHaveProperty("id");
expect(res.body).toHaveProperty("email");
expect(res.body).toHaveProperty("name");
expect(res.body).toHaveProperty("role");
});
});
it("should reject invalid responses", async () => {
// Mock endpoint that returns invalid data
app.get("/api/invalid", (req, res) => {
res.json({
id: "123",
// Missing required fields!
});
});
// Should fail validation
await request(app).get("/api/invalid").expect(500);
});
});
JSON Schema Validation
// schemas/user.schema.ts
export const userSchema = {
type: "object",
required: ["id", "email", "name", "role"],
properties: {
id: { type: "string" },
email: { type: "string", format: "email" },
name: { type: "string", minLength: 1 },
role: { type: "string", enum: ["USER", "ADMIN"] },
createdAt: { type: "string", format: "date-time" },
},
additionalProperties: false,
};
// tests/schema-validation.test.ts
import Ajv from "ajv";
import addFormats from "ajv-formats";
const ajv = new Ajv();
addFormats(ajv);
describe("User Schema Validation", () => {
const validate = ajv.compile(userSchema);
it("should validate correct user object", () => {
const user = {
id: "123",
email: "john@example.com",
name: "John Doe",
role: "USER",
createdAt: "2024-01-01T00:00:00Z",
};
expect(validate(user)).toBe(true);
});
it("should reject missing required fields", () => {
const user = {
id: "123",
email: "john@example.com",
// Missing name and role
};
expect(validate(user)).toBe(false);
expect(validate.errors).toContainEqual(
expect.objectContaining({
message: "must have required property 'name'",
})
);
});
it("should reject invalid email format", () => {
const user = {
id: "123",
email: "invalid-email",
name: "John Doe",
role: "USER",
};
expect(validate(user)).toBe(false);
});
});
CI Integration
# .github/workflows/contract-tests.yml
name: Contract Tests
on: [push, pull_request]
jobs:
consumer-tests:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: actions/setup-node@v4
- name: Run consumer tests
run: npm run test:pact
- name: Publish pacts
run: |
npx pact-broker publish \
./pacts \
--consumer-app-version=${{ github.sha }} \
--broker-base-url=${{ secrets.PACT_BROKER_URL }} \
--broker-token=${{ secrets.PACT_BROKER_TOKEN }}
provider-tests:
runs-on: ubuntu-latest
needs: consumer-tests
steps:
- uses: actions/checkout@v4
- uses: actions/setup-node@v4
- name: Verify provider
run: npm run test:pact:verify
env:
PACT_BROKER_URL: ${{ secrets.PACT_BROKER_URL }}
PACT_BROKER_TOKEN: ${{ secrets.PACT_BROKER_TOKEN }}
Breaking Change Detection
// tests/breaking-changes.test.ts
describe("Breaking Change Detection", () => {
it("should not remove required fields", async () => {
const v1Response = {
id: "123",
email: "john@example.com",
name: "John Doe",
role: "USER",
};
const v2Response = {
id: "123",
email: "john@example.com",
// Missing 'name' - BREAKING CHANGE!
role: "USER",
};
// Validate v2 still has all v1 required fields
const v1Keys = Object.keys(v1Response);
const v2Keys = Object.keys(v2Response);
const missingFields = v1Keys.filter((key) => !v2Keys.includes(key));
expect(missingFields).toHaveLength(0);
});
it("should not change field types", async () => {
const v1Response = {
id: "123", // string
age: 25, // number
};
const v2Response = {
id: 123, // number - BREAKING CHANGE!
age: "25", // string - BREAKING CHANGE!
};
expect(typeof v2Response.id).toBe(typeof v1Response.id);
expect(typeof v2Response.age).toBe(typeof v1Response.age);
});
});
Contract Documentation
# API Contract Documentation
## User API Contract
### Consumer: UserWebApp
### Provider: UserAPI
### Interactions
#### Get User by ID
**Request:**
```http
GET /api/users/{id}
Authorization: Bearer {token}
```
Response (200):
{
"id": "string",
"email": "string (email format)",
"name": "string",
"role": "USER | ADMIN",
"createdAt": "string (ISO 8601)"
}
Response (404):
{
"error": "User not found"
}
Provider States
- user {id} exists: User with given ID exists in database
- user {id} does not exist: User with given ID does not exist
Breaking Change Policy
- Cannot remove required fields
- Cannot change field types
- Cannot remove enum values
- Can add optional fields
- Can deprecate with 6-month notice
## Best Practices
1. **Consumer-driven**: Consumers define expectations
2. **Test early**: Run in CI on every commit
3. **Use Pact Broker**: Central contract repository
4. **Provider states**: Setup test data properly
5. **Version contracts**: Track API versions
6. **Document changes**: Clear migration guides
7. **Monitor compliance**: Track contract violations
## Output Checklist
- [ ] Contract test framework chosen (Pact/OpenAPI)
- [ ] Consumer tests written
- [ ] Provider verification configured
- [ ] Provider states implemented
- [ ] Schema validation added
- [ ] Breaking change detection
- [ ] CI integration configured
- [ ] Contract documentation
- [ ] Pact Broker setup (if using Pact)
- [ ] Versioning strategy defined
More from monkey1sai/openai-cli
eslint-prettier-config
Configures ESLint and Prettier for consistent code quality with TypeScript, React, and modern best practices. Use when users request "ESLint setup", "Prettier config", "linting configuration", "code formatting", or "lint rules".
9api-security-hardener
Hardens API security with rate limiting, input validation, authentication, and protection against common attacks. Use when users request "API security", "secure API", "rate limiting", "input validation", or "API protection".
9secure-headers-csp-builder
Implements security headers and Content Security Policy with safe rollout strategy (report-only → enforce), testing, and compatibility checks. Use for "security headers", "CSP", "HTTP headers", or "XSS protection".
9security-incident-playbook-generator
Creates response procedures for security incidents with containment steps, communication templates, and evidence collection. Use for "incident response", "security playbook", "breach response", or "IR plan".
9bruno-collection-generator
Generates Bruno collection files (.bru) from Express, Next.js, Fastify, or other API routes. Creates organized collections with environments, authentication, and folder structure for the open-source Bruno API client. Use when users request "generate bruno collection", "bruno api testing", "create bru files", or "bruno import".
9mermaid-diagram-generator
Creates Mermaid diagrams for flowcharts, sequence diagrams, ERDs, and architecture visualizations in markdown. Use when users request "Mermaid diagram", "flowchart", "sequence diagram", "ERD diagram", or "architecture diagram".
9