在 Istio 最新的 Ambient 模式中,使用了 tproxy 做透明流量劫持(见此博客),这与 Sidecar 模式中基于 IPtables 的流量劫持方式有些许不同,这篇文文章,我们就就一起来探究下什么是 tproxy。
在介绍透明代理之前,我们先了解下什么是代理。
代理在互联网中的用途非常广泛,例如:
代理的分类方式有很多,下图根据代理的位置将其划分为了两类:
代理可能与客户端或服务器位于同一节点(或网络空间,如 Kubernetes 中的 Pod),也可以位于远端。另外还可以根据代理对客户端或服务端是否可见(visible)来分为透明代理和非透明代理。下图展示了客户端(A)通过代理(B)向服务端(C)发送请求的过程。
tproxy
是 Linux 的内核模块(自 Linux 2.2 版本开始引入),用于实现透明代理,其名称中的字母 t
即代表透明(transparent)。
要使用透明代理首先需要把指定的数据包使用 iptables 拦截到指定的网卡上,然后在该网卡监听并转发数据包。
使用 tproxy
实现透明代理的步骤如下:
PREROUTING
链的 mangle
表中创建一个规则,拦截流量发送给 tproxy 处理,例如 iptables -t mangle -A PREROUTING -p tcp -dport 9080 -j TPROXY --on-port 15001 --on-ip 127.0.0.1 --tproxy-mark 0x1/0x1
,给所有目的地为 9080
端口的 TCP 数据包打上标记 1
,你还可以指定来源 IP 地址或者 IP 集,进一步缩小标记范围,tproxy 监听在 15001
端口;1
的数据包查找特定的路由表:例如 ip rule add fwmark 1 lookup 100
,让所有 fwmark
为 1 的数据包查找 100
路由表;ip rule add local 0.0.0.0/0 dev lo table 100
,在 100
路由表中将所有 IPv4 地址声明为本地,当然这只是一个例子,实际使用时需要请将特定的 IP 的数据包转发到本地的 lo
回环网卡;15001
(从 Linux 内核空间进入用户空间),你可以编写网络应用处理数据包或使用 Squid 或 Envoy 等支持 tproxy 的软件来处理数据包;透明代理具有以下优点:
透明代理提供更高的带宽并减少传输延迟,从而提高服务质量;
用户无需配置网络和主机;
企业可以控制对其网络服务的访问;
用户可以通过透明代理连接互联网以绕过一些监管;
透明代理有以下缺点:
透明代理作为代理中的一类重要类型,它的用途广泛,不论是 xray、clash 等代理软件,还是 Istio 服务网格中得使用了应用。了解它的原理和工作方式有助于我们科学正确的使用代理,而是否使用透明代理取决于你对它的信任和了解程度。
最后更新于 2025/01/10