Skip to content

The Docker Compose structure for running the application suite.

License

Notifications You must be signed in to change notification settings

freezingsaddles/freezing-compose

Repository files navigation

Freezing Saddles Docker Compose Files

This project is for deploying the Freezing Saddles application for either development or production.

Before you get started, we assume you have already installed Docker and Docker Compose. Older versions of Docker may not work properly, as they may not support some features this uses, such as buildsteps. This has been tested with these versions:

  • Docker version 18.06.3-ce, build d7080c1, on Container Linux by CoreOS stable (2303.3.0)
  • Docker version 19.03.5, build 633a0ea, on macOS High Sierra 10.13.6 (17G9016)
  • Docker version 19.03.5, build 633a0ea, on CentOS Linux release 7.7.1908 (Core)
  • Docker version 24.0.6, build ed223bc, on macOS Sequoia 15.1.1 (24B91)
  • Docker version 26.1.3, build b72abbb, on Rocky Linux release 8.10 (Green Obsidian)
  • Docker version 27.4.1, build b9d17ea, on Rocky Linux release 9.4 (Blue Onyx)

It definitely will not work with the Docker 1.13 that ships with CentOS 7, for example. If you want to use this with CentOS or another Red Hat Enterprise Linux compatible system, please follow the Docker Engine CE instructions from the vendor. A provisioning script for Rocky Linux 9 is in bin/provision-server.sh that will install required tooling on a fresh Rocky Linux 9 system.

1. Development

You can use the docker-compose.dev.yml file, in conjunction with the main docker-compose.yml file, to spin up services that might be needed during development, but not production.

Note: The commands that follow assume that you are running on a system running a command processor in the Bourne shell family such as bash or zsh, found in Linux and UNIX operating systems (Ubuntu or macOS, for example). If your system does not include bash by default (such as Windows), you can probably adapt the commands to work, or run them through the msys2 packages or WSL, though hacks will be required for running WSL and docker together.

The docker-compose.yml file does not define a service for the database, but the docker-compose.dev.yml file does.

To make it easier to deal with both of these files at once, please create an alias in your shell:

alias docker-compose-dev='docker-compose -f docker-compose.yml -f docker-compose.dev.yml'

Add this to your $HOME/.profile or $HOME/.bash_profile to make it permanent.

1.1 Clone Repository

Start by cloning this repository on your development workstation.

For example:

git clone https://github.com/freezingsaddles/freezing-compose

Now you can confirm that docker-compose is working correctly by changing to that directory and executing docker-compose commands.

cd freezing-compose
docker-compose ps

You should see lots of warnings about undefined configuration variables. Good! We will get to that next.

1.2 Configure Environment Variables for docker-compose in .env file

Copy the example.env file to a file named .env. This is where docker-compose will look for environment variables.

cp sample-env .env
# edit the environment
vi .env

See sample.env for a complete annotated example of a docker-compose .env file.

These environment variables will be passed in to the various services that need them. Look through them the docker-compose.yml file to see how that works.

For development, you won't need all the values, so don't worry about needing to set up a Strava app or Datadog account before you begin developing.

1.3 Configure and Start MySQL Only

We recommend that for development, you run MySQL through Docker and docker-compose. To start up start up MySQL using docker-compose, follow these steps:

  1. Make sure that you have edited your .env file to have different passwords for MYSQL_ROOT_PASSWORD and MYSQL_PASSWORD
  2. Create the named volume Docker will use for the MySQL database
  3. Start the MySQL container
# Edit your .env file and pick new passwords
vi .env

# If you have not already, you first need to create these named volumes:

docker volume create --name=freezing-data docker volume create --name=beanstalkd-data


# Then you can start MySQL container
docker-compose -f docker-compose.yml -f docker-compose.dev.yml up -d mysql

# MySQL will take up to 60 seconds to become ready, you can monitor 
docker logs freezing-mysql 2>&1 | tail

The output from the docker logs command will look something like this once the server is up and running:

