Featured image of post iptables 命令、规则、参数详解

iptables 命令、规则、参数详解

防火墙相关概念

此处先描述一些相关概念。

从逻辑上讲。防火墙可以大体分为主机防火墙和网络防火墙。
主机防火墙: 针对于单个主机进行防护。
网络防火墙: 往往处于网络入口或边缘, 针对于网络入口进行防护, 服务于防火墙背后的本地局域网。

网络防火墙和主机防火墙并不冲突, 可以理解为, 网络防火墙主外 (集体), 主机防火墙主内 (个人)。

从物理上讲, 防火墙可以分为硬件防火墙和软件防火墙。
硬件防火墙: 在硬件级别实现部分防火墙功能, 另一部分功能基于软件实现, 性能高, 成本高。
软件防火墙: 应用软件处理逻辑运行于通用硬件平台之上的防火墙, 性能低, 成本低。

那么在此处, 我们就来聊聊 Linux 的 iptables

iptables 其实不是真正的防火墙, 我们可以把它理解成一个客户端代理, 用户通过 iptables 这个代理, 将用户的安全设定执行到对应的"安全框架"中, 这个"安全框架"才是真正的防火墙, 这个框架的名字叫 netfilter

netfilter 才是防火墙真正的安全框架 (framework), netfilter 位于内核空间。
iptables 其实是一个命令行工具, 位于用户空间, 我们用这个工具操作真正的框架。

netfilter/iptables (下文中简称为iptables) 组成 Linux 平台下的包过滤防火墙, 与大多数的Linux软件一样, 这个包过滤防火墙是免费的, 它可以代替昂贵的商业防火墙解决方案, 完成封包过滤、封包重定向和网络地址转换 (NAT) 等功能。

Netfilter 是 Linux 操作系统核心层内部的一个数据包处理模块, 它具有如下功能:
网络地址转换 (Network Address Translate)
数据包内容修改
以及数据包过滤的防火墙功能

所以说, 虽然我们使用 service iptables start 启动 iptables “服务”, 但是其实准确的来说, iptables 并没有一个守护进程, 所以并不能算是真正意义上的服务, 而应该算是内核提供的功能。

iptables 基础

我们知道 iptables 是按照规则来办事的, 我们就来说说规则 (rules), 规则其实就是网络管理员预定义的条件, 规则一般的定义为"如果数据包头符合这样的条件, 就这样处理这个数据包"。规则存储在内核空间的信息包过滤表中, 这些规则分别指定了源地址、目的地址、传输协议 (如TCP、UDP、ICMP) 和服务类型 (如HTTP、FTP和SMTP) 等。当数据包与规则匹配时, iptables就根据规则所定义的方法来处理这些数据包, 如放行 (accept) 、拒绝 (reject) 和丢弃 (drop) 等。配置防火墙的主要工作就是添加、修改和删除这些规则。

这样说可能并不容易理解, 我们来换个容易理解的角度, 从头说起.

当客户端访问服务器的web服务时, 客户端发送报文到网卡, 而 tcp/ip 协议栈是属于内核的一部分, 所以, 客户端的信息会通过内核的 TCP 协议传输到用户空间中的 web 服务中, 而此时, 客户端报文的目标终点为 web 服务所监听的套接字 (IP: Port) 上, 当web服务需要响应客户端请求时, web服务发出的响应报文的目标终点则为客户端, 这个时候, web服务所监听的IP与端口反而变成了原点, 我们说过, netfilter 才是真正的防火墙, 它是内核的一部分, 所以, 如果我们想要防火墙能够达到"防火"的目的, 则需要在内核中设置关卡, 所有进出的报文都要通过这些关卡, 经过检查后, 符合放行条件的才能放行, 符合阻拦条件的则需要被阻止, 于是, 就出现了 input 关卡和 output 关卡, 而这些关卡在 iptables 中不被称为"关卡", 而被称为"链"。

021217_0051_1.png

其实我们上面描述的场景并不完善, 因为客户端发来的报文访问的目标地址可能并不是本机, 而是其他服务器, 当本机的内核支持 IP_FORWARD 时, 我们可以将报文转发给其他服务器, 所以, 这个时候, 我们就会提到iptables中的其他"关卡", 也就是其他"链", 他们就是 “路由前”、“转发”、“路由后”, 他们的英文名是

