Linux Namespace系列(06):network namespace (CLONE_NEWNET)

Linux Namespace系列(06):network namespace (CLONE_NEWNET)networknames 用来隔离网络设备 IP 地址 端口等 每个 namespace 将会有自己独立的网络栈 路由表 防火墙规则 socket 等 每个新的 networknames 默认有一个本地环回接口 除了 lo 接口外 所有的其他网络设备 物理 虚拟网络接口 网桥等 只能属于一个 networknames 每个 socket 也只能属于一个 networknames

network namespace用来隔离网络设备, IP地址, 端口等. 每个namespace将会有自己独立的网络栈,路由表,防火墙规则,socket等。

每个新的network namespace默认有一个本地环回接口,除了lo接口外,所有的其他网络设备(物理/虚拟网络接口,网桥等)只能属于一个network namespace。每个socket也只能属于一个network namespace。

当新的network namespace被创建时,lo接口默认是关闭的,需要自己手动启动起

标记为”local devices”的设备不能从一个namespace移动到另一个namespace,比如loopback, bridge, ppp等,我们可以通过ethtool -k命令来查看设备的netns-local属性。

#这里“on”表示该设备不能被移动到其他network namespace dev@ubuntu:~$ ethtool -k lo|grep netns-local netns-local: on [fixed]

本篇所有例子都在ubuntu-server-x86_64 16.04下执行通过

示例

本示例将演示如何创建新的network namespace并同外面的namespace进行通信。

#--------------------------第一个shell窗口---------------------- #记录默认network namespace ID dev@ubuntu:~$ readlink /proc/$$/ns/net net:[] #创建新的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:[] #运行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网络做一个更详细的介绍。

ip netns

在单独操作network namespace时,ip netns是一个很方便的工具,并且它可以给namespace取一个名字,然后根据名字来操作namespace。那么给namespace取名字并且根据名字来管理namespace里面的进程是怎么实现的呢?请看下面的脚本(也可以直接看它的源代码):

#开始之前,获取一下默认network namespace的ID dev@ubuntu:~$ readlink /proc/$$/ns/net net:[] #创建一个用于绑定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:[] #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  /var/run/netns/netnamespace1 #退出新创建的bash root@ubuntu:~# exit exit #可以看出netnamespace1的inode没变,说明我们使用了bind mount后 #虽然新的namespace中已经没有进程了,但这个新的namespace还存在 dev@ubuntu:~$ ls -i /var/run/netns/netnamespace1  /var/run/netns/netnamespace1 #上面的这一系列操作等同于执行了命令: ip netns add netnamespace1 #下面的nsenter命令等同于执行了命令: ip netns exec netnamespace1 bash #我们可以通过nsenter命令再创建一个新的bash,并将它加入netnamespace1所关联的namespace(net:[]) dev@ubuntu:~$ sudo nsenter --net=/var/run/netns/netnamespace1 bash root@ubuntu:~# readlink /proc/$$/ns/net net:[]

从上面可以看出,给namespace取名字其实就是创建一个文件,然后通过mount –bind将新创建的namespace文件和该文件绑定,就算该namespace里的所有进程都退出了,内核还是会保留该namespace,以后我们还可以通过这个绑定的文件来加入该namespace。

通过这种办法,我们也可以给其他类型的namespace取名字(有些类型的 namespace可能有些特殊,本人没有一个一个的试过)。

参考

  • Namespaces in operation, part 7: Network namespaces
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/199463.html原文链接:https://javaforall.net

(0)
上一篇 2026年3月20日 下午12:56
下一篇 2026年3月20日 下午12:56


相关推荐

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号