前言
目前我们公司使用的基本上都是java开发的后端,本文详细的介绍了公司java程序docker 包构建的演变过程,这里面不对java包本身的构建做过多的赘述。
docker 镜像的演变过程
最初的时候我们只想着给java包怎么放到docker 镜像中,我们使用了如下的Dockerfile
FROM openjdk:8u212-jre-alpine
ENV TZ="Asia/Shanghai"
ENV JVM="-XX:MetaspaceSize=256m -XX:MaxMetaspaceSize=512m -Xms1024m -Xmx2048m -Xmn512m -Xss512k -XX:SurvivorRatio=8 -XX:+UseConcMarkSweepGC"
RUN ln -snf /usr/share/zoneinfo/$TZ /etc/localtime ; \
echo $TZ > /etc/timezone ; \
mkdir -p /var/upload_tmp ;\
mkdir -p /home/guiyu-v2/config ; \
mkdir -p /home/guiyu-v2/jar ; \
mkdir -p /home/guiyu-v2/jar/logxml ;\
mkdir -p /home/temp_download/oss
CMD ["sh","-c","java `echo $JVM` -Djava.io.tmpdir=/var/upload_tmp -Dfile.encoding=utf-8 -Dspring.config.location=/home/guiyu-v2/config/oss.yml -Dlogging.config=/home/guiyu-v2/jar/logxml/log-oss.xml -jar /home/guiyu-v2/jar/guiyu-oss-web-2.0-SNAPSHOT.jar"]
COPY ./guiyu-oss-web/src/main/resources/application.yml /home/guiyu-v2/config/oss.yml
COPY ./guiyu-oss-web/target/guiyu-oss-web-2.0-SNAPSHOT.jar /home/guiyu-v2/jar/
COPY ./guiyu-oss-web/src/main/resources/logback.xml /home/guiyu-v2/jar/logxml/log-oss.xml
使用上面的Dockerfile 显着运维人员太没有技术水平了,然后吧启动参数变成可变的变量,于是乎演变成下面
FROM openjdk:8u212-jre-alpine
ENV TZ="Asia/Shanghai"
RUN ln -snf /usr/share/zoneinfo/$TZ /etc/localtime ; \
echo $TZ > /etc/timezone ; \
mkdir -p /var/upload_tmp ;\
mkdir -p /home/work/ ; \
mkdir -p /home/work/ ; \
mkdir -p //home/work/ ;\
mkdir -p /home/temp_download/oss ;\
sed -i 's/dl-cdn.alpinelinux.org/mirrors.aliyun.com/g' /etc/apk/repositories
ENV JVM_OPTS -server -Xms1g -Xmx1g -XX:+UseG1GC
ENV JAVA_OPTS -Dfile.encoding=UTF-8 -Djava.security.egd=file:/dev/urandom -Dspring.profiles.active=prod -Dlogging.config=/home/work/logback-spring.xml
ENV JAVA_AGENT=
ENV APP_NAME=guiyu-oss-web-2.0-SNAPSHOT.jar
CMD java ${JVM_OPTS} ${JAVA_OPTS} ${JAVA_AGENT} -jar ${APP_NAME}
COPY ./guiyu-oss-web/src/main/resources/application.yml /home/work/application.yml
COPY ./guiyu-oss-web/target/guiyu-oss-web-2.0-SNAPSHOT.jar /home/work/
COPY ./guiyu-oss-web/src/main/resources/logback.xml /home/work/logback-spring.xml
发展到这个地步的时候我们启动就只需要改 JVM_OPTS和JAVA_OPTS 参数即可,稍微有点人性化了,但是好景不长,线上出了一点问题,需要通过jvm分析工具来看看那个地方出问题了,我们采用上面这种方式构建的docker镜像无法使用jvm分析工具。
因为上面这种方式构建的Docker镜像里面所有的java进程都是 PID 为1,jvm分析PId 为1 的有点问题,而且好多服务并不能处理系统发送的kill 指令,这所是我不能容忍的,所以就进化成了如下样子,
然后给Docker 添加一个init 进程放在主进程
FROM openjdk:8u212-jre-alpine
ENV TZ="Asia/Shanghai"
RUN ln -snf /usr/share/zoneinfo/$TZ /etc/localtime ; \
echo $TZ > /etc/timezone ; \
mkdir -p /var/upload_tmp ;\
mkdir -p /home/work/ ; \
mkdir -p /home/work/ ; \
mkdir -p //home/work/ ;\
mkdir -p /home/temp_download/oss ;\
sed -i 's/dl-cdn.alpinelinux.org/mirrors.aliyun.com/g' /etc/apk/repositories ;\
RUN apk add --no-cache tini
ENV JVM_OPTS -server -Xms1g -Xmx1g -XX:+UseG1GC
ENV JAVA_OPTS -Dfile.encoding=UTF-8 -Djava.security.egd=file:/dev/urandom -Dspring.profiles.active=prod -Dlogging.config=/home/work/logback-spring.xml
ENV JAVA_AGENT=
ENV APP_NAME=guiyu-oss-web-2.0-SNAPSHOT.jar
ENTRYPOINT ["/sbin/tini", "--"]
CMD java ${JVM_OPTS} ${JAVA_OPTS} ${JAVA_AGENT} -jar ${APP_NAME}
COPY ./guiyu-oss-web/src/main/resources/application.yml /home/work/application.yml
COPY ./guiyu-oss-web/target/guiyu-oss-web-2.0-SNAPSHOT.jar /home/work/
COPY ./guiyu-oss-web/src/main/resources/logback.xml /home/work/logback-spring.xml
到这个时候为止,大部分问题都已经解决,但是进去到docker 容器里面发现没有 jmap等指令,需要通过如下命令去安装即可
apk add openjdk8
本着docker 镜像最小原则,就没有把它安装到所有的docker镜像中去。
到此公司的java包docker 镜像构建完毕