Quick Start Guide with Github Actions

Tip

WELCOME TO THE QUICK START GUIDE

If you want to get hands-on and learn about Design First API Design with Contract testing, and see it come to life, then you are in the right place.

This guide shows you how to integrate your development workflow.

To setup an integration within the API Hub for Design UI see the API Hub for Design integration guide.

Note

GUIDES

These testing guides will help provide describe each step in more detail and should serve as a handy reference point

Provider Testing Guide

Consumer Testing Guide

Agenda

In this workshop, you will get to see the full development workflow in action. Following the scenario below, we will simplify it for demonstration by using GitHub to fork the example repositories and use GitHub actions to run the CI pipelines

  1. Create and document an API using OpenAPI Specification

  2. Publish the provider contract (an OpenAPI document) to API Hub for Contract Testing

  3. Write the API consumer

  4. Write tests for an API client using tools such as Mountebank,Nock,Wiremock,Nock,Cypress and Mock-Service-Worker or traditional Pact .NET to mock the API, and convert those mocks into a consumer contract

  5. Publish the consumer contract to API Hub for Contract Testing

  6. Learn about API Hub for Contract Testing's breaking change detection system.

Pre-requisites

Get an API Hub for Design account

Note

You'll need an account, don't worry, we will be using the free tier.

  • Don't have one - get a subscription on the API Hub pageAccessing API Hub

  • I've already got one!

Tip

You have got your API Hub for Design account, and are successfully logged in.

Caution

WHERE IS MY EMAIL?

Not got an email? Don't forget to check your spam folder.

Get a GitHub account

Note

All our examples run on Github Actions CI pipelines. You'll need an account. Don't worry its free.

  • Don't have one - sign up 👉 here.

  • I've already got one!

Tip

You are logged in to your GitHub account.

Get a API Hub for Contract Testing account

Note

The Bi-Directional Feature is only supported by API Hub for Contract Testing, so you'll need an account, don't worry, the developer tier is free.

Using a shared company API Hub for Contract Testing account?

You can use a shared company API Hub for Contract Testing account, but it will make things a bit fiddly, as you'll need to change the identifiers of the various resources that get created so that they don't clash with those from other workshop participants. We've found from past experience running workshops that it's much simpler if everyone has their own account.

Tip

You have got your API Hub for Contract Testing account, and are successfully logged in.

Caution

WHERE IS MY EMAIL?

Not got an email? Don't forget to check your spam folder, otherwise reach out to the team at .

Provider side

Design the API

Create and document an API using OpenAPI.

As we are following a specification or design first approach to API development, we start by creating an OpenAPI description document, that describes how our API should work.

Authoring an OAS document is beyond the scope of this tutorial, but you can find plenty of resources on the internet (such as at swagger.io).

Our products reference openAPI spec is here

  1. POST /products - create a new product

  2. GET /products - gets all products

  3. GET /products/:id - gets a single product

The product schema is as follows:

product-schema-pf.png
  • Create OpenAPI Document in API Hub for Design

    1. Select Create New - > Create New API

    2. Select Owner

    3. Select Specification: OpenAPI 3.0.x

    4. Select Template: --None--

    5. Select Name: Enter the name of your API here

    6. Select Version: 1.0.0

    7. Select Auto Mock API: off

    8. Select Create API

    9. Paste the contents of the reference OpenAPI specification

PF_Screenshot_SwaggerHub_01
swaggerhub-pactflow-demo.png
swaggerhub-pactflow-demo-1.png
swaggerhub-pactflow-demo-2.png

Tip

SUCCESS

You should have created your first API definition in API Hub for Design. For fuller information on API Hub for Design Authoring - Follow this guide to create your first API in API Hub for Design.

Publish your provider design spec to API Hub for Contract Testing

Important

See our Publishing contracts docs for more information.

Now that we have created our provider specification, we need to share it to our consumers. This is where API Hub for Contract Testing comes in to the picture. This step is referred to as "publishing" the provider contract.

The publishing step takes two key components:

  • The provider contract itself (in our case, the OAS document)

  • The test results (in our case, we will use our OAS document, as we are just testing the design, we will use a separate flow to test our implementation when it is created)

