Skip to content

Commit

Permalink
flow: Add test for excluding pkt recursion from flow
Browse files Browse the repository at this point in the history
Add tests for verifying matching packet flows when including and
excluding pkt recursion from flow matching.

Bug: #6260
  • Loading branch information
coledishington committed Aug 31, 2023
1 parent e338b7b commit 0257a6a
Show file tree
Hide file tree
Showing 8 changed files with 114 additions and 0 deletions.
25 changes: 25 additions & 0 deletions tests/flow-pkt-recursion/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
# Test Purpose

Tests comparing flows with and without recursion level set. Ignoring
recursion level in flows is useful for devices that run inline IPS and
terminate an unencrypted tunnel, like an IPv6 tunnel. Terminating the
tunnel causes ingress request and reply traffic to have different
headers. e.g.

request: IPv4]ICMP] -> |IPS| -> IPv6]IPv4]ICMP]
reply: <- |IPS| <- IPv6]IPv4]ICMP]

The terminated tests are checking when the suricata is an inline IPS
device that is terminating a tunnel. Both flow directions are tested,
first packet ingress on non-tunneled interface and first packet ingress
on tunneled interface.
This is the case where recursion level can affect the flows when IPS is
running inline IPS and terminating a tunnel.

Middleware tests check when the suricata device is analysing tunneled
packets and is not a tunnel terminator.
This case should not be affected by recursion level in flows.

## PCAP

This PCAP was generated with scapy.
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
# Exclude recursion-level from flow matching
# Re-use the pcap from the middleware-included test.
command: |
${SRCDIR}/src/suricata -c "${SRCDIR}/suricata.yaml" -l "${OUTPUT_DIR}" \
-r "${TEST_DIR}/../flow-pkt-recursion-middleware-included/test.pcap" \
--set "threshold-file=${SRCDIR}/threshold.config" \
--set "classification-file=${SRCDIR}`[ -f ${SRCDIR}/etc/classification.config ] && printf '/etc'`/classification.config" \
--set "reference-config-file=${SRCDIR}`[ -f ${SRCDIR}/etc/reference.config ] && printf '/etc'`/reference.config" \
--set 'decoder.recursion-level.use-for-tracking=false'
checks:
- filter:
count: 1
match:
event_type: flow
proto: ICMP
flow.pkts_toserver: 1
flow.pkts_toclient: 1
Binary file not shown.
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
checks:
- filter:
count: 1
match:
event_type: flow
proto: ICMP
flow.pkts_toserver: 1
flow.pkts_toclient: 1
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
# Exclude recursion-level from flow matching
# Re-use the pcap from the middleware-included test.
command: |
${SRCDIR}/src/suricata -c "${SRCDIR}/suricata.yaml" -l "${OUTPUT_DIR}" \
-r "${TEST_DIR}/../flow-pkt-recursion-terminated-included/test.pcap" \
--set "threshold-file=${SRCDIR}/threshold.config" \
--set "classification-file=${SRCDIR}`[ -f ${SRCDIR}/etc/classification.config ] && printf '/etc'`/classification.config" \
--set "reference-config-file=${SRCDIR}`[ -f ${SRCDIR}/etc/reference.config ] && printf '/etc'`/reference.config" \
--set 'decoder.recursion-level.use-for-tracking=false'
checks:
# All packets should be caught as being in one flow
- filter:
count: 2
match:
event_type: flow
proto: ICMP
flow.pkts_toserver: 1
flow.pkts_toclient: 1
Binary file not shown.
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
checks:
# None of the flows are joined due to different recursion levels
- filter:
count: 4
match:
event_type: flow
proto: ICMP
flow.pkts_toserver: 1
flow.pkts_toclient: 0
35 changes: 35 additions & 0 deletions tests/flow-pkt-recursion/test.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
from pathlib import Path

from scapy.all import ICMP, IP, Ether, IPv6, PcapWriter, Raw

mac_1, mac_2 = 'cb:cf:2b:50:a7:61', '49:a2:25:1a:07:4a'

request = Ether(src=mac_1, dst=mac_2)
reply = Ether(src=mac_2, dst=mac_1)

ip_1, ip_2 = '1.1.1.1', '2.2.2.2'
ipv6_1, ipv6_2 = 'fd01::1.1.1.1', 'fd02::2.2.2.2'

payload = Raw(b'#JSb[abR^79aV(kDAN,(C\n\\A+p V+MF7\rd9Z&&9D31.;T%\x0ct.#')
icmp_echo = ICMP(type=8, seq=1) / payload
icmp_reply = ICMP(type=0, seq=1) / payload

test = 'flow-pkt-recursion'

middleware_pcap = Path.cwd() / f'{test}-middleware-included' / 'test.pcap'
with PcapWriter(str(middleware_pcap)) as pcap:
# Flow of IPv6 tunneled packets in both directions
pcap.write(request / IPv6(src=ipv6_1, dst=ipv6_2) / IP(src=ip_1, dst=ip_2) / icmp_echo)
pcap.write(reply / IPv6(src=ipv6_2, dst=ipv6_1) / IP(src=ip_2, dst=ip_1) / icmp_reply)

terminated_pcap = Path.cwd() / f'{test}-terminated-included' / 'test.pcap'
with PcapWriter(str(terminated_pcap)) as pcap:
# Flow of tunnel terminated on Suricata device, echo originates
# from Suricata device
pcap.write(request / IP(src=ip_1, dst=ip_2) / icmp_echo)
pcap.write(reply / IPv6(src=ipv6_2, dst=ipv6_1) / IP(src=ip_2, dst=ip_1) / icmp_reply)

# Flow of tunnel terminated on Suricata device, reply originates
# from Suricata device
pcap.write(reply / IPv6(src=ipv6_2, dst=ipv6_1) / IP(src=ip_2, dst=ip_1) / icmp_echo)
pcap.write(request / IP(src=ip_1, dst=ip_2) / icmp_reply)

0 comments on commit 0257a6a

Please sign in to comment.