gpt4 book ai didi

docker - 我应该如何为开发数据库构建 docker 镜像?

转载 作者:行者123 更新时间:2023-12-05 04:43:50 28 4
gpt4 key购买 nike

我有一个开发数据库,​​我想将其提交到 docker 镜像中,然后推送到私有(private)存储库并用于本地开发和 CI 构建。

数据库保存为SQL备份,我可以通过将备份文件映射到官方镜像的/docker-entrypoint-initdb.d/目录下成功恢复到MariaDB容器中,将在第一次运行容器时执行此备份文件。发生这种情况时,MariaDB 会将数据库恢复到位于容器内 /var/lib/mysql 中的数据文件中。

理论上,我可以停止容器,将其提交到一个新图像,推送它,然后我就完成了。然而,MariaDB 的 Dockerfile 声明了 VOLUME/var/lib/mysql,这意味着一旦创建容器,此路径将映射到主机上的自动命名卷,即使 --volume 不用于显式挂载到该路径。由于该路径是从宿主机挂载的,恢复后的数据库不会被docker commit捕获。似乎无法覆盖此自动卷安装。

一个解决方法是 docker create 一个新容器而不运行它,docker cp 我的数据库备份到 /docker-entrypoint-initdb.d/,然后 docker commit 将其提交到新图像。这个镜像可以推送,因为它包含备份数据,当它第一次运行时,MariaDB 会把它恢复到数据库中。但是,无论何时创建容器,您都必须等待恢复完成,而不是与已经恢复的数据库一起运送图像。

另一种解决方法是 fork MariaDB Dockerfile 以删除 VOLUME 指令,或者通过在基础镜像上自行安装 MariaDB 来创建我自己的。

第三种选择可能是创建我自己的 Dockerfile 以从 mariadb 作为基础镜像构建,并尝试将恢复作为该构建过程的一部分,但我还没有完全调查这个。

实现此目标最有效的方法是什么?

最佳答案

我找到了以下方法:

体积定义存储在图像元数据中,因此可以手动更改此元数据以删除体积声明并将结果保存到新图像中。此过程已封装在一个名为 docker-copyedit 的有用 Python 包中。 .

$ docker pull mariadb
$ pip install docker-copyedit
$ docker-copyedit.py FROM mariadb INTO mariadb:novolumes REMOVE ALL VOLUMES

现在我有了 MariaDB 的本地 :novolumes 图像,我可以在 Dockerfile 构建中自动化数据库。为了使用图像的内置功能来创建新数据库并从 /docker-entrypoint-initdb.d/ 目录中恢复,我创建了一个 bash 脚本,它提供了 docker-entrypoint .sh 然后执行 relevant portions of _main()执行恢复然后退出而不直接执行 _main()(这将导致数据库重新启动,因此构建永远不会退出)。然后我使用第二个构建阶段从恢复阶段仅导入 /var/lib/mysql 目录。

docker 文件
FROM mariadb:novolumes AS restorer

ENV MARIADB_ALLOW_EMPTY_ROOT_PASSWORD=yes

COPY data/source/database.sql /docker-entrypoint-initdb.d/

COPY restore_db.sh restore_db.sh
RUN ["./restore_db.sh", "mysqld"]

FROM mariadb:novolumes

COPY --from=restorer /var/lib/mysql /var/lib/mysql
restore_db.sh
#!/bin/bash

# Import helper functions
source docker-entrypoint.sh

# Direct copy-paste of lines 356-392 from docker-entrypoint.sh:

mysql_note "Entrypoint script for MariaDB Server ${MARIADB_VERSION} started."

mysql_check_config "$@"
# Load various environment variables
docker_setup_env "$@"
docker_create_db_directories

# If container is started as root user, restart as dedicated mysql user
if [ "$(id -u)" = "0" ]; then
mysql_note "Switching to dedicated user 'mysql'"
exec gosu mysql "$BASH_SOURCE" "$@"
fi

# there's no database, so it needs to be initialized
if [ -z "$DATABASE_ALREADY_EXISTS" ]; then
docker_verify_minimum_env

# check dir permissions to reduce likelihood of half-initialized database
ls /docker-entrypoint-initdb.d/ > /dev/null

docker_init_database_dir "$@"

mysql_note "Starting temporary server"
docker_temp_server_start "$@"
mysql_note "Temporary server started."

docker_setup_db
docker_process_init_files /docker-entrypoint-initdb.d/*

mysql_note "Stopping temporary server"
docker_temp_server_stop
mysql_note "Temporary server stopped"

echo
mysql_note "MariaDB init process done. Ready for start up."
echo
fi

最后,我可以构建并运行容器以快速创建本地开发数据库,​​和/或将镜像推送到存储库。

$ docker build -t NReilingh/devdb .
$ docker run --rm -d NReilingh/devdb

关于docker - 我应该如何为开发数据库构建 docker 镜像?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/69499298/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com