PREROUTING、FORWARD、POSTROUTING

也就是说, 当我们启用了防火墙功能时, 报文需要经过如下关卡, 也就是说, 根据实际情况的不同, 报文经过"链"可能不同。如果报文需要转发, 那么报文则不会经过 input 链发往用户空间, 而是直接在内核空间中经过 forward 链和 postrouting 链转发出去的。

021217_0051_2.png

所以, 根据上图, 我们能够想象出某些常用场景中, 报文的流向:

到本机某进程的报文: PREROUTING –> INPUT
由本机转发的报文: PREROUTING –> FORWARD –> POSTROUTING
由本机的某进程发出报文 (通常为响应报文): OUTPUT –> POSTROUTING

链的概念

现在, 我们想象一下, 这些"关卡"在 iptables 中为什么被称作"链"呢?我们知道, 防火墙的作用就在于对经过的报文匹配"规则", 然后执行对应的"动作",所以, 当报文经过这些关卡的时候, 则必须匹配这个关卡上的规则, 但是, 这个关卡上可能不止有一条规则, 而是有很多条规则, 当我们把这些规则串到一个链条上的时候, 就形成了"链",所以, 我们把每一个"关卡"想象成如下图中的模样 , 这样来说, 把他们称为"链"更为合适, 每个经过这个"关卡"的报文, 都要将这条"链"上的所有规则匹配一遍, 如果有符合条件的规则, 则执行规则对应的动作。

021217_0051_3.png

表的概念

我们再想想另外一个问题, 我们对每个"链"上都放置了一串规则, 但是这些规则有些很相似, 比如, A类规则都是对IP或者端口的过滤, B类规则是修改报文, 那么这个时候, 我们是不是能把实现相同功能的规则放在一起呢, 必须能的。

我们把具有相同功能的规则的集合叫做"表", 所以说, 不同功能的规则, 我们可以放置在不同的表中进行管理, 而iptables已经为我们定义了4种表, 每种表对应了不同的功能, 而我们定义的规则也都逃脱不了这4种功能的范围, 所以, 学习iptables之前, 我们必须先搞明白每种表 的作用。

iptables 为我们提供了如下规则的分类, 或者说, iptables为我们提供了如下"表"

  • filter表: 负责过滤功能, 防火墙;内核模块: iptables_filter
  • nat表: network address translation, 网络地址转换功能;内核模块: iptable_nat
  • mangle表: 拆解报文, 做出修改, 并重新封装 的功能;iptable_mangle
  • raw表: 关闭nat表上启用的连接追踪机制;iptable_raw

也就是说, 我们自定义的所有规则, 都是这四种分类中的规则, 或者说, 所有规则都存在于这4张"表"中。

表链关系

但是我们需要注意的是, 某些"链"中注定不会包含"某类规则", 就像某些"关卡"天生就不具备某些功能一样, 比如, A"关卡"只负责打击陆地敌人, 没有防空能力, B"关卡"只负责打击空中敌人, 没有防御步兵的能力, C"关卡"可能比较NB, 既能防空, 也能防御陆地敌人, D"关卡"最屌, 海陆空都能防。

那让我们来看看, 每个"关卡"都有哪些能力, 或者说, 让我们看看每个"链"上的规则都存在于哪些"表"中。
我们还是以图为例, 先看看prerouting"链"上的规则都存在于哪些表中。

注意: 下图只用于说明prerouting链上的规则存在于哪些表中, 并没有描述表的顺序。

021217_0051_4.png

这幅图是什么意思呢?它的意思是说, prerouting"链"只拥有nat 表、raw 表和 mangle 表所对应的功能, 所以, prerouting 中的规则只能存放于 nat 表、raw 表和 mangle 表中。

那么, 根据上述思路, 我们来总结一下, 每个"关卡"都拥有什么功能, 或者说, 每个"链"中的规则都存在于哪些"表"中。

  • PREROUTING 的规则可以存在于: raw表, mangle表, nat表。
  • INPUT 的规则可以存在于: mangle表, filter表, (centos7 中还有 nat 表, centos6 中没有)。
  • FORWARD 的规则可以存在于: mangle表, filter表。
  • OUTPUT 的规则可以存在于: raw表mangle表, nat表, filter表。
  • POSTROUTING 的规则可以存在于: mangle表, nat表。

