Yes, you read that right. I deleted almost 80% of the code from the Redis bitnami container image on Docker Hub and it still does everything I need. No tricks, no rabbits up my sleeve. It turns out there is a huge amount of unused code in the bitnami distribution — and this is a common problem for all of the popular container images.
It’s also way more secure. Let me show you why and how.
Unused code => lots of CVEs
Cloud-native infrastructure is built around open source technologies, which is great but comes with inherent security trade-offs. We all use several Docker containers to create cloud-native applications, but we often overlook how many vulnerabilities these open source images have. To counter this, Google, AWS, and DockerHub have all developed tools to scan containers and show CVE reports. Unfortunately, nothing has been done to fix the large number of CVEs in open source images, and it’s a well-kept secret that no one wants to talk about.
A few months back, I spoke to some former colleagues who were building RapidFort, a platform for removing unnecessary software packages from containers with minimal developer effort. While this solution works well for the code enterprises generate, I wondered about open source containers I was running at my own company. They might be carrying thousands of vulnerabilities yet to be discovered and it made me uneasy.
I was intrigued by the large number of CVEs in open source containers and started playing with RapidFort’s tech. I wanted to build a repository of popular open-source images that are slim, fast, and contain only the packages necessary to achieve common tasks.
For example: When running a production database, I might not need networking tools or several bash commands until and unless I plan to debug something. Why give malicious actors easy opportunities to land and expand in my infrastructure?
With all this in mind, I used RapidFort’s technology to build a completely open-source GitHub project: https://github.com/rapidfort/community-images. And I’m pleased to announce that as of this writing, you can download daily-updated hardened images for:
Before we get into the technical details, let me tell you how these images are generated.
Building container stubs to test in the CI/CD pipeline
When developers deliver a product, they write some code, build a container image, and deploy that image to production. Most of the time, no one scans that container to make sure it’s secure. This is where RapidFort makes a difference.
With RapidFort, a developer can generate an instrumented image called a “stub.” This stub can run in a test environment that exercises (or covers) expected workflows using a coverage script. (Please note that the coverage script is not the same as the testing script. The coverage script needs to cover primary functionality and doesn’t need to worry about the output or functional/integration testing.) RapidFort tracks what code was actually used in the coverage test and gives a developer the opportunity to delete anything unused.
Stubs are very small and easy to put together. Once the coverage script uses the container image, RapidFort can provide a hardened image. This hardened image contains only the software required to run the workload. Voila, you have a lean, secure image that can be tested in a CI/CD pipeline and released/deployed to production.
How I shrank the Redis image by 78%
To create our repository of the most commonly used images, I started with Redis, one of the most frequently downloaded images on the internet at more than a billion downloads.
$ docker pull redis:latest
After downloading the image, I created a stub (instrumented) container through rapidfort cli.
$ rfstub redis:latest
I quickly pulled together a coverage script that exercises the most commonly used Redis commands.
After using the commands, I wanted to exercise TLS connections, for which I just ran the same tests in the TLS mode. I also wanted to build an image for Redis Cluster that can handle clustering. However, the Redis-cli won’t allow a file to be passed for Redis commands as the keys can move around to different nodes. I quickly built a bash script that picks one command at a time and runs through redis-cli.
#!/bin/bash
set -x
REDIS_PASSWORD=$1
input="${REDIS_TEST_FILE}"
while IFS= read -r line
do
# shellcheck disable=SC2086
REDISCLI_AUTH="${REDIS_PASSWORD}" redis-cli -h "${HELM_RELEASE}" "${TLS_PREFILX[@]}" -c $line
done < "$input"
I also covered three runtimes for Redis containers:
- Kubernetes
- Docker container
- Docker-compose
These coverage scripts cover most use cases. After running the coverage scripts, I created a hardened image from RapidFort cli.
$ rfharden redis:latest-rfstub
To ensure the hardened image works in most scenarios, I used https://redis.io/docs/reference/optimization/benchmarks/ as benchmark testing. This tool allowed me to exercise different commands at scale on Redis and proved the image is ready for testing on production.
I also built a completely automated CI/CD pipeline for the community images using GitHub actions. This ensures I create a hardened version for source images within one hour of publishing.
Shrinking Redis by 78% removed 66% of known vulnerabilities
The results from hardening these images were impressive:
- The Redis image shrank from 96 MB to 21 MB.
- 66% of the known CVE(s) were removed.
- Of 74 critical and high CVEs, the CVE scanner found only thirty-six in the hardened image.
- All 4 CVEs with published proofs of concepts (POCs) were removed. CVEs with a published POC present a higher risk and a Rapid Risk Score (RRS) is a measure of this risk. More about RRS (POC) here.
When my team is faced with 74 critical, high CVEs to remediate, it requires a lot of additional work. Using my coverage test, RapidFort was able to eliminate 38 of those CVEs, meaning my team has much less work to do to prepare the image for production.
Join the community and try these hardened images
After months of testing, I am excited to release our first versions of the hardened Community Images.
I convinced my colleagues to use these hardened images to build the RapidFort system, demonstrating our confidence that the rest of the world can use these images.
We need your help with what’s next for our Community Images project
While I am happy to release our first version of images, the work is not done yet until we have covered most images available for everyone to use. I am looking for volunteers to join our Community Images project and contribute. If you are interested in reducing the number of vulnerabilities your infrastructure carries, please download these hardened images.
If you run into any issues, please feel free to contribute or open an issue. https://github.com/rapidfort/community-images/issues/new/choose
If you have any questions, I’d love to hear them in the comments.