This information will be helpful later on, when we need to check compatibility with its consumers.

  1. Fork the example provider projects in to your own Github account (click the 'Fork' button in the top right).

    example-provider.png
    1. Open your forked example provider project (https://github.com/<your-username>/example-provider)

    2. Open .github/workflows/ProviderDesignFeedback.yml

    3. In the upper right corner of the file view, click 🖊️ to open the file editor.

    4. Update the value of PACT_BROKER_BASE_URL to the base URL of your own API Hub for Contract Testing account. You can easily get this by clicking the COPY PACTFLOW BASE URL button on the API Tokens page in API Hub for Contract Testing.

      provider-design-feedback.png
    5. Remove the line if: ${{ github.repository_owner == 'pactflow' }}

    6. Press the green Commit changes button

  2. Create a Github Secret to store your API Hub for Contract Testing API token in.

    1. In API Hub for Contract Testing:

      1. Log in to your API Hub for Contract Testing account (https://<your-subdomain>.pactflow.io) and go to Settings > API Tokens - See here for the docs.

      2. Click the Copy button for the read/write CI token (make sure it's the read write one, not the read only one).

    2. In Github:

      1. In your forked example provider project (https://github.com/<your-username>/example-provider)

      2. Click on the Settings tab

      3. Select Secrets from the side menu.

      4. Click New repository secret (the button is to the right of the "Actions secrets" heading)

      5. Set the name of the secret to PACTFLOW_TOKEN_FOR_CI_CD_WORKSHOP

      6. Paste in the API Hub for Contract Testing API token value you copied in the previous step.

      7. Click Add Secret

        action-secrets.png
  3. Click on the Actions tab

  4. Click the button with the text "I understand my workflows, go ahead and enable them"

  5. Sync OpenAPI - > SCM with Github Sync -

    https://support.smartbear.com/swaggerhub/docs/integrations/github-sync.html

    1. Open the API page in API Hub for Design.

    2. Click the API name, switch to the Integrations tab, and click Add New Integrations:

    3. Select GitHub Sync.

    4. In the subsequent dialog, specify the integration parameters:

      1. Name – Required. A display name for the integration. gh-design-to-pactflow

      2. GitHub Token – Required. The GitHub access token that API Hub for Design will use to access the target GitHub repository.

      3. The easiest way to get the token is to click Connect to GitHub and allow API Hub for Design to retrieve information from your GitHub account:

      4. Click Next in the GitHub Token edit box to continue. API Hub for Design will validate the token and then display other parameters.

      5. Repository Owner – Select you GitHub user or organization that owns the repository you created in the previous step

      6. Repository – Select the repository you setup earlier to push the code to

    5. Sync Method – Select the synchronization type: Basic Sync

    6. Branch – Required. The repository branch to push the code to. If this branch does not exist, it will be created based on the repository’s default branch. Choose swaggerhub

    7. Generated API Code – Required. Select what you want to generate: YAML(Resolved)

    8. Output Folder - Select oas

    9. Output File - Select swagger.yaml

    10. Click Create And Execute - > Done

      PF_Screenshot_SwaggerHub_05

After you run through the process your dashboard should look like this:

PF_Screenshot_CreateExampleProject_01

Tip

SUCCESS

Your should have uploaded your first API definition to API Hub for Contract Testing.

Check if we could deploy our design candidate safely to production

Now that we have designed our provider specification and published our provider contract, we can check if we deploy the application to production.

Whilst we don't currently have any consumers to worry about, we want to be prepared for when we do. API Hub for Contract Testing has a tool called can_i_deploy to help us.

The can-i-deploy command is an important part of a CI/CD workflow, adding stage gates to prevent deploying incompatible applications to environments such as production.

This diagram shows an illustrative CI/CD pipeline as it relates to our progress to date:

PF_Screenshot_DeployToProduction_01.png

Check if it is safe to deploy the provider to production

We run the command:

can-i-deploy

This should pass, because as we discussed above, there are no consumers:

api-definition.png

$ pact-broker can-i-deploy --pacticipant pactflow-example-consumer --version="1.0.0-21b1f7fdf6428bfb0f583e151d9893c230a1c555-design" --to-environment production Computer says yes \o/ There are no missing dependencies

Later on, when consumers start to use our API, we will be prevented from releasing a change that results in a backwards incompatible change for our consumers. Consumers will also use this command to ensure they are compatible with the Provider API in the target environment (in this case, production).

If this was our actual implementation, this is where we would deploy our provider to production. Once we have deployed, we would API Hub for Contract Testing know that the new version of the Provider has been promoted to that environment, allowing API Hub for Contract Testing to communicate to any future consumers of the provider, that the OAS associated with this version of the provider is supported in production. If a consumer adds functionality that uses a subset of the OAS, they will be free to deploy safely!

By using the can-i-deploy against our design, we will be able check that our future design candidate changes will be compatible with any of the future consumers that are deployed or released. Giving us visibility and feedback, at the earliest opportunity.

Expected state by the end of this step

  • The provider build is passing and can-i-deploy states it is free to be deployed to production ✅.

Consumer side

Setup and trigger the example consumer project

  1. Fork the [example-consumer] [https://github.com/pactflow/example-consumer] project in to your own Github account (click the 'Fork' button in the top right).

  2. Open your forked example-consumer project (https://github.com/<your-username>/example-consumer)

  3. Create a Github Secret to store your API Hub for Contract Testing API token in.

    1. In API Hub for Contract Testing:

      1. Log in to your API Hub for Contract Testing account (https://<your-subdomain>.pactflow.io), and go to Settings > API Tokens - See here for the docs.

      2. Click the Copy button for the read/write CI token (make sure it's the read write one, not the read only one).

    2. In Github:

      1. In your forked example-consumer project (https://github.com/<your-username>/example-consumer)

      2. Click on the Settings tab.

      3. Select Secrets from the side menu.

      4. Click New repository secret (the button is to the right of the "Actions secrets" heading)

      5. Set the name of the secret to PACTFLOW_TOKEN_FOR_CI_CD_WORKSHOP

      6. Paste in the API Hub for Contract Testing API token value you copied in the previous step.

      7. Click on the Actions tab.

      8. Click the button with the text "I understand my workflows, go ahead and enable them".

      9. Open .github/workflows/Build.yml

      10. In the upper right corner of the file view, click 🖊️ to open the file editor.

      11. Update the value of PACT_BROKER_BASE_URL o the base URL of your own API Hub for Contract Testing account. You can easily get this by clicking the COPY PACTFLOW BASE URL button on the API Tokens page in API Hub for Contract Testing.

      12. Press the green Commit changes button.

Build triggered

The consumer build will trigger

cancel-workflow.png

Pact Contract Tests passed

The consumers unit tests will pass

build-yml.png

You can click into the build job to see the output

deploy.png

Pact Contract files uploaded

The pact files are uploaded to the API Hub for Contract Testing Broker

complete-job.png
Pact successfully published for pactflow-example-consumer version 253c165958d15624ce7245d5739689860439d24b and provider pactflow-example-provider.
  View the published pact at https://saflow.pactflow.io/pacts/provider/pactflow-example-provider/consumer/pactflow-example-consumer/version/253c165958d15624ce7245d5739689860439d24b
  Events detected: contract_published, contract_content_changed (first time untagged pact published)
  No enabled webhooks found for the detected events
  Next steps:
    * Add Pact verification tests to the pactflow-example-provider build. See https://docs.pact.io/go/provider_verification

Consumer can-i-deploy check

It will then call the can-i-deploy command before it tries to deploy, it passes. This is because the consumers pact is a valid subset of the OpenAPI specification.

can-i-deploy.png
========== STAGE: can-i-deploy? ==========

Computer says yes \o/ 

CONSUMER                  | C.VERSION  | PROVIDER                  | P.VERSION               | SUCCESS? | RESULT#
--------------------------|------------|---------------------------|-------------------------|----------|--------
pactflow-example-consumer | 253c165... | pactflow-example-provider | 1.0.0-21b1f7f...-design | true     | 1      

VERIFICATION RESULTS
--------------------
1. https://saflow.pactflow.io/contracts/bi-directional/provider/pactflow-example-provider/version/1.0.0-21b1f7fdf6428bfb0f583e151d9893c230a1c555-design/consumer/pactflow-example-consumer/version/253c165958d15624ce7245d5739689860439d24b/cross-contract-verification-results (success)

All required verification results are published and successful

Consumer record deployment

can-i-deploy is successful, so we can deploy our consumer, once complete, we record the consumers application version as deployed to the production environment.

record-deployment.png
========== STAGE: deploy ==========

Deploying to production
touch .env
Recorded deployment of pactflow-example-consumer version 253c165958d15624ce7245d5739689860439d24b to production environment in API Hub for Contract Testing.

Expected state by the end of this step

  • Both consumer and provider builds passing ✅

    PF_Screenshot_ExampleConsumer_01
    PF_Screenshot_ConsumerAndProvider_02

See a breaking change in action

Let's go a bit deeper and introduce a breaking change into the system. Breaking changes come in two main ways:

  1. A consumer can add a new expectation (e.g., a new field/endpoint) on a provider that doesn't exist.

  2. A provider might make a change (e.g., remove or rename a field, or change an endpoint) that breaks an existing consumer.

API Hub for Contract Testing will detect such situations using the can-i-deploy ool. When it runs, it performs a contract comparison that checks if the consumer contract is a valid subset of the provider contract in the target environment.

Remove a field the consumer requires

As we are concentrating on our design specification, we will propose 2 changes in API Hub for Design. One will be an unsafe operation that renames a field the consumer requires.

Let's see it in action.

  • In API Hub for Design:

    1. Open your API definition we created earlier

    2. We are going to update the name field in the Product schema to the firstName

      swaggerhub-remove-name.png
    3. Click Save

    4. Click Sync

    5. Update the commit message feat: removing name field

    6. Press Push

      sync-management.png
pact-publish-oas-actions.png
Run pactflow/actions/publish-provider-contract@v1.0.1
Successfully published provider contract for pactflow-example-provider version 1.0.0-3a694151f8837a8f38ed80124e9d81530d614dde-design to API Hub for Contract Testing

Breaking change prevention with API Hub for Contract Testing

can-i-deploy fails

can-i-deploy-failed.png
pact-broker can-i-deploy --pacticipant pactflow-example-provider --version="1.0.0-3a694151f8837a8f38ed80124e9d81530d614dde-design" --to-environment production

Computer says no ¯_(ツ)_/¯

CONSUMER                  | C.VERSION  | PROVIDER                  | P.VERSION               | SUCCESS? | RESULT#
--------------------------|------------|---------------------------|-------------------------|----------|--------
pactflow-example-consumer | 253c165... | pactflow-example-provider | 1.0.0-3a69415...-design | false    | 1      

VERIFICATION RESULTS
--------------------
1. https://saflow.pactflow.io/contracts/bi-directional/provider/pactflow-example-provider/version/1.0.0-3a694151f8837a8f38ed80124e9d81530d614dde-design/consumer/pactflow-example-consumer/version/253c165958d15624ce7245d5739689860439d24b/cross-contract-verification-results (failure)

The cross contract verification between the pact for the version of pactflow-example-consumer currently deployed or released to production (253c165958d15624ce7245d5739689860439d24b) and the oas for version 1.0.0-3a694151f8837a8f38ed80124e9d81530d614dde-design of pactflow-example-provider failed

Danger

DANGER

This build will fail, API Hub for Contract Testing knows all of the consumers' needs down to the field level.

If you head into the API Hub for Contract Testing UI and drill down into the "contract comparison" tab, you'll see the output from comparing the consumer and provider contracts. It's alerting us to the fact that the consumer needs a field, but the provider doesn't support it.

Read more about how to interpret failures.

Graceful change

Let's see a non-breaking change in action, we want to rename the name field to firstName. We are going to use the expand and contract pattern, where we support our consumers using the name field, until they have changed their code to use firstName. We will do this by adding both properties into the Product schema.

  • In API Hub for Design:

    1. Open your API definition we created earlier

    2. We are going to add the name field back in the Product schema, so it should contain both name and firstName

      swaggerhub-add-firstname.png
    3. Click Save

    4. Click Sync

    5. Update the commit message feat: adding firstName field

    6. Press Push

      swaggerhub-add-firstname-push-change.png
provider-build-triggered.png
provider-contract-published.png
Run pactflow/actions/publish-provider-contract@v1.0.1
Successfully published provider contract for pactflow-example-provider version 1.0.0-4bff20fab7b711b66a508c999690547d9559d12b-design to API Hub for Contract Testing

Design verified as compatible (expand & contract)

can-i-deploy-passed.png
pact-broker can-i-deploy --pacticipant pactflow-example-provider --version="1.0.0-4bff20fab7b711b66a508c999690547d9559d12b-design" --to-environment production
Computer says yes \o/ 

CONSUMER                  | C.VERSION  | PROVIDER                  | P.VERSION               | SUCCESS? | RESULT#
--------------------------|------------|---------------------------|-------------------------|----------|--------
pactflow-example-consumer | 253c165... | pactflow-example-provider | 1.0.0-4bff20f...-design | true     | 1      

VERIFICATION RESULTS
--------------------
1. https://saflow.pactflow.io/contracts/bi-directional/provider/pactflow-example-provider/version/1.0.0-4bff20fab7b711b66a508c999690547d9559d12b-design/consumer/pactflow-example-consumer/version/253c165958d15624ce7245d5739689860439d24b/cross-contract-verification-results (success)

All required verification results are published and successful

Tip

This build should pass, API Hub for Contract Testing knows all of the consumers needs down to the field level. Because the consumers request is a subset of the product schema, this is a safe operation.

We could now update our consumer code to use firstName, allowing the consumer to deprecate the name field.

github-workflow-overview.png

Important

WHAT HAVE WE LEARNED?

  1. It is always safe to remove a field from a provider, if no consumers are currently using it

  2. It is not safe to remove a field/endpoint from a provider, if an existing consumer is using it, and API Hub for Contract Testing will detect this situation.

  3. API Hub for Contract Testing will prevent a consumer from deploying a change that a Provider has yet to support

🚀 If you've made it this far, you should now have a good understanding of how API Hub for Design &amp; API Hub for Contract Testing's bi-directional contract testing feature works to make it safe to release software into production quickly and reliably.

You may be interested in one of:

Publication date: