In this tutorial, we are going to explore how to migrate from MetalLB to Cilium.
Before I go into the technical details, let me explain the reasoning behind the cover above. It was actually the illustration of the booth we had at KubeCon US in 2023.
It was originally inspired by an illustration that my colleague Bill posted on Twitter.
Let me describe it – not only for accessibility purposes but also in the likelihood Twitter/X will be gone by the time you read this post. The original tweet by @theburningmonk shows two people on a deserted surface, with one asking the other: “I’m pretty sure the application is somewhere around here”. Unbeknownst to them, the application is buried under them, under several layers of cloud native tools – Load-balancer, Ingress, Kube-proxy, Service Mesh and Side-car.
In response, Bill replied: “Fixed it, just needed one Cilium” and pasted a logo of Cilium on top of all the cloud native tools – as if to say, Cilium can do it all.
I’ve talked several times before about how Cilium can help folks deal with the overwhelming volume of cloud native networking tools (for example, with Cilium’s Gateway API support) and even made one of my predictions that users will look at simplifying their stack of tools.
As highlighted in the Cilium 1.14 release blog post, one tool that Cilium users might not need any longer is MetalLB.
Use cases for MetalLB
Let me start by saying – MetalLB is a very popular and reliable bare metal load-balancer tool.
MetalLB is indeed used to connect a cluster to the rest of the network – with BGP or, most commonly in home labs, with Layer 2 announcements (I will explain what this refers to later). It also provides IP Address Management (IPAM) capabilities for Kubernetes Services of the type LoadBalancer and generally acts as a Load Balancer for self-managed clusters (the cloud managed Kubernetes offerings tend to automatically do IPAM and LoadBalancing for you).
And MetalLB was used by Cilium in its early support of BGP, it was used in Cilium’s own CI/CD pipelines and was even used in our popular Isovalent Cilium labs.
I used the past tense purposely in the previous sentence: Cilium no longer leverages MetalLB for BGP (we use GoBGP instead), it has been removed from the GitHub Actions CI/CD testing and is no longer used in our labs. As one of the lab maintainers, I had no issue with MetalLB – I simply wanted to reduce the number of tools and dependencies.
Let’s walk through how Cilium supports these features and how we migrated from MetalLB in our labs.
Load-Balancer IPAM
First, we are going to look at one of the key use cases supported by MetalLB – IP address management – and we will walk through how Cilium addresses it.
As described in the Kubernetes documentation, Kubernetes Services of the type LoadBalancer “expose the Service externally using an external load balancer. Kubernetes does not directly offer a load balancing component; you must provide one, or you can integrate your Kubernetes cluster with a cloud provider.”
One of the tasks operated by such external load balancer is to assign an external IP address to the Kubernetes Services. As mentioned, the tool of choice for this had often been MetalLB. This is particularly relevant in self-managed Kubernetes clusters as, when using cloud managed Kubernetes clusters, an IP and DNS entry will automatically be assigned to Kubernetes Services of the type LoadBalancer.
Cilium introduced support for Load-Balancer IP Address Management in Cilium 1.13 and we’ve seen a quick adoption of this feature. Let’s review with an example.
Load Balancer IP Address Management with Cilium
First, note this feature is enabled by default but dormant until the first IP Pool is added to the cluster. From this pool of IP addresses (defined in a CiliumLoadBalancerIPPool
) will be allocated IPs to Kubernetes Services of the type LoadBalancer
.
LoadBalancer Services can be automatically created when exposing Services externally via the Ingress or Gateway API Resources but for this example, we’ll create a Service of the type LoadBalancer manually.
Let’s review a simple IP Pool:
IP addresses from the 20.0.10.0/24
range will be assigned to a LoadBalancer Service.
Before we deploy the pool, check what happens when we deploy a Service of the type LoadBalancer
. The service-blue
Service is a simple Service that is labelled color:blue
and is in the tenant-a
namespace.
Let’s deploy it. At first, no IP address has been allocated yet (the External-IP
is still <pending>
).
Next, we deploy the Cilium IP Pool:
Let us check again the Service – an IP address from the 20.0.10.0/24
pool has been assigned in the field EXTERNAL-IP
.
Service Selectors
Operators may want to have applications and services assigned IP addresses from specific ranges. This might be useful to then enforce network security rules on an traditional border firewall.
For example, you may want test or production services to get IP addresses from different ranges and apply different rules on the firewall as permissions to the test network might be looser than to the production one.
In our example, we will keep using colors. Services (in the tenant-b
namespace) tagged with blue
, yellow
or red
will be allowed to pick up IP addresses from a primary
colors IP pool.
This would be done by using a Service Selector, with a regular expression such as the one defined in pool-primary.yaml
:
The expression - {key: color, operator: In, values: [yellow, red, blue]}
checks whether the Service requesting the IP has a label with the key color
with a value of either yellow
, red
or blue
.
Let’s deploy it:
This time, we’ll try this with a Service that hasn’t got the right label and one with the right label.
Review their assigned IPs and labels in the output above. Unlike service-red
, service-green
does not get an IP address. It’s simply because it doesn’t match the regular expression defined in the previously defined pool (green is not one of the primary colors).
Let’s now use an IP Pool that matches the color:green
label:
This time, an IP address from the 40.0.10.0/24
was assigned to service-green
.
Requesting a specific IP
Users might want to request a specific set of IPs from a particular range. Cilium supports two methods to achieve this: the .spec.loadBalancerIP
method (legacy, deprecated in K8S 1.24) and an annotation-based method.
The former method has now been deprecated because it did not support dual-stack services (only a single IP could be requested). With the latter annotation method, a list of requested (v4 or v6) IPs will be specified instead. We will look at an Dual Stack example later on.
Review this yellow
service. Note the annotation used to requested specific IPs. In this scenario, we are requesting 4 IP addresses:
We’re also going to use another pool, this time, matching on the namespace (tenant-c
in our example).
Let’s deploy this service and pool and observe which IP addresses have been allocated:
Note that the Service was allocated two of the requested IP addresses:
50.0.10.100
because it matches the namespace (tenant-c
).60.0.10.100
because it matches the primary colors labels.
Two other IP addresses were not assigned:
30.0.10.100
is not part of any defined pools.40.0.10.100
is part of an existing pool but itsserviceSelector
doesn’t match with the Service (remember that this is part of thepool-green
pool which matched on a green label).
L3 Announcement over BGP
Now that we have one (or multiple) IP address(es) assigned to our Services, we need to advertise them to the rest of the network so that external clients can reach them (and the Service it’s fronting).
We covered BGP in more details in a recent blog post (“Connecting your Kubernetes island to your network with Cilium BGP“) but to keep it short, know that BGP on Cilium enables you simply to set up BGP peering sessions between Cilium-managed nodes and Top-of-rack (ToR) devices and to tell the rest of the network about the networks and IP addresses used by your pods and your services.
Setting up BGP can be done by using a CiliumBGPPeeringPolicy
such as the one below:
As you can see in the YAML above, we can specify which services are advertised, using a service selector. In our example, we only want to advertise to our peers Services with the color:yellow
label and located in the tenant-c
namespace.
Once this peering policy is applied (and assuming that the peer has been correctly configured), the peer will see routes such as:
The peer can see the Pod CIDR 10.244.0.0/24
advertised by Cilium but also the LoadBalancer Service IPs that match the filters defined in the Service Selector.
IPv6 Support
The examples so far focused on IPv4 but both LB-IPAM and BGP on Cilium support IPv6.
Here is an example of an LB-IPAM pool that would assign both an IPv4 and IPv6 address:
When deploying a DualStack Service with the right label, the Service receives both an IPv4 and IPv6 address:
We can peer over IPv6 with our BGP neighbor and advertise our IPv6 address accordingly:
When we log onto our neighbour, we can see that the Service IPv6 address (alongside the IPv6 PodCIDR) has been received:
Note how our Top of Rack device is actually peering with multiple Cilium nodes.
Here is an illustration of the network topology highlighted in this paragraph:
All this is nice if you love routing and networking but, for those of you with a difficult relationship with BGP, we have another option for you.
L2 Announcement with ARP
While using BGP to advertise our Service IPs works great, not every device in the network will be BGP-capable and BGP might be overkill for some use cases, for example, in home labs.
One popular MetalLB use case that is now supported by Cilium is Layer 2 announcement, using ARP.
If you have local clients on the network that need access to the Service, they need to know which MAC address to send their traffic to. This is the role of ARP – the Address Resolution Protocol is used to discover the MAC address associated with a particular IP.
In the image above, our clients Client-A and Client-B are located on the same network as the Kubernetes LoadBalancer Service. In this case, BGP is superfluous; the clients simply need to understand which MAC address is associated with the IP address of our service so that they know to which host they need to send the traffic to.
What Cilium can do is reply to ARP requests from clients for LoadBalancer IPs or External IPs and enable these clients to access their services.
The feature can be tested in our Cilium LB-IPAM and L2 Announcements lab:
To enable this feature, use the following flags:
Let’s verify it’s enabled with the following command:
Why don’t we walk through another Star Wars-inspired example? We want to advertise the following Death Star service locally. Notice how the DeathStar service has an external IP address (12.0.0.101
) and the org:empire
label.
Let’s advertise this Service locally with a simple CiliumL2AnnouncementPolicy
. The policy below is pretty simple to understand. It will advertise locally the IP addresses of any Services with the label org: empire
, whether they are External IPs or LoadBalancer IPs. Only the worker nodes will be replying to ARP requests (the node selector expression excludes the control plane node).
Once deployed, a client from outside the cluster can access the service:
If you really want to know what happens under the hood, here is a screenshot of termshark showing the ARP request and reply. In this instance, the client (with an IP of 12.0.0.1
) wants to access a Service at 12.0.0.101
and sends an ARP request as a broadcast asking “Who has 12.0.0.101? Tell 12.0.0.1”.
The Cilium node will reply back to the client with its own MAC address (“12.0.0.101 is at aa:c1:ab:05:ed:67”).
This works great and is a fantastic alternative when BGP is unwanted or unneeded.
The only drawback with Layer 2 Announcement is that it does not currently support IPv6.
Translating MetalLB configuration to Cilium
So far, we’ve reviewed how many of the common MetalLB use cases can be natively addressed by Cilium. To help you migrate from MetalLB, here is a sample configuration of our own migration from MetalLB to Cilium’s LB-IPAM and L2 Announcement features.
Here is the MetalLB configuration we used in our popular Service Mesh and Gateway API labs:
Here is the equivalent Cilium configuration:
Both configurations would allocate an IP address from the 172.18.255.192/28
IP range and would announce the LoadBalancer IP from the eth0
network interfaces of the NodeA
node.
Conclusion
With every release, Cilium becomes so much more than just a Container Network Interface (CNI). It is now an Ingress controller, a Gateway API implementation, a VPN and, as you saw in this blog post, it can also act as a bare metal load balancer. And while many users will continue to happily use MetalLB, those of you who already use Cilium may decide that one fewer tool to manage already helps reducing the operational fatigue.
Thanks for reading.
Learn More
Prior to joining Isovalent, Nico worked in many different roles—operations and support, design and architecture, and technical pre-sales—at companies such as HashiCorp, VMware, and Cisco.
In his current role, Nico focuses primarily on creating content to make networking a more approachable field and regularly speaks at events like KubeCon, VMworld, and Cisco Live.