Dockerfile无法执行CMD命令(适用于Windows的Docker)

时间:2018-10-09 13:57:51

标签: docker containers dockerfile docker-container docker-windows

对于dockerfile中的Windows,无法在docker上执行CMD命令。

我想使用CMD运行几个命令,但是甚至不能运行单个命令。

我尝试了以下操作,但没有任何效果...

1。

CMD "sqlcmd -S database -i C:\db\db_scripts\upgradescript.sql -U sa -P mypassword"

2。

RUN ["powershell", "-NoProfile", "-Command", "c:\db\db_scripts\script.ps1"]

3。

CMD ["cmd", "sqlcmd", "-S", "database", "-i", "C:\db\db_scripts\upgradescript.sql", "-U", "sa", "-P", "mypassword"]

4。

ARG arg1=database
ARG arg2=C:\db\db_scripts\upgradescript.sql
ARG arg3=sa
ARG arg4=mypassword!

RUN ["cmd", "-NoProfile", "-Command", "sqlcmd -S $env:arg1 -i $env:arg2 -U $env:arg3 -P $env:arg4"]

5。

RUN powershell.exe  c:\db\db_scripts\script.ps1

6。

ENTRYPOINT ["powershell", "-NoProfile", "-Command", "sqlcmd"]
CMD ["-S database -i C:\db\db_scripts\upgradescript.sql -U sa -P mypassword"]

支持文件:

A。 Script.ps1(供2和5使用)

cmd /c "sqlcmd -S database -i C:\db\db_scripts\upgradescript.sql -U sa -P mypassword"

B。 Dockerfile

# escape=`

FROM microsoft/mssql-server-windows-developer
COPY db\* c:\db\
COPY db_scripts\* c:\db\db_scripts\
ENV attach_dbs="[{'dbName':'ABC','dbFiles':['C:\db\ABC.mdf','C:\db\ABC_Log.ldf']},{'dbName':'XYZ','dbFiles':['C:\db\XYZ.mdf','C:\db\XYZ_Log.ldf']}]"
CMD "sqlcmd -S database -i C:\db\db_scripts\upgradescript.sql -U sa -P mypassword"

请注意,由于转义字符,我不需要使用“ \\”。

观察

容器启动并在几秒钟后消失。

如果我卸下CMD部件,则容器工作正常。我可以进入容器并可以在sqlcmd shell中运行上面的cmd命令。

我在做什么错,缺少什么,有什么更好的方法等。有人可以给我一些指导吗?谢谢!

编辑:

根据Josh的回答,此ENTRYPOINT命令对我有用的唯一方法是(共享以便其他人可以受益,甚至发布更好的方法)...

ENV arg1 database
ENV arg2 C:\db\db_scripts\upgradescript.sql
ENV arg3 sa
ENV arg4 mypassword
ENTRYPOINT ["powershell sleep(60); sqlcmd"]
CMD ["-S $env:arg1 -i $env:arg2 -U $env:arg3 -P $env:arg4"]

1 个答案:

答案 0 :(得分:4)

我认为您可能误解了CMD,ENTRYPOINT和RUN指令的性质。 CMD和ENTRYPOINT都在容器启动/运行时生效,而RUN是构建时命令。有点令人困惑的是,您在docker构建过程中使用RUN来准备容器映像,而CMD是在容器中运行的。

通过documentation

  

CMD的主要目的是为执行中的容器提供默认值...

任何给定的容器映像将永远只有一条CMD指令。

  

在Dockerfile中只能有一条CMD指令。如果您列出多个CMD,则只有最后一个CMD才会生效。

如果要确保可执行文件在容器启动时运行,则可以考虑使用ENTRYPOINT指令。将ENTRYPOINT视为提供CMD指令默认值的命令。从文档中:

  

Dockerfile应该指定CMD或ENTRYPOINT命令中的至少一个。   使用容器作为可执行文件时,应定义ENTRYPOINT。   应将CMD用作为ENTRYPOINT命令定义默认参数或在容器中执行自定义命令的方式。   使用替代参数运行容器时,CMD将被覆盖。

RUN命令是一条构建时指令,可用于准备容器映像以及必要的文件,配置等。

有关ENTRYPOINT与CMD的更多信息,请参见this SO question

ED:您正在研究在容器中运行SQL脚本的情况是一个棘手的问题,因为在数据库引擎准备好接受命令之前不得运行该脚本。第一步,您可以在ENTRYPOINT指令中运行升级脚本。您也可以考虑通过docker exec命令运行SQL升级脚本容器后置备。

ED2:提供的示例:

ENTRYPOINT ["powershell", "-NoProfile", "-Command", "sqlcmd"] CMD ["sqlcmd -S database -i C:\db\db_scripts\upgradescript.sql -U sa -P mypassword"]

不起作用,因为它可能导致无意义的命令行指令。请记住,CMD为ENTRYPOINT指令提供了参数,因此您只需在ENTRYPOINT中指定Powershell sqlcmd,而让CMD提供参数(未经测试,OTOMH):

ENTRYPOINT powershell sqlcmd -S database -U sa -P mypassword -i

CMD C:\db\db_scripts\upgradescript.sql

ED 3: 通过组合两个语句并向ENTRYPOINT添加语句终止符(;),您可以允许运行标准SQL容器。\ Start.ps1脚本,该脚本进入无限循环,以防止容器立即停止(容器仅在它们中已执行的进程正在运行时才运行)。即使用户在运行时覆盖CMD指令,这也可以确保启动脚本得以执行:

ENTRYPOINT powershell "sqlcmd -S database -U sa -P mypassword -i 'C:\db\db_scripts\upgradescript.sql';"

CMD .\Start.ps1