23.1. Firewall Mode Design
Note
In Suricata 8 the firewall mode is experimental and subject to change.
The firewall mode in Suricata allows the use of a ruleset that has different properties than the default "threat detection" rulesets:
default policy is
drop, meaning a firewall ruleset needs to specify what is allowedfirewall rules are loaded from separate files
firewall rules use a new action
acceptfirewall rules are required to use explicit action scopes and rule hooks (see below)
evaluation order is as rules are in the file(s), per protocol state
23.1.1. Concepts
23.1.1.1. Tables
A table is a collection of rules with different properties. These tables are built-in.
No custom tables can be created. Tables are available within the scope of packet layer
and application layer (if available). Each rule can define its own action scope.
23.1.1.1.1. Packet layer tables
Rules categorized in the following tables apply to all packets.
Table |
Description |
Default Policy |
Rule Order |
|---|---|---|---|
|
Firewall rules to be evaluated before flow is created/updated |
|
As appears in the rule file |
|
Firewall rules to be evaluated before stream is updated |
|
As appears in the rule file |
|
Firewall rules to be evaluated against every packet after decoding |
|
As appears in the rule file |
|
Generic IDS/IPS threat detection rules |
|
Internal IDS/IPS rule ordering |
23.1.1.1.2. Application layer tables
If applayer is available, rules from the following tables apply. The tables for the
application layer are per app layer protocol and per protocol state. e.g. http:request_line.
Table |
Description |
Default Policy |
Rule Order |
|---|---|---|---|
|
Firewall rules to be evaluated per applayer protocol and state |
|
As appears in the rule file |
|
Generic IDS/IPS threat detection rules |
|
Internal IDS/IPS rule ordering |
23.1.1.2. Actions and Action Scopes
Firewall rules require action scopes to be explicitly specified.
23.1.1.2.1. accept
accept is used to issue an accept verdict to the packet, flow or hook.
packetaccept this packetflowaccept the rest of the packets in this flowhookaccept rules for the current hook/state, evaluate the next tablestxaccept rules for the current transaction, evaluate the next tables
The accept action is only available in firewall rules.
Note
some protocol implementations like dns use a transaction per direction.
For those accept:tx will only accept packets that are part of that direction.
23.1.1.2.2. drop
drop is used to drop either the packet or the flow
packetdrop this packet directly, don't eval any further rulesflowdrop this packet as withpacketand drop all future packets in this flow
Note
the action pass is not available in firewall rules due to ambiguity around
the existing meaning for threat detection rules.
23.1.1.3. Explicit rule hook (states)
In the regular IDS/IPS rules the engine infers from the rule's matching logic where the rule should be "hooked" into the engine. While this works well for these types of rules, it does lead to many edge cases that are not acceptable in a firewall ruleset. For this reason in the firewall rules the hook needs to be explicitly set.
There are two types of hooks available based on the layer.
23.1.1.3.1. Packet layer hooks
flow_start: evaluate the rule only on the first packet in both the directionspre_flow: evaluate the rule before the flow is created/updatedpre_stream: evaluate the rule before the stream is updatedall: evaluate the rule on every packet
23.1.1.3.2. Application layer hooks
The application layer states / hooks are defined per protocol. Each of the hooks has its own
default-drop policy, so a ruleset needs an accept rule for each of the states to allow
the traffic to flow through.
This is done in the protocol field of the rule. Where in threat detection a rule might look like:
alert http ... http.uri; ...
In the firewall case it will be:
accept:hook http1:request_line ... http.uri; ...
All available applayer hooks are available via commandline option --list-app-layer-hooks.
23.1.1.3.2.1. general
Each protocol has at least the default states.
Request (to_server) side:
request_startedrequest_complete
Response (to_client) side:
response_startedresponse_complete
23.1.1.3.2.2. http
For the HTTP protocol there are a number of states to hook into. These apply to HTTP 0.9, 1.0 and 1.1. HTTP/2 uses its own state machine.
Available states:
Request (to_server) side:
request_startedrequest_linerequest_headersrequest_bodyrequest_trailerrequest_complete
Response (to_client) side:
response_startedresponse_lineresponse_headersresponse_bodyresponse_trailerresponse_complete
23.1.1.3.2.3. tls
Available states:
Request (to_server) side:
client_in_progressclient_hello_doneclient_cert_doneclient_handshake_doneclient_finished
Response (to_client) side:
server_in_progressserver_helloserver_cert_doneserver_hello_doneserver_handshake_doneserver_finished
23.1.1.3.2.4. ssh
Available states are listed in Hooks.
23.1.1.4. Firewall pipeline
The firewall pipeline works in the detection engine, and is invoked after packet decoding, flow update, stream tracking and reassembly and app-layer parsing are all done in the context of a single packet.
For each packet rules in the first firewall hook packet:filter are then evaluated. Assuming
the verdict of this hook is accept:hook, the next hook is evaluated: packet:td (packet
threat detection). In this hook the IDS/IPS rules are evaluated. Rule actions here are not
immediate, as they can still be modified by alert postprocessing like rate_filter, thresholding, etc.
The default drop for the packet:filter table is drop:packet. Thus the drop is
only applied to the current packet.
If the packet has been marked internally as a packet with an application layer update, then the
next table is app:*:*.
In app:*:* the per application layer states are all evaluated at least once. At each of
these states an accept:hook is required to progress to the next state. When all available states
have been accepted, the pipeline moves to the final table app:td (application layer threat
detection). A drop in the app:filter table is immediate, however and accept is
conditional on the verdict of the app:td table.
The default drop in one of the app:*:* tables is a drop:flow. This means that the
current packet as well as all future packets from that flow are dropped.
In app:td the IDS/IPS rules for the application layer are evaluated. drop actions in this
table are queued in the alert queue.
When all tables have been evaluated, the alert finalize process orders threat detection alerts
by action-order logic. It can then apply a drop or default to accept-ing.
23.1.1.5. Pass rules with Firewall mode
In IDS/IPS mode, a pass rule with app-layer matches will bypass the detection engine for the
rest of the flow. In firewall mode, this bypass no longer happens in the same way, as pass rules
do not affect firewall rules. So the detection engine is still invoked on packets of such a flow,
but the packet:td and app:td tables are skipped.
23.1.2. Firewall rules
Firewall rules are loaded first and separately from the following section of suricata.yaml:
firewall-rule-path: /etc/suricata/firewall/
firewall-rule-files:
- fw.rules
One can optionally, also load firewall rules exclusively from commandline using the
--firewall-rules-exclusive option.
Firewall rules are available in the file firewall.json as a part of the output
of engine analysis.