如果我用下面的.yml运行$ docker-compose down,我似乎不能让MySQL数据持久

version: '2'
services:
  # other services

  data:
    container_name: flask_data
    image: mysql:latest
    volumes:
      - /var/lib/mysql
    command: "true"

  mysql:
    container_name: flask_mysql
    restart: always
    image: mysql:latest
    environment:
      MYSQL_ROOT_PASSWORD: 'test_pass' # TODO: Change this
      MYSQL_USER: 'test'
      MYSQL_PASS: 'pass'
    volumes_from:
      - data
    ports:
      - "3306:3306"

我的理解是,在我的数据容器中使用卷:- /var/lib/mysql将它映射到我的本地机器目录,mysql将数据存储到容器中,由于这种映射,即使容器被破坏,数据也应该持久化。mysql容器只是一个进入db的客户端接口,可以看到本地目录,因为volumes_from: - data

试着回答这个问题,但没有成功。Docker-Compose持久数据故障

EDIT

更改了我的.yml如下所示,并创建了一个dir ./data,但现在当我运行docker-compose up -build mysql容器不会开始抛出错误说

  data:
    container_name: flask_data
    image: mysql:latest
    volumes:
      - ./data:/var/lib/mysql
    command: "true"

  mysql:
    container_name: flask_mysql
    restart: always
    image: mysql:latest
    environment:
      MYSQL_ROOT_PASSWORD: 'test_pass' # TODO: Change this
      MYSQL_USER: 'test'
      MYSQL_PASS: 'pass'
    volumes_from:
      - data
    ports:
      - "3306:3306"


flask_mysql | mysqld: Can't create/write to file '/var/lib/mysql/is_writable' (Errcode: 13 - Permission denied)
flask_mysql | 2016-08-26T22:29:21.182144Z 0 [Warning] TIMESTAMP with implicit DEFAULT value is deprecated. Please use --explicit_defaults_for_timestamp server option (see documentation for more details).
flask_mysql | 2016-08-26T22:29:21.185392Z 0 [ERROR] --initialize specified but the data directory exists and is not writable. Aborting.

您必须为mysql数据创建一个单独的卷。

所以它看起来是这样的:

volumes_from:
  - data
volumes:
  - ./mysql-data:/var/lib/mysql

不,/var/lib/mysql是mysql容器内的路径,与主机上的路径无关。你的主机甚至可能根本没有mysql。因此,目标是持久化一个来自mysql容器的内部文件夹。


有3种方法:

第一个方法

您需要指定在主机上存储mysql数据的目录。然后可以删除数据容器。您的mysql数据将保存在本地文件系统。

Mysql容器定义必须是这样的:

mysql:
  container_name: flask_mysql
  restart: always
  image: mysql:latest
  environment:
    MYSQL_ROOT_PASSWORD: 'test_pass' # TODO: Change this
    MYSQL_USER: 'test'
    MYSQL_PASS: 'pass'
volumes:
 - /opt/mysql_data:/var/lib/mysql
ports:
  - "3306:3306"

第二种方式

在输入docker-compose down之前提交数据容器:

docker commit my_data_container
docker-compose down

第三条道路

你也可以使用docker-compose stop而不是docker-compose down(这样你就不需要提交容器了)


数据容器是一个多余的解决方案。数据卷可以帮到你。改变你的船坞结构。yml:

version: '2'
services:
  mysql:
    container_name: flask_mysql
    restart: always
    image: mysql:latest
    environment:
      MYSQL_ROOT_PASSWORD: 'test_pass' # TODO: Change this
      MYSQL_USER: 'test'
      MYSQL_PASS: 'pass'
    volumes:
      - my-datavolume:/var/lib/mysql
volumes:
  my-datavolume:

Docker将在/var/lib/docker/volumes文件夹中为您创建卷。只要你不输入docker-compose down -v,这个卷就会一直存在


实际上这就是路径你应该提到一个有效的路径。如果你的数据目录在current directory,那么你应该提到。/my-data而不是my-data,否则在mysql和mariadb中也会给你这个错误。

volumes:
 ./my-data:/var/lib/mysql

在@Ohmen的回答之外,你还可以添加一个外部标志来创建docker compose之外的数据卷。这样,docker compose就不会尝试创建它。此外,你也不必担心在$ docker-compose down -v事件中丢失数据卷中的数据。 下面的例子来自官方页面。

version: "3.8"

services:
  db:
    image: postgres
    volumes:
      - data:/var/lib/postgresql/data

volumes:
  data:
    external: true

首先,你需要删除所有旧的mysql数据使用

docker-compose down -v

然后在docker-compose.yml中添加两行

volumes:
  - mysql-data:/var/lib/mysql

and

volumes:
  mysql-data:

你最后的码头写手。Yml将看起来像

version: '3.1'

services:
  php:
    build:
      context: .
      dockerfile: Dockerfile
    ports:
      - 80:80
    volumes:
      - ./src:/var/www/html/
  db:
    image: mysql
    command: --default-authentication-plugin=mysql_native_password
    restart: always
    environment:
      MYSQL_ROOT_PASSWORD: example
    volumes:
      - mysql-data:/var/lib/mysql

  adminer:
    image: adminer
    restart: always
    ports:
      - 8080:8080
volumes:
  mysql-data:

之后使用此命令

docker-compose up -d

现在您的数据将被持久化,即使在使用此命令后也不会被删除

docker-compose down

额外:-但如果你想删除所有的数据,那么你将使用

docker-compose down -v

可行的绑定挂载方案:

  mariadb:
    image: mariadb:latest
    restart: unless-stopped
    environment:
      - MARIADB_ROOT_PASSWORD=${MARIADB_ROOT_PASSWORD}
    volumes:
      - type: bind
        source: /host/dir
        target: /var/lib/mysql