但是, 我们在实际的使用过程中, 往往是通过"表"作为操作入口, 对规则进行定义的, 之所以按照上述过程介绍 iptables, 是因为从 “关卡” 的角度更容易从入门的角度理解, 但是为了以便在实际使用的时候, 更加顺畅的理解它们, 此处我们还要将各"表"与"链"的关系罗列出来

表 (功能) <–> 链 (钩子):

raw 表中的规则可以被哪些链使用: PREROUTING, OUTPUT
mangle 表中的规则可以被哪些链使用: PREROUTING, INPUT, FORWARD, OUTPUT, POSTROUTING
nat 表中的规则可以被哪些链使用: PREROUTING, OUTPUT, POSTROUTING (centos7 中还有 INPUT, centos6中没有)
filter 表中的规则可以被哪些链使用: INPUT, FORWARD, OUTPUT

其实我们还需要注意一点, 因为数据包经过一个"链"的时候, 会将当前链的所有规则都匹配一遍, 但是匹配时总归要有顺序, 我们应该一条一条的去匹配, 而且我们说过, 相同功能类型的规则会汇聚在一张"表"中, 那么, 哪些"表"中的规则会放在"链"的最前面执行呢, 这时候就需要有一个优先级的问题, 我们还拿 prerouting “链"做图示。

021217_0051_5.png

prerouting 链中的规则存放于三张表中, 而这三张表中的规则执行的优先级如下:
raw –> mangle –> nat

但是我们知道, iptables为我们定义了4张"表”,当他们处于同一条"链"时, 执行的优先级如下。
优先级次序 (由高而低):
raw –> mangle –> nat –> filter

但是我们前面说过, 某些链天生就不能使用某些表中的规则, 所以, 4张表中的规则处于同一条链的目前只有output链, 它就是传说中海陆空都能防守的关卡。

为了更方便的管理, 我们还可以在某个表里面创建自定义链, 将针对某个应用程序所设置的规则放置在这个自定义链中, 但是自定义链接不能直接使用, 只能被某个默认的链当做动作去调用才能起作用, 我们可以这样想象, 自定义链就是一段比较"短"的链子, 这条"短"链子上的规则都是针对某个应用程序制定的, 但是这条短的链子并不能直接使用, 而是需要"焊接"在 iptables 默认定义链子上, 才能被 IPtables 使用, 这就是为什么默认定义的"链"需要把"自定义链"当做"动作"去引用的原因。这是后话, 后面再聊, 在实际使用时我们即可更加的明白。

数据经过防火墙的流程

结合上述所有的描述, 我们可以将数据包通过防火墙的流程总结为下图:

021217_0051_6.png

我们在写Iptables规则的时候, 要时刻牢记这张路由次序图, 灵活配置规则。

我们将经常用到的对应关系重新写在此处, 方便对应图例查看。

链的规则存放于哪些表中 (从链到表的对应关系):

  • PREROUTING 的规则可以存在于: raw表, mangle表, nat表。
  • INPUT 的规则可以存在于: mangle表, filter表, (centos7 中还有 nat 表, centos6 中没有)。
  • FORWARD 的规则可以存在于: mangle表, filter表。
  • OUTPUT 的规则可以存在于: raw表mangle表, nat表, filter表。
  • POSTROUTING 的规则可以存在于: mangle表, nat表。

表中的规则可以被哪些链使用 (从表到链的对应关系):

  • raw 表中的规则可以被哪些链使用: PREROUTING, OUTPUT
  • mangle 表中的规则可以被哪些链使用: PREROUTING, INPUT, FORWARD, OUTPUT, POSTROUTING
  • nat 表中的规则可以被哪些链使用: PREROUTING, OUTPUT, POSTROUTING (centos7中还有INPUT, centos6中没有)
  • filter 表中的规则可以被哪些链使用: INPUT, FORWARD, OUTPUT