2019-12-01 04:04:23 1 [Note] RSA public key file not found: /var/lib/mysql//public_key.pem. Some authentication plugins will not work.
2019-12-01 04:04:23 1 [Note] Server hostname (bind-address): '*'; port: 3306
2019-12-01 04:04:23 1 [Note] IPv6 is available.
2019-12-01 04:04:23 1 [Note]   - '::' resolves to '::';
2019-12-01 04:04:23 1 [Note] Server socket created on IP: '::'.
2019-12-01 04:04:23 1 [Warning] Insecure configuration for --pid-file: Location '/var/run/mysqld' in the path is accessible to all OS users. Consider choosing a different directory.
2019-12-01 04:04:23 1 [Warning] 'proxies_priv' entry '@ root@freezing-mysql' ignored in --skip-name-resolve mode.
2019-12-01 04:04:23 1 [Note] Event Scheduler: Loaded 0 events
2019-12-01 04:04:23 1 [Note] mysqld: ready for connections.
Version: '5.6.46'  socket: '/var/run/mysqld/mysqld.sock'  port: 3306  MySQL Community Server (GPL)

1.3.1 Connecting to the database

To connect to the database, you can run a local mysql client if you already have one installed.

You can use Docker to connect to the database. The commands required are long and tedious, so create a shell alias for it. Substitute your values for the password and other configurable settings into the aliases below and run these at a shell prompt:

export MYSQL_VERSION=8.0
export MYSQL_HOST=localhost
export MYSQL_PORT=3306
export MYSQL_USER=freezing
export MYSQL_PASSWORD="please-change-me-as-this-is-a-default"
export MYSQL_ROOT_USER=root
export MYSQL_ROOT_PASSWORD="terrible-root-password-which-should-be-changed"
export MYSQL_DATABASE=freezing
alias mysqldump-freezing='docker run -i --rm --network=host mysql:$MYSQL_VERSION  mysqldump --host=$MYSQL_HOST --port=$MYSQL_PORT --user=$MYSQL_ROOT_USER --password="$MYSQL_ROOT_PASSWORD" $MYSQL_DATABASE --default-character-set=utf8mb4'
alias mysql-freezing='docker run -it --rm --volume $HOME/.mysql_history:/root/.mysql_history --network=host mysql:$MYSQL_VERSION  mysql --host=$MYSQL_HOST --port=$MYSQL_PORT --user=$MYSQL_USER --password="$MYSQL_PASSWORD" --database=$MYSQL_DATABASE --default-character-set=utf8mb4'
alias mysql-freezing-non-interactive='docker run -i --rm --volume $HOME/.mysql_history:/root/.mysql_history --network=host mysql:$MYSQL_VERSION mysql --host=$MYSQL_HOST --port=$MYSQL_PORT --user=$MYSQL_USER --password="$MYSQL_PASSWORD" --database=$MYSQL_DATABASE --default-character-set=utf8mb4'
alias mysql-freezing-root='docker run -it --rm --volume $HOME/.mysql_history:/root/.mysql_history --network=host mysql:$MYSQL_VERSION  mysql --host=$MYSQL_HOST --port=$MYSQL_PORT --user=$MYSQL_ROOT_USER --password="$MYSQL_ROOT_PASSWORD" --database=$MYSQL_DATABASE --default-character-set=utf8mb4'
alias mysql-freezing-non-interactive-root='docker run -i --rm --volume $HOME/.mysql_history:/root/.mysql_history --network=host mysql:$MYSQL_VERSION mysql --host=$MYSQL_HOST --port=$MYSQL_PORT --user=$MYSQL_ROOT_USER --password="$MYSQL_ROOT_PASSWORD" --database=$MYSQL_DATABASE --default-character-set=utf8mb4'

You can put these aliases in your shell configuration files such as $HOME/.profile or $HOME/.bashrc files to make them stick.

 $  mysql-freezing
mysql: [Warning] Using a password on the command line interface can be insecure.
Reading table information for completion of table and column names
You can turn off this feature to get a quicker startup with -A

Welcome to the MySQL monitor.  Commands end with ; or \g.
Your MySQL connection id is 12
Server version: 8.0.40 MySQL Community Server - GPL

Copyright (c) 2000, 2023, Oracle and/or its affiliates.

Oracle is a registered trademark of Oracle Corporation and/or its
affiliates. Other names may be trademarks of their respective
owners.

Type 'help;' or '\h' for help. Type '\c' to clear the current input statement.

mysql> status
--------------
mysql  Ver 14.14 Distrib 5.7.44, for Linux (x86_64) using  EditLine wrapper

