Test containers with Python and Conu

More and more developers are using containers to develop and deploy their applications. This means that easily testing containers is also becoming important. Conu (short for container utilities) is a Python library that makes it easy to write tests for your containers. This article shows you how to use it to test your containers.

Getting started

First you need a container application to test. For that, the following commands create a new directory with a container Dockerfile, and a Flask application to be served by the container.

$ mkdir container_test
$ cd container_test
$ touch Dockerfile
$ touch app.py

Copy the following code inside the app.py file. This is the customary basic Flask application that returns the string “Hello Container World!”

from flask import Flask
app = Flask(__name__)

@app.route('/')
def hello_world():
    return 'Hello Container World!'

if __name__ == '__main__':
    app.run(debug=True,host='0.0.0.0')

Create and Build a Test Container

To build the test container, add the following instructions to the Dockerfile.

FROM registry.fedoraproject.org/fedora-minimal:latest
RUN microdnf -y install python3-flask && microdnf clean all
ADD ./app.py /srv
CMD ["python3", "/srv/app.py"]

Then build the container using the Docker CLI tool.

$ sudo dnf -y install docker
$ sudo systemctl start docker 
$ sudo docker build . -t flaskapp_container

Note : The first two commands are only needed if Docker is not installed on your system.

After the build use the following command to run the container.

$ sudo docker run -p 5000:5000 --rm flaskapp_container
* Running on http://0.0.0.0:5000/ (Press CTRL+C to quit)
* Restarting with stat
* Debugger is active!
* Debugger PIN: 473-505-51

Finally, use curl  to check that the Flask application is correctly running inside the container:

$ curl http://127.0.0.1:5000
Hello Container World!

With the flaskapp_container now running and ready for testing, you can stop it using Ctrl+C.

Create a test script

Before you write the test script, you must install conu. Inside the previously created container_test  directory run the following commands.

$ python3 -m venv .venv
$ source .venv/bin/activate
(.venv)$ pip install --upgrade pip
(.venv)$ pip install conu

$ touch test_container.py

Then copy and save the following script in the test_container.py file.

import conu

PORT = 5000

with conu.DockerBackend() as backend:
  image = backend.ImageClass("flaskapp_container")
  options = ["-p", "5000:5000"]
  container = image.run_via_binary(additional_opts=options)
  
  try:
    # Check that the container is running and wait for the flask application to start.
    assert container.is_running()
    container.wait_for_port(PORT)
    
    # Run a GET request on / port 5000.
    http_response = container.http_request(path="/", port=PORT)
    
    # Check the response status code is 200
    assert http_response.ok
    
    # Get the response content
    response_content = http_response.content.decode("utf-8")

    # Check that the "Hello Container World!" string is served.
    assert "Hello Container World!" in response_content

    # Get the logs from the container
    logs = [line for line in container.logs()]
    # Check the the Flask application saw the GET request.
    assert b'"GET / HTTP/1.1" 200 -' in logs[-1]

  finally:
    container.stop()
    container.delete()

Test Setup

The script starts by setting conu to use Docker as a backend to run the container. Then it sets the container image to use the flaskapp_container you built in the first part of this tutorial.

The next step is to configure the options needed to run the container. In this example, the Flask application serves the content on port 5000. Therefore you need to expose this port and map it to the same port on the host.

Finally, the script starts the container, and it’s now ready to be tested.

Testing methods

Before testing a container, check that the container is running and ready. The example script is using container.is_running and container.wait_for_port. These methods ensure the container is running and the service is available on the expected port.

The container.http_request is a wrapper around the requests library which makes it convenient to send HTTP requests during the tests. This method returns a requests.Response object, so it’s easy to access the content of the response for testing.

Conu also gives access to the container logs. Once again, this can be useful during testing. In the example above, the container.logs method returns the container logs. You can use them to assert that a specific log was printed, or for example that no exceptions were raised during testing.

Conu provides many other useful methods to interface with containers. A full list of the APIs is available in the documentation. You can also consult the examples available on GitHub.

All the code and files needed to run this tutorial are available on GitHub as well.  For readers who want to take this example further, you can look at using pytest to run the tests and build a container test suite.

For Developers Using Software

2 Comments

  1. Pavan Sapkal

    can we test other application using conu except container of docker ?

Comments are Closed

The opinions expressed on this website are those of each author, not of the author's employer or of Red Hat. Fedora Magazine aspires to publish all content under a Creative Commons license but may not be able to do so in all cases. You are responsible for ensuring that you have the necessary permission to reuse any work on this site. The Fedora logo is a trademark of Red Hat, Inc. Terms and Conditions