下图中 nat 表在 centos7 中的情况就不再标明。

021217_0051_7.png

规则的概念

说了一圈又说回来了, 在上述描述中我们一直在提规则, 可是没有细说, 现在说说它。

先说说规则的概念, 然后再通俗的解释它。

规则: 根据指定的匹配条件来尝试匹配每个流经此处的报文, 一旦匹配成功, 则由规则后面指定的处理动作进行处理;

那么我们来通俗的解释一下什么是 iptables 的规则, 之前打过一个比方, 每条"链"都是一个"关卡", 每个通过这个"关卡"的报文都要匹配这个关卡上的规则, 如果匹配, 则对报文进行对应的处理, 比如说, 你我二人此刻就好像两个"报文", 你我二人此刻都要入关, 可是城主有命, 只有器宇轩昂的人才能入关, 不符合此条件的人不能入关, 于是守关将士按照城主制定的"规则", 开始打量你我二人, 最终, 你顺利入关了, 而我已被拒之门外, 因为你符合"器宇轩昂"的标准, 所以把你"放行"了, 而我不符合标准, 所以没有被放行, 其实, “器宇轩昂"就是一种"匹配条件”, “放行"就是一种"动作”, “匹配条件"与"动作"组成了规则。

了解了规则的概念, 那我们来聊聊规则的组成部分,此处只是大概的将规则的结构列出, 后面的文章中会单独对规则进行总结。

规则由匹配条件和处理动作组成。

匹配条件

匹配条件分为基本匹配条件与扩展匹配条件

基本匹配条件:
源地址 Source IP, 目标地址 Destination IP

上述内容都可以作为基本匹配条件。

扩展匹配条件:

除了上述的条件可以用于匹配, 还有很多其他的条件可以用于匹配, 这些条件泛称为扩展条件, 这些扩展条件其实也是 netfilter 中的一部分, 只是以模块的形式存在, 如果想要使用这些条件, 则需要依赖对应的扩展模块。

源端口 Source Port, 目标端口 Destination Port

上述内容都可以作为扩展匹配条件

处理动作

处理动作在iptables中被称为target (这样说并不准确, 我们暂且这样称呼), 动作也可以分为基本动作和扩展动作。

此处列出一些常用的动作, 之后的文章会对它们进行详细的示例与总结:

  • ACCEPT: 允许数据包通过。
  • DROP: 直接丢弃数据包, 不给任何回应信息, 这时候客户端会感觉自己的请求泥牛入海了, 过了超时时间才会有反应。
  • REJECT: 拒绝数据包通过, 必要时会给数据发送端一个响应的信息, 客户端刚请求就会收到拒绝的信息。
  • SNAT: 源地址转换, 解决内网用户用同一个公网地址上网的问题。
  • MASQUERADE: 是SNAT的一种特殊形式, 适用于动态的、临时会变的ip上。
  • DNAT: 目标地址转换。
  • REDIRECT: 在本机做端口映射。
  • LOG: 在/var/log/messages文件中记录日志信息, 然后将数据包传递给下一条规则, 也就是说除了记录以外不对数据包做任何其他操作, 仍然让下一条规则去匹配。

表 (table)

包含4个表:

4个表的优先级由高到低: raw–>mangle–>nat–>filter

  • raw: RAW 表只使用在 PREROUTING 链 和 OUTPUT 链上,因为优先级最高, 从而可以对收到的数据包在连接跟踪前进行处理。一但用户使用了 RAW 表,在某个链上,RAW 表处理完后,将跳过 NAT 表和 ip_conntrack 处理,即不再做地址转换和数据包的链接跟踪处理了.
  • filter: 这个规则表是预设规则表, 拥有 INPUT、FORWARD 和 OUTPUT 三个规则链, 这个规则表顾名思义是用来进行封包过滤的理动作
  • net: 此规则表拥有 prerouting 和 postrouting 两个规则链, 主要功能为进行一对一、一对多、多对多等网址转译工作 (SNATDNAT)
  • mangle: 此规则表拥有 prerouting、FORWARD、postrouting 三个规则链, 除了进行网址转译工作会改写封包外, 在某些特殊应用可能也必须去改写封包(ITL、TOS)或者是设定 MARK (将封包作记号, 以进行后续的过滤)这时就必须将这些工作定义在 mangles 规则表中

