PTF Packet Testing Framework
PTF is a Python based dataplane test framework. It is based on unittest, which is included in the standard Python distribution.
This document is meant to provide an introduction to the framework, discuss the basics of running tests and to provide examples of how to add tests.
Most of the code was taken from the OFTest framework. However, PTF focuses on the dataplane and is independent of OpenFlow. We also made a few additions to oftest.
The following software is required to run OFTest:
- Python 2.7
- Scapy
- pypcap (optional - VLAN tests will fail without this)
- tcpdump (optional - Scapy will complain if it's missing)
We recommend that you install your extension of Scapy, which you can obtain
here. It adds support for additional
header types: VXLAN
, ERSPAN
, GENEVE
, MPLS
and NVGRE
.
Root/sudo privilege is required on the host, in order to run ptf
.
Once you have written tests and your switch is running, ypu can run 'ptf'. Use
--help
to see command line switches.
For example:
sudo ./ptf --test-dir mytests/ --pypath $PWD \
--interface 0@veth1 --interface 1@veth3 --interface 2@veth5 \
--interface 3@veth7 --interface 4@veth9 --interface 5@veth11 \
--interface 6@veth13 --interface 7@veth15 --interface 8@veth17
This will run all the tests included in the mytests
directory. The --pypath
option can be used to easily add directories to the Python PATH. This is useful
if you use a Python interface to configure your data plane (as part of your
tests). The --interface
option (or -i
) can be used to specify the interfaces
on which to inject packets (along with the corresponding port number).
Take a look at the example
directory. This is not a working example as it is
(the switch is not included), but it will show you how to write tests. This
directory contains the following:
run_client.sh
: a wrapper aroundptf
switch_sai_thrift
: empty directory, this is where the Python bindings to program the switch's control plane would be copiedmytests/sai_base_test.py
: a wrapper Python class around PTF's BaseTest class. It is the base class for all the tests we added tomytests/switch.py
mytests/switch.py
: some example tests
If you want to run the example, you will need to obtain
p4factory. For the following, I will
assume that you cloned the repository and installed the dependencies. I will
assume that environment variable P4FACTORY
contains the path to the cloned
repository.
First, you need to create the required veths:
cd $P4FACTORY/tools/
sudo ./veth_setup.sh
The next step is to compile the target switch and to run it:
cd $P4FACTORY/targets/switch/
make bm-switchsai
sudo ./behavioral-model
Finally, you can run the example tests:
cd <ptf-dir>/example/
sudo ../ptf --test-dir mytests/ \
--pypath $P4FACTORY/targets/switch/of-tests/pd_thrift/
--interface 0@veth1 --interface 1@veth3 --interface 2@veth5 \
--interface 3@veth7 --interface 4@veth9 --interface 5@veth11 \
--interface 6@veth13 --interface 7@veth15 --interface 8@veth17
We added the following features to the base OFTest framework:
They can be used to discard some of the packets received from the switch. Take a
look at sai_base_test.py for an example. You
will see the following code testutils.add_filter(testutils.not_ipv6_filter)
which tells PTF to discard received IPv6 packets. You can add your own filters
(they have to be callable Python objects which take a Scapy packet as input).
A PTF test -just like an OFTest test- matches the received packets against expected packets. This is an exact match. However, sometimes one does not care about all the fields in the packets. PTF introduces the Mask class which lets you specified some field you do not care about when performing the match. For example:
import mask
m = mask.Mask(expected_pkt)
m.set_do_not_care_scapy(IP, 'ttl')
verify_packets(<test>, m, <port list>)
The "platform" is a configuration file (written in Python) that tells PTF how to send packets to and receive packets from the dataplane of the switch.
The default platform, eth
, uses Linux Ethernet interfaces and is configured
with the -i
option (or --interface
). Pass the option as -i ofport@interface
, for example -i 1@eth1
. If no -i
options are given the the
default configuration uses vEths.
Another common platform, remote
, provides support for testing of switches on a
different host. This can be useful for cases where interfaces are not available
on one host (i.e. they're not bound to a Linux interface driver) or where PTF
cannot run on the same host (unsupported OS, missing software, etc.).
This can be enable by modifying the platforms/remote.py
file to point to 4
NICs on the host running PTF, like so:
remote_port_map = {
(0, 23) : "eth2", # port 23 of device 0 is connected to physical port on the server eth2
(0, 24) : "eth3", # port 24 of device 0 is connected to physical port on the server eth3
(0, 25) : "eth4",
(0, 26) : "eth5"
}
There is a facility for passing test-specific parameters into tests that works as follows. On the command line, give the parameter
--test-params="key1=17;key2=True"
You can then access these parameters in your tests' Pyhton code using the following code:
import ptf.testutils as testutils
# Returns a dictionary which includes all your parameters
test_params = testutils.test_params_get()
# Returns the value of the parameter "param", or None if not found
param_value = testutils.test_param_get("param")
Take a look at sai_base_test.py for an example.
It is very easy to create groups of tests, using the provided group
Python
decorator. Simply decorate your test with @group(<name of group>)
.
Take a look at switch.py for an example.
One given test can belong to several groups. You can choose to run only the tests belonging to a given group using a command like this one:
sudo ./ptf --test-dir mytests/ --pypath $PWD <name of group>
We also provide a convenient disabled
decorator for tests.
The original OFTest was meant to unit test a single OF-compliant switch. With PTF, we tried to add support for testing a network of several devices. If you do not intend to use this multi-device feature, you do not need to worry about it, it should not impact you. If you want to leverage this feature, here is what you need to do:
- when adding interfaces, instead of writing
<port_number>@<interface_name>
, you need to write<device_number>-<port_number>@<interface_name>
- when sending a packet, the port number becomes a tuple (device, port):
send_packet(self, (<device_number>, <port_number>), pkt)
- the
verify_*
functions where also updated to include device information. For example:verify_packets(self, pkt, device_number=<device_number>, ports=<port_list>)
. For more information, you can take a look at these functions in src/ptf/dataplane.py.