Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/maelvls/cloudroutesync

♾ Sync netlink routes with your cloud routing table (author: @networkop)
https://github.com/maelvls/cloudroutesync

netlink-protocol routing

Last synced: 2 months ago
JSON representation

♾ Sync netlink routes with your cloud routing table (author: @networkop)

Awesome Lists containing this project

README

        

# cloudroutesync
An add-on for cloud-hosted routers that reads local routes via [rtnetlink](https://man7.org/linux/man-pages/man7/rtnetlink.7.html) and synchronizes them with a cloud routing table.

![](./image.png)

The main use cases are:

* Overlay-free self-managed Kuberentes clusters in the cloud - you can run BGP to establish reachability between PodCIDRs (e.g. KubeRouter, Calico)
* Multi/Hybrid cloud - sync local route table with subnets configured in a different environments (on-prem, another cloud).

> Note: The application relies on another routing daemon to program local netlink routes. This can be [FRR](http://docs.frrouting.org/en/latest/), [Quagga](https://www.nongnu.org/quagga/docs/quagga.html), [Bird](https://bird.network.cz/) or any other routing software suite.

## Currently Supported Clouds

* AWS
* Azure
* GCP*
* Openstack (maybe)

> Due to limitations of GCP's networking stack, the only supported mode is syncronization of routes received from outside of the local subnet. These routes will be set with nextHop of the router VM running cloudroutesync.

## Prerequisites

The application must be running on a cloud VM with enough IAM permissions to create/update cloud route table.

For example, in AWS this would require:

* Create/Delete RouteTable
* Associate/Deassociate RouteTables
* Create/Delete Routes
* Create/Delete Tags
* Describe NetworkInterfaces and Instances

In addition to that, any VM that needs to send packets not natively known to the cloud SDN, need to have IP source check disabled.

See Terraform [directory](./terraform) for examples for:

* [AWS](./terraform/aws/main.tf)
* [Azure](./terraform/azure/main.tf)
* [GCP](./terraform/gcp/main.tf)

## Installation

To build a binary:

```
go get -v github.com/networkop/cloudroutesync
```

Alternatively, it is available as a docker image at `networkop/cloudroutesync`.

## Usage

```
Usage of ./cloudroutesync:
-cleanup
cleanup any created objects
-cloud string
public cloud providers [azure|aws|gcp]
-debug
enable debug logging
-event
enable event-based sync (default is periodic, controlled by 'sync')
-netlink int
netlink polling interval in seconds (default 10)
-sync int
cloud routing table sync interval in seconds (default 10)
```

It can run in two modes:

* Event-driven mode - cloud route table is only updated whenever there was a change detected in the netlink routing table. This mode is enabled with a `-event` flag.

* Periodic mode (default) - cloud route table is synced periodically based on the interval defined in the `-sync` flag.

## Demo

Demonstration can be done using any of the supported providers from the terraform [directory](./terraform).
Here we'll use AWS as an example.

1. Spin up a test environment with two VMs

```
cd ./terraform/aws
terraform init && terraform apply -auto-approve

```

3. SSH into both VMs and bring up the demo application.

Router VM will run both the FRR and the `cloudroutesync`:

```
router_ip=$(terraform output -json | jq -r '.public_address_router.value[0]')
ssh ubuntu@$router_ip
ubuntu@ip-10-0-1-31:~$ sudo CLOUD=aws docker-compose up -d
```

Second, non-router VM will run only the FRR container:

```
vm_ip=$(terraform output -json | jq -r '.public_address_vm.value[0]')
ssh ubuntu@$vm_ip
ubuntu@ip-10-0-1-195:~$ sudo docker-compose up -d frr
```

3. From a non-router VM and configure a BGP peering towards the cloud router (replace 10.0.1.31 with the private IP of the router VM)

```
ubuntu@ip-10-0-1-195:~$ sudo docker-compose exec frr vtysh
conf
router bgp
neighbor 10.0.1.31 peer-group PEERS
```

4. From the same VM configure a new loopback IP and redistribute it into BGP

```
interface lo
ip address 198.51.100.100/32
!
router bgp
redistribute connected
```

5. From a non-router VM start a ping towards router VM sourced from the new interface (replace 10.0.1.31 with the private IP of the router VM)

```
ubuntu@ip-10-0-1-195:~$ ping 10.0.1.31 -I 198.51.100.100
```

6. From a router VM observe the logs of the `cloudroutesync` service:

```
ubuntu@ip-10-0-1-31:~$ sudo docker-compose logs crs
crs_1 | time="2020-10-12T21:11:08Z" level=info msg="Starting Virtual Cloud Router"
crs_1 | time="2020-10-12T21:11:08Z" level=info msg="Running on AWS"
crs_1 | time="2020-10-12T21:11:08Z" level=info msg="Checking routing table"
crs_1 | time="2020-10-12T21:11:19Z" level=info msg="Checking routing table"
crs_1 | time="2020-10-12T21:11:29Z" level=info msg="Checking routing table"
crs_1 | time="2020-10-12T21:11:39Z" level=info msg="Checking routing table"
crs_1 | time="2020-10-12T21:11:39Z" level=info msg="Route change detected"
crs_1 | time="2020-10-12T21:11:39Z" level=info msg="Creating route 198.51.100.100/32 in rtb-007864623346bee20"
crs_1 | time="2020-10-12T21:11:49Z" level=info msg="Checking routing table"
```

7. Observe how route table gets populated with the new prefix.

The ping from step #5 should now receive responses.

8. At the end of the test, to cleanup any leftover state like route tables and their association run:

```
cloudroutesync -cloud aws cleanup
INFO[0000] Starting Virtual Cloud Router
INFO[0000] Running on AWS
INFO[0000] Deleting own route table
```