在具有'source'的Dockerfile中使用RUN指令不起作用

我有一个Dockerfile,用于安装一个vanilla python环境(我将在其中安装一个应用程序,但在以后的日期)。

FROM ubuntu:12.04

# required to build certain python libraries
RUN apt-get install python-dev -y

# install pip - canonical installation instructions from pip-installer.org
# http://www.pip-installer.org/en/latest/installing.html
ADD https://bitbucket.org/pypa/setuptools/raw/bootstrap/ez_setup.py /tmp/ez_setup.py
ADD https://raw.github.com/pypa/pip/master/contrib/get-pip.py /tmp/get-pip.py
RUN python /tmp/ez_setup.py
RUN python /tmp/get-pip.py
RUN pip install --upgrade pip 

# install and configure virtualenv
RUN pip install virtualenv 
RUN pip install virtualenvwrapper
ENV WORKON_HOME ~/.virtualenvs
RUN mkdir -p $WORKON_HOME
RUN source /usr/local/bin/virtualenvwrapper.sh

生成运行正常,直到最后一行,我得到以下异常:

[previous steps 1-9 removed for clarity]
...
Successfully installed virtualenvwrapper virtualenv-clone stevedore
Cleaning up...
 ---> 1fc253a8f860
Step 10 : ENV WORKON_HOME ~/.virtualenvs
 ---> Running in 8b0145d2c80d
 ---> 0f91a5d96013
Step 11 : RUN mkdir -p $WORKON_HOME
 ---> Running in 9d2552712ddf
 ---> 3a87364c7b45
Step 12 : RUN source /usr/local/bin/virtualenvwrapper.sh
 ---> Running in c13a187261ec
/bin/sh: 1: source: not found

如果我ls到该目录中(只是为了测试先前步骤提交)我可以看到存在的文件按预期:

$ docker run 3a87 ls /usr/local/bin
easy_install
easy_install-2.7
pip
pip-2.7
virtualenv
virtualenv-2.7
virtualenv-clone
virtualenvwrapper.sh
virtualenvwrapper_lazy.sh

如果我试着运行source命令,我会得到和上面一样的'not found'错误。 如果我运行交互式shell会话,则源代码工作:

$ docker run 3a87 bash
source
bash: line 1: source: filename argument required
source: usage: source filename [arguments]

我可以从这里运行脚本,然后高兴地访问workonmkvirtualenv等。

我已经做了一些挖掘,初步看起来好像问题可能在于作为Ubuntu系统外壳bash作为Ubuntu的登录shell和破折号之间的区别, 不支持source命令。

但是,对此的答案似乎是使用'。'。 而不是source ,但这只会导致Docker运行时崩溃,出现恐慌异常。

从Dockerfile RUN指令运行shell脚本以解决此问题的最佳方法是(运行Ubuntu 12.04 LTS的默认基本映像)。


RUN /bin/bash -c "source /usr/local/bin/virtualenvwrapper.sh"


原始答复

FROM ubuntu:14.04
RUN rm /bin/sh && ln -s /bin/bash /bin/sh

这应该适用于每个Ubuntu Docker基本映像。 我通常为我写的每个Dockerfile添加这一行。

由有关旁观者编辑

如果你想得到“在整个Dockerfile中使用bash而不是sh ”的效果,而不改变容器内部的操作系统,并且可能损坏容器内的操作系统,那么你可以告诉Docker你的意图。 这样做是这样的:

SHELL ["/bin/bash", "-c"]

*可能的损害是Linux中的许多脚本(在新安装的Ubuntu上安装grep -rHInE '/bin/sh' /返回超过2700个结果)期望在/bin/sh有完整的POSIX shell。 bash shell不仅仅是POSIX加上额外的内建函数。 有些内建(和更多)的行为与POSIX中的行为完全不同。 我完全支持避免POSIX(以及任何你没有在另一个shell上测试的脚本都会工作,因为你认为你避免了基础主义)的谬论,并且仅仅使用了bashism。 但是你在脚本中使用适当的shebang来做到这一点。 不要从整个操作系统下拉出POSIX外壳。 (除非您有时间验证Linux附带的所有2700 plus脚本以及您安装的任何软件包中的所有脚本。)

下面的答案更详细。 https://stackoverflow.com/a/45087082/117471


我有同样的问题,为了在virtualenv内执行pip install,我必须使用以下命令:

RUN pip install virtualenv virtualenvwrapper
RUN mkdir -p /opt/virtualenvs
ENV WORKON_HOME /opt/virtualenvs
RUN /bin/bash -c "source /usr/local/bin/virtualenvwrapper.sh 
    && mkvirtualenv myapp 
    && workon myapp 
    && pip install -r /mycode/myapp/requirements.txt"

我希望它有帮助。

链接地址: http://www.djcxy.com/p/25709.html

上一篇: Using the RUN instruction in a Dockerfile with 'source' does not work

下一篇: ^? in stty output mean?