1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
|
#!/usr/bin/env python3
# Copyright (c) 2021 Cisco and/or its affiliates.
#
# SPDX-License-Identifier: Apache-2.0 OR GPL-2.0-or-later
#
# Licensed under the Apache License 2.0 or
# GNU General Public License v2.0 or later; you may not use this file
# except in compliance with one of these Licenses. You
# may obtain a copy of the Licenses at:
#
# http://www.apache.org/licenses/LICENSE-2.0
# https://www.gnu.org/licenses/old-licenses/gpl-2.0-standalone.html
#
# Note: If this file is linked with Scapy, which is GPLv2+, your use of it
# must be under GPLv2+. If at any point in the future it is no longer linked
# with Scapy (or other GPLv2+ licensed software), you are free to choose
# Apache 2.
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
"""Traffic script that sends an IP ICMPv4/ICMPv6 packet from one interface to
the other one. Dot1q or Dot1ad tagging of the ethernet frame can be set.
"""
import sys
from scapy.layers.inet import IP, UDP
from scapy.layers.l2 import Ether
from scapy.packet import Raw
from .PacketVerifier import RxQueue, TxQueue
from .TrafficScriptArg import TrafficScriptArg
from . import vxlan
def main():
"""Send IP ICMPv4/ICMPv6 packet from one traffic generator interface to
the other one. Dot1q or Dot1ad tagging of the ethernet frame can be set.
"""
args = TrafficScriptArg(
[
u"tx_src_mac", u"tx_dst_mac", u"tx_src_ip", u"tx_dst_ip", u"tx_vni",
u"rx_src_ip", u"rx_dst_ip", u"rx_vni"
]
)
tx_if = args.get_arg(u"tx_if")
rx_if = args.get_arg(u"rx_if")
tx_src_mac = args.get_arg(u"tx_src_mac")
tx_dst_mac = args.get_arg(u"tx_dst_mac")
tx_src_ip = args.get_arg(u"tx_src_ip")
tx_dst_ip = args.get_arg(u"tx_dst_ip")
tx_vni = args.get_arg(u"tx_vni")
rx_src_ip = args.get_arg(u"rx_src_ip")
rx_dst_ip = args.get_arg(u"rx_dst_ip")
rx_vni = args.get_arg(u"rx_vni")
rxq = RxQueue(rx_if)
txq = TxQueue(tx_if)
sent_packets = []
tx_pkt_p = (Ether(src=u"02:00:00:00:00:01", dst=u"02:00:00:00:00:02") /
IP(src=u"192.168.1.1", dst=u"192.168.1.2") /
UDP(sport=12345, dport=1234) /
Raw(u"raw data"))
pkt_raw = (Ether(src=tx_src_mac, dst=tx_dst_mac) /
IP(src=tx_src_ip, dst=tx_dst_ip) /
UDP(sport=23456) /
vxlan.VXLAN(vni=int(tx_vni)) /
tx_pkt_p)
pkt_raw /= Raw()
# Send created packet on one interface and receive on the other
sent_packets.append(pkt_raw)
txq.send(pkt_raw)
ether = rxq.recv(2, ignore=sent_packets)
# Check whether received packet contains layers Ether, IP and VXLAN
if ether is None:
raise RuntimeError(u"Packet Rx timeout")
ip = ether.payload
if ip.src != rx_src_ip:
raise RuntimeError(f"IP src mismatch {ip.src} != {rx_src_ip}")
if ip.dst != rx_dst_ip:
raise RuntimeError(f"IP dst mismatch {ip.dst} != {rx_dst_ip}")
if ip.payload.dport != 4789:
raise RuntimeError(
f"VXLAN UDP port mismatch {ip.payload.dport} != 4789"
)
vxlan_pkt = ip.payload.payload
if int(vxlan_pkt.vni) != int(rx_vni):
raise RuntimeError(u"vxlan mismatch")
rx_pkt_p = vxlan_pkt.payload
if rx_pkt_p.src != tx_pkt_p.src:
raise RuntimeError(
f"RX encapsulated MAC src mismatch {rx_pkt_p.src} != {tx_pkt_p.src}"
)
if rx_pkt_p.dst != tx_pkt_p.dst:
raise RuntimeError(
f"RX encapsulated MAC dst mismatch {rx_pkt_p.dst} != {tx_pkt_p.dst}"
)
if rx_pkt_p[IP].src != tx_pkt_p[IP].src:
raise RuntimeError(
f"RX encapsulated IP src mismatch {rx_pkt_p[IP].src} != "
f"{tx_pkt_p[IP].src}"
)
if rx_pkt_p[IP].dst != tx_pkt_p[IP].dst:
raise RuntimeError(
f"RX encapsulated IP dst mismatch {rx_pkt_p[IP].dst} != "
f"{tx_pkt_p[IP].dst}"
)
# TODO: verify inner Ether()
sys.exit(0)
if __name__ == u"__main__":
main()
|