Reverse traceroute is an extension to ICMP, enabling hosts to request traceroute measurements back to themselves from a remote target.
For detailed information about reverse traceroute and its concepts, have a look at the resources.
This repository contains reference implementations for reverse traceroute client and server applications.
Both the protocol and tools are able to support IPv4 and IPv6.
Traceroute provides information on the forward path towards a target. As such, it is popular for troubleshooting problems on the forward path. Should a problem arise on the reverse path traceroute's output offers no help and may even be misleading.
Thats why we developed reverse traceroute. Reverse traceroute allows you to determine the reverse path from a target host, which runs our server program, back to you, enabling you to detect and (hopefully) troubleshoot said issues.
Reverse traceroute was built in accordance with the following principles:
- No direct control over the remote host ✅
Access to the target host is not required. The target merely has to run our server. - Safe to use ✅
Reverse traceroute must not introduce possible attack vectors inside a network. - Deployable in todays Internet ✅
The messages sent by reverse traceroute should be able to traverse the Internet unaltered. - Policable by network operators ✅
Network operators who do not want reverse traceroute traffic inside their administrative domain can easily enforce such restrictions. - Awareness of load balancing ✅
Reverse traceroute knows how to keep packets on a single path. - No hackery (IP Spoofing / IP Options) ✅
We refrain from using unorthodox means and practices, such as IP spoofing. - No changes to routers required ✅
The chances of deploying reverse traceroute in the wild increase when leaving routers untouched. - Mimic classic traceroute ✅
Reverse traceroute behaves just like regular traceroute. You can send UDP, TCP and ICMP probes and expect both node address and Round-Trip-Time as a measurement.
The client is implemented as a Python script and requires at least Python 3.10
.
It supports supports both single- and multipath discovery.
When run in the singlepath mode a fixed flow identifier has to be specified, which
determines the path that the traceroute probes will illuminate.
When run in the multipath mode, a variation of the Diamond Miner
algorithm ensures that all nodes for a hop will be detected with a certainty specified by the user.
Both modes of operation can be used in the forward and, since we are talking about reverse traceroute, in the reverse direction.
As the client renders the discovered paths, you need the graphviz binary accessible through your
PATH
variable.
To discover multiple paths towards a destination with TCP, use the following command:
augsburg-traceroute forward tcp multipath <target>
To discover only a single reverse path identified by flow 1234 with UDP back from a target:
augsburg-traceroute reverse udp singlepath --flow 1234 <target>
To discover both reverse and forward paths with TCP and submit the results to our measurement study:
augsburg-traceroute --transmit two-way tcp multipath <target>
The client provides a few more fine-grained controls to influence the probing behaviour.
Run augsburg-traceroute -h
to learn more.
The server application is written as an eBPF program. In order to parse reverse traceroute requests and responses before they are processed by the kernel, it is attached to a traffic control ingress hook.
As the application makes use of recent eBPF features such as timers,
at least a recent linux kernel version of 5.15.0
is required.
We successfully tested the server on an x86-64 machine running Ubuntu 22.04 and a 5.15.0-52
kernel.
You can specify both the size of the session buffer and the timeout value,
after which unanswered sessions are dropped.
The interface name on which the server will process reverse traceroute traffic
is a mandatory argument.
sudo augsburg-traceroute-server-v4 [-n MAX_SESSIONS] [-t TIMEOUT_NS] ifname
sudo augsburg-traceroute-server-v6 [-n MAX_SESSIONS] [-t TIMEOUT_NS] ifname
To run the IPv4 and IPv6 servers on the interface eth0, with at most 50.000 sessions and and a session timeout of 5 seconds:
augsburg-traceroute-server-v4 -n 50000 -t 5000000000 eth0
augsburg-traceroute-server-v6 -n 50000 -t 5000000000 eth0
This example creates a service for the IPv4 version of the server. You can create a similar IPv6 version with minimal adjustments.
In order to persist the server applications across reboots,
you can create a systemd service.
First you have to create a file /etc/systemd/system/reverse-traceroute-v4@.service
:
[Unit]
Description=reverse-traceroute-v4
After=network-online.target
[Service]
Type=simple
ExecStart=<path-to-server> -n <max-sessions> -t <timeout-ns> %I
[Install]
WantedBy=multi-user.target
Make sure to replace the path to the server and arguments with your own configuration. Before starting the service, you have to make systemd aware of it by running:
sudo systemctl daemon-reload
Then you can start the service with the following command:
sudo systemctl start reverse-traceroute-v4@<ifname>
In order to persist the service across reboots, run:
sudo systemctl enable reverse-traceroute-v4@<ifname>
Note that you have to replace <ifname>
with the name of the interface
the server should run on.
This repository includes dependencies in the form of git submodules,
which must be initialized before the software can be built.
To initialize said submodules when cloning the repository, run:
git clone --recurse-submodules https://github.com/HSAnet/reverse-traceroute
In order to build the client Poetry has to be installed.
In the client/
directory run:
poetry build -f wheel
The built package can be found in the dist/
subdirectory.
To install the built client globally, run:
sudo pip3 install dist/<package-name>.whl
Make sure to replace <package-name>
with the actual name of the package.
In the server/
directory run:
bash build_docker.sh
The executables will be built in a docker container that includes the needed dependencies.
We are trying to collect data about traceroute paths for our measurement study.
If you want to participate, you can use the --transmit
switch in the client
to transmit your data to our server.
Please be aware that the data includes hostnames by default.
If you do not want to transmit resolved hostnames as part of the trace
you can use the --no-resolve
client switch.
Reverse traceroute was designed as a distributed service. Hence it lives from the people who decide to host publicly available server endpoints.
Should you decide to host such a reverse traceroute server,
then please let us know so that we can add your server to the list of endpoints,
which are maintained inside the ENDPOINTS
file.
- Introduce alias resolution to match nodes on both the forward and reverse path
Both the client and the server are subject to change, as they are still early in development. As such, you may encounter bugs.
Project website: https://net.hs-augsburg.de/en/project/reverse-traceroute/
Protocol draft: https://datatracker.ietf.org/doc/html/draft-heiwin-intarea-reverse-traceroute
Talk at DENOG: https://youtu.be/Y7NtqLEtgjU
Valentin Heinrich <valentin.heinrich@hs-augsburg.de>
Rolf Winter <rolf.winter@hs-augsburg.de>