Docker基礎(chǔ)知識之Linux namespace圖文詳解
前言
Docker 是“新瓶裝舊酒”的產(chǎn)物,依賴于 Linux 內(nèi)核技術(shù) chroot 、namespace 和 cgroup。本篇先來看 namespace 技術(shù)。
Docker 和虛擬機技術(shù)一樣,從操作系統(tǒng)級上實現(xiàn)了資源的隔離,它本質(zhì)上是宿主機上的進程(容器進程),所以資源隔離主要就是指進程資源的隔離。實現(xiàn)資源隔離的核心技術(shù)就是 Linux namespace。這技術(shù)和很多語言的命名空間的設(shè)計思想是一致的(如 C++ 的 namespace)。
隔離意味著可以抽象出多個輕量級的內(nèi)核(容器進程),這些進程可以充分利用宿主機的資源,宿主機有的資源容器進程都可以享有,但彼此之間是隔離的,同樣,不同容器進程之間使用資源也是隔離的,這樣,彼此之間進行相同的操作,都不會互相干擾,安全性得到保障。
為了支持這些特性,Linux namespace 實現(xiàn)了 6 項資源隔離,基本上涵蓋了一個小型操作系統(tǒng)的運行要素,包括主機名、用戶權(quán)限、文件系統(tǒng)、網(wǎng)絡(luò)、進程號、進程間通信。

這 6 項資源隔離分別對應(yīng) 6 種系統(tǒng)調(diào)用,通過傳入上表中的參數(shù),調(diào)用 clone() 函數(shù)來完成。
int clone(int (*child_func)(void *), void *child_stack, int flags, void *arg);
clone() 函數(shù)相信大家都不陌生了,它是 fork() 函數(shù)更通用的實現(xiàn)方式,通過調(diào)用 clone() ,并傳入需要隔離資源對應(yīng)的參數(shù),就可以建立一個容器了(隔離什么我們自己控制)。
一個容器進程也可以再 clone() 出一個容器進程,這是容器的嵌套。

如果想要查看當(dāng)前進程下有哪些 namespace 隔離,可以查看文件 /proc/[pid]/ns (注:該方法僅限于 3.8 版本以后的內(nèi)核)。

可以看到,每一項 namespace 都附帶一個編號,這是唯一標(biāo)識 namespace 的,如果兩個進程指向的 namespace 編號相同,則表示它們同在該 namespace 下。同時也注意到,多了一個 cgroup,這個 namespace 是 4.6 版本的內(nèi)核才支持的。Docker 目前對它的支持普及度還不高。所以我們暫時先不考慮它。
下面通過簡單的代碼來實現(xiàn) 6 種 namespace 的隔離效果,讓大家有個直觀的印象。
UTS namespace
UTS namespace 提供了主機名和域名的隔離,這樣每個容器就擁有獨立的主機名和域名了,在網(wǎng)絡(luò)上就可以被視為一個獨立的節(jié)點,在容器中對 hostname 的命名不會對宿主機造成任何影響。
首先,先看總體的代碼骨架:
#define _GNU_SOURCE
#include <sys/types.h>
#include <sys/wait.h>
#include <stdio.h>
#include <sched.h>
#include <signal.h>
#include <unistd.h>
#define STACK_SIZE (1024 * 1024)
static char container_stack[STACK_SIZE];
char* const container_args[] = {
"/bin/bash",
NULL
};
// 容器進程運行的程序主函數(shù)
int container_main(void *args)
{
printf("在容器進程中!\n");
execv(container_args[0], container_args); // 執(zhí)行/bin/bash return 1;
}
int main(int args, char *argv[])
{
printf("程序開始\n");
// clone 容器進程
int container_pid = clone(container_main, container_stack + STACK_SIZE, SIGCHLD, NULL);
// 等待容器進程結(jié)束
waitpid(container_pid, NULL, 0);
return 0;
}
該程序骨架調(diào)用 clone() 函數(shù)實現(xiàn)了子進程的創(chuàng)建工作,并定義子進程的執(zhí)行函數(shù),clone() 第二個參數(shù)指定了子進程運行的??臻g大小,第三個參數(shù)即為創(chuàng)建不同 namespace 隔離的關(guān)鍵。
對于 UTS namespace,傳入 CLONE_NEWUTS,如下:
int container_pid = clone(container_main, container_stack + STACK_SIZE, SIGCHLD | CLONE_NEWUTS, NULL);
為了能夠看出容器內(nèi)和容器外主機名的變化,我們子進程執(zhí)行函數(shù)中加入:
sethostname("container", 9);
最終運行可以看到效果如下:

IPC namespace
IPC namespace 實現(xiàn)了進程間通信的隔離,包括常見的幾種進程間通信機制,如信號量,消息隊列和共享內(nèi)存。我們知道,要完成 IPC,需要申請一個全局唯一的標(biāo)識符,即 IPC 標(biāo)識符,所以 IPC 資源隔離主要完成的就是隔離 IPC 標(biāo)識符。
同樣,代碼修改僅需要加入?yún)?shù) CLONE_NEWIPC 即可,如下:
int container_pid = clone(container_main, container_stack + STACK_SIZE, SIGCHLD | CLONE_NEWUTS | CLONE_NEWIPC, NULL);
為了看出變化,首先在宿主機上建立一個消息隊列:

然后運行程序,進入容器查看 IPC,沒有找到原先建立的 IPC 標(biāo)識,達到了 IPC 隔離。

PID namespace
PID namespace 完成的是進程號的隔離,同樣在 clone() 中加入 CLONE_NEWPID 參數(shù),如:
int container_pid = clone(container_main, container_stack + STACK_SIZE, SIGCHLD | CLONE_NEWUTS | CLONE_NEWIPC | CLONE_NEWPID, NULL);
效果如下,echo $$ 輸出 shell 的 PID 號,發(fā)生了變化。

但是對于 ps/top 之類命令卻沒有改變:

具體的原因和接下來的內(nèi)容(包括 mount namespace,network namespace 和 user namespace),大家可以關(guān)注我的公眾號閱讀,那里的閱讀體驗會更好一些。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關(guān)文章
docker容器下配置jupyter notebook的操作
這篇文章主要介紹了docker容器下配置jupyter notebook的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-03-03
解決docker CMD/ENTRYPOINT執(zhí)行sh腳本報: not found/run.sh:
這篇文章主要介紹了解決docker CMD/ENTRYPOINT執(zhí)行sh腳本報: not found/run.sh:的問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-11-11
Docker容器與主機間的文件傳輸方法(復(fù)制/上傳/下載)
這篇文章主要介紹了Docker容器與主機間的文件傳輸方法(復(fù)制/上傳/下載),需要的朋友可以參考下2018-02-02
jenkins+gitlab+nginx部署前端應(yīng)用實現(xiàn)
在日常開發(fā)中,往往可能同時多個項目并行進行開發(fā),本文介紹了jenkins+gitlab+nginx部署前端應(yīng)用實現(xiàn),感興趣的可以了解一下2021-05-05

