From 74aa80e2b362ec0bc11251c17d4c8859443dc3e5 Mon Sep 17 00:00:00 2001 From: "g.nardiello" Date: Thu, 5 Jan 2023 11:59:16 +0100 Subject: [PATCH] Added EXIT_BACKUP feature with gracefully shutdown --- README.md | 64 ++++++++++++++++++++++++++++++++++++++++++++++++++++++- run.sh | 21 +++++++++++++++++- 2 files changed, 83 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index 923ec5e..1786703 100644 --- a/README.md +++ b/README.md @@ -31,6 +31,7 @@ docker container run -d \ - `MAX_BACKUPS`: The number of backups to keep. When reaching the limit, the old backup will be discarded. No limit by default. - `INIT_BACKUP`: If set, create a backup when the container starts. - `INIT_RESTORE_LATEST`: If set, restores latest backup. +- `EXIT_BACKUP`: If set, create a backup when the container stops. - `TIMEOUT`: Wait a given number of seconds for the database to be ready and make the first backup, `10s` by default. After that time, the initial attempt for backup gives up and only the Cron job will try to make a backup. - `GZIP_LEVEL`: Specify the level of gzip compression from 1 (quickest, least compressed) to 9 (slowest, most compressed), default is 6. - `USE_PLAIN_SQL`: If set, back up and restore plain SQL files without gzip. @@ -184,4 +185,65 @@ mysql-cron-backup: docker container exec /restore.sh /backup/ ``` -if no database name is specified, `restore.sh` will try to find the database name from the backup file. \ No newline at end of file +if no database name is specified, `restore.sh` will try to find the database name from the backup file. + +### Automatic backup and restore on container starts and stops + +Set `INIT_RESTORE_LATEST` to automatic restore the last backup on startup. +Set `EXIT_BACKUP` to automatic create a last backup on shutdown. + +```yaml + mysql-cron-backup: + image: fradelg/mysql-cron-backup + depends_on: + - mariadb + volumes: + - ${VOLUME_PATH}/backup:/backup + environment: + - MYSQL_HOST=my_mariadb + - MYSQL_USER=${MYSQL_USER} + - MYSQL_PASS=${MYSQL_PASSWORD} + - MAX_BACKUPS=15 + - INIT_RESTORE_LATEST=1 + - EXIT_BACKUP=1 + # Every day at 03:00 + - CRON_TIME=0 3 * * * + # Make it small + - GZIP_LEVEL=9 + restart: unless-stopped + +volumes: + data: +``` + +Docker database image could expose a directory you could add files as init sql script. + +```yaml + mysql: + image: mysql + expose: + - 3306 + volumes: + - data:/var/lib/mysql + # If there is not scheme, restore using the init script (if exists) + - ./init-script.sql:/docker-entrypoint-initdb.d/database.sql.gz + environment: + - MYSQL_ROOT_PASSWORD=${MYSQL_ROOT_PASSWORD} + - MYSQL_DATABASE=${DATABASE_NAME} + restart: unless-stopped +``` + +```yaml + mariadb: + image: mariadb + expose: + - 3306 + volumes: + - data:/var/lib/mysql + # If there is not scheme, restore using the init script (if exists) + - ./init-script.sql:/docker-entrypoint-initdb.d/database.sql.gz + environment: + - MYSQL_ROOT_PASSWORD=${MARIADB_ROOT_PASSWORD} + - MYSQL_DATABASE=${DATABASE_NAME} + restart: unless-stopped +``` \ No newline at end of file diff --git a/run.sh b/run.sh index 95f8009..ea11ae7 100755 --- a/run.sh +++ b/run.sh @@ -14,7 +14,26 @@ elif [ -n "${INIT_RESTORE_LATEST}" ]; then find /backup -maxdepth 1 -name '*.sql.gz' | tail -1 | xargs /restore.sh fi +function final_backup { + echo "=> Captured trap for final backup" + DATE=$(date +%Y%m%d%H%M) + echo "=> Requested last backup at $(date "+%Y-%m-%d %H:%M:%S")" + exec /backup.sh + exit 0 +} + +if [ -n "${EXIT_BACKUP}" ]; then + echo "=> Listening on container shutdown gracefully to make last backup before close" + trap final_backup SIGHUP SIGINT SIGTERM +fi + echo "${CRON_TIME} /backup.sh >> /mysql_backup.log 2>&1" > /tmp/crontab.conf crontab /tmp/crontab.conf echo "=> Running cron task manager in foreground" -exec crond -f -l 8 -L /mysql_backup.log +crond -f -l 8 -L /mysql_backup.log & + +echo "Listening on crond, and wait..." + +while : ; do sleep 1 ; done + +echo "Script is shutted down." \ No newline at end of file