1
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

scapyでトラフィックジェネレーター(笑)

Posted at

はじめに

ネットワーク評価の際にトラフィック負荷試験をする必要があり、簡易的なトラフィックジェネレーターソフトが必要でした。
せっかくなのでiperfよりも細かくパケットが定義できそうなscapyを使用して、簡易的なトラフィックジェネレーターを作成してみます。
スループットのパフォーマンスを向上させるため、scapyのsendpfast()を使用するのでtcpreplayもインストールします。
※ちなみに本スクリプトでは大量のショートパケットや1Gを超える帯域負荷は環境によって安定出力ができません(なので(笑))

環境の準備

本記事の環境

  • Rocky Linux 9
    • 4core 8GB memory
--------------------      --------      ------------
| Rocky linux(src) | -1G- | L2SW | -1G- | receiver |
--------------------      --------      ------------

関連ソフトのインストール

python

sudo dnf install -y python3.11

# インストールできたらvenvも作っておく
python3.11 -m venv .venv

scapy

python3.11 -m pip install scapy

tcpreplay

dnf install -y --enablerepo=epel tcpreplay

各プロトコルのトラフィック生成、送信

以下は送信したいプロトコル(TCP、UDP、ICMP)を選択し、記述したIPアドレス、ポート番号を使用して50000pps(およそ600Mbps)の負荷で送信し続けるコードです。
データヘッダには適当なバイト列を入れています。(ここでデータヘッダサイズを調整できます)

traffgen.py
from scapy.all import *

def info():
    # IPアドレスとポートの設定
    iface = 'ens18' # 送信元インターフェースを指定
    src_ip = "192.168.1.18" # 送信元のIPアドレスを指定
    src_port = 10000 # 送信元ポート番号を指定

    dst_ip = "192.168.1.3" # 宛先のIPアドレスを指定
    dst_port = 80 # 宛先ポート番号を指定

    modeselect = input('select traffic gen mode: ')
    if modeselect == 'tcp':
        mode = tcppkt(src_ip,src_port,dst_ip,dst_port)
    elif modeselect == 'udp':
        mode = udppkt(src_ip,src_port,dst_ip,dst_port)
    elif modeselect == 'icmp':
        mode = icmppkt(src_ip,dst_ip)

    pkt = mode
    print(pkt)
    sendpfast(pkt, iface=iface, loop=1, file_cache=True, pps=50000)

def tcppkt(src_ip,src_port,dst_ip,dst_port):
    # 1パケット1500バイトのTCP SYNパケットを作成
    pkt = Ether()/\
        IP(src=src_ip, dst=dst_ip, flags="DF") /\
        TCP(sport=src_port, dport=dst_port, flags="S") /\
        (b"A" * 1460) # データヘッダサイズを指定
    return pkt

def udppkt(src_ip,src_port,dst_ip,dst_port):
    # 1パケット1500バイトのUDPパケットを作成
    pkt = Ether()/\
        IP(src=src_ip, dst=dst_ip, flags="DF") /\
        UDP(sport=src_port, dport=dst_port) /\
        (b"A" * 1472) # データヘッダサイズを指定
    return pkt

def icmppkt(src_ip,dst_ip):
    # 1パケット1500バイトのICMP echo requestパケットを作成
    pkt = Ether()/\
        IP(src=src_ip, dst=dst_ip, flags="DF") /\
        ICMP() /\
        (b"A" * 1472) # データヘッダサイズを指定
    return pkt

if __name__ == '__main__':
    info()
# venvに移行
[user@localhost traffgen]$ source .venv/bin/activate
(.venv) [user@localhost traffgen]$

# scapy使用時はroot権限で実行
(.venv) [user@localhost traffgen]$ sudo -s
[sudo] user のパスワード:
(.venv) [root@localhost traffgen]#

# traffgen.py実行(tcp or udp or icmpを入力)
(.venv) [root@localhost traffgen]# python3.11 traffgen.py 
select traffic gen mode: tcp
Ether / IP / TCP 192.168.1.18:ndmp > 192.168.1.3:http S / Raw
# パケットを送信し続けるため、停止したいタイミングでKeyboard interruptして停止します
  • 受信側
    • TCP
      スクリーンショット 2025-01-30 225139.png
      スクリーンショット 2025-01-30 225252.png

    • UDP
      image.png
      image.png

    • ICMP
      image.png
      image.png

TCPパケットでは安定した600Mbpsの受信ができておらず、波形にばらつきが見られます。
UDP・ICMPパケットではおおよそ綺麗な波形に見えます。

pcapファイルを作成して送信

先程のコードでは、事前に定義したポート番号のみを使用してパケットを生成しますが、複数ポートの番号を使用することができません。
そこで、ポート1024-65535までを宛先としたパケットを格納したpcapファイルを生成します。ここでは送信元ポート番号も1024-65535からランダムな値を用いてみます。