常用命令:

-A 追加规则-->iptables -A INPUT
-D 删除规则-->iptables -D INPUT 1(编号)
-R 修改规则-->iptables -R INPUT 1 -s 192.168.12.0 -j DROP 取代现行规则, 顺序不变(1是位置)
-I 插入规则-->iptables -I INPUT 1 --dport 80 -j ACCEPT 插入一条规则, 原本位置上的规则将会往后移动一个顺位
-L 查看规则-->iptables -L INPUT 列出规则链中的所有规则
-N 新的规则-->iptables -N allowed 定义新的规则

通用参数:

-p 协议  例: iptables -A INPUT -p tcp
-s源地址 例: iptables -A INPUT -s 192.168.1.1
-d目的地址 例: iptables -A INPUT -d 192.168.12.1
-sport源端口 例:iptables -A INPUT -p tcp --sport 22
-dport目的端口 例:iptables -A INPUT -p tcp --dport 22
-i指定入口网卡 例:iptables -A INPUT -i eth0
-o指定出口网卡 例:iptables -A FORWARD -o eth0

-j 指定要进行的处理动作

常用的ACTION:
DROP: 丢弃
REJECT: 明示拒绝
ACCEPT: 接受
SNAT基于原地址的转换
source--指定原地址
    比如我们现在要将所有192.168.10.0网段的IP在经过的时候全都转换成172.16.100.1这个假设出来的外网地址:
iptables -t nat -A POSTROUTING -s 192.168.10.0/24 -j SNAT --to-source 172.16.100.1(外网有效ip)
这样, 只要是来自本地网络的试图通过网卡访问网络的, 都会被统统转换成172.16.100.1这个IP.
MASQUERADE(动态伪装) --家用带宽获取的外网ip, 就是用到了动态伪装
iptables -t nat -A POSTROUTING -s 192.168.10.0/24 -j MASQUERADE
DNAT目标地址转换
destination-指定目标地址
iptables -t nat -A PREROUTING -d 192.168.10.18 -p tcp --dport 80 -j DNAT --to-destination 172.16.100.2
10.18访问80端口转换到100.2上
MASQUERADE: 源地址伪装
REDIRECT: 重定向: 主要用于实现端口重定向
MARK: 打防火墙标记的
RETURN: 返回 在自定义链执行完毕后使用返回, 来返回原规则链。

链 (chain)

每个表都有自己的一组内置链, 可以对链进行自定义, 这样就可以建立一组规则,
filter 表中的 input、output 和 forward 链

匹配(match)

每个iptables规则都包含一组匹配以及一个目标, iptables匹配指的是数据包必须匹配的条件, 只有当
数据包满足所有的匹配条件时, iptables才能根据由该规则的目标所指定的动作来处理该数据包
匹配都在iptable的命令行中指定
source--匹配源ip地址或网络
destination (-d)--匹配目标ip地址或网络
protocol (-p)--匹配ip值
in-interface (-i)--流入接口(例如, eth0)
out-interface (-o)--流出接口
state--匹配一组连接状态
string--匹配应用层数据字节序列
comment--在内核内存中为一个规则关联多达256个字节的注释数据

目标(target)

iptables支持一组目标, 用于数据包匹配一条规则时触发一个动作
ACCEPT--允许数据包通过
DROP--丢弃数据包, 不对该数据包做进一步的处理, 对接收栈而言, 就好像该数据包从来没有被接收一样
LOG--将数据包信息记录到syslog
REJECT--丢弃数据包, 同时发送适当的响应报文(针对TCP连接的TCP重要数据包或针对UDP数据包的ICMP端口不可达消息)
RETURN--在调用链中继续处理数据包

默认配置文件

vim /etc/sysconfig/iptables 策略文件
vim /etc/sysconfig/iptables-config  配置文件

iptables 缺省具有5条规则链

prerouting(内到外) forward(转发) postrouting(外到内) input(输入) output(输出)

链管理命令 (这都是立即生效的)

