Skip to main content

Python implementation for Pact (http://pact.io/)

Project description

https://img.shields.io/badge/license-MIT-brightgreen.svg https://img.shields.io/badge/python-2.7,%203.3,%203.4,%203.5,%203.6,%203.7,%203.8-brightgreen.svg https://img.shields.io/badge/pypi-0.1.1-brightgreen.svg https://img.shields.io/pypi/wheel/Django.svg https://travis-ci.org/Kalimaha/pact-test.svg?branch=master https://coveralls.io/repos/github/Kalimaha/pact-test/badge.svg?branch=development https://img.shields.io/badge/Say%20Thanks-!-1EAEDB.svg

Pact Test for Python

This repository contains a Python implementation for Pact. Pact is a specification for Consumer Driven Contracts Testing. For further information about Pact project, contracts testing, pros and cons and useful resources please refer to the Pact website.

There are two phases in Consumer Driven Contracts Testing: a Consumer sets up a contract (it’s consumer driven after all!), and a Provider honours it. But, before that…

Installation

Pact Test is distributed through PyPi so it can be easily included in the requirements.txt file or normally installed with pip:

$ pip install pact-test

Providers Tests (Set the Contracts)

https://img.shields.io/badge/Pact-1.0-brightgreen.svg https://img.shields.io/badge/Pact-1.1-red.svg https://img.shields.io/badge/Pact-2.0-red.svg https://img.shields.io/badge/Pact-3.0-red.svg https://img.shields.io/badge/Pact-4.0-red.svg

Consumers run Provider Tests to create pacts and establish a contract between them and service providers. An example of a Python client using pact test is available at here. Consumers define all the interactions with their providers in the following way:

@service_consumer('PythonEats')
@has_pact_with('PyzzaHut')
class PyzzaHutTest(ServiceProviderTest):

    @given('some pizzas exist')
    @upon_receiving('a request for a pepperoni pizza')
    @with_request({'method': 'get', 'path': '/pizzas/pepperoni/'})
    @will_respond_with({'status': 200, 'body': {'id': 42, 'type': 'pepperoni'}})
    def test_get_pepperoni_pizza(self):
        pizza = get_pizza('pepperoni')
        assert pizza['id'] == 42
        assert pizza['type'] == 'pepperoni'

This test verifies, against a mock server, the expected interaction and creates a JSON file (the pact) that will be stored locally and also sent to the Pact Broker, if available. It is possible to define multiple tests for the same state in order to verify all the scenarios of interest, For example, we can test an unhappy 404 situation:

@given('some pizzas exist')
@upon_receiving('a request for an hawaiian pizza')
@with_request({'method': 'get', 'path': '/pizzas/hawaiian/'})
@will_respond_with({'status': 404, 'body': {'message': 'we do not serve pineapple with pizza'}})
def test_get_hawaiian_pizza(self):
    pizza = get_pizza('hawaiian')
    assert pizza.status_code == 404
    assert pizza.json()['message'] == 'we do not serve pineapple with pizza'

Consumers Tests (Honour Your Contracts)

https://img.shields.io/badge/Pact-1.0-brightgreen.svg https://img.shields.io/badge/Pact-1.1-red.svg https://img.shields.io/badge/Pact-2.0-red.svg https://img.shields.io/badge/Pact-3.0-red.svg https://img.shields.io/badge/Pact-4.0-red.svg

Providers run Consumer Tests to verify that they are honouring their pacts with the consumers. There are few examples of an hypothetical restaurant service implemented with the most popular Python web frameworks:

There are few things required to setup and run consumer tests.

Pact Helper

This helper class is used by Pact Test to start and stop the web-app before and after the test. It also defines the ports and endpoint to be used by the test. The following is an example of Pact Helper:

class RestaurantPactHelper(PactHelper):
    process = None

    def setup(self):
        self.process = subprocess.Popen('gunicorn start:app -w 3 -b :8080 --log-level error', shell=True)

    def tear_down(self):
        self.process.kill()

There are few rules for the helper:

  • it must extend PactHelper class from pact_test

  • it must define a setup method

  • it must define a tear_down method

It is also possible to override default url (localhost) and port (9999):

class RestaurantPactHelper(PactHelper):
    test_url = '0.0.0.0'
    test_port = 5000

States

When a consumer sets a pact, it defines certain states. States are basically pre-requisites to your test. Before honouring the pacts, a provider needs to define such states. For example:

@honours_pact_with('UberEats')
@pact_uri('http://Kalimaha.github.io/src/pacts/pact.json')
class UberEats(ServiceConsumerTest):

    @state('some menu items exist')
    def test_get_menu_items(self):
        DB.save(MenuItem('spam'))
        DB.save(MenuItem('eggs'))

In this example, the provider stores some test data in its DB in order to make the system ready to receive mock calls from the consumer and therefore verify the pact.

Configuration

The default configuration of Pact Test assumes the following values:

  • consumer_tests_path: tests/service_consumers

  • provider_tests_path: tests/service_providers

  • pact_broker_uri: None

It is possible to override such values by creating a file named .pact.json in the project’s root. The following is an example of a valid configuration file:

{
  "consumer_tests_path": "mypath/mytests",
  "provider_tests_path": "mypath/mytests",
  "pact_broker_uri": "http://example.com/"
}

All fields are optional: only specified fields will override default configuration values.

Development

Setup

python3 setup.py install

Test

It is possible to run the tests locally with Docker through the following command:

$ ./bin/test

By default this command tests the library against Python 3.6. It is possible to specify the Python version as follows:

$ ./bin/test <ENV>

Available values for ENV are: py27, py33, py34, py35 py36, py37 and py38. It is also possible to test all the versions at once with:

$ ./bin/test all

Upload New Version

$ python3 setup.py sdist upload

With Python Wheels:

$ python3 setup.py sdist bdist_wheel
$ twine upload dist/*

Project details


Release history Release notifications | RSS feed

This version

1.0.4

Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

pact-test-1.0.4.tar.gz (18.1 kB view details)

Uploaded Source

Built Distribution

pact_test-1.0.4-py2.py3-none-any.whl (25.1 kB view details)

Uploaded Python 2 Python 3

File details

Details for the file pact-test-1.0.4.tar.gz.

File metadata

  • Download URL: pact-test-1.0.4.tar.gz
  • Upload date:
  • Size: 18.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.1.1 pkginfo/1.5.0.1 requests/2.23.0 setuptools/41.0.1 requests-toolbelt/0.9.1 tqdm/4.46.1 CPython/3.7.4

File hashes

Hashes for pact-test-1.0.4.tar.gz
Algorithm Hash digest
SHA256 91b185e188af96de8b0b9e4879fda22c708c9eb1f10967f1b4b06768e23bd404
MD5 dc9b5dd5b03e6a0bdb7acfcb05bf088d
BLAKE2b-256 2814a598404a3b64f90a6cbe6a217240c6aa6b84f25b99dbd7b25089e140f19a

See more details on using hashes here.

File details

Details for the file pact_test-1.0.4-py2.py3-none-any.whl.

File metadata

  • Download URL: pact_test-1.0.4-py2.py3-none-any.whl
  • Upload date:
  • Size: 25.1 kB
  • Tags: Python 2, Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.1.1 pkginfo/1.5.0.1 requests/2.23.0 setuptools/41.0.1 requests-toolbelt/0.9.1 tqdm/4.46.1 CPython/3.7.4

File hashes

Hashes for pact_test-1.0.4-py2.py3-none-any.whl
Algorithm Hash digest
SHA256 9c819d778417cc0fdd28a46dc0d553e45d88035de56cde22780a8fa72b109071
MD5 f8504f08f6be8ab4374f4eddb83a79d8
BLAKE2b-256 b0449769a32588f7f6d8567174b1e035a6a5802da03d01d3d837521c386b0640

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page