Docker之構建上下文詳解

来源:https://www.cnblogs.com/soilge/archive/2019/11/05/11799827.html
-Advertisement-
Play Games

昨天寫了使用 Dockerfile 定製鏡像。其中構建上下文這一塊沒有寫,今天把這一塊單獨拿出來寫一下。 Docker鏡像構建 簡單說下構建鏡像步驟: 1. cd Dockerfile 所在目錄; 2. 執行 docker build 構建命令: docker build t . 通過上面的工作流, ...


昨天寫了使用 Dockerfile 定製鏡像。其中構建上下文這一塊沒有寫,今天把這一塊單獨拿出來寫一下。

Docker鏡像構建

簡單說下構建鏡像步驟:

  1. cd Dockerfile 所在目錄;
  2. 執行 docker build 構建命令:

    docker build -t .

通過上面的工作流,很容易形成這樣的理解誤區:

  1. docker build 後面的 . 為 Dockerfile 所在的目錄;
  2. Dockerfile 文件名 必須為 Dockerfile;

其實上面這種理解是錯誤的,要想準確理解其含義,首先我們需要先瞭解下 Docker 的架構和 docker build 的工作原理。

Docker架構

Docker 使用C/S (客戶端/伺服器)體系的架構,Docker 客戶端與 Docker 守護進程通信,Docker 守護進程負責構建,運行和分發 Docker 容器。Docker 客戶端和守護進程可以在同一個系統上運行,也可以將 Docker 客戶端連接到遠程 Docker 守護進程。Docker 客戶端和守護進程使用 REST API 通過UNIX套接字或網路介面進行通信。

https://docs.docker.com/engine/images/architecture.svg

docker build 的工作原理

  1. Client端執行 docker build . 命令 ;
  2. Docker 客戶端會將構建命令後面指定的路徑(.)下的所有文件打包發送給 Docker 服務端;
  3. Docker 服務端收到客戶端發送的包,然後解壓,根據 Dockerfile 裡面的指令進行鏡像的分層構建;

鏡像構建上下文(Context)

當我們進行鏡像構建的時候,並非所有定製都會通過 RUN 指令完成,經常會需要將一些本地文件複製進鏡像,比如通過 COPY 指令、ADD 指令等。而 docker build 命令構建鏡像,其實並非在本地構建,而是在服務端,也就是 Docker 引擎中構建的。那麼在這種客戶端/服務端的架構中,如何才能讓服務端獲得本地文件呢?

這就引入了上下文的概念。當構建的時候,用戶會指定構建鏡像上下文的路徑,docker build 命令得知這個路徑後,會將路徑下的所有內容打包,然後上傳給 Docker 引擎。這樣 Docker 引擎收到這個上下文包後,展開就會獲得構建鏡像所需的一切文件。如果在 Dockerfile 中這麼寫:

COPY ./package.json /app/

這並不是要複製執行 docker build 命令所在的目錄下的 package.json,也不是複製 Dockerfile 所在目錄下的 package.json,而是複製 上下文(context) 目錄下的 package.json。

因此,COPY這類指令中的源文件的路徑都是相對路徑。這也是初學者經常會問的為什麼 COPY ../package.json /app 或者 COPY /opt/xxxx /app 無法工作的原因,因為這些路徑已經超出了上下文的範圍,Docker 引擎無法獲得這些位置的文件。如果真的需要那些文件,應該將它們複製到上下文目錄中去。

示例1 :

[root@192 test]# ls
Dockerfile
[root@192 test]# cat Dockerfile
FROM alpine:latest
ADD  /root/mydocker/apache-tomcat-9.0.27.tar.gz /data/soft
[root@192 test]# ls /root/mydocker/apache-tomcat-9.0.27.tar.gz
/root/mydocker/apache-tomcat-9.0.27.tar.gz
[root@192 test]# docker build  .
Sending build context to Docker daemon  3.072kB
Step 1/2 : FROM alpine:latest
 ---> 965ea09ff2eb
Step 2/2 : ADD /root/mydocker/apache-tomcat-9.0.27.tar.gz /data/soft
ADD failed: stat /var/lib/docker/tmp/docker-builder904012777/root/mydocker/apache-tomcat-9.0.27.tar.gz: no such file or directory

可以看出:

  1. 鏡像構建上下文路徑並不是 Dockerfile文件所在的路徑;
  2. Dockerfile 中指令的工作目錄是服務端解壓客戶端傳輸包的路徑,因為 ADD 指令失敗了,意味著當前目錄並沒有 apache-tomcat 文件;

理解構建上下文對於鏡像構建是很重要的,可以避免犯一些不應該的錯誤。比如有些初學者在發現 COPY /opt/xxxx /app 不工作後,於是乾脆將 Dockerfile 放到了硬碟根目錄去構建,結果發現 docker build 執行後,在發送一個幾十 GB 的東西,極為緩慢而且很容易構建失敗。那是因為這種做法是在讓 docker build 打包整個硬碟,這顯然是使用錯誤。

一般來說,應該會將 Dockerfile 置於一個空目錄下,或者項目根目錄下。如果該目錄下沒有所需文件,那麼應該把所需文件複製一份過來。如果目錄下有些東西確實不希望構建時傳給 Docker 引擎,那麼可以用 .gitignore 一樣的語法寫一個.dockerignore,該文件是用於剔除不需要作為上下文傳遞給 Docker 引擎的。

