【问题标题】:MultiStage Docker Image多阶段 Docker 映像
【发布时间】:2020-04-07 17:39:59
【问题描述】:

我创建了一个多阶段 dockerfile。

FROM dot-portal.de.pri.o2.com:8079/centos:centos7 as base

# Install a basic SSH server GIT, UNZIP, LSOF and JDK 8

ADD myrepo.repo /etc/yum.repos.d/myrepo.repo

RUN rm -rf /etc/yum.repos.d/CentOS*

COPY docker /opt/docker

RUN  yum repolist refresh && yum clean all && yum repolist && yum install -y openssh-server git unzip lsof epel-release bzip2 fontconfig && yum clean all && rm -rf /var/cache/yum && cd /opt/docker && rpm -ivh *.rpm && rm -rf /opt/docker

ADD docker-compose-Linux-x86_64 /usr/local/bin/docker-compose

RUN ln -s /usr/local/bin/docker-compose /usr/bin/docker-compose

# update sshd settings, create jenkins user, set jenkins user pw, generate ssh keys
RUN sed -i 's|session    required     pam_loginuid.so|session    optional     pam_loginuid.so|g' /etc/pam.d/sshd \
    && mkdir -p /var/run/sshd \
    && useradd -u 1000 -m -s /bin/bash jenkins \
    && echo "jenkins:jenkins" | chpasswd \
    && usermod -aG wheel jenkins \
    && /usr/bin/ssh-keygen -A \
    && sed -i "s/#UsePrivilegeSeparation.*/UsePrivilegeSeparation no/g" /etc/ssh/sshd_config \
    && sed -i 's/PermitRootLogin without-password/PermitRootLogin yes/' /etc/ssh/sshd_config \
    && echo export JAVA_HOME="/`alternatives  --display java | grep best | cut -d "/" -f 2-6`" >> /etc/environment \
    && usermod -a -G docker jenkins \
        && sed 's@session\s*required\s*pam_loginuid.so@session optional pam_loginuid.so@g' -i /etc/pam.d/sshd


RUN git config --global http.sslVerify false && mkdir -p /opt/app/jenkins

COPY --chown=jenkins:jenkins jenkins_tools /opt/app/jenkins/jenkins_tools/

ADD node-v10.13.0-linux-x64.tar.gz /opt/app/jenkins/jenkins_tools

ENV CI_TOOLS_HOME /opt/app/jenkins/jenkins_tools

ENV ARCH Linux-x86_64
ENV CI_TOOLS_HOME /opt/app/jenkins/jenkins_tools
ENV JAVA_HOME /usr/java/jdk1.8.0_241-amd64/

COPY dot-portal.de.pri.o2.com.crt $CI_TOOLS_HOME/dot-portal.de.pri.o2.com.crt
RUN cd $CI_TOOLS_HOME && echo yes | keytool -importcert -file dot-portal.de.pri.o2.com.crt -alias "dot-portal.de.pri.o2.com" -keystore "$JAVA_HOME/jre/lib/security/cacerts" -storepass changeit

COPY --chown=jenkins:jenkins authorized_keys /home/jenkins/.ssh/

RUN mkdir /root/.ssh && mkdir -p /opt/app/home/jenkins && chown -R jenkins:jenkins /opt/app/home/jenkins

COPY dockerslave /etc/default/docker

RUN ln -s $CI_TOOLS_HOME/phantomjs-2.1.1-linux-x86_64/bin/phantomjs /usr/bin/ && chown -R jenkins:jenkins /opt/app/jenkins/jenkins_tools

FROM base
COPY --from=base /opt/app/jenkins/jenkins_tools /opt/app/jenkins/jenkins_tools
COPY --from=base /etc/default/docker /etc/default/docker
COPY --from=base /usr/bin/phantomjs /usr/bin/phantomjs

ENV CI_TOOLS_HOME /opt/app/jenkins/jenkins_tools
ENV ARCH Linux-x86_64
ENV CI_TOOLS_HOME /opt/app/jenkins/jenkins_tools
ENV JAVA_HOME /usr/java/jdk1.8.0_241-amd64/
ENV MAVEN_HOME /opt/app/jenkins/jenkins_tools/nonarch/maven-3.3.9
RUN echo "export PATH=$CI_TOOLS_HOME/node-v10.13.0-linux-x64/bin/:$CI_TOOLS_HOME/phantomjs-2.1.1-linux-x86_64/bin/:$JAVA_HOME/bin:$MAVEN_HOME/bin:$PATH" >> ~/.bashrc

# Standard SSH port
EXPOSE 22

CMD ["/usr/sbin/sshd", "-D", "-e"]

我的问题是,我在基本图像中安装/创建的包和用户不会反映在最终图像中。仅从 Base 创建最终映像,但是当我连接到容器时,我无法看到在基本映像中创建的用户“jenkins”,并且包也丢失了。

【问题讨论】:

  • 正如我所见,最终图像不包含任何有关创建用户的说明。为什么您希望最终图像中有用户 jenkins?

标签: docker dockerfile docker-multi-stage-build


【解决方案1】:

您想在这里通过多阶段 docker 构建实现什么目标?如果您的第二阶段从具有两个阶段的构建中的第一阶段的结果开始,那么您还不如只有一个阶段...

通常使用多阶段构建,因为使用第一阶段(例如“构建器”)来构建一些需要大量开发工具的人工制品。在第二阶段,只有该工件将被复制到一个小的基础图像中以获得一个非常小的最终图像,因为它不包含构建工件所需的工具。

来自 docker docs 的示例:

FROM golang:1.7.3 as builder
WORKDIR /go/src/github.com/alexellis/href-counter/
RUN go get -d -v golang.org/x/net/html  
COPY app.go .
RUN CGO_ENABLED=0 GOOS=linux go build -a -installsuffix cgo -o app .

FROM alpine:latest  
RUN apk --no-cache add ca-certificates
WORKDIR /root/
COPY --from=builder /go/src/github.com/alexellis/href-counter/app .
CMD ["./app"]  

在您的情况下,我很确定您的最终图像确实包含 jenkins 用户,因为您的第二阶段是从基础阶段开始的。但是,这些从基础映像复制的复制命令将在您的第二阶段将权限重置为 root

COPY --from=base /opt/app/jenkins/jenkins_tools /opt/app/jenkins/jenkins_tools
COPY --from=base /etc/default/docker /etc/default/docker
COPY --from=base /usr/bin/phantomjs /usr/bin/phantomjs

无论如何,当您启动FROM base 时,此复制命令是多余的。如果您删除这些行,jenkins_tools 目录将正确地归 jenkins 用户所有。

请注意,您可以使用 USER <username> 更改用于在 docker 构建期间执行命令的用户

【讨论】:

  • 如果我删除 COPY 行,我的 docker 镜像大小会减小吗?现在 Jenkins SSH slave docker 镜像大小为 2.5GB
  • 是的,它会更小。在这种情况下,我可能也会删除 FROM base 行,因为这张图片没有必要。
  • 如果我删除 FROM 基线,它会创建一个大小为 2GB 的图像
  • 我要缩小图片尺寸
  • 你删除了三个复制行吗?此外,您似乎正在将一堆工具复制到容器中,您真的需要所有这些工具吗?这条线COPY docker /opt/docker 应该做什么?你不需要你的docker引擎在docker中的docker镜像中,只需要cli
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2020-09-08
  • 1970-01-01
  • 2018-09-25
  • 2020-06-05
  • 1970-01-01
  • 2020-02-23
  • 1970-01-01
相关资源
最近更新 更多