docker_practice/advanced_network/port_mapping.md

1.8 KiB
Raw Blame History

##映射容器端口到宿主主机的实现

默认情况下,容器可以主动访问到外部网络的连接,但是外部网络无法访问到容器。

容器访问外部实现

容器所有到外部网络的连接源地址都会被NAT成本地系统的IP地址。这是使用iptables的源地址伪装操作实现的。

查看主机的NAT规则。

$ sudo iptables -t nat -nL
...
Chain POSTROUTING (policy ACCEPT)
target     prot opt source               destination
MASQUERADE  all  --  172.17.0.0/16       !172.17.0.0/16
...

其中,上述规则将所有源地址在172.17.0.0/16网段目标地址为其他网段外部网络的流量动态伪装为从系统网卡发出。MASQUERADE跟传统SNAT的好处是它能动态从网卡获取地址。

外部访问容器实现

容器允许外部访问,可以在docker run时候通过-p-P参数来启用。

不管用那种办法,其实也是在本地的iptable的nat表中添加相应的规则。

使用-P时:

$ iptables -t nat -nL
...
Chain DOCKER (2 references)
target     prot opt source               destination
DNAT       tcp  --  0.0.0.0/0            0.0.0.0/0            tcp dpt:49153 to:172.17.0.2:80

使用-p 80:80时:

$ iptables -t nat -nL
Chain DOCKER (2 references)
target     prot opt source               destination
DNAT       tcp  --  0.0.0.0/0            0.0.0.0/0            tcp dpt:80 to:172.17.0.2:80

注意:

  • 这里的规则映射了0.0.0.0,意味着将接受主机来自所有接口的流量。用户可以通过-p IP:host_port:container_port-p IP::port来指定允许访问容器的主机上的IP、接口等以制定更严格的规则。
  • 如果希望永久绑定到某个固定的IP地址可以在Docker 配置文件/etc/default/docker中指定DOCKER_OPTS="--ip=IP_ADDRESS"之后重启Docker服务即可生效。