自動化構建
是應用發佈過程當中必不可少的環節, 經常使用的構建工具備jenkins
,walle
等。而這些工具在構建應用時一般會有如下問題:php
- 須要直接或間接的寫一坨用於構建的shell命令等,不易管理、兼容性較差
- 上面一點可能還比較容易解決,但最爲致命的是:重度依賴如
jenkins
宿主機或打包機上的軟件環境,如git
,maven
,java
等
理想狀況是: 不一樣的應用如java應用、go應用、php應用等等,均可以在某臺負責構建的宿主機上並行無干擾的執行構建操做,且構建中依賴的軟件環境、構建流程等均可以由開發人員控制。 html
到目前爲止,能很好的完成以上使命的,可能非docker莫屬了!java
在docker的世界裏,構建交付的是鏡像
,而可以產生鏡像的是Dockerfile
(手動使用docker commit
的另當別論). git
在docker ce 17.05
以後,出現了一個很重要的特性Multi-Stage Build
(多階段構建) , 它將顯著提高你的運維生產力!github
下文將用實戰案例來詳細解讀
Multi-Stage Build
這一特性
如下演示以java
hello world 爲例,完整代碼在: https://github.com/zhouzhipeng/docker-multi-stage-demoweb
這是一個標準的maven 項目,僅有個HelloWorld主類。大致構建思路爲:docker
Dockerfile.build 用於編譯和打包jarshell
FROM maven:3.5.2-alpine MAINTAINER zhouzhipeng <admin@zhouzhipeng.com> WORKDIR /app COPY . . # 編譯打包 RUN mvn package -Dmaven.test.skip=true
Dockerfile.old 用於運行jar中的主類bash
FROM openjdk:8-jre-alpine MAINTAINER zhouzhipeng <admin@zhouzhipeng.com> WORKDIR /app COPY docker-multi-stage-demo-1.0-SNAPSHOT.jar . # 運行main類 CMD java -cp docker-multi-stage-demo-1.0-SNAPSHOT.jar com.zhouzhipeng.HelloWorld
注意到,兩個dockerfile之間關聯的 docker-multi-stage-demo-1.0-SNAPSHOT.jar 文件,須要另一個build.sh 腳原本串起來.app
build.sh
#!/usr/bin/env bash # 1. 先構建出帶有產物jar的鏡像 docker build -t zhouzhipeng/dockermultistagedemo-build -f Dockerfile.build . # 2. 臨時建立 dockermultistagedemo-build 容器 docker create --name build zhouzhipeng/dockermultistagedemo-build # 3. 將上面容器中的jar拷貝出來 docker cp build:/app/target/docker-multi-stage-demo-1.0-SNAPSHOT.jar ./ # 4. 構建java執行的鏡像 docker build -t zhouzhipeng/dockermultistagedemo -f Dockerfile.old . # 5. 刪除臨時jar文件 rm -rf docker-multi-stage-demo-1.0-SNAPSHOT.jar
對Dockerfile和shell也瞭解的朋友相信應該都看得懂,在此不作過多贅述.
看過上一節後,你也許會感受是否是有點麻煩呢? 是的,麻煩之處在於不只要寫多個dockerfile,並且還須要一個build.sh 腳原本額外執行。 無疑是增大了構建應用的複雜度!
將上面的Dockerfile.build 和Dockerfile.old 結合起來,稍加修飾,獲得以下全新的Dockerfile:
FROM maven:3.5.2-alpine as builder MAINTAINER zhouzhipeng <admin@zhouzhipeng.com> WORKDIR /app COPY src . COPY pom.xml . # 編譯打包 (jar包生成路徑:/app/target) RUN mvn package -Dmaven.test.skip=true FROM openjdk:8-jre-alpine MAINTAINER zhouzhipeng <admin@zhouzhipeng.com> WORKDIR /app COPY --from=builder /app/target/docker-multi-stage-demo-1.0-SNAPSHOT.jar . # 運行main類 CMD java -cp docker-multi-stage-demo-1.0-SNAPSHOT.jar com.zhouzhipeng.HelloWorld
而後,仍然是熟悉的docker build命令
docker build -t zhouzhipeng/dockermultistagedemo-new .
便可。
細心的你應該不難發現,上面的Dockerfile 中有兩處地方不同,
FROM
語句COPY
命令後多了--from=builder
這就是今天的主咖 Multi-Stage Build
, 先來經過一張圖來直觀感覺下什麼是所謂的Multi-Stage Build
(多階段構建 ):
經過多階段構建,既能夠保持Dockerfile簡潔易讀,又可讓最終的產物鏡像很「乾淨」。
仍是以上文中的Dockerfile爲例, 以下圖所示:
紅框中的部分能夠看做是一個個獨立的「stage」 ,能夠粗略想象成就是一個獨立的Dockerfile內容。
你們知道鏡像構建是一層一層疊加的,按照Dockerfile的命令行順序,由上至下依次執行疊加。 因此,下層的stage才能夠引用到上層的stage,爲了方便引用到上層的stage,故須要給其取一個名字, 用as
操做符。
FROM
命令的完整格式以下:
FROM <image>[:<tag>] [AS <name>]
stage之間交互的是文件,故COPY
命令須要擴展,經過--from=<name>
來指定須要從上方的哪一個"stage" 拷貝文件, 其完整命令格式以下:
COPY --from=<name|index> <src>... <dest> # 注意--from 是可選的,當上層的stage沒有名字時能夠按照index(從0開始)的順序引用,eg. --from=0
值得一提的是,默認狀況下使用docker build
命令構建一個包含多個stage的dockerfile時,最終的產物是最下方的一個stage 所產生的鏡像。
固然,若是出於調試緣由或其餘需求,docker也是支持構建到指定的stage的,使用 --target builder
就能夠只構建builder鏡像。
docker build -t zhouzhipeng/builder --target builder .
到目前爲止,咱們已經有了一個可以一鍵構建的Dockerfile 文件,接下來就只差讓它可以自動構建了!
你能夠用你熟悉的jenkins
結合github的webhook來實現提交一次代碼,就執行一次docker build命令。
固然,我推薦我的體驗的話就用官方的docker hub 吧,由於這樣你構建的鏡像還能夠與他人共享。
具體的用Docker hub 的 automated build 功能就不詳細說明了, 下面用一張gif圖快速演示下,感興趣的朋友能夠自行去探索下。
Multi-Stage Build
這一特性很是適合作構建管道流,對於那些依賴環境複雜、流程也複雜的應用來講最合適不過了。
能夠clone下上面的源碼試下哦: https://github.com/zhouzhipeng/docker-multi-stage-demo
by zhouzhipeng from https://blog.zhouzhipeng.com/...
本文可全文轉載,但須要保留原做者和出處。
https://docs.docker.com/v17.09/engine/userguide/eng-image/multistage-build/