1MinDocker #11 - Advanced compose example
Intermediate ·In the last article, we introduced the compose
file reference, and we went through numerous top-level elements and their attributes. In this article, we will present an advanced example using docker compose
to deploy a multi-container application on the cloud.
We will refer, for this tutorial, to a slightly modified version of the compose.yaml
file proposed for ElasticSearch-Logstash-Kibana by the awesome-compose repository by Docker on GitHub.
Background
Why would we build a multi-container environment with ElasticSearch, Logstash and Kibana? Well, the idea is that this stack (also known as the ELK stack), can help us monitor logs and data sent by third party services (Logstash), store and index them for fast search/retrieval (ElasticSearch) and visualize statistics in real time (Kibana).
This is optimal when we have real-time data flows (such as with IoT devices, social media or web traffic), we want to track logs from big servers and analyze them and/or we want to search data-heavy applications, such as e-commerce ones.
If you want to know more, please refer to:
As you can see, these three services are all provided by Elastic.
Set-up
Getting the needed files
First of all, we clone the awesome-compose
repository:
git clone https://github.com/docker/awesome-compose.git
And we head over to our folder of interest:
cd awesome-compose/elasticsearch-logstash-kibana/
TIPđŸ’¡: you can use all the folders in this repository to experiment with different
compose
settings
And we can take a look at the structure of the repository:
# You might need to install tree before using it
tree . -L 2
We will get out this structure:
.
|__ logstash/
| |__nginx.log
| |__pipeline/
|__compose.yaml
The logstash
folder contains a pipeline
subfolder and a nginx.log
file: their purpose is not important for our scopes, but we have to keep in mind that they are there.
Adding some modifications
To showcase more of the compose
file elements, we introduce a .env
file, which we can create in this way:
touch .env
And then modify with our favorite text editor (for me it’s VSCode):
code .env
In the .env
file, let’s create the following keys and values:
JAVA_OPTS="-Xms512m -Xmx512m"
DISCOVERY_SEED_HOSTS="logstash"
API_TOKEN="super-secret-token"
We will use them in the compose
file (see below).
The compose
file
Let’s take a look to the compose file, which is slightly different from the one proposed by awesome-compose
:
services:
elasticsearch:
image: elasticsearch:7.16.1
container_name: es
environment:
discovery.type: single-node
ES_JAVA_OPTS: $JAVA_OPTS # as we mentioned in the last article, compose can access the env variables we set in our .env file
ports:
- "9200:9200"
- "9300:9300"
healthcheck:
test: ["CMD-SHELL", "curl --silent --fail localhost:9200/_cluster/health || exit 1"]
interval: 10s
timeout: 10s
retries: 3
networks:
- elastic
logstash:
image: logstash:7.16.1
container_name: log
environment:
discovery.seed_hosts: $DISCOVERY_SEED_HOSTS
LS_JAVA_OPTS: $JAVA_OPTS
secrets:
- api_token
volumes:
- ./logstash/pipeline/logstash-nginx.config:/usr/share/logstash/pipeline/logstash-nginx.config
- ./logstash/nginx.log:/home/nginx.log
ports:
- "5000:5000/tcp"
- "5000:5000/udp"
- "5044:5044"
- "9600:9600"
depends_on:
- elasticsearch
networks:
- elastic
command: logstash -f /usr/share/logstash/pipeline/logstash-nginx.config
kibana:
image: kibana:7.16.1
container_name: kib
volumes:
- kibana-cache:/etc/logs/cache
ports:
- "5601:5601"
depends_on:
- elasticsearch
networks:
- elastic
networks:
elastic:
driver: bridge
secrets:
api_token:
environment: "API_TOKEN"
volumes:
kibana-cache:
external: true
name: kibana_cache_volume
We will now break this compose
file down service by service.
ElasticSearch
ElasticSearch is based on the Docker image elasticsearch:7.16.1, and the container that is launched within the service is named es.
It accesses two environment variables: ES_JAVA_OPTS, that is read from the .env
file, and discovery.type, which is set in-line.
It is connected to two ports: 9200 and 9300, accessible to the local device host on the same port addresses.
There is an healthcheck, that is performed for a maximum of three times with 10s intervals and a timeout (maximum time of execution of the test command) of 10s.
The service is bound to the elastic network, whose driver is the most common: bridge (simply connects all the containers together, bridging among them).
This container does not depend on anything, so it is the first to be started.
Logstash
ElasticSearch is based on the Docker image logstash:7.16.1, and the container that is launched within the service is named log.
It accesses two environment variables: LS_JAVA_OPTS, that is read from the .env
file, and discovery.seed_hosts, which is also read from the .env` file.
It also has access to a secret, api_token, which is read from the environment file also as the API_TOKEN variable.
Inside the container, two volumes are mounted from the local file system (the pipeline
subfolder and the nginx.log
file) into the container’s system.
It is connected to three ports: 5600, 5044 and 9300, accessible to the local device host on the same port addresses.
There is no healthcheck, but the service depends on ElasticSearch so it is the second one to be started: as soon as the container is started, the command logstash -f /usr/share/logstash/pipeline/logstash-nginx.config
is executed, overriding the eventual CMD
entries in the original Dockerfile from the logstash
image.
The service is bound to the elastic network.
Kibana
Kibana is based on the Docker image kibana:7.16.1, and the container that is launched within the service is named kib.
It does not access environment variables or secrets.
Inside the container, there is one volume mounted, kibana-cache, the depends on a volume whose life cycle is externally managed, and which is named kibana_cache_volume.
It is connected to one ports: 5061, accessible to the local device host on the same port address.
There is no healthcheck, but the service depends on ElasticSearch so it is the second one to be started, along with Logstash.
The service is bound to the elastic network.
Launch and stop the service
Now, we can just launch our multi-container application with:
docker compose up
Or we can use:
docker compose up --detach
If we do not want the services logs to be displayed (and, potentially swamp) on our terminal.
If we want to see what is currently running, we can use:
docker compose ps
And if we want to stop the services, or take down the compose
app, we can simply run:
# Stop the services without removing containers
docker compose stop
# Stop the services and remove containers
docker compose down
We will stop here for this article, and in the next one we will explore continuous-integration/continuous-deployment solution for Docker images. See you in 2025!đŸ¥°