Connection id:		12
Current database:	freezing
Current user:		[email protected]
SSL:			Cipher in use is ECDHE-RSA-AES128-GCM-SHA256
Current pager:		stdout
Using outfile:		''
Using delimiter:	;
Server version:		8.0.40 MySQL Community Server - GPL
Protocol version:	10
Connection:		127.0.0.1 via TCP/IP
Server characterset:	utf8mb4
Db     characterset:	utf8mb4
Client characterset:	utf8mb4
Conn.  characterset:	utf8mb4
TCP port:		3306
Uptime:			4 min 20 sec

Threads: 4  Questions: 399  Slow queries: 0  Opens: 265  Flush tables: 3  Open tables: 184  Queries per second avg: 1.534
--------------

mysql> show databases;
+--------------------+
| Database           |
+--------------------+
| freezing           |
| information_schema |
| performance_schema |
+--------------------+
3 rows in set (0.00 sec)

mysql> quit
Bye

Due to a quirk of Docker, you will need to use the mysql-freezing-non-interactive and mysql-freezing-root-non-interactive aliases when you redirect the input of MySQL, such as when loading a database dump.

1.4 Optional: Loading a MySQL data dump

This is probably the most convenient way to get started with Freezing Saddles development, but it is optional; if you don't have a dump file, the first time the freezing-web container is run it will populate the schema and baseline tables using the Alembic database migration toolkit.

If you have a MySQL dump (e.g. of the production database), you can load it into a fresh database like this:

$ mysql-freezing-root-non-interactive < freezing-2019-03-20-fixed.sql
$ # Or if you have configured the mysql-freezing-root-non-interactive alias:
$ mysql -h 127.0.0.1 -u root -p freezing < freezing-2019-03-20-fixed.sql
Enter password: <type in the root password you configured above>

Note: if you have trouble with using 127.0.0.1 try localhost. Different combinations of Docker and host operating systems and MySQL may have quirks that make one or the other fail. MySQL may assume that you are using a server that is listening on a local unix socket, which won't work since MySQL in the container only has an exposed listener via TCP/IP.

1.4.1 Optional: Upgrading MySQL

Previously, docker-compose.dev.yml used MySQL 5.6, but we upgraded to MySQL 5.7 in 2022. If you have a volume with a MySQL 5.6 database on your development system, you can upgrade it with these commands:

docker-compose-dev stop mysql
docker-compose-dev up -d mysql
sleep 10 # wait for mysql to come up
docker-compose-dev exec mysql mysql_upgrade -p"$(grep MYSQL_ROOT_PASSWORD .env | cut -d= -f2)"
docker-compose-dev stop mysql
docker-compose-dev up -d mysql

This will complete the upgrade to MySQL 5.7.

1.5 Recreating the MySQL database

If you ever want to destroy and recreate your MySQL database, just remove the container and volume and re "up" it:

docker-compose -f docker-compose.yml -f docker-compose.dev.yml stop mysql
docker-compose -f docker-compose.yml -f docker-compose.dev.yml rm mysql

docker volume rm freezing-data && \
    docker volume create --name=freezing-data && \
    docker-compose -f docker-compose.yml -f docker-compose.dev.yml up -d mysql
# If you have trouble, you might need to do a `docker rm XXXXXX` where XXXXXX is
# the ID of a stopped freezing-mysql container, then repeat the above commands.

1.6 Running freezing-web and Other Containers via docker-compose

Now that MySQL is running, you can also start up the freezing-web container to actually have the website running locally.

