Otá*_*udo 7 bash shell stdin zsh docker
我有一个 Docker 容器,需要传递一个包含当前目录中文件名的字符串(字符串数组也可以,但据我所知这是不可能的)。
我获取容器内文件名的第一个策略是运行命令docker build -t doc-validate . && docker run doc-validate (printf "%s " $(ls -p | grep -v /)),将输出(printf "%s " $(ls -p | grep -v /))直接发送到容器的标准输入中,但是,我zsh: bad pattern: (printf %s Dockerfile作为错误返回;在我看来,容器或 shell 试图在某处执行“Dockerfile”,我不知道为什么会发生这种情况,因为printf直接在终端中运行相同的命令按预期工作(仅打印当前目录中的文件名) .
我的第二种方法是尝试将这些文件名作为环境变量发送到容器。运行PROJECT_FILES=(printf "%s " $(ls -p | grep -v /))在预期的终端工作,$PROJECT_FILES输出这些文件名。但是,如果我尝试像这样将其直接传递到容器中:docker build -t doc-validate . && docker run --env PROJECT_FILES=(printf "%s " $(ls -p | grep -v /)) doc-validate我得到的结果与zsh: bad pattern: PROJECT_FILES=(printf %s Dockerfile错误相同。
在我看来,由于 'zsh:' 在错误输出中,命令甚至在进入容器之前就失败了,但是,我不明白为什么在终端中独立运行它们是可行的,以及为什么将它们作为参数传递给容器真是头疼。
如何将(printf "%s " $(ls -p | grep -v /))在本地机器上运行的输出作为容器内可访问的值?
Dockerfile:
FROM node:16
WORKDIR /moduleRoot
COPY package.json /moduleRoot
RUN npm install
COPY . /moduleRoot/
# Authorize execution of entrypoint bash script
RUN chmod +x /moduleRoot/docker-container-entrypoint.sh
# these values below will be set later, just register them explicitly here
ENV PROJECT_FILES=""
CMD [ "/moduleRoot/docker-container-entrypoint.sh" ]
Run Code Online (Sandbox Code Playgroud)
Bash 入口点(CMD 中的 docker-container-entrypoint.sh 文件):
#!/bin/bash
echo "Args are: $@"
echo "PROJECT_FILES=$PROJECT_FILES"
Run Code Online (Sandbox Code Playgroud)
首先,您正在使用node:16并根据项目的入口点node,它将调用来执行您作为参数传递的任何内容。
在您上,Dockerfile您正在调用 aCMD并且没有更改ENTRYPOINT,因此当您执行容器时,它将尝试使用节点执行 bash 脚本,并且您一定会收到如下错误:
/moduleRoot/docker-container-entrypoint.sh:2\necho "Args are: $@"\n ^^^^^^^^^^^^^^\n\nSyntaxError: Unexpected string\n at Object.compileFunction (node:vm:353:18)\n at wrapSafe (node:internal/modules/cjs/loader:1039:15)\n at Module._compile (node:internal/modules/cjs/loader:1073:27)\n at Object.Module._extensions..js (node:internal/modules/cjs/loader:1138:10)\n at Module.load (node:internal/modules/cjs/loader:989:32)\n at Function.Module._load (node:internal/modules/cjs/loader:829:14)\n at Function.executeUserEntryPoint [as runMain] (node:internal/modules/run_main:76:12)\n at node:internal/main/run_main_module:17:47\nRun Code Online (Sandbox Code Playgroud)\n如果您的 bash 脚本只是为了帮助您排除故障,那么您可以使用在线heredoc传递您需要的参数:
\xc2\xbb docker run -t --rm doc-validate <<< echo $(ls -a1)\n\nArgs are: docker-container-entrypoint.sh Dockerfile package.json\nPROJECT_FILES=\nRun Code Online (Sandbox Code Playgroud)\n另一方面,如果您需要 bash 脚本,那么您必须CMD将ENTRYPOINT. 另一件值得注意的事情是 Linuxls没有该-p标志。