From 08f0dc0daaad7953decb0af97c8406b9ff290ab7 Mon Sep 17 00:00:00 2001 From: Fabio Pulvirenti Date: Sun, 5 Sep 2021 22:49:51 +0200 Subject: [PATCH] feat: added support for message provider using pact broker (#257) * feat: added support for message provider using pact broker * fix: added new line * fix: update tests to support pact-broker runs fix: updated message_pact to wait writing contract process to finish * fix: removed time from import --- Makefile | 2 +- examples/message/README.md | 81 +++++++++++++++++-- examples/message/conftest.py | 52 ++++++++++++ examples/message/run_pytest.sh | 2 +- examples/message/tests/conftest.py | 6 -- .../tests/consumer/test_message_consumer.py | 2 + .../tests/provider/test_message_provider.py | 32 ++++++++ pact/message_pact.py | 1 + pact/message_provider.py | 19 +++++ pact/verifier.py | 1 + tests/test_message_provider.py | 19 +++++ tests/test_verifier.py | 3 + 12 files changed, 206 insertions(+), 14 deletions(-) create mode 100644 examples/message/conftest.py delete mode 100644 examples/message/tests/conftest.py diff --git a/Makefile b/Makefile index decc028c1c..1e82f5f0d3 100644 --- a/Makefile +++ b/Makefile @@ -45,7 +45,7 @@ define messaging cd examples/message pip install -r requirements.txt pip install -e ../../ - pytest + ./run_pytest.sh endef export messaging diff --git a/examples/message/README.md b/examples/message/README.md index 55081fa39b..13c7380576 100644 --- a/examples/message/README.md +++ b/examples/message/README.md @@ -99,9 +99,6 @@ def test_throw_exception_handler(pact): ## Provider -Note: The current example only tests the consumer side. -In the future, provider tests will also be included. - ``` +-------------------+ +-----------+ |(Message Provider) | message | (Pact) | @@ -110,10 +107,82 @@ In the future, provider tests will also be included. +-------------------+ +-----------+ ``` -## E2E Messaging +```python +import pytest +from pact import MessageProvider + +def document_created_handler(): + return { + "event": "ObjectCreated:Put", + "documentName": "document.doc", + "creator": "TP", + "documentType": "microsoft-word" + } + + +def test_verify_success(): + provider = MessageProvider( + message_providers={ + 'A document created successfully': document_created_handler + }, + provider='ContentProvider', + consumer='DetectContentLambda', + pact_dir='pacts' + + ) + with provider: + provider.verify() +``` + + +### Provider with pact broker +```python +import pytest +from pact import MessageProvider + + +PACT_BROKER_URL = "http://localhost" +PACT_BROKER_USERNAME = "pactbroker" +PACT_BROKER_PASSWORD = "pactbroker" +PACT_DIR = "pacts" + + +@pytest.fixture +def default_opts(): + return { + 'broker_username': PACT_BROKER_USERNAME, + 'broker_password': PACT_BROKER_PASSWORD, + 'broker_url': PACT_BROKER_URL, + 'publish_version': '3', + 'publish_verification_results': False + } + +def document_created_handler(): + return { + "event": "ObjectCreated:Put", + "documentName": "document.doc", + "creator": "TP", + "documentType": "microsoft-word" + } -Note: The current example only tests the consumer side. -In the future, provider tests will also be included. +def test_verify_from_broker(default_opts): + provider = MessageProvider( + message_providers={ + 'A document created successfully': document_created_handler, + }, + provider='ContentProvider', + consumer='DetectContentLambda', + pact_dir='pacts' + + ) + + with pytest.raises(AssertionError): + with provider: + provider.verify_with_broker(**default_opts) + +``` + +## E2E Messaging ``` +-------------------+ +-----------+ +-------------------+ diff --git a/examples/message/conftest.py b/examples/message/conftest.py new file mode 100644 index 0000000000..1ee4c7dc29 --- /dev/null +++ b/examples/message/conftest.py @@ -0,0 +1,52 @@ + +from testcontainers.compose import DockerCompose + +import pytest + + +def pytest_addoption(parser): + parser.addoption( + "--publish-pact", type=str, action="store", + help="Upload generated pact file to pact broker with version" + ) + + parser.addoption( + "--provider-url", type=str, action="store", + help="The url to our provider." + ) + + parser.addoption( + "--run-broker", type=bool, action="store", + help="Whether to run broker in this test or not." + ) + + +# This fixture is to simulate a managed Pact Broker or Pactflow account +# Do not do this yourself but setup one of the above +# https://github.com/pact-foundation/pact_broker +@pytest.fixture(scope='session', autouse=True) +def broker(request): + version = request.config.getoption('--publish-pact') + publish = True if version else False + + # yield + if not publish: + yield + return + + run_broker = request.config.getoption('--run-broker') + + if not run_broker: + yield + return + else: + print('Starting broker') + with DockerCompose("../broker", + compose_file_name=["docker-compose.yml"], + pull=True) as compose: + + stdout, stderr = compose.get_logs() + if stderr: + print("Errors\\n:{}".format(stderr)) + print(stdout) + yield diff --git a/examples/message/run_pytest.sh b/examples/message/run_pytest.sh index 15c3b1fdbd..35dade4fb1 100755 --- a/examples/message/run_pytest.sh +++ b/examples/message/run_pytest.sh @@ -1,7 +1,7 @@ #!/bin/bash set -o pipefail -pytest +pytest --run-broker True --publish-pact 2 # publish to broker assuming broker is active # pytest tests/consumer/test_message_consumer.py::test_publish_to_broker --publish-pact 2 diff --git a/examples/message/tests/conftest.py b/examples/message/tests/conftest.py deleted file mode 100644 index 7873fddf43..0000000000 --- a/examples/message/tests/conftest.py +++ /dev/null @@ -1,6 +0,0 @@ - -def pytest_addoption(parser): - parser.addoption( - "--publish-pact", type=str, action="store", - help="Upload generated pact file to pact broker with version" - ) diff --git a/examples/message/tests/consumer/test_message_consumer.py b/examples/message/tests/consumer/test_message_consumer.py index b5671da66c..d8376a4e61 100644 --- a/examples/message/tests/consumer/test_message_consumer.py +++ b/examples/message/tests/consumer/test_message_consumer.py @@ -123,6 +123,8 @@ def test_publish_to_broker(pact): `pytest tests/consumer/test_message_consumer.py::test_publish_pact_to_broker --publish-pact 2` """ + cleanup_json(PACT_FILE) + expected_event = { "event": "ObjectCreated:Delete", "documentName": "document.doc", diff --git a/examples/message/tests/provider/test_message_provider.py b/examples/message/tests/provider/test_message_provider.py index f7eab414dc..ae20ab3a13 100644 --- a/examples/message/tests/provider/test_message_provider.py +++ b/examples/message/tests/provider/test_message_provider.py @@ -1,6 +1,22 @@ import pytest from pact import MessageProvider +PACT_BROKER_URL = "http://localhost" +PACT_BROKER_USERNAME = "pactbroker" +PACT_BROKER_PASSWORD = "pactbroker" +PACT_DIR = "pacts" + + +@pytest.fixture +def default_opts(): + return { + 'broker_username': PACT_BROKER_USERNAME, + 'broker_password': PACT_BROKER_PASSWORD, + 'broker_url': PACT_BROKER_URL, + 'publish_version': '3', + 'publish_verification_results': False + } + def document_created_handler(): return { @@ -49,3 +65,19 @@ def test_verify_failure_when_a_provider_missing(): with pytest.raises(AssertionError): with provider: provider.verify() + + +def test_verify_from_broker(default_opts): + provider = MessageProvider( + message_providers={ + 'A document created successfully': document_created_handler, + 'A document deleted successfully': document_deleted_handler + }, + provider='ContentProvider', + consumer='DetectContentLambda', + pact_dir='pacts' + + ) + + with provider: + provider.verify_with_broker(**default_opts) diff --git a/pact/message_pact.py b/pact/message_pact.py index 7f67dc14f6..b8190d56b1 100644 --- a/pact/message_pact.py +++ b/pact/message_pact.py @@ -171,6 +171,7 @@ def write_to_pact_file(self): ] self._message_process = Popen(command) + self._message_process.wait() def _insert_message_if_complete(self): """ diff --git a/pact/message_provider.py b/pact/message_provider.py index d3da7ee572..4774fc84af 100644 --- a/pact/message_provider.py +++ b/pact/message_provider.py @@ -115,6 +115,25 @@ def verify(self): return_code, _ = verifier.verify_pacts(pact_files, verbose=False) assert (return_code == 0), f'Expected returned_code = 0, actual = {return_code}' + def verify_with_broker(self, enable_pending=False, include_wip_pacts_since=None, **kwargs): + """Use Broker to verify. + + Args: + broker_username ([String]): broker username + broker_password ([String]): broker password + broker_url ([String]): url of broker + enable_pending ([Boolean]) + include_wip_pacts_since ([String]) + publish_version ([String]) + + """ + verifier = Verifier(provider=self.provider, + provider_base_url=self._proxy_url()) + + return_code, _ = verifier.verify_with_broker(enable_pending, include_wip_pacts_since, **kwargs) + + assert (return_code == 0), f'Expected returned_code = 0, actual = {return_code}' + def __enter__(self): """ Enter a Python context. diff --git a/pact/verifier.py b/pact/verifier.py index fcd13a0d50..447a756e38 100644 --- a/pact/verifier.py +++ b/pact/verifier.py @@ -3,6 +3,7 @@ from pact.verify_wrapper import VerifyWrapper, path_exists, expand_directories + class Verifier(object): """A Pact Verifier.""" diff --git a/tests/test_message_provider.py b/tests/test_message_provider.py index dd38c06d30..8b71c3bdd3 100644 --- a/tests/test_message_provider.py +++ b/tests/test_message_provider.py @@ -6,6 +6,7 @@ from pact.message_provider import MessageProvider from pact import message_provider as message_provider + class MessageProviderTestCase(TestCase): def _mock_response( self, @@ -33,6 +34,13 @@ def setUp(self): 'a document created successfully': self.message_handler } ) + self.options = { + 'broker_username': "test", + 'broker_password': "test", + 'broker_url': "http://localhost", + 'publish_version': '3', + 'publish_verification_results': False + } def test_init(self): self.assertIsInstance(self.provider, MessageProvider) @@ -50,6 +58,17 @@ def test_verify(self, mock_verify_pacts): assert mock_verify_pacts.call_count == 1 mock_verify_pacts.assert_called_with(f'{self.provider.pact_dir}/{self.provider._pact_file()}', verbose=False) + @patch('pact.Verifier.verify_with_broker', return_value=(0, 'logs')) + def test_verify_with_broker(self, mock_verify_pacts): + self.provider.verify_with_broker(**self.options) + + assert mock_verify_pacts.call_count == 1 + mock_verify_pacts.assert_called_with(False, None, broker_username="test", + broker_password="test", + broker_url="http://localhost", + publish_version='3', + publish_verification_results=False) + class MessageProviderContextManagerTestCase(MessageProviderTestCase): def setUp(self): diff --git a/tests/test_verifier.py b/tests/test_verifier.py index 3822e46215..7b0a46f058 100644 --- a/tests/test_verifier.py +++ b/tests/test_verifier.py @@ -7,12 +7,14 @@ from pact.verifier import Verifier from pact.verify_wrapper import VerifyWrapper + def assertVerifyCalled(mock_wrapper, *pacts, **options): tc = unittest.TestCase() tc.assertEqual(mock_wrapper.call_count, 1) mock_wrapper.assert_called_once_with(*pacts, **options) + class VerifierPactsTestCase(TestCase): def setUp(self): @@ -130,6 +132,7 @@ def test_passes_include_wip_pacts_since_value(self, mock_path_exists, mock_wrapp mock_wrapper.call_args.kwargs, ) + class VerifierBrokerTestCase(TestCase): def setUp(self):