At Bitnami, we are committed to ensuring the quality of the apps we deliver, and as such, tests play a fundamental role in the bitnami/containers
repository. Bear in mind that every contribution to our containers is ultimately published to our container registries, where it is made available for the rest of the community to benefit from. Before this happens, different checks are required to succeed. More precisely, tests are run when a new contribution (regardless of its author) is made through a GitHub Pull Request.
In this section, we will discuss:
- Where to find the tests
- VMware Image Builder (VIB)
- VIB pipeline definition file
- Testing strategy
- Generic acceptance criteria
- GOSS
All the apps have an associated folder inside /.vib with their custom tests implementation (the goss
subfolder) and the file containing their test plan (vib-verify.json
).
The service that powers the verification of the thousands of monthly tests performed in the repository is VMware Image Builder. VMware Image Builder (VIB) is a platform-agnostic, API-first modular service that allows large enterprises and independent software vendors to automate the packaging, verification, and publishing processes of software artifacts on any platform and cloud.
The CI pipeline in this repository will be used to verify the changes proposed in a PR and triggered by any new commits once said PR is ready to be verified. But as every application is different, VIB needs to be supplied with a definition of the set of actions and configurations that precisely describe the verification process to perform in each case. This is the role of the aforementioned vib-verify.json
file, which every app defines and can be found alongside its tests inside the /.vib
folder. Keeping it simple, the vib-verify.json
file defines what VIB should do.
Let's take a look at an example and try to understand it!
{
"context": {
"resources": {
"url": "{SHA_ARCHIVE}",
"path": "{VIB_ENV_PATH}"
},
"runtime_parameters": "Y29tbWFuZDogWyJ0YWlsIiwgIi1mIiwgIi9kZXYvbnVsbCJd"
},
...
"phases": {
...
"verify": {
"actions": [
{
"action_id": "goss",
"params": {
"resources": {
"path": "/.vib"
},
"tests_file": "wordpress/goss/goss.yaml",
"vars_file": "wordpress/goss/vars.yaml",
"remote": {
"pod": {
"workload": "deploy-wordpress"
}
}
}
},
{
"action_id": "trivy",
"params": {
"threshold": "LOW",
"vuln_type": [
"OS"
]
}
},
...
]
}
}
}
This guide will focus on the verify
phase section, of which there are some things to remark on:
-
For the testing of containers, VIB will take the container built in the previous
package
phase and include it in a basic Helm chart template composed of a deployment and service template. -
VIB does only allow to modify the
ENTRYPOINT/CMD
of the image (throughruntime_parameters
). Consequently, this both simplifies and limits the configurability of the template chart and container image tested underneath. -
A container's testing phase will usually include a single
goss
testing action, followed by additional security-related actions.
Note
Some containers with per-branch ARM support use separate per-branch vib-verify.json
pipelines. Remember to replicate changes performed on the main pipeline definition file to those pipelines.
This strategy has to be understood together with the VIB limitations for the containers mentioned above. These restraints prevent us from setting up complex multi-container testing scenarios, which is a necessity for most of our containers to be initialized properly. To work around this, we will only test up to the postunpack phase of the container (where the initial filesystem changes are done). Essentially, we are assuming that most apps’ integration and functional tests are performed in the related chart’s test suite and thus are not required to be duplicated for the containers catalog. As a consequence, we will concentrate on the verification of the app’s compilation logic and the container’s filesystem itself.
Some examples of the suitability of tests for the bitnami/wordpress
container:
- ✅ Checking Apache config added in Wordpress' postunpack stage
- ❌ Manually finishing the container initialization to run functional tests
- ✅ Checking the image filesystem (created dirs existence, changed permissions, etc) related to the compilation/postunpack stages
- ❌ Verifying bash logic performed at
libwordpress.sh
, as we'll run Goss before that logic is executed - ✅ Testing binaries' existence and usability
Something of note is the equality of scope for the whole container catalog. Though some apps do not require additional containers to run and can be fully initialised as they are, we will be using the same testing setup for every container.
As of now, and linking with the scope definition we saw previously, the runtime_parameters
field is only used to "stop" a container's initialization logic after its postunpack
has been executed. The runtime_parameters
value is a base64 encoded string going by command: ["tail", "-f", "/dev/null"]
and is the same for every VIB pipeline defined in bitnami/containers
.
For your test code PR to be accepted the following criteria must be fulfilled:
- Test scope needs to be focused on installation of the app and not testing the app
- Key features of the app should be covered, when possible
- Tests need to contain assertions
- Tests need to be stateless
- Tests need to be independent
- Tests need to be retry-able
- Tests need to be executable in any order
- Test code needs to be peer-reviewed
- Tests need to be as minimalistic as possible
- Tests should run properly for future versions without major changes
- Avoid hardcoded values
- Include only necessary files
- Test code needs to be maintainable
- Test names should be descriptive
- Test data should be generated dynamically
GOSS is the framework used to implement integration tests and the only testing tool presently used in our VIB pipelines. It is the reference tool to use when tests require interaction with a specific pod, with its tests being executed from within the pod.
For VIB to execute GOSS tests, the following block of code needs to be defined in the corresponding VIB pipeline definition file (/.vib/app/vib-verify.json
).
Note
Values denoted withing dollar signs ($$VALUE$$
) should be treated as placeholders
{
"action_id": "goss",
"params": {
"resources": {
"path": "/.vib"
},
"tests_file": "$$app$$/goss/goss.yaml",
"vars_file": "$$app$$/goss/vars.yaml",
"remote": {
"pod": {
"workload": "deploy-$$app$$" // As explained previously, the used template is always a deployment
}
}
}
}
Related files should be located under /.vib/app/goss
.
A GOSS test suite for a given application can be divided in two. One half contains the tests specifically manufactured to verify different aspects of the application and the other half is intended to verify that the container complies with Bitnami's best practices. As some of these tests are oftentimes almost identical between different apps, we have compiled some of them in a series of GOSS templates to unify and simplify their usage.
At their core, GOSS templates are just tests of similar nature put together on a .yaml
file under the .vib/common/goss/templates
folder.
Every template is composed of:
- A brief description of the tests' nature
- Any needed vars it may use to run its tests
- One or more tests
To better understand them, let's see one of these templates:
########################
# Checks binaries are added to the $PATH
# Needed vars:
# - .Vars.binaries (Array)
########################
command:
{{ range $binary := .Vars.binaries }}
check-{{ $binary }}-binary:
exec: which {{ $binary }}
exit-status: 0
{{ end }}
In the example above, we can see the template will execute a which
command for every binary included in the .Vars.binaries
array. These variables may be optional or required and must be defined in the /.vib/app/goss/vars.yaml
file.
There are many different templates, some of them focusing on verifying specific types of apps (those that use PHP, NGINX, etc.) or a particular best practice. In the same vein, new templates can be added if a particular group of tests is going to be used by several apps.
There are instances where it is not needed to create custom tests for a given app, where using the templates will suffice. There will also be suites that may require the use of testing files to properly verify the app. Generally, a test suite can be composed of the following files:
.vib/java/goss
├── goss.yaml
├── java.yaml
├── testfiles
│ └── HelloTest.jar
└── vars.yaml
- The optional
app.yaml
file is where custom tests created just for the related app verification are included. - The
goss.yaml
file should include the list of used GOSS templates as well as theapp.yaml
file (if necessary). - The
vars.yaml
file should include every variable used in the templates. - The
testfiles
folder should include any external files used in theapp.yaml
tests.
Not every suite will be composed of the same tests, as it will depend on the type of application, its Dockerfile, and the used compilation/configuration logic. The list below details each pillar that should be checked when creating a test suite as well as when to use some of the most common templates:
- Dockerfile check:
- Does it include bitnami components which contain binaries? If so, use the
check-binaries.yaml
GOSS template. - Does it include the
ca-certificates
package? If so, use thecheck-ca-certificates.yaml
template.
- Does it include bitnami components which contain binaries? If so, use the
- Compilation logic check:
- Are there files or directories created and/or with permission changes? If so, use the
check-directories.yaml
and/orcheck-files.yaml
templates. - Are there additional files/dirs modifications? If so, use custom filesystem tests.
- Add tests for any compilation options or flags used.
- Are there files or directories created and/or with permission changes? If so, use the
- Check postunpack script:
- Are there files or directories created and/or with permission changes? If so, use the
check-directories.yaml
and/orcheck-files.yaml
templates. - Are there additional files/dirs modifications? If so, use custom filesystem tests.
- Are there files or directories created and/or with permission changes? If so, use the
- Check apps version:
- If
$APP_VERSION
follows semver version, use thecheck-app-version.yaml
template.
- If
- Check apps-dependant tests:
- If the app is a runtime, test if the runtime can run a compatible file.
- If the app requires yet-to-run initialization logic:
- No complex configuration nor testing environment is created manually.
- If possible, test the app's basic features.
- If the app is just a part of a bigger setup (exporter, multi-container apps, etc.):
- No testing environment is created manually.
- If possible, test the app's basic features.
- If the app uses subcomponents (java/php, apache, etc.):
- If possible, test a subcomponent when there is a custom config added to them.
- Verify whether the subcomponent is capable of running the app.
- Must-have templates to be added to every suite:
check-linked.libraries.yaml
check-broken-symlinks.yaml
check-sed-in-place.yaml
check-spdx.yaml
- Final checks:
- When possible, NO per-branch tests are used.
- Every GOSS template is included in
goss.yaml
and the needed vars are in place.
- No distro-specific tests are included.
- Prioritise using the tests included in the templates over creating custom tests.
- Due to GOSS limitations, there can't be two tests with the same name or checking the same file/directory. In those cases, only one of them will be run.
- For clarity purposes, the vars needed for the GOSS templates shouldn't be used in the custom tests at
app.yaml
. - Tests should not rely on system packages (e.g.
curl
). Favor built-in GOSS primitives instead. - Prefer checking the exit status of a command rather than looking for a specific output. This will avoid most of the potential flakiness.
Sometimes it is of interest to run the tests locally, for example during development. Though there may be different approaches, you may follow the steps below to execute the tests locally:
-
Download the GOSS binary for Linux
-
Launch the container using some command that ensures it will not exit immediately. Find two examples below:
docker run --rm --name app_name -d -it bitnami/app_name bash -c "tail -f /dev/null"
or for a scratch container (e.g. Node.js minimal):
docker run --rm --name app_name -d -it --entrypoint node bitnami/app_name --eval "setTimeout(() => {}, 3600 * 1000);"
-
Add the binary and test files to the tested container as volumes
chmod +x /local/path/to/binary/goss-linux-amd64 docker cp /local/path/to/binary/goss-linux-amd64 app_name:/usr/local/bin/gossctl docker cp /local/path/to/repo/containers/.vib app_name:/vib
-
Launch the tests
$ docker exec --workdir /vib app_name goss --gossfile /vib/app_name/goss/goss.yaml --vars /vib/app_name/goss/vars.yaml validate ......... Total Duration: 1.203s Count: 11, Failed: 0, Skipped: 0