Skip to content

Evaluating SoNiC switches by creating a PQC-secure inter-switch communication

License

Notifications You must be signed in to change notification settings

EMCL-Research-ITSecLab/SONiC-interswitch-communication

Repository files navigation

rosenpass-docker

Rosenpass Docker Implementation

Build and Test Release Pipeline Docker Image Version (latest semver) Docker Image Version (latest semver)

For more information on rosenpass and how it works, please refer to either the Homepage or the GitHub Repository

1. Getting started

To use the image in a custom setup, the following steps need to be applied:

For demonstration purposes we will use 4 virtual machines: 1 will act as server, 3 as clients, with the goal to connect the clients via the server with each other. Please note that in this setup the machines have no shared storage, hence the setup is not fully automated as with the docker-compose files (which are described here), since the public keys cannot be exchanged that easily. Furthermore, we assume the following:

Configuration Value
Server public IP 172.31.100.100
Client 1 public IP 172.31.100.101
Client 2 public IP 172.31.100.102
Client 3 public IP 172.31.100.103
Server VPN IP 10.11.12.100/24
Client 1 VPN IP 10.11.12.101/24
Client 2 VPN IP 10.11.12.102/24
Client 3 VPN IP 10.11.12.103/24
Allowed VPN CIDR 10.11.12.0/24
IPv6 enabled False
Docker network name: br0, subnet: 172.28.0.0/24

If you'd like to use this example on one machine with 4 docker containers, don't forget to create a network so that the containers can reach each other.

1.1 Start the server and client instances

First start the clients by running for client1:

client 1:

docker run -d \
 --cap-add=NET_ADMIN \
 -e MODE=client \
 -e SERVER_PUBLIC_IP="172.31.100.100" \
 -e SERVER_PORT="9999" \
 -e CLIENT_VPN_IP=10.11.12.101/24 \
 --privileged \
 --name client1 \
 stefan96/rosenpass:latest

client 2:

docker run -d \
 --cap-add=NET_ADMIN \
 -e MODE=client \
 -e SERVER_PUBLIC_IP="172.31.100.100" \
 -e SERVER_PORT="9999" \
 -e CLIENT_VPN_IP=10.11.12.102/24 \
 --privileged \
 --name client2 \
 stefan96/rosenpass:latest

client 3:

docker run -d \
 --cap-add=NET_ADMIN \
 -e MODE=client \
 -e SERVER_PUBLIC_IP="172.31.100.100" \
 -e SERVER_PORT="9999" \
 -e CLIENT_VPN_IP=10.11.12.103/24 \
 --privileged \
 --name client3 \
 stefan96/rosenpass:latest

client 4:

docker run -d \
 --cap-add=NET_ADMIN \
 -e MODE=client \
 -e SERVER_PUBLIC_IP="172.31.100.100" \
 -e SERVER_PORT="9999" \
 -e CLIENT_VPN_IP=10.11.12.104/24 \
 --privileged \
 --name client4 \
 stefan96/rosenpass:latest

and for the server:

docker run -d \
 --cap-add=NET_ADMIN \
 -e MODE="server" \
 -e SERVER_PUBLIC_IP="172.31.100.100" \
 -e SERVER_PORT="9999" \
 -e SERVER_VPN_IP="10.11.12.100/24" \
 --privileged \
 --name "server" \
 stefan96/rosenpass

This will create key pairs on each machine (public and private) which we will be using to authenticate the machines with each other. Furthermore, with the information provided by the environment variables, a startup script will be created that contains all the commands necessary to execute, once the keys are exchanged

Please note: Carefully review your input information as a startup script will be generated from them. So if you type in something wrong you will either need to adjust the startup script or redo all the steps

1.2 Exchange the public keys

Now, the server needs the public keys of every client, and the clients need the public-key of the server. This part is not automated yet and requires you to manually exchange the keys ( i.e. copy the public key directories to the other machine(s) ). For a server, the public key will be created at /keys/rosenpass-server-public. You can copy the key to your host machine using

docker cp server:/keys/rosenpass-server-public ./

From there you can copy the key to the destined container(s). For clients, the key will be located at: /keys/rosenpass-client-public

