Docker 用于 Windows:SQL 服务器数据库位于具有 Windows-容器的持久卷上

Docker for Windows: SQL Server Database on persistent Volume with Windows-Container

在 Docker 上 Windows 并使用 windows 容器,我无法让我的持久卷在主数据库目录上工作windows 容器的。这将是 C:\Program Files\Microsoft SQL Server\MSSQL14.MSSQLSERVER\MSSQL\DATA 我如何才能获得数据库持久卷的好处,而不必将备份和恢复弄乱到 mssql-server-container 中?

这可能是因为数据目录在我尝试挂载持久卷的文件夹中存储了主数据库和系统数据库。

在 SQL 服务器中 linux 容器这很简单,您可以将持久卷连接到 /var/opt/mssql 并让您的数据库持久化。

我知道我可以将数据库从备份恢复到容器中,但这有两个主要缺点:我必须有一个大容器,因为我正在使用一个大数据库。所以我将容器的 20 GB 限制扩展到 60 GB 但是......每次从备份重建数据库非常耗时。 第二个缺点是,如果 mssql-dev 容器被终止,数据库也会丢失。在此数据库上的任何工作都将消失。如果数据库可以驻留在持久卷上,这将有所不同。

docker run -d -e sa_password=<Password> -e ACCEPT_EULA=Y -v "C:\mylocalfolder:C:\Program Files\Microsoft SQL Server\MSSQL14.MSSQLSERVER\MSSQL\DATA" microsoft/mssql-server-windows-developer

错误消息是“Windows 系统调用失败:虚拟计算机或容器意外关闭。 (0xc0370106)

解决方法 1

将持久卷连接到另一个位置,例如 c:\mydata 以防止上面的错误消息。 然后在不使用标准数据库文件夹的情况下将数据库连接到服务器。

提取数据库.bak文件,所以有mdf和log文件

--Get the name of your DB
RESTORE FILELISTONLY
FROM DISK = 'c:\mydata'
GO
--do the extraction of the bak file to certain folder
RESTORE DATABASE mydatabase
FROM DISK = 'c:\mydata'
WITH REPLACE,
MOVE 'mydatabase' TO 'c:\mydata\extractedDb.mdf',
MOVE 'mydatabase_log' TO 'c:\mydata\extractedLog.ldf'
GO

完成此操作后,您现在应该在持久卷上准备好数据库文件。现在将数据库附加到服务器。这必须通过创建一个新的数据库来完成但是这个过程只需要几纳秒就可以完成!

CREATE DATABASE WhosebugIsGreat
ON (FILENAME = 'c:\mydata\extractedDb.mdf'),
(FILENAME = 'c:\mydata\extractedLog.ldf')
FOR ATTACH;

现在数据库在持久卷中是安全的。如果 db-server 容器出现故障或正在重建,您只需再次 运行 最后一条语句(或者甚至更好地在您的 docker-compose 或 docker 文件中实现它):

CREATE DATABASE WhosebugIsGreat
ON (FILENAME = 'c:\mydata\extractedDb.mdf'),
(FILENAME = 'c:\mydata\extractedLog.ldf')
FOR ATTACH;

解决方法 2

-attach_dbs 参数似乎以同样的方式工作。 Docker 运行:

docker run -p 1433:1433 --name mssql-dev -e sa_password=<yourpassword> -e ACCEPT_EULA=Y -e attach_dbs="[{'dbName':'PowerSlide_SQLDB','dbFiles':['C:\your\path\database.mdf','C:\sqldata\databaselog.ldf']}]" -v "d:\sqldata:C:\sqldata" microsoft/mssql-server-windows-developer

或者如果您更喜欢 Docker-Compose,这有点棘手。我不得不省略括号外的前导和结束 ' 并将括号内的双引号替换为 ' 以使其工作。

version: '3.2'
services:
  mssql-dev:
    container_name: mssql-dev
    image: 'microsoft/mssql-server-windows-developer'
    volumes:
      - "d:\sqldata:C:\sqldata"
    ports:
      - "1433:1433"
    restart: always
    environment:
      - "ACCEPT_EULA=Y"
      - "sa_password=yourpassword"
      - attach_dbs=[{"dbName":"<yourDbName>","dbFiles":["C:\<your>\path\database.mdf","C:\your\path\databaselog.ldf"]}]
volumes:
  mssql-dev-data:

看来这个问题可以用上面的解决方法 1 和 2 来回答。

  1. 将持久卷连接到另一个位置,如 c:\mydata 以防止上面的错误消息。然后在不使用标准数据库文件夹的情况下将数据库连接到服务器。
  2. 提取数据库.bak文件,所以有mdf和log文件
--Get the name of your DB
RESTORE FILELISTONLY
FROM DISK = 'c:\mydata'
GO
--do the extraction of the bak file to certain folder
RESTORE DATABASE mydatabase
FROM DISK = 'c:\mydata'
WITH REPLACE,
MOVE 'mydatabase' TO 'c:\mydata\extractedDb.mdf',
MOVE 'mydatabase_log' TO 'c:\mydata\extractedLog.ldf'
GO
  1. 通过以下三种方式之一将数据库附加到服务器:

Docker 运行 示例:

docker run -p 1433:1433 --name mssql-dev -e sa_password=<yourpassword> -e ACCEPT_EULA=Y -e attach_dbs="[{'dbName':'PowerSlide_SQLDB','dbFiles':['C:\your\path\database.mdf','C:\sqldata\databaselog.ldf']}]" -v "d:\sqldata:C:\sqldata" microsoft/mssql-server-windows-developer

如果您更喜欢 Docker-Compose,这有点棘手。我不得不省略括号外的前导和结束 ' 并将括号内的双引号替换为 ' 以使其工作。 docker-撰写示例:

version: '3.2'
services:
  mssql-dev:
    container_name: mssql-dev
    image: 'microsoft/mssql-server-windows-developer'
    volumes:
      - "d:\sqldata:C:\sqldata"
    ports:
      - "1433:1433"
    restart: always
    environment:
      - "ACCEPT_EULA=Y"
      - "sa_password=yourpassword"
      - attach_dbs=[{"dbName":"<yourDbName>","dbFiles":["C:\<your>\path\database.mdf","C:\your\path\databaselog.ldf"]}]
volumes:
  mssql-dev-data:

或使用 SQL 命令附加数据库

CREATE DATABASE WhosebugIsGreat
ON (FILENAME = 'c:\mydata\extractedDb.mdf'),
(FILENAME = 'c:\mydata\extractedLog.ldf')
FOR ATTACH;