容器基础(二): 使用Namespace进行边界隔离
Linux Namespace
容器技术可以认为是一种沙盒(sandbox), 为了实现沙盒/容器/应用间的隔离,就需要一种技术来对容器界定边界,从而让容器不至于互相干扰。当前使用的技术就是Namespace。Namespace定义如下:
Namespace是Linux 内核用来隔离内核资源的方式, 是对全局系统资源的一种封装隔离,使得处于不同namespace的进程拥有独立的全局系统资源,改变一个namespace中的系统资源只会影响当前namespace里的进程,对其他namespace中的进程没有影响。
目前, Linux内核里面实现了7种不同类型的namespace:
名称 宏定义 隔离内容
Cgroup CLONE_NEWCGROUP Cgroup root directory (since Linux 4.6)
IPC CLONE_NEWIPC System V IPC, POSIX message queues (since Linux 2.6.)
Network CLONE_NEWNET Network devices, stacks, ports, etc. (since Linux 2.6.)
Mount CLONE_NEWNS Mount points (since Linux 2.4.)
PID CLONE_NEWPID Process IDs (since Linux 2.6.)
User CLONE_NEWUSER User and group IDs (started in Linux 2.6. and completed in Linux 3.8)
UTS CLONE_NEWUTS Hostname and NIS domain name (since Linux 2.6.)
和namespace相关的函数有三个:
clone: 创建一个新的进程并把他放到新的namespace中
int clone(int (*child_func)(void *), void *child_stack
, int flags, void *arg); flags:
指定一个或者多个上面的CLONE_NEW*(当然也可以包含跟namespace无关的flags),
这样就会创建一个或多个新的不同类型的namespace, 并把新创建的子进程加入新创建的这些namespace中。
setns: 将当前进程加入到已有的namespace中
int setns(int fd, int nstype); fd:
指向/proc/[pid]/ns/目录里相应namespace对应的文件,
表示要加入哪个namespace nstype:
指定namespace的类型(上面的任意一个CLONE_NEW*):
. 如果当前进程不能根据fd得到它的类型,如fd由其他进程创建,并通过UNIX domain socket传给当前进程,那么就需要通过nstype来指定fd指向的namespace的类型
. 如果进程能根据fd得到namespace类型,比如这个fd是由当前进程打开的,那么nstype设置为0即可
unshare: 使当前进程退出指定类型的namespace,并加入到新创建的namespace(相当于创建并加入新的namespace)
int unshare(int flags); flags:
指定一个或者多个上面的CLONE_NEW*, 这样当前进程就退出了当前指定类型的namespace并加入到新创建的namespace
clone和unshare的功能都是创建并加入新的namespace, 他们的区别是:
> unshare是使当前进程加入新的namespace
> clone是创建一个新的子进程,然后让子进程加入新的namespace,而当前进程保持不变
简单的示例程序如下,clone时指定CLONE_NEWIPC/CLONE_NEWPID等参数, 由于没有自定义文件系统, 这个示例程序不使用chroot/pivot_root:
➜ cat namespace.c
#define _GNU_SOURCE
#include <sched.h>
#include <sys/wait.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/mount.h> #define CHECK_EXIST(ret_val, error_msg) \
do { \
if (ret_val == -) { \
perror(error_msg); \
exit(-); \
} \
} while () static int child_func(void *hostname) {
// set hostname
CHECK_EXIST(sethostname((char *)hostname, strlen((char *)hostname)), "sethostname"); // 挂载proc, 后续ps命令则只能看到自身进程
CHECK_EXIST(mount("proc", "/proc", "proc", , NULL), "mount-proc"); // exec /bin/bash and wait input
CHECK_EXIST(execlp("bash", "bash", (char *) NULL), "execlp"); return ;
} //子进程的栈空间
static char s_child_stack[ * ]; int main(int argc, char *argv[]) {
pid_t child_pid; if (argc < ) {
printf("Usage: %s <child-hostname>\n", argv[]);
return -;
} /*
* 创建并启动子进程并传入hostname给子进程
*
* flags参数如下:
* 设置CLONE_NEWPID设置pid namespace
* 设置CLONE_NEWUTS设置新的UTS namespace,
* 设置子进程退出后返回给父进程的信号为SIGCHLD, 跟namespace无关
*/
child_pid = clone(
child_func,
//栈是从高位向低位增长,所以这里要指向高位地址
s_child_stack + sizeof(s_child_stack),
CLONE_NEWNS | CLONE_NEWUTS | CLONE_NEWPID | CLONE_NEWIPC | SIGCHLD,
argv[]); CHECK_EXIST(child_pid, "clone"); waitpid(child_pid, NULL, ); return ;
}
➜
从结果来看:
指定CLONE_NEWPID, 然后子进程挂载proc,子进程里面就只能看到自己的进程,且第一个bash的pid=1;
指定CLONE_NEWIPC, 父子进程的IPC也被隔离了;
➜ ipcs -q
--------- 消息队列 -----------
键        msqid      拥有者  权限     已用字节数 消息
➜ ipcmk -Q
消息队列 id:0
➜ ipcs -q
--------- 消息队列 -----------
键        msqid      拥有者  权限     已用字节数 消息      
0x34d432ae 0          root       644        0            0
➜ ./namespace sdocker
root@sdocker:/home/jason/demo/namespace# hostname
sdocker
root@sdocker:/home/jason/demo/namespace# ps -ef
UID        PID  PPID  C STIME TTY          TIME CMD
root         1     0  0 00:39 pts/0    00:00:00 bash
root         3     1  0 00:39 pts/0    00:00:00 ps -ef
root@sdocker:/home/jason/demo/namespace# ipcs -q
--------- 消息队列 -----------
键        msqid      拥有者  权限     已用字节数 消息
root@sdocker:/home/jason/demo/namespace# exit
exit
➜ ipcs -q
--------- 消息队列 -----------
键        msqid      拥有者  权限     已用字节数 消息      
0x34d432ae 0          root       644        0            0
➜
容器基础(二): 使用Namespace进行边界隔离的更多相关文章
- C++ 顺序容器基础知识总结
		0.前言 本文简单地总结了STL的顺序容器的知识点.文中并不涉及具体的实现技巧,对于细节的东西也没有提及.一来不同的标准库有着不同的实现,二来关于具体实现<STL源码剖析>已经展示得全面细 ... 