-P :设置默认策略的 (设定默认门是关着的还是开着的)
    默认策略一般只有两种
    iptables -P INPUT (DROP|ACCEPT)
    默认是关的/默认是开的
    比如:
    iptables -P INPUT DROP
这就把默认规则给拒绝了。并且没有定义哪个动作, 所以关于外界连接的所有规则包括Xshell连接之类的, 远程连接都被拒绝了
-F: FLASH, 清空规则链的(注意每个链的管理权限)
    iptables -t nat -F PREROUTING
    iptables -t nat -F 清空nat表的所有链
    -N:NEW 支持用户新建一个链
    iptables -N inbound_tcp_web 表示附在tcp表上用于检查web的。
-X: 用于删除用户自定义的空链
    使用方法跟-N相同, 但是在删除之前必须要将里面的链给清空昂了
-E: 用来Rename chain主要是用来给用户自定义的链重命名
    -E oldname newname
-Z: 清空链, 及链中默认规则的计数器的 (有两个计数器, 被匹配到多少个数据包, 多少个字节)
    iptables -Z :清空

规则管理命令

-A: 追加, 在当前链的最后新增一个规则
-I num : 插入, 把当前规则插入为第几条。
   -I 3 :插入为第三条
-R num: Replays替换/修改第几条规则
   格式: iptables -R 3 …………
-D num: 删除, 明确指定删除第几条规则

查看管理命令 “-L”

    附加子命令
-n: 以数字的方式显示ip, 它会将ip直接显示出来, 如果不加-n, 则会将ip反向解析成主机名。
-v: 显示详细信息
-vv
-vvv :越多越详细
-x: 在计数器上显示精确值, 不做单位换算
--line-numbers : 显示规则的行号
-t nat: 显示所有的关卡的信息

扩展匹配

隐含扩展: 对协议的扩展

-p tcp :TCP协议的扩展。一般有三种扩展
--dport XX-XX: 指定目标端口,不能指定多个非连续端口,只能指定单个端口, 比如
--dport 21  或者 --dport 21-23 (此时表示21,22,23)
--sport: 指定源端口
--tcp-fiags: TCP的标志位 (SYN,ACK, FIN,PSH, RST,URG)
    对于它, 一般要跟两个参数:
    1.检查的标志位
    2.必须为1的标志位
    --tcpflags syn,ack,fin,rst syn   =    --syn
    表示检查这4个位, 这4个位中syn必须为1, 其他的必须为0。所以这个意思就是用于检测三次握手的第一次包的。对于这种专门匹配第一包的SYN为1的包, 还有一种简写方式, 叫做--syn
-p udp: UDP协议的扩展
    --dport
    --sport
-p icmp: icmp数据报文的扩展
    --icmp-type:
    echo-request(请求回显), 一般用8 来表示
    所以 --icmp-type 8 匹配请求回显数据包
    echo-reply (响应的数据包) 一般用0来表示

系统CentOS 7

# 启动指令
systemctl start iptables

# 重启指令
systemctl restart iptables

# 关闭指令
systemctl stop iptables

命令笔记

*filter
:INPUT ACCEPT [0:0]
:FORWARD ACCEPT [0:0]
:OUTPUT ACCEPT [0:0]
-A INPUT -m state --state RELATED,ESTABLISHED -j ACCEPT
-A INPUT -p icmp -j ACCEPT
-A INPUT -i lo -j ACCEPT
-A INPUT -p tcp -m state --state NEW -m tcp --dport 22 -j ACCEPT
-A INPUT -p tcp -m state --state NEW -m tcp --dport 21 -j ACCEPT
-A INPUT -m state --state NEW -m tcp -p tcp --dport 80 -j ACCEPT
-A INPUT -m state --state NEW -m tcp -p tcp --dport 443 -j ACCEPT
-A INPUT -m state --state NEW -m tcp -p tcp --dport 1080 -j ACCEPT
-A INPUT -j REJECT --reject-with icmp-host-prohibited
-A FORWARD -j REJECT --reject-with icmp-host-prohibited
COMMIT

原文

iptables详解 (1): iptables概念
iptables命令、规则、参数详解

最后更新于 2020-02-16
使用 Hugo 构建
主题 StackJimmy 设计