(For local development, you don't need to bother with running nginx, letsencrypt, etc, as those expect real domain names and actual account credentials.)

The website expects some minimum configuration to be specified, so you may need to check your .env file:

# Set this to something random. For example, generate a UUID and paste it in (without any quotes)
# python -c 'import uuid; print(uuid.uuid4())'
SECRET_KEY=deadbeef

# For the SA URL, you need to use the user & password you configured for your app db above.
# The host should be 'mysql.container' which will be resolved by docker to the MySQL container's IP address
SQLALCHEMY_URL=mysql+pymysql://freezing:[email protected]/freezing?charset=utf8mb4&binary_prefix=true

# Some of the website reports expect the start date and end date and teams to be configured.

Now you can start up the freezing-web container!

These commands will start it, wait 20 seconds for it to boot, and then tail the logs of the container:

docker-compose -f docker-compose.yml -f docker-compose.dev.yml up -d freezing-web
docker ps

After this starts, docker ps should yield:

$ docker ps
CONTAINER ID        IMAGE                                 COMMAND                  CREATED             STATUS              PORTS                    NAMES
225322a8332d        freezingsaddles/freezing-web:latest   "/bin/sh -c 'gunicor…"   28 minutes ago      Up 28 minutes       0.0.0.0:8000->8000/tcp   freezing-web
a4587dc270d1        mysql:5.6                             "docker-entrypoint.s…"   29 minutes ago      Up 29 minutes       0.0.0.0:3306->3306/tcp   freezing-mysql
6f6e57c475d5        freezing-compose_beanstalkd           "beanstalkd -p 11300…"   4 hours ago         Up 4 hours          11300/tcp                beanstalkd

Then, inspect the logs of the web container:

docker logs freezing-web

If this starts correctly, it will look like:

[2019-12-01 18:56:09 +0000] [6] [INFO] Starting gunicorn 19.7.1
[2019-12-01 18:56:09 +0000] [6] [INFO] Listening at: http://0.0.0.0:8000 (6)
[2019-12-01 18:56:09 +0000] [6] [INFO] Using worker: sync
[2019-12-01 18:56:09 +0000] [9] [INFO] Booting worker with pid: 9
[2019-12-01 18:56:59 +0000] [6] [CRITICAL] WORKER TIMEOUT (pid:9)
[2019-12-01 18:56:59 +0000] [9] [INFO] Worker exiting (pid: 9)
[2019-12-01 18:57:00 +0000] [11] [INFO] Booting worker with pid: 11

If it works, you can visit the local development web site at: http://127.0.0.1:8000/

If it fails, it will have a stack trace that complains about the problem. The first thing to check is the SQLALCHEMY_URL, as it tries to connect to the database early in the boot process.

Note: if you just want to do website development, it is probably easier to just set up a Python 3 virtual environment on your host instead.

See the README for freezing-web for those instructions.

1.7 Using a local container instead of a repository-pulled container

In order to test a locally created container in Docker before pushing it up to the repository, follow this procedure:

# Build the docker image in the freezing-web or other project
cd ../freezing-web
docker build . -t local/freezing-web
cd ../freezing-compose
export FREEZING_WEB_IMAGE=local/freezing-web:latest
docker-compose-dev up -d freezing-web

Here is a terminal session showing the before and after or switching the image to a local one:

[freezing-compose (master)]$ docker-compose-dev up -d
Creating network "freezing-compose_default" with the default driver
Creating freezing-mysql    ... done
Creating beanstalkd        ... done
Creating nginx-letsencrypt ... done
Creating nginx-docker-gen  ... done
Creating logspout          ... done
Creating nginx             ... done
Creating dd-agent          ... done
Creating freezing-sync     ... done
Creating freezing-web      ... done
Creating freezing-nq       ... done
[freezing-compose (master)]$ docker ps
CONTAINER ID        IMAGE                                    COMMAND                  CREATED              STATUS                                 PORTS                                      NAMES
0a222643de63        freezingsaddles/freezing-web:latest      "/bin/sh -c 'gunicor…"   About a minute ago   Up 59 seconds                          0.0.0.0:8000->8000/tcp                     freezing-web
e1dda065040b        freezingsaddles/freezing-nq:latest       "/bin/sh -c 'gunicor…"   About a minute ago   Up 59 seconds                          8000/tcp                                   freezing-sync
d80a268e405b        freezingsaddles/freezing-nq:latest       "/bin/sh -c 'gunicor…"   About a minute ago   Up About a minute                      8000/tcp                                   freezing-nq
23cb4024f0d2        nginx:stable                             "nginx -g 'daemon of…"   About a minute ago   Up About a minute                      0.0.0.0:80->80/tcp, 0.0.0.0:443->443/tcp   nginx
1fe08132af7f        freezing-compose_beanstalkd              "beanstalkd -p 11300…"   About a minute ago   Up About a minute                      11300/tcp                                  beanstalkd
13e6427fb5c4        gliderlabs/logspout                      "/bin/logspout syslo…"   About a minute ago   Restarting (1) 13 seconds ago                                                     logspout
d6b5cdc8d640        datadog/docker-dd-agent:latest           "/entrypoint.sh supe…"   About a minute ago   Up About a minute (health: starting)   8125/udp, 8126/tcp                         dd-agent
a3a1b3b3e9cd        jwilder/docker-gen                       "/usr/local/bin/dock…"   About a minute ago   Up About a minute                                                                 nginx-docker-gen
84fc1f85a1ee        jrcs/letsencrypt-nginx-proxy-companion   "/bin/bash /app/entr…"   About a minute ago   Up 59 seconds                                                                     nginx-letsencrypt
39f7700118fb        mysql:5.6                                "docker-entrypoint.s…"   About a minute ago   Up About a minute                      0.0.0.0:3306->3306/tcp                     freezing-mysql
[freezing-compose (master)]$ export FREEZING_WEB_IMAGE=local/freezing-web:latest
[freezing-compose (master)]$ docker-compose-dev up -d
nginx-letsencrypt is up-to-date
logspout is up-to-date
nginx is up-to-date
nginx-docker-gen is up-to-date
freezing-mysql is up-to-date
beanstalkd is up-to-date
dd-agent is up-to-date
Recreating freezing-web ...
freezing-nq is up-to-date
Recreating freezing-web ... done
[freezing-compose (master)]$ docker ps | grep freezing-web
eb2c13a109ba        local/freezing-web:latest                "/bin/sh -c 'gunicor…"   15 seconds ago      Up 13 seconds                     0.0.0.0:8000->8000/tcp                     freezing-web
[freezing-compose (master)]$

By using these techniques you can test locally in a fashion that is very similar to production.

2. Deploy to Production

This repository can be used to deploy the Freezing Saddles application for production. Before you get started, we assume you have already installed Docker and Docker Compose.

The current production setup assumes that you will use an external MySQL server, but future work is contemplated to support running MySQL as a Docker container in production.

2.1 Clone Repository

Clone the repository this repository onto a server that runs Docker and docker-compose. freezingsaddles.org ran on CoreOS from 2018 to 2020, and today runs on Rocky Linux 9. Ths should works but any modern operating system distribution that has Docker support will probably work. That includes RHEL 7+ and open source derivatives such as Centos, Rocky Linux and Alma Linux, all Ubuntu LTS versions from 16.04 on, and so on.

For example:

git clone https://github.com/freezingsaddles/freezing-compose
ln -s "$PWD/freezing-compose" /opt/compose

2.2 Create Persistent Docker Volumes

Create the persistent volumes needed for production:

docker volume create --name=freezing-data
docker volume create --name=beanstalkd-data
docker volume create --name=wordpress-data

2.3 Configure MySQL production server

Note: The production configuration assumes you will run a MySQL server outside of the environment managed by docker-compose. Please configure an external MySQL server, for example an AWS RDS MySQL server, in the .env file for production use.

Connect to your production database as the root user and issue these commands to create the database, putting in a real password instead of the one below:

drop database if exists freezing;
create database freezing character set utf8mb4;
create user freezing identified by 'please-change-me-as-this-is-a-default';
grant all on freezing.* to freezing;

Adapt the shell aliases in section 1.3 above so that they connect to the production database from your production host.

If you set up the shell aliases suggested above (test them by issuing a mysql command), you should be able to load a snapshot of the production database with:

mysql-non-interactive < freezing-2019-12-31.sql

2.4 Configure Environment

# Create and edit the .env file, filling it in completely with real values
cd /opt/compose
cp example.env .env
vi .env

# Verify Docker compose is working
docker-compose ps

2.5 Run Containers

cd /opt/compose
docker-compose up -d
# wait a minute, then verify that the containers are working
sleep 60
docker-compose ps

If any containers are not started, troubleshoot with docker ps and docker logs container-name. You might need to tweak the configuration multiple times before all containers come up cleanly. Restart the containers after tweaking the .env file each time with docker-compose up -d until things work.

Legal

Legal

This software is a community-driven effort, and as such the contributions are owned by the individual contributors:

Copyright 2018 Hans Lillelid
Copyright 2018 Richard Bullington-McGuire
Copyright 2023 Adam Sloan

This software is licensed under the Apache 2.0 license, with some marked portions available under compatible licenses (such as the MIT-licensed bin/provision-server.sh and bin/recover-container-name-conflict.sh.)

About

The Docker Compose structure for running the application suite.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Contributors 3

  •  
  •  
  •  

Languages