- docker容器基础
		一.docker容器基础6种名称空间:UTS.MOunt.IPC.PID.User.Net (1) Linux Namespaces:namespace 系统调用参数 隔离内容 内核版本 UTS ... 
- 容器的进程与namespace、rootfs
		一:容器是什么 容器的本质是一种特殊的进程. 在linux容器中有三个重要的概念:Namespace.Cgroups.rootfs. Namespace做隔离,让进程只能看到Namespace中的世界 ... 
- 【原创】探索容器底层知识之Namespace
		一.先谈谈进程 在正式介绍Namespace之前,先介绍下进程,因为容器本质上是进程,但是在介绍进程之前,先理清下“程序”和“进程”的关系,这是IT从业人员在日常工作中经常碰到的两个词汇,举个通俗点的 ... 
- Bootstrap <基础二十九>面板(Panels)
		Bootstrap 面板(Panels).面板组件用于把 DOM 组件插入到一个盒子中.创建一个基本的面板,只需要向 <div> 元素添加 class .panel 和 class .pa ... 
- Bootstrap <基础二十二>超大屏幕(Jumbotron)
		Bootstrap 支持的另一个特性,超大屏幕(Jumbotron).顾名思义该组件可以增加标题的大小,并为登陆页面内容添加更多的外边距(margin).使用超大屏幕(Jumbotron)的步骤如下: ... 
- Bootstrap<基础二> 网格系统
		Bootstrap 提供了一套响应式.移动设备优先的流式网格系统,随着屏幕或视口(viewport)尺寸的增加,系统会自动分为最多12列. 什么是网格(Grid)? 摘自维基百科: 在平面设计中,网格 ... 
- 2、JavaScript 基础二 (从零学习JavaScript)
		11.强制转换 强制转换主要指使用Number.String和Boolean三个构造函数,手动将各种类型的值,转换成数字.字符串或者布尔值. 1>Number强制转换 参数为原始类型值的转换规 ... 
- day  66  Django基础二之URL路由系统
		Django基础二之URL路由系统 本节目录 一 URL配置 二 正则表达式详解 三 分组命名匹配 四 命名URL(别名)和URL反向解析 五 命名空间模式 一 URL配置 Django 1.11 ... 
随机推荐
- 的NodeJS异步数据库函数需要同步的答案 +
			我是新来的NodeJS和我写,需要从我去过的所有的函数应该是在这种情况下,读QUERY我的MySQL数据库,并返回代码,我非常希望服务器能够对其他事件作出回应而这个地方是轨迹查询请求.然而,它并不特别 ... 
- app上线
			不管第一次还是第二次APP上线都需要三样东西:开发者证书,appID,描述文件 
- JS将unicode码转中文方法
			原理,将unicode的 \u 先转为 %u,然后使用unescape方法转换为中文. ? 1 2 3 4 <script type="text/javascript"> ... 
- IE 兼容模式 设置 Meta Compatible 和 Iframe 子页面的关系
			背景 因为历史原因,之前很多的系统都会是 顶级页面+Iframe来加载子级页面的这种模式构件系统,而且系统都只能运行在IE6或者IE 高版本兼容模式下(IE 7模式). 随着现在的审美原来越高,脚本能 ... 
- sudo 密码超时时间
			Centos 没有默认超时时间,所以用一次sudo就需要输入密码. vi /etc/sudoers 添加下面的内容,2表示分钟数(看自己需求更改). Defaults timestamp_timeou ... 
- 谷歌angle库使用心得
			通过谷歌的angle库可以在项目中,调用opengl接口渲染时,选择调用directx或者webgl来渲染,避免机器没有安装opengl驱动启动异常的问题. 这个库的使用可以不修改原有使用opengl ... 
- BZOJ1053: [HAOI2007]反素数ant(爆搜)
			Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 4163 Solved: 2485[Submit][Status][Discuss] Descript ... 
- java使用apache-poi生成excel表格
			public static void main(String[] args) throws IOException { // TODO Auto-generated method stub //创建一 ... 
- MySQL超大表如何提高count速度
			经常用到count统计记录数,表又超级大,这时候sql执行很慢,就是走索引,也是很慢的,怎么办呢? 1.这个时候我们就要想为什么这么慢:根本原因是访问的数据量太大,就算只计算记录数也是很慢的. 2.如 ... 
- javascript遍历方法总结
			forEach 循环 JavaScript诞生已经有20多年了,我们一直使用的用来循环一个数组的方法是这样的: for (var index = 0; index < myArray.lengt ... 
