Skip to content

This repository show how Consumer-driven Contract Testing (CDCT) works in a serverless scenario (two asynchronous AWS Lambdas that communicate through an Amazon SNS topic) with Pact. Even though the AWS orientation the principles behind this is cross-provider/language!

Notifications You must be signed in to change notification settings

mariano-calandra-xp/serverless-cdct

 
 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

1 Commit
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Introduction

This Git repository is the code companion for this Medium blogpost. Aim of the article is to demonstrate how challenging can be the testing of service oriented architectures (in our example two micro/nanoservices deployed on AWS Lambda).

From "unit" to "consumer-driven contract" testing

With good reason, unit tests are seen as a guiding light by every good developer. As the name suggest, this type of test involves just a unit of the overall codebase, without any external dependencies like databases, HTTP calls, queues, topics or something like that. This characteristics makes unit testing deterministic and, above all, fast; so we could run an entire suite of thousands of tests in a bunch of seconds. Back in the days of big fat monolithic projects, a good suite of unit tests may validate the entire system right after a refactoring work.

Today, with the raise of microservices architectures, this is simply undoable! What we used to call the system, in fact, is now spread into hundreds or thousands of tiny little projects, completely unaware one another. For this reason, unit testing (alone) is not enough when it comes to testing microservices oriented architectures!

Beware: I’m not saying that unit testing isn’t useful! What I’m saying, instead, is that even though unit tests are a great tool for validating microservices’ internal behaviour; what we really need is a complementary tool for testing their external behaviour too. Generally speaking, we need to check that a modification of a provider service does not impact its consumers. As this type of tests are consumer-first they are called Consumer-Driven Contract Testing (CDCT).

Read more here.

Filetree

.
├── README.md                       # this file
├── list-of-orders                  # ListOfOrders service
│   ├── package-lock.json
│   ├── package.json
│   ├── src
│   │   └── add-order-lambda.js     # Lambda's source
│   └── test
│       └── consumer.test.js        # Consumer's test
├── orders                          # Orders service
│   ├── package-lock.json
│   ├── package.json
│   ├── src
│   │   └── create-order-lambda.js  # Lambda's source
│   └── test
│       └── provider.test.js        # Provider's test
├── pacts
│   └── listoforders-orders.json    # contract built after cdct run
└── template.yml                    # IaC definition

7 directories, 12 files

CDCT in action

To run your test you have to install all the dependencies (including testing frameworks: Pact for CDCT and Jest) for both microservices list-of-orders and orders:

cd orders
rm -rf node_modules
npm install --development

cd ../list-of-orders
rm -rf node_modules
npm install --development

once this activity has been done we have to create our contracts (you can read the aforementioned article for details). As we said, these tests are consumer-first, so we have to run the ListOfOrders tests first, so if you are not already there, move into list-of-orders directory and run npm run test (it is an alias, defined in package.json):

cd list-of-orders
npm run test

asciicast

If everything is OK, you should see a /pacts directory with a JSON file in it that represent the contract (i.e. the expectations of ListOfOrders service).

From now on, every time we need to modify the provider service (i.e. Orders) we can run its test on it and verify if it broke the contract or not. For a better understanding, move into orders directory and run the provider's test:

cd orders
npm run test

asciicast

Everything is OK!

Suppose now a junior developer is working with Orders service and, he thinks that is a good idea to change
the line 47 of method makeOrderConfirmedEvt in create-order-lambda.js, so the message produced will be:

{
    "type": "order_confirmed",
    "order_id": "3123",
    "timestamp": "2019-12-10T17:16:06.656Z",
    "customer_name": "peter",
    "address": "14 North Moore Street, New York"
}

instead of

{
    "type": "order_confirmed",
    "order_id": "3123",
    "timestamp": "2019-12-10T17:16:06.656Z",
    "customer_id": "peter",
    "address": "14 North Moore Street, New York"
}

With a small work of refactoring, this change to the internal of Orders service will be easily accepted by unit test. We have a great problem now, because ListOfOrders service (i.e. the consumer) still waits for customer_id otherwise an exception will be thrown.

With CDCT in place we avoid this risk, as we have a tool to validate ahead of time if the contract between producer and consumer has been broken.

asciicast

Deploy on AWS

If CDCT doesn't return any error, you could safely deploy this application using the following command:

# satisfy orders dependencies
cd orders
rm -rf node_modules
npm install --production

# satisfy list-of-orders dependencies
cd ../list-of-orders
rm -rf node_modules
npm install --production

# deploy the solution
cd ..
sam deploy --profile <target profile> # how to create a target profile? https://amzn.to/2sI6s7i

If this is the first deploy of the project, some information will be asked through a wizard.

Note

Concepts and ideas expressed in that article remains valid whatever the technological stack! We used Pact as CDCT framework and AWS Lambda as deployment, but you could reimplement the solution on-prem using Kubernetes and testing the contracts manually.

About

This repository show how Consumer-driven Contract Testing (CDCT) works in a serverless scenario (two asynchronous AWS Lambdas that communicate through an Amazon SNS topic) with Pact. Even though the AWS orientation the principles behind this is cross-provider/language!

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • JavaScript 100.0%