那麼為什麼會有人誤以為 . 是指定 Dockerfile 所在目錄呢?這是因為在預設情況下,如果不額外指定 Dockerfile 的話,會將上下文目錄下的名為 Dockerfile 的文件作為 Dockerfile。

這隻是預設行為,實際上 Dockerfile 的文件名並不要求必須為 Dockerfile,而且並不要求必須位於上下文目錄中,比如可以用-f ../Dockerfile.php參數指定某個文件作為 Dockerfile。

當然,一般大家習慣性的會使用預設的文件名 Dockerfile,以及會將其置於鏡像構建上下文目錄中。

示例2

[root@192 test]# docker images
REPOSITORY                         TAG                         IMAGE ID            CREATED             SIZE
tomcat                             jdk8-adoptopenjdk-hotspot   1a2bfb3e6eee        15 hours ago        318MB
openjdk                            8-jre-slim                  d0cfe439ce3d        13 days ago         184MB
btnguyen2k/oraclejdk8_jre-alpine   latest                      fab475620d00        4 years ago         208MB
[root@192 test]# cat ../mynginx/test
FROM alpine:latest
[root@192 test]# ls
apache-tomcat-9.0.27.tar.gz
[root@192 test]#  docker build -f ../mynginx/test  -t test:v1 .
Sending build context to Docker daemon  10.99MB
Step 1/1 : FROM alpine:latest
latest: Pulling from library/alpine
89d9c30c1d48: Already exists
Digest: sha256:c19173c5ada610a5989151111163d28a67368362762534d8a8121ce95cf2bd5a
Status: Downloaded newer image for alpine:latest
 ---> 965ea09ff2eb
Successfully built 965ea09ff2eb
Successfully tagged test:v1
[root@192 test]# docker images
REPOSITORY                         TAG                         IMAGE ID            CREATED             SIZE
tomcat                             jdk8-adoptopenjdk-hotspot   1a2bfb3e6eee        15 hours ago        318MB
alpine                             latest                      965ea09ff2eb        11 days ago         5.55MB
test                               v1                          965ea09ff2eb        11 days ago         5.55MB
openjdk                            8-jre-slim                  d0cfe439ce3d        13 days ago         184MB
btnguyen2k/oraclejdk8_jre-alpine   latest                      fab475620d00        4 years ago         208MB     

可以看出:

Dockerfile 的文件名並不要求必須為 Dockerfile,而且並不要求必須位於上下文目錄中,比如可以用-f ../mynginx/test參數指定某個文件作為 Dockerfile。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • Made By Herolh 目錄 { index} [TOC] Centos7開放及查看埠 1、開放埠 2、查看防火牆所有開放的埠 3.、關閉防火牆 如果要開放的埠太多,嫌麻煩,可以關閉防火牆,安全性自行評估 4、查看防火牆狀態 5、查看監聽的埠 6、檢查埠被哪個進程占用 7、查看進程 ...
  • 01 uptime命令 通常我們發現系統變慢時,我們都會執行top或者uptime命令,來查看當前系統的負載情況,比如像下麵,我執行了uptime,系統返回的了結果。 前幾列的信息,相信大家都很熟悉,它們分別是當前時間、系統運行時間和正在登陸的用戶個數,最後一個就是系統平均負載的情況。 Load A ...
  • 一.介紹 服務(service) 本質就是進程,但是是運行在後臺的,通常都會監聽某個埠,等待其它程式的請求,比如(mysql , sshd 防火牆等),因此我們又稱為守護進程,是Linux 中非常重要的知識點。 二.基本使用 三. 查看服務名 此文件夾下放著系統都有哪些服務所以 即可 四.chkc ...
  • 一.rpm介紹 二.rpm相關指令 三.rpm包結構 例如一個 rpm 包名:firefox 45.0.1 1.el6.centos.x86_64.rpm 四.yum介紹 Yum是一個 "Shell " 前端軟體包管理器。基於 "RPM " 包管理,能夠從指定的伺服器自動下載 RPM 包並且安裝,可 ...
  • 一.查看ip 二.png 測試主機之間網路連通 三.linux配置網路環境 ` 修改配置文件後,要重啟網路服務才能生效 ...
  • 本文系轉載,著作權歸作者所有。商業轉載請聯繫作者獲得授權,非商業轉載請註明出處。 作者: 陳莉君 來源: 微信公眾號linux閱碼場(id: linuxdev) 初次踏入Linux 幾多耕耘,幾多收穫,不小心在Linux這個自由王國暢游了多年。依稀記得剛進入這個世界時的那份恐慌和無知,更慶幸那時的無 ...
  • 修改文件許可權——chmod 修改文件擁有者——chown 修改文件所屬組群——chgrp ...
  • Linux 操作系統在伺服器領域廣泛的使用到,作為一個後臺開發工程師很有必要瞭解Linux相關的知識。 本篇日誌是我學習Linux過程中的簡單記錄和總結。本著“理論夠用,實踐為主”的原則,這裡沒有對Linux底層原理方面的知識做深入研究,只是記錄了相關的基礎概念,有興趣的同學可以參考Linux相關的 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...