When you transfer all the client keys to the server, remember to either rename them to be able to distinguish between them or to save them to different directories

1.3 Start the server script

After transferring the public key directories from your clients to the server, log into the docker container of your server and run the start script with bash /etc/open_server_connection.sh You will be prompted for the location where you saved each public key directory of the clients. Type in one at a time and submit with "enter". Afterward you will need to type in the IP that is allowed for this key (in other words: which VPN IP shall be assigned to this key. This is the client VPN IP of the machine that will be using the according private key to authenticate). Submit with "enter". Repeat the process for each key and IP. Afterward, the server will start and allow the connections from the clients.

Example: 
From client1 the prompts could look like this
1. path/to/the/pubkey-of-client1/
2. 10.11.12.101

1.4 Start the client scripts

After transferring the public key directory from your server to the clients, log into the each client docker container and run the start script with bash /etc/connect_to_server.sh You will be prompted for the location of the servers public key and afterward for the allowed IP range the tunnel shall be established for. In this example, use 10.11.12.0/24 to be able to connect to all clients. Afterward, the clients will establish a tunnel to the setup server Repeat this on each client machine.

Example:
1. path/to/the/pubkey-of-server/
2. 10.11.12.0/24

Here, we are using the CIDR that contains all the clients and the server in the VPN to be able to reach them. Your setup could look different depending on your use case.

1.5 Connection test

From an arbitrary client, ping the server by using ping 10.11.12.100 This should ping the server within the VPN. Afterward try to connect to another client (e.g. client 3) by running: ping 10.11.12.103 This will send the requests to the server and from there to the other client in the network. You can confirm this by running tcpdump -i rosenpass0 on the server during the ping. There should appear now entries for each ping.

(optional) 1.6 Set routes

When you are running the container in a native Unix environment or inside wsl but without docker desktop (installation guide) then you can also set routes via ip route add and point your desired network traffic through the docker network interface that is attached to the container. This will allow you to use the tunnel for all traffic sent from your device.
The following example will demonstrate how to route the traffic for the IP ranges of the VPN subnet (10.11.12.0/24) from the host of client1 through wireguard to server1. This will enable us to ping all other clients as well with their VPN IPs from the host machine of client1.

To do that, you will need to create a network with docker network create -d bridge --subnet=172.28.0.0/24 br0 and attach it to the container with docker network connect -ip 172.28.0.10 br0 client1 By doing this a network interface will be created which you should see with ip a. Afterward, run ip route add 10.11.12.0/24 via 172.28.0.10 to route all traffic with an IP of 10.11.12.0/24 through the wiregaurd container, which is setup to forward these addresses by using the tunnel to the VPN server (Allowed addresses in step 4).

To test the setup, try to ping e.g. the VPN IP of client2 (10.11.12.102) from the host of client1 ping 10.11.12.102. You should now be able to get a proper response.

Alternatively you can run the docker containers in host network mode, so that they share the sam network devices and routes with the host. Then you won't need to add the route manually as it will be created automatically on executing the script in the container.

2. Fully automated setup example with docker compose

In order to build and start the containers in a local environment, simply run from the root of the project:

nix build . 
docker load < result
cd tests
docker compose up --wait

This spins up 2 docker containers with the minimal rosenpass image, which are connected via a bridge network of docker.

To stop and cleanup, run:

docker compose down

To be able to ping the server container from the host via its VPN ip, follow these steps

2.1 Manual Tests

To manually test the containers, log into the client container and run: ping -6 -I rosenpass0 fe70::3 (ipv6 version) ping -I rosenpass0 172.26.0.3 (ipv4 version)

This will try to ping the server container via the rosenpass interface. On the server container you can run watch -n 0.2 'wg show all; wg show all preshared-keys' To see, that a peer has been connected. If you wait 2 minutes, you will also see, that the PSK will be replaced by rosenpass.

2.2 Automated Tests

The above test is also automated via pytest. To execute it, install the dependencies first via pip install -r tests/requirements.txt Then execute the tests by switching into the tests directory and run pytest -s

3. Manual setup without docker compose

It is also possible to start the images in 3 different modes, without using the docker compose file. For the modes available, please refer to the manual here

