network namespace用来隔离网络设备, IP地址, 端口等. 每一个namespace将会有本身独立的网络栈,路由表,防火墙规则,socket等。node
每一个新的network namespace默认有一个本地环回接口,除了lo接口外,全部的其余网络设备(物理/虚拟网络接口,网桥等)只能属于一个network namespace。每一个socket也只能属于一个network namespace。 git
当新的network namespace被建立时,lo接口默认是关闭的,须要本身手动启动起github
标记为"local devices"的设备不能从一个namespace移动到另外一个namespace,好比loopback, bridge, ppp等,咱们能够经过ethtool -k命令来查看设备的netns-local属性。docker
#这里“on”表示该设备不能被移动到其余network namespace dev@ubuntu:~$ ethtool -k lo|grep netns-local netns-local: on [fixed]
本篇全部例子都在ubuntu-server-x86_64 16.04下执行经过shell
本示例将演示如何建立新的network namespace并同外面的namespace进行通讯。ubuntu
#--------------------------第一个shell窗口---------------------- #记录默认network namespace ID dev@ubuntu:~$ readlink /proc/$$/ns/net net:[4026531957] #建立新的network namespace dev@ubuntu:~$ sudo unshare --uts --net /bin/bash root@ubuntu:~# hostname container001 root@ubuntu:~# exec bash root@container001:~# readlink /proc/$$/ns/net net:[4026532478] #运行ifconfig啥都没有 root@container001:~# ifconfig root@container001:~# #启动lo (这里不详细介绍ip这个tool的用法,请参考man ip) root@container001:~# ip link set lo up root@container001:~# ifconfig lo Link encap:Local Loopback inet addr:127.0.0.1 Mask:255.0.0.0 inet6 addr: ::1/128 Scope:Host UP LOOPBACK RUNNING MTU:65536 Metric:1 RX packets:0 errors:0 dropped:0 overruns:0 frame:0 TX packets:0 errors:0 dropped:0 overruns:0 carrier:0 collisions:0 txqueuelen:1 RX bytes:0 (0.0 B) TX bytes:0 (0.0 B) root@container001:~# ping 127.0.0.1 PING 127.0.0.1 (127.0.0.1) 56(84) bytes of data. 64 bytes from 127.0.0.1: icmp_seq=1 ttl=64 time=0.070 ms 64 bytes from 127.0.0.1: icmp_seq=2 ttl=64 time=0.015 ms #获取当前bash进程的PID root@container001:~# echo $$ 15812 #--------------------------第二个shell窗口---------------------- #建立新的虚拟以太网设备,让两个namespace能通信 dev@ubuntu:~$ sudo ip link add veth0 type veth peer name veth1 #将veth1移动到上面第一个窗口中的namespace #这里15812是上面bash的PID dev@ubuntu:~$ sudo ip link set veth1 netns 15812 #为veth0分配IP并启动veth0 dev@ubuntu:~$ sudo ip address add dev veth0 192.168.8.1/24 dev@ubuntu:~$ sudo ip link set veth0 up dev@ubuntu:~$ ifconfig veth0 veth0 Link encap:Ethernet HWaddr 9a:4d:d5:96:b5:36 inet addr:192.168.8.1 Bcast:0.0.0.0 Mask:255.255.255.0 inet6 addr: fe80::984d:d5ff:fe96:b536/64 Scope:Link UP BROADCAST RUNNING MULTICAST MTU:1500 Metric:1 RX packets:8 errors:0 dropped:0 overruns:0 frame:0 TX packets:8 errors:0 dropped:0 overruns:0 carrier:0 collisions:0 txqueuelen:1000 RX bytes:648 (648.0 B) TX bytes:648 (648.0 B) #--------------------------第一个shell窗口---------------------- #为veth1分配IP地址并启动它 root@container001:~# ip address add dev veth1 192.168.8.2/24 root@container001:~# ip link set veth1 up root@container001:~# ifconfig veth1 veth1 Link encap:Ethernet HWaddr 6a:dc:59:79:3c:8b inet addr:192.168.8.2 Bcast:0.0.0.0 Mask:255.255.255.0 inet6 addr: fe80::68dc:59ff:fe79:3c8b/64 Scope:Link UP BROADCAST RUNNING MULTICAST MTU:1500 Metric:1 RX packets:8 errors:0 dropped:0 overruns:0 frame:0 TX packets:8 errors:0 dropped:0 overruns:0 carrier:0 collisions:0 txqueuelen:1000 RX bytes:648 (648.0 B) TX bytes:648 (648.0 B) #链接成功 root@container001:~# ping 192.168.8.1 PING 192.168.8.1 (192.168.8.1) 56(84) bytes of data. 64 bytes from 192.168.8.1: icmp_seq=1 ttl=64 time=0.098 ms 64 bytes from 192.168.8.1: icmp_seq=2 ttl=64 time=0.023 ms
到目前为止,两个namespace之间能够网络通讯了,但在container001里仍是不能访问外网。下面将经过NAT的方式让container001可以上外网。这部份内容彻底是网络相关的知识,跟namespace已经没什么关系了。安全
#--------------------------第二个shell窗口---------------------- #回到上面示例中的第二个窗口 #确认IP forward是否已经开通,这里1表示开通了 #若是你的机器上是0,请运行这个命令将它改成1: sudo sysctl -w net.ipv4.ip_forward=1 dev@ubuntu:~$ cat /proc/sys/net/ipv4/ip_forward 1 #添加NAT规则,这里ens32是机器上链接外网的网卡 #关于iptables和nat都比较复杂,这里不作解释 dev@ubuntu:~$ sudo iptables -t nat -A POSTROUTING -o ens32 -j MASQUERADE #--------------------------第一个shell窗口---------------------- #回到第一个窗口,添加默认网关 root@container001:~# ip route add default via 192.168.8.1 root@container001:~# route -n Kernel IP routing table Destination Gateway Genmask Flags Metric Ref Use Iface 0.0.0.0 192.168.8.1 0.0.0.0 UG 0 0 0 veth1 192.168.8.0 0.0.0.0 255.255.255.0 U 0 0 0 veth1 #这样就能够访问外网了 #因为测试环境的限制,因此采用下面的方式检测网络是否畅通 #若是网络没有什么限制的话,随便ping一个外部的IP测试就能够了 root@container001:~# curl -I www.google.com HTTP/1.1 200 OK Date: Fri, 15 Jul 2016 08:12:03 GMT
network namespace的概念比较简单,但如何作好网络的隔离和连通却比较难,包括性能和安全相关的考虑,须要很好的Linux网络知识。后续在介绍docker网络管理的时候会对Linux网络作一个更详细的介绍。bash
在单独操做network namespace时,ip netns是一个很方便的工具,而且它能够给namespace取一个名字,而后根据名字来操做namespace。那么给namespace取名字而且根据名字来管理namespace里面的进程是怎么实现的呢?请看下面的脚本(也能够直接看它的源代码):网络
#开始以前,获取一下默认network namespace的ID dev@ubuntu:~$ readlink /proc/$$/ns/net net:[4026531957] #建立一个用于绑定network namespace的文件, #ip netns将全部的文件放到了目录/var/run/netns下, #因此咱们这里重用这个目录,而且建立一个咱们本身的文件netnamespace1 dev@ubuntu:~$ sudo mkdir -p /var/run/netns dev@ubuntu:~$ sudo touch /var/run/netns/netnamespace1 #建立新的network namespace,并在新的namespace中启动新的bash dev@ubuntu:~$ sudo unshare --net bash #查看新的namespace ID root@ubuntu:~# readlink /proc/$$/ns/net net:[4026532448] #bind当前bash的namespace文件到上面建立的文件上 root@ubuntu:~# mount --bind /proc/$$/ns/net /var/run/netns/netnamespace1 #经过ls -i命令能够看到文件netnamespace1的inode号和namespace的编号相同,说明绑定成功 root@ubuntu:~# ls -i /var/run/netns/netnamespace1 4026532448 /var/run/netns/netnamespace1 #退出新建立的bash root@ubuntu:~# exit exit #能够看出netnamespace1的inode没变,说明咱们使用了bind mount后 #虽然新的namespace中已经没有进程了,但这个新的namespace还存在 dev@ubuntu:~$ ls -i /var/run/netns/netnamespace1 4026532448 /var/run/netns/netnamespace1 #上面的这一系列操做等同于执行了命令: ip netns add netnamespace1 #下面的nsenter命令等同于执行了命令: ip netns exec netnamespace1 bash #咱们能够经过nsenter命令再建立一个新的bash,并将它加入netnamespace1所关联的namespace(net:[4026532448]) dev@ubuntu:~$ sudo nsenter --net=/var/run/netns/netnamespace1 bash root@ubuntu:~# readlink /proc/$$/ns/net net:[4026532448]
从上面能够看出,给namespace取名字其实就是建立一个文件,而后经过mount --bind将新建立的namespace文件和该文件绑定,就算该namespace里的全部进程都退出了,内核仍是会保留该namespace,之后咱们还能够经过这个绑定的文件来加入该namespace。curl
经过这种办法,咱们也能够给其余类型的namespace取名字(有些类型的 namespace可能有些特殊,本人没有一个一个的试过)。