pcapgen.py
from scapy.all import *
import random

def info():
    src_ip = "192.168.1.18" # 送信元のIPアドレスを指定
    dst_ip = "192.168.1.3" # 宛先ポート番号を指定
    dst_port_range = range(1024,65535)

    pkts = []

    protocol = input('select protocol: ')
    if protocol == 'tcp':
        tcppcap(src_ip, dst_ip, dst_port_range, pkts)
    elif protocol == 'udp':
        udppcap(src_ip, dst_ip, dst_port_range, pkts)
    if protocol == 'icmp':
        icmppcap(src_ip, dst_ip, pkts)

def tcppcap(src_ip, dst_ip, dst_port_range, pkts):
    # 1パケット1500バイトのTCP SYNパケットを作成
    for dst_port in dst_port_range:
        src_port = random.randint(1024, 65535)
        pkt = Ether() /\
            IP(src=src_ip, dst=dst_ip, flags="DF") /\
            TCP(sport=src_port, dport=dst_port, flags="S") /\
            (b"A" * 1460)
        pkts.append(pkt)
    wrpcap('tcp_port_scan.pcap', pkts)
    print("TCP port scan pcap file was created successfully.")

def udppcap(src_ip, dst_ip, dst_port_range, pkts):
    # 1パケット1500バイトのUDPパケットを作成
    for dst_port in dst_port_range:
        src_port = random.randint(1024, 65535)
        pkt = Ether() /\
            IP(src=src_ip, dst=dst_ip, flags="DF") /\
            UDP(sport=src_port, dport=dst_port) /\
            (b"A" * 1472)
        pkts.append(pkt)
    wrpcap('udp_port_scan.pcap', pkts)
    print("UDP port scan pcap file was created successfully.")

def icmppcap(src_ip, dst_ip, pkts):
    # 1パケット1500バイトのICMPパケットを作成
    pkt = Ether() /\
        IP(src=src_ip, dst=dst_ip, flags="DF") /\
        ICMP() / (b"A" * 1472)
    pkts.append(pkt)
    wrpcap('icmp_port_scan.pcap', pkts)
    print("ICMP port scan pcap file was created successfully.")

if __name__ == '__main__':
    info()

以下のコードを実行し、pcapファイルからパケットを送信します。

traffgen-pcap.py
from scapy.all import *

mode = input('select traffic gen mode: ')
if mode == 'tcp':
    pkt = rdpcap('tcp_port_scan.pcap')
if mode == 'udp':
    pkt = rdpcap('udp_port_scan.pcap')
if mode == 'icmp':
    pkt = rdpcap('icmp_port_scan.pcap')

print(pkt)

iface = 'ens18' # 送信元インターフェースを指定
sendpfast(pkt, iface=iface, loop=1, file_cache=True, pps=50000)
# 宛先port 1024-65535のpcapファイルを作成(tcp or udp or icmpを入力)
(.venv) [root@localhost traffgen]# python3.11 pcapgen.py 
select protocol: tcp
TCP port scan pcap file was created successfully.

# traffgen-pcap.py実行(tcp or udp or icmpを入力)
(.venv) [root@localhost traffgen]# python3.11 traffgen-pcap.py 
select traffic gen mode: tcp
<tcp_port_scan.pcap: TCP:64511 UDP:0 ICMP:0 Other:0>
  • 受信側
    • TCP
      image.png
      image.png
    • UDP
      image.png
      image.png

pcapファイルからパケットを送信した場合の方では、TCP・UDPどちらもおおよそ600Mbpsを安定して受信できており、こちらの方が安定した波形に見えます。

その他

  • sendpfast()のfile_cacheオプションはパケットをramにキャッシュする(らしい)ので、オプションを有効にするとスループットが安定する。

  • mbpsオプションを使用していないのは、上記オプションを並装すると10~30秒程トラフィックを送信し続けると指定した値を超える過剰な負荷が発生した。
    image.png

    • ppsオプションの場合でも100秒程トラフィックを送信し続けた際に過剰な負荷が発生した。
      image.png
  • pcapファイルを生成するならわざわざscapyからsendpfastを叩かずにtcpreplay+netmapすればさらに安定しそう。その検証はまた次回。

最後に

環境やシステム、設計によってネットワーク負荷試験の重要度は変わります。
特にトラフィックジェネレーターアプライアンスが必要でない今回のような試験の場面も多くあると思いますので、要件にマッチしたものをご使用いただければと思います。

また、本記事はTCP syn flood、UDP floodを行うスクリプトとなっているため、悪用厳禁でお願いいたします。

1
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
1
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?