【问题标题】:docker container exits on entry pointsdocker 容器在入口点退出
【发布时间】:2019-07-30 03:47:03
【问题描述】:

我正在尝试使用 docker run 命令通过 Dockerfile 运行 docker 容器。以下是我的 Dockerfile。

FROM python:3.6
# for imaging stuff
RUN apt-get update
RUN apt install libmagickwand-dev
# Create  app directory
RUN mkdir -p /home/test/app
# Install Libre Office and ghostscript for pdf conversion and repairing
RUN apt-get update -qq \
    && apt-get install -y -q libreoffice \
    && apt-get remove -q -y libreoffice-gnome \
    && apt-get update \
    && apt-get -y install ghostscript \
    && apt-get -y install nano \
    && apt-get -y install poppler-utils \
    && apt-get install -y nginx
# Cleanup after apt-get commands
RUN apt-get clean \
    && rm -rf /var/lib/apt/lists/* /tmp/* /var/tmp/* \
    /var/cache/apt/archives/*.deb /var/cache/apt/*cache.bin
# Activate WORKING DIR
WORKDIR /home/test/app
# Initiating devnull directory
RUN mkdir -p dev_null
# Copying requirements
COPY requirements/local.txt /tmp/requirements.txt
# Install the app dependencies
RUN pip install -r /tmp/requirements.txt
COPY id_rsa /root/.ssh/id_rsa
COPY requirements/private.txt /tmp/private.txt
RUN ssh-keyscan -T 60 bitbucket.org >> /root/.ssh/known_hosts \
    && chmod 600  /root/.ssh/id_rsa \
    && pip install -r /tmp/private.txt
# Envs
ENV DJANGO_SETTINGS_MODULE app.settings.local
ENV ENVIORNMENT local
# ADD the source code and entry point into the container
ADD . /home/test/app
ADD docker-entrypoint.sh /home/test/app/docker-entrypoint.sh
# Making entry point executable
RUN apt-get update && apt-get install -y supervisor \
    && apt-get install -y nginx
#RUN mkdir -p /var/log/test
#COPY supervisor.conf /etc/supervisor/conf.d/supervisor.conf
RUN chmod +x docker-entrypoint.sh
RUN mkdir -p /var/log/test
COPY supervisor.conf /etc/supervisor/conf.d/supervisor.conf
# Exposing port


# Copy entrypoint script into the image
COPY ./docker-entrypoint.sh /
COPY ./django_nginx.conf /etc/nginx/
RUN chmod +x start.sh

#RUN chmod +x /docker-entrypoint.sh
EXPOSE 8000
ENTRYPOINT ["bash", "/docker-entrypoint.sh"]
CMD ["bash", "/home/test/app/start.sh"]

以下是我的切入点。

#!/usr/bin/env bash
set -e

# ToDo Need to enable this
#until psql $DATABASE_URL -c '\l'; do
#  >&2 echo "Postgres is unavailable - sleeping"
#  sleep 1
#done
#
#>&2 echo "Postgres is up - continuing"
cd app
mkdir -p app/keys
if [[ ! -e /var/log/gunicorn-access.log ]]; then
    touch /var/log/gunicorn-access.log
fi
if [[ ! -e /var/log/gunicorn-error.log ]]; then
    touch /var/log/gunicorn-error.log
fi

if [ "x$DJANGO_MANAGEPY_MIGRATE" = 'xon' ]; then
    echo "Django starting to migrate un-applied migrations"
    python manage.py migrate --noinput
fi

if [ "x$DJANGO_MANAGEPY_COLLECTSTATIC" = 'xon' ]; then
    echo "Django starting to collect static data"
    python manage.py collectstatic --noinput
fi

if [ "x$DJANGO_LOADDATA" = 'xon' ]; then
    python manage.py loaddata taxing/fixtures/province-taxing-table-initial-data.json
fi

if [ "x$LOAD_TEMPLATE_FROM_S3" = 'xtrue' ]; then
    echo "loading s3"
    python manage.py loadindex

fi


# Start Gunicorn processes
echo Starting Gunicorn.
exec gunicorn test.wsgi:application \
    --name test \
    --workers 3 \
    --log-level=info \
    --log-file=/srv/logs/gunicorn.log \
    --access-logfile=/srv/logs/access.log &

exec service nginx start

下面是我的start.sh

#!/bin/bash

echo "I am running"
echo "Belive me"
/usr/bin/supervisord -n

我正在使用docker build -t test . 构建容器,它构建得很好。但是当我尝试使用docker run --name test --env-file ./env test 运行这个容器时,我的容器在 Dockerfile 的["bash", "/docker-entrypoint.sh"] 命令上退出,没有任何错误/消息,但是如果我从 Dockerfile 中删除入口点命令,它工作得很好。我无法找出错误。任何帮助表示赞赏。

【问题讨论】:

    标签: docker dockerfile docker-build docker-entrypoint docker-run


    【解决方案1】:

    这是我看到的这个序列中发生的事情:

    ENTRYPOINT ["bash", "/docker-entrypoint.sh"]
    CMD ["bash", "/home/test/app/start.sh"]
    

    当你启动容器时,Docker 运行bash /docker-entrypoint.sh bash /home/test/app/start.sh。但是,入口点脚本根本不会查看其命令行参数,因此您在此处指定的任何 CMD 或您在 docker run 命令行末尾给出的任何命令都会被完全忽略。

    当该入口点脚本运行时:

    exec gunicorn ... &
    exec service nginx start
    # end of file
    

    它将gunicorn作为后台进程启动并继续到下一行;然后它用service 命令替换自己。 service 命令成为主容器进程,进程 ID 为 1。它启动 nginx,并立即返回。现在主容器进程已经返回,容器退出。


    对于您编写的这段代码,您应该删除入口点脚本末尾的 exec 行并将它们替换为 just

    exec "$@"
    

    这将导致 shell 运行其命令行参数(即 Dockerfile CMD)。

    但是,有一个现成的nginx Docker image。通常,如果您需要多个进程,将它们作为两个单独的容器在同一个 Docker 网络上运行会更容易也更好。这避免了尝试在同一个容器中运行多个事物的复杂性。

    【讨论】:

      【解决方案2】:

      如果你的镜像构建成功,那么你可以尝试在你的 start.sh 中附加这一行

      tail -f /etc/issue
      

      您的 start.sh 可能如下所示:

      #!/bin/bash
      
      echo "I am running"
      echo "Belive me"
      /usr/bin/supervisord -n
      tail -f /etc/issue
      

      【讨论】:

      • 试过了,但还是没有改善。容器在执行最后一行入口点后退出,即启动 nginx。在运行容器时打印以下消息。启动 Gunicorn。启动 nginx:nginx.
      • 另一个修复CMD ["/bin/bash", "/home/test/app/start.sh"]
      • 不走运维杰。仍在退出。如果我从 Dockerfile 中删除 entrypoints 命令,那么它工作得很好。
      • 尝试以分离模式运行运行命令docker run -dit --name test --env-file ./env test,然后尝试执行/home/test/app/start.sh以验证脚本是否可执行。按 ctrl+d 退出
      • 试过 docker run -dit --name test --env-file ./env test 但容器仍然在入口点退出。
      猜你喜欢
      • 1970-01-01
      • 2017-05-20
      • 2018-01-18
      • 1970-01-01
      • 2021-07-26
      • 1970-01-01
      • 2022-01-25
      • 2023-01-25
      • 2015-04-30
      相关资源
      最近更新 更多