4. Setup on GNS3

To setup a minimal example on gns3, it is recommended to use 3 sonic switches. One will connect the other 2 leaf switches. Each leaf switch gets assigned a virtual client as children. This setup will allow us to communicate from one client with another one via leaf and spine switches, by using a rosenpass VPN to secure the communication. Please note that the clients will not be part of the VPN, only the switches. The topology is displayed below:

Architecture in GNS3

For this topology we assume the following:

  • Each switch is in a management VLAN called VLAN 100

  • Each leaf has its own VLAN to connect its clients (VLAN 200 and VLAN 250)

  • DHCP for the VLANs will be configured in the switches responsible

  • The following configuration will be applied:

    Configuration Value
    Spine VLan 100 IP 10.0.100.1
    Spine VPN IP 10.11.12.100
    Leaf 1 VPN IP 10.11.12.101
    Leaf 1 VLAN 100 10.0.100.2
    Leaf 1 VLAN 200 10.0.200.1
    Leaf 2 VPN IP 10.11.12.102
    Leaf 2 VLAN 100 10.0.100.3
    Leaf 2 VLAN 250 10.0.250.1
    Client 1 IP 10.0.200.2
    Client 2 IP 10.0.250.2

4.1 Setup GNS3 and configure DHCP

First you will need to replicate the architecture. Make sure to connect the internet sources on the eth0 ports, so for a each switch, eth0 should point to an arbitrary port of the hub that is connected to the GNS3 NAT.

4.1.1 Create Management VLAN 100

Now, in order to allow the the communication between the switches directly, it is necessary to setup a VLAN. Furthermore, a DHCP service is needed to assign IP addresses, since the sonic switches only come with a DHCP relay. Execute the following steps on the spine switch to setup the necessary VLAN for the Management connection:

sonic-cli
configure t
interface Vlan 100
no shutdown
ip address 10.0.100.1/24
exit
configure
interface Ethernet 0
no shutdown
switchport access Vlan 100
configure
interface Ethernet 1
no shutdown
switchport access Vlan 100

Assuming that the leaf spines are connected on the ports 1/1 and 1/2 on the spine, as depicted above, this will setup the VLAN in the spine and attach the respective ports to this. This VLAN is used as management connection.

does not work correctly at the moment

At last, save the dhcpd.conf file which can be found here to /home/admin/data/dhcpd.conf Then you can execute sudo tpcm install name dhcp pull networkboot/dhcpd --args "--network host -v /home/admin/data/:/data" to start the dhcp container in order to assign the vlan ips automatically to all clients. Using tpcm, this container will restart on switch restarts as well.

workaround

go to the leaf swtiches and execute the following (depending on the leaf x = 2 or 3):

sonic-cli
configure t
interface Ethernet0
no sh
ip addr 10.0.100.x/24
4.1.2 setup VLANs on the leafs

Now on the leaf switches run the following (Assuming the spine is connected to the leaf on the leafs port 1/1 and the kali linux (client of the leaf) is connect to port 1/2 ) Leaf1:

sonic-cli
configure t
interface vlan 200
ip address 10.0.200.1/24
no shutdown
exit
interface Ethernet1
no shutdown
switchport access Vlan 200

Leaf2:

sonic-cli
configure t
interface vlan 250
ip address 10.0.250.1/24
no shutdown
exit
interface Ethernet1
no shutdown
switchport access Vlan 250

This will also create the VLAN on the leafs and assign the ports to attach to it.

4.1.3 Start DHCP on the leafs

In order to use the created VLANs, the DHCP servers need to be setup as well. In order to do that, copy the according configurations, which can be found here for leaf 1 and leaf 2. Copy these to the directory /home/data and name them dhcpd.conf

Afterwards, to start the dhcp server, run:

sudo tpcm install name dhcp pull networkboot/dhcpd --args "--network host -v /home/admin/data/:/data"

When you now start the Kali linux instance (which should be connected on eth0 to the leaf switch) then you will notice that it gets an IPv4 address assigned.

4.2 Start the containers on the nodes

Start the rosenpass containers for each node accordingly. For the spine switch, run:

