-
Notifications
You must be signed in to change notification settings - Fork 6
deprecated proposals
Tests are written in subsequent Dockerfiles that start FROM tool_image
, which is the original image for the tool. Each subsequent Dockerfile contains all the necessary arrangements (additional testing dependencies, pull data, run commands, possibly compare with expected output, or simple call a defined test suite). Testing is done when you do docker build -t tool_test1 Dockerfile.test1
, image is deleted upon test completion, CI reports success or failure per test, including build log. Tests are triggered when the docker image building & pushing is successful at the CI, on the “latest” image, as a chained job.
Consider a docker image built and tagged as:
docker build -t container-registry.phenomenal-h2020.eu/phnmnl/myapp .
docker push container-registry.phenomenal-h2020.eu/phnmnl/myapp
Then we would have a second image for test 1:
FROM container-registry.phenomenal-h2020.eu/phnmnl/myapp:latest
MAINTAINER PhenoMeNal-H2020 Project ( [email protected] )
# We get the data
RUN wget http://url/to/my/data.test1.tar.gz
# We run the test command
RUN exec_myapp -i data.test1.tar.gz
A real example of this is available at the midcor testing job on the PhenoMeNal Jenkins. Container for that test is available here.
- The
tool_image
only keeps whatever is necessary for the tool and is not augmented in size by any testing dependencies/data. - You could have even one separate container per unit test if wanted.
- These tests can be naturally chained in any hierarchical way within the CI (although if this hierarchy gets very complicated, this becomes an issue).
- One Test, one Jenkins jobs (so you can make it very granular).
- You have one or more additional dockerfiles per tool.
- Test dependency hierarchy could become complicated and require a CI plugin on its own.
- Since these are build processes and not run processes, is difficult to outsource them to a container orchestrator (as you probably want to outsource these builds to some sort of cluster, because they might entails some heavy computation in some cases)
I found very recently as well that you could use several CMD
lines in the dockerfile, and then when it is built, it generates separate images at once (so the first one is the "production" one, and all the subsequents ones are tests). See details here, on the comment by Dimitri Toubelis from Alkeron Multimedia. He proposes something like:
FROM ubuntu:16.04
MAINTAINER PhenoMeNal-H2020 Project ( [email protected] )
# install packages required to run app
RUN apt-get update && apt-get install -y \
"<app runtime=""> and <dependencies>" \ # add app runtime and required packages
&& rm -rf /var/lib/apt/lists/*
# copy app files
COPY app app
COPY run.sh run.sh
# main app command
CMD ["run.sh", "<app arguments="">"]
#--- add tests after this line ---
# install packages required to run tests
RUN apt-get update && apt-get install -y \
"<test tools=""> and <dependencies>" \ # add testing tools and required packages
&& rm -rf /var/lib/apt/lists/*
COPY tests tests
COPY test.sh test.sh
VOLUME "/var/tests/results"
# main test command
CMD ["test.sh", "<test arguments="">"]
- This avoids having more than a Dockerfile per tool.
- The
tool_image
only keeps whatever is necessary for the tool and is not augmented in size by any testing dependencies/data. - You could have even one separate container per unit test if wanted.
- It might complicate things at the CI level as you need to recognise the different images being created and tag them.
- What happens when the first test fails? probably build process stops there and no other tests are not run.
- Lose granularity of the tests probably, as the CI will see them as a single entity.
This would entail a combination of (proposals 1 or 2) and proposal 3. It means, generate a separate Docker image for the test (through its own Dockerfile, based on the proposal 1; provision data and dependencies), yet execute the test commands itself as a run in the container orchestration.
- Can be naturally outsourced to the container orchestrator for heavy computation.
- The
tool_image
only keeps whatever is necessary for the tool and is not augmented in size by any testing dependencies/data. - You could have even one separate container per unit test if wanted.
- You have one or more additional dockerfiles per tool.
- Test dependency hierarchy could become complicated and require a CI plugin on its own.
- Might require additional effort to add as a job in the CI, specially since we have now both build and run jobs chained.
We only have one Dockerfile without any test environment and test data. Thus, we build the docker containers as we would without testing. However, in github we create a test.sh script alongside where the Dockerfile is located. The test.sh script contains all the commands needed for running the tests from within the docker. Needed input and output files are supplied via the -v
at the commandline when starting the docker alongside the test.sh script which is executed instead of the CMD
. For testing, the docker is started with a custom but formalized command.
docker run -v phnmnl-test:/test-in:ro -v /tmp/phnmnl-test:/test-out:rw -ti phnmnl/msconvert test.sh
test.sh
#!/bin/sh
msconvert -i /test-in/raw-file.bruker -o /test-out/output.mzml --mzml
if [[ $(cat /test-in/raw-file.check.mzml) != $(cat /test-out/output.mzml) ]]; then
echo "test failed"
exit 1
else
echo "test ok"
fi
- We only have one Dockerfile which does not need any modifications.
- We could have separate test1.sh and test2.sh files for different types of tests.
- We could easily have different tests for integration testing.
- Different test1.sh and test2.sh could be run in parallel.
- The docker is being started with a rather lengthy command which could be hard to handle with CI.
- The volumes have to be managed somehow (where are the test-data located and how do we get them to the docker?).
Funded by the EC Horizon 2020 programme, grant agreement number 654241 |
---|