pact-contract-testing
Authors and verifies Pact consumer-driven contract tests across the full Pact lifecycle - consumer tests producing pact files, publishing to the Pact Broker, provider verification, and `can-i-deploy` deployment gates. Use when introducing a new HTTP/JSON API contract between two services, diagnosing breaking changes, or wiring contract verification into CI.
pact-contract-testing
Overview
Pact is a code-first tool for testing HTTP and message integrations using consumer-driven contract tests (pact-overview). The contract is generated as a side-effect of the consumer's automated tests - each test case documents one request/response pair, and only the parts the consumer actually uses get tested.
The full lifecycle has five steps (pact-how-it-works):
When to use
If the API has no consumers under your control (public APIs, third-party integrations), prefer openapi-contract-diff - schema- based diffs need no provider/consumer coordination.
Authoring (consumer side)
Install (Node example)
npm install --save-dev @pact-foundation/pact(Per pact-js.)
Consumer test with PactV3
const { PactV3, MatchersV3 } = require('@pact-foundation/pact');
const { like, eachLike } = MatchersV3;
const provider = new PactV3({
consumer: 'web-app',
provider: 'pet-service',
dir: path.resolve(process.cwd(), 'pacts'),
});
describe('Pet Service consumer', () => {
it('returns a list of dogs', async () => {
provider
.given('I have a list of dogs')
.uponReceiving('a request for all dogs with the builder pattern')
.withRequest({
method: 'GET',
path: '/dogs',
headers: { Accept: 'application/json' },
})
.willRespondWith({
status: 200,
headers: { 'Content-Type': 'application/json' },
body: eachLike({
id: like(1),
name: like('Rex'),
}),
});
await provider.executeTest(async (mockServer) => {
const response = await fetch(`${mockServer.url}/dogs`);
expect(response.status).toBe(200);
});
});
});(Adapted from pact-js.)
The DSL lifecycle is given() → uponReceiving() → withRequest() → willRespondWith(). given() describes a provider state the verifier will set up later. Matchers like like() and eachLike() let the contract assert type/shape rather than exact values, so the provider isn't bound to fixture-specific data.
Where pact files are written
By default, PactV3 writes pact files to ./pacts/ relative to the process CWD (pact-js). Each consumer/provider pair produces one JSON file: <consumer>-<provider>.json.
Publishing to the Pact Broker
The Broker is the central registry for pact files and verification results. Publishing happens after the consumer tests pass:
npx pact-broker publish ./pacts \
--consumer-app-version=$(git rev-parse HEAD) \
--branch=$(git rev-parse --abbrev-ref HEAD) \
--broker-base-url=$PACT_BROKER_BASE_URL \
--broker-token=$PACT_BROKER_TOKENThe Broker stores (overview):
Tagging by branch + the Git SHA as consumer-app-version is the canonical pattern - can-i-deploy queries the matrix using these identifiers.
Provider verification
const { Verifier } = require('@pact-foundation/pact');
new Verifier({
providerBaseUrl: 'http://localhost:8081',
pactBrokerUrl: process.env.PACT_BROKER_BASE_URL,
pactBrokerToken: process.env.PACT_BROKER_TOKEN,
provider: 'pet-service',
providerVersion: process.env.GIT_SHA,
providerVersionBranch: process.env.GIT_BRANCH,
publishVerificationResult: true,
consumerVersionSelectors: [
{ mainBranch: true },
{ deployedOrReleased: true },
],
}).verifyProvider();(Adapted from pact-js.)
The provider replays every request from each pact file against a running provider instance and compares responses (pact-how-it-works). consumerVersionSelectors controls which consumer pacts get verified - mainBranch plus deployedOrReleased ensures the provider stays compatible with both the latest consumer work and what's currently in production.
publishVerificationResult: true is what makes the matrix update - without it, the broker has no record of this provider version's verification status.
Provider states
For each consumer interaction with a given(<state>), the provider test setup must register a hook that puts the system into that state before replay. Using Express:
new Verifier({
...
stateHandlers: {
'I have a list of dogs': async () => {
await db.dogs.bulkInsert([{ id: 1, name: 'Rex' }]);
return { description: 'dogs seeded' };
},
},
}).verifyProvider();State setup that fails causes the matching interaction to fail verification.
can-i-deploy - deployment gate
Per can-i-deploy:
pact-broker can-i-deploy \
--pacticipant pet-service \
--version $(git rev-parse HEAD) \
--to-environment production| Flag | Required | Effect |
|---|---|---|
--pacticipant | yes | Application (consumer or provider) name. |
--version | yes | App version string (typically the Git SHA). |
--to-environment | optional | Target environment to check against. Omit to check against latest. |
Exit codes (can-i-deploy):
The output includes a markdown-style matrix of consumer/provider version pairs with verification status and links to detailed verification results.
Recording deployments
After deploying, record the deployment so subsequent can-i-deploy queries see the new "currently deployed" baseline:
pact-broker record-deployment \
--pacticipant pet-service \
--version $(git rev-parse HEAD) \
--environment productionSkipping this step is the most common reason can-i-deploy returns spurious "no" verdicts.
CI integration
A complete CI flow per side:
Consumer pipeline
- name: Run consumer tests + publish pact
env:
PACT_BROKER_BASE_URL: ${{ secrets.PACT_BROKER_BASE_URL }}
PACT_BROKER_TOKEN: ${{ secrets.PACT_BROKER_TOKEN }}
run: |
npm test # writes ./pacts/<consumer>-<provider>.json
npx pact-broker publish ./pacts \
--consumer-app-version=$GITHUB_SHA \
--branch=${GITHUB_REF##*/}
- name: Can I deploy?
env:
PACT_BROKER_BASE_URL: ${{ secrets.PACT_BROKER_BASE_URL }}
PACT_BROKER_TOKEN: ${{ secrets.PACT_BROKER_TOKEN }}
run: |
pact-broker can-i-deploy \
--pacticipant=web-app \
--version=$GITHUB_SHA \
--to-environment=productionProvider pipeline
- name: Verify pacts from broker
env:
PACT_BROKER_BASE_URL: ${{ secrets.PACT_BROKER_BASE_URL }}
PACT_BROKER_TOKEN: ${{ secrets.PACT_BROKER_TOKEN }}
run: |
npm run start:provider &
npx wait-on http://localhost:8081
npm run verify:pact # publishVerificationResult: true
- name: Can I deploy?
env:
PACT_BROKER_BASE_URL: ${{ secrets.PACT_BROKER_BASE_URL }}
PACT_BROKER_TOKEN: ${{ secrets.PACT_BROKER_TOKEN }}
run: |
pact-broker can-i-deploy \
--pacticipant=pet-service \
--version=$GITHUB_SHA \
--to-environment=productioncan-i-deploy is the actual gate - pact verification happens in step 1, but a green verification alone doesn't prove every paired version is compatible.