docker run -d --cap-add=NET_ADMIN --network host -e MODE=server -e SERVER_PUBLIC_IP="10.0.100.1" -e SERVER_PORT=9999 -e SERVER_VPN_IP="10.11.12.100/24" --privileged --name=server --restart always -v /home/admin/keys:/keys stefan96/rosenpass:v0.2.1-SNAPSHOT

Then we will need to change the access mode on the generated keys in the keys directory that we mounted on container start. Also we will preemptively create directories for the client keys to come:

sudo chmod -R 777 /home/admin/keys/rosenpass-server-public/
sudo mkdir /home/admin/keys/rosenpass-client1-public/
sudo mkdir /home/admin/keys/rosenpass-client2-public/
sudo chmod -R 777 /home/admin/keys/rosenpass-client2-public/ /home/admin/keys/rosenpass-client1-public/

For Leaf1, run:

docker run -d --cap-add=NET_ADMIN --network host -e MODE=client -e SERVER_PUBLIC_IP="10.0.100.1" -e SERVER_PORT=9999 -e CLIENT_VPN_IP="10.11.12.101/24" --privileged --name=client --restart always -v /home/admin/keys:/keys stefan96/rosenpass:v0.2.1-SNAPSHOT

sudo chmod -R 777 /home/admin/keys/rosenpass-client-public/
sudo mkdir /home/admin/keys/rosenpass-server-public/
sudo chmod -R 777 /home/admin/keys/rosenpass-server-public/

For Leaf2, run:

docker run -d --cap-add=NET_ADMIN --network host -e MODE=client -e SERVER_PUBLIC_IP="10.0.100.1" -e SERVER_PORT=9999 -e CLIENT_VPN_IP="10.11.12.102/24" --privileged --name=client --restart always -v /home/admin/keys:/keys stefan96/rosenpass:v0.2.1-SNAPSHOT

sudo chmod -R 777 /home/admin/keys/rosenpass-client-public/
sudo mkdir /home/admin/keys/rosenpass-server-public/
sudo chmod -R 777 /home/admin/keys/rosenpass-server-public/

Here we do not use tpcm since it would prevent us from executing commands in the container. Instead we are using dockers restart policy to restart the containers automatically. By executing these steps, the container will be set up on the switch and the public and private keys will be copied to the host machine (This is helpful for the subsequent key exchange but also for bringing your own keys in the container). Also we preemptively took care on permission errors when trying to copy the keys.

4.3 Use scp to get the keys for the server/clients

Use scp to login to the other switch(es) and download the public keys from there.

For the spine switch: Assuming the IPs of the leaf switches are 10.0.100.2 and 10.0.100.3

scp [email protected]:/home/admin/keys/rosenpass-client-public/* /home/admin/keys/rosenpass-client1-public
scp [email protected]:/home/admin/keys/rosenpass-client-public/* /home/admin/keys/rosenpass-client2-public

For Leaf 1 and 2: Assuming that the IP of the spine is 10.0.100.1 scp [email protected]:/home/admin/keys/rosenpass-server-public/* /home/admin/keys/rosenpass-server-public

This will exchange the public keys of the switches and mount them directly into the rosenpass container for further usage.

4.4 Execute the scripts as mentioned above

Execute the startup scripts for the clients and the server like mentioned above start the server start the client

Please note however, that there are several differences regarding the allowed IPs: On the spine (VPN Server) you will assign for each leaf switch 2 allowed IPs. One for the VPN connection and one for routing the VLAN connection to the children of the leaf. For instance, in the container of the spine, to add Leaf 1 as a peer, the config you will need to enter during the scripted setup is:

key location: /keys/rosenpass-client1-public
allowed IPs: 10.11.12.101/32,10.0.200.0/24

For the second leaf respectively:

key location: /keys/rosenpass-client2-public
allowed IPs: 10.11.12.102/32,10.0.250.0/24

On the leaf switches, you will also need to setup the connection a bit differently. Here the allowed IP section will be composed of the VPN IP range and the VLAN IP range of the VLAN of the other switch. So for Leaf 1, enter:

key location: /keys/rosenpass-server-public
allowed IPs: 10.11.12.0/24,10.0.250.0/24

for Leaf 2, enter:

key location: /keys/rosenpass-server-public
allowed IPs: 10.11.12.0/24,10.0.200.0/24

This will ensure, that the necessary traffic can be routed over the VPN

4.7 Add routing for the VLAN over VPN

At last,the nodes themselves need to be able to route the traffic between each other. Therefore, on the Spine, execute:

sudo ip route add 10.0.250.0/24 via 10.11.12.102
sudo ip route add 10.0.200.0/24 via 10.11.12.101

on Leaf1 run:

sudo ip route add 10.0.250.0/24 via 10.11.12.102

on Leaf2 run:

sudo ip route add 10.0.200.0/24 via 10.11.12.101

4.8 Test ping from client 1 to client 2

To test the connection, run from kali 1: ping 10.0.250.2 The ping should work fine. However, if the connection did not work, there might be an issue during the conduction of the setup or the rosenpass clients could not yet establish the connection to the server. This is crucial, as the server at the beginning does not know where the clients are and needs to wait for them to authenticate to it. To circumvent that, run a simple ping from each leaf node to the spine:

ping 10.11.12.100

If successful, this authenticates the clients with the server

To verify that the traffic is routed over the VPN, log into the spine and execute: docker exec -it server watch -n 2 'wg show rosenpass0' You should see a small increase in the traffic section on each ping

5. Connecting clients with stateful rules instead of vlans

In order to setup a connection without the need of the management VLAN, one could pursue one of the following ways:

  1. Use Vlans in the leafs and spine to have the kali instances receive IPs to ping each other. Then use ip routes to route the traffic between the sonic devices, e.g. ip route add 10.0.200.0/24 dev Ethernet1 in combination with iptables Rule of the nat table to masquerade the traffic so that the source IP is adapted on each hop

  2. Use no VLAN at all and bridge the ethernet interfaces on each switch together (e.g. Ethernet 1/1 and Ethernet 1/2 in one bridge of the Leaf Switch 1) so that the traffic is directly forwarded to the next switch. However, this for one won't work with the VPN setup and second would essentially downgrade the router to a regular switch.

  3. Use of a firewall of any sort to alter/accept the packages by stateful rules. This approach however, cannot be used to redirect the traffic along the interfaces of the routers since they have no capability in changing the outgoing layer2 interface as they operate on layer 3

Since the other 2 approaches are not as recommendable, the following describes the setup for approach Number 1

Leaf 1

iptables -t nat -A POSTROUTING -o Ethernet0 -j MASQUERADE
iptables -A FORWARD -i Ethernet1 -o Ethernet0 -m state --state RELATED,ESTABLISHED -j ACCEPT
iptables -A FORWARD -i Ethernet1 -o Ethernet0 -j ACCEPT

ip route add 10.0.250.0/24 dev Ethernet0

Leaf 2

iptables -t nat -A POSTROUTING -o Ethernet0 -j MASQUERADE
iptables -A FORWARD -i Ethernet1 -o Ethernet0 -m state --state RELATED,ESTABLISHED -j ACCEPT
iptables -A FORWARD -i Ethernet1 -o Ethernet0 -j ACCEPT

ip route add 10.0.200.0/24 dev Ethernet0

Spine

iptables -t nat -A POSTROUTING -o Ethernet0 -j MASQUERADE
iptables -t nat -A POSTROUTING -o Ethernet1 -j MASQUERADE

iptables -A FORWARD -i Ethernet1 -o Ethernet0 -j ACCEPT
iptables -A FORWARD -i Ethernet0 -o Ethernet1 -j ACCEPT

iptables -A FORWARD -i Ethernet1 -o Ethernet0 -m state --state RELATED,ESTABLISHED -j ACCEPT
iptables -A FORWARD -i Ethernet0 -o Ethernet1 -m state --state RELATED,ESTABLISHED -j ACCEPT

ip route add 10.0.250.0/24 dev Ethernet1
ip route add 10.0.200.0/24 dev Ethernet0

6. Load tests

In order to further evaluate the functionality of the solution, load tests were conducted in order to visualize the data flow. The therefor used scripts are located here. The results are depicted below: Visualization of the load distribution As you can see once the load generator started from kali1, all other systems also faced a significant increase in CPU load. This is to be expected and shows that the traffic flew along all the nodes. Via tcpdump it could be seen that the traffic also flew along the correct interfaces once again.

7. Tools used

The following section describes the Tools that were used to build, bundle and test the image, as well as tools which supported the development process.

7.1 Nix

The package management tool Nix is used to build rosenpass from source and to inject it into a minimal docker container

The following packages are included by using Nix:

Nix-package Package Description Reason for Inclusion Required
rosenpass.packages.x86_64-linux.rosenpass The rosenpass functionalities, build from source To be able to use rosenpass in the final container True
pkgs.coreutils-full The GNU Core Utilities are the basic file, shell and text manipulation utilities of the GNU operating system. This is the full version that includes additional utilities. Adds different necessary functionalities for practical work with the container (debugging, rosenpass-script needs certain cli tools as well) True
pkgs.iana-etc Package to include protocols for communication Needed to include /etc/protocols in the image to be able to communicate e.g. via imcp for a ping True
pkgs.gnugrep GNU grep, a tool for searching text using regular expressions. The rosenpass implementation relies on grep, otherwise it cannot function True
pkgs.iproute2 A collection of utilities for Linux networking. ip command is necessary to create network devices for rosenpass and wireguard True
pkgs.procps Package to include sysctl sysctl is needed to be able to configure ipv6 correctly False
pkgs.wireguard-tools Package to include wireguard Is needed to establish a VPN connection between servers False
pkgs.iputils Package to include ping and other network tools Is needed to test the connection between the servers False
pkgs.bash Bash is the GNU Project's shell—the Bourne Again SHell. This is an sh-compatible shell that incorporates useful features from the Korn shell (ksh) and C shell (csh). Bash to be able to debug and login to the container via cli False
pkgs.tcpdump tcpdump utility to watch network traffic Really useful to include for debugging purposes False

7.1.1 Package size

Since it is not possible to retrieve the package sizes from an official source, it was tested, what size a minimum image will have. By tests, a plain Nix image will have roughly 1 MB of Space occupied. To have only the rosenpass tool included (which is not functional on its own), 67MB of space are required. All packages that are strictly required for rosenpass to work, consume 94MB of space (Wireguard is not included). When including Wireguard as well, the image takes up to 132MB. The image with all the packages listed above, needs 132MB as well. These results were obtained by using each combination of packages as mentioned above with a plain image and observing the size properties of the results in the nix path. via nix path-info -Sh ./result

7.1.2 Flake

Since rosenpass is relying on the experimental functionality of "flakes", this project also harnesses its capabilities. Flake allows even more reproducibility than normal Nix.

7.2 Docker

As container runtime environment, docker is used as the defacto standard solution. With these, the resulting image can easily be integrate into the SONic switches, which are already using Docker for their functionalities.

7.2.2 Docker configuration

In order to be able to setup rosenpass correctly in docker containers, a few things need to be considered:

  • In order to prevent permission and RETNETLINK errors, the container needs:
    • root privileges
    • the NET_ADMIN capability
  • Additionally for ipv6 compatibility the container need:
    • sysctl options enabled to allow ipv6

7.3 Testing Frameworks

In order to prevent errors and enable users to quickly test out the images, small tests are created to ensure that.

7.3.1 Pytest

As testing framework, Pytest was used, because Python offers a small and lightweight solution for creating tests with docker containers. At the moment the tests directory contains 2 test scenarios that are checked each time in the CI Pipeline (on PR and on Release). These Tests cover the following topics:

  • Setting up a client and a server container, using IPv4 and running a connection Test
  • Setting up a client and a server container, using IPv6 and running a connection Test

On top of these, more complex and exhaustive testing can be implemented.

7.3.2 Act

Act is a CLI tools to test out github action workflows locally in docker containers, without them running on the remote github actions runner and thus causing spam for everyone subscribed to the Repository.

About

Evaluating SoNiC switches by creating a PQC-secure inter-switch communication

Topics

Resources

License

Stars

Watchers

Forks