Setting up Cluster Mesh
- PodCIDR ranges in all clusters must be non-conflicting.
- This guide and the referenced scripts assume that Cilium was installed using the Installation with managed etcd instructions which leads to etcd being managed by Cilium using etcd-operator. You can use any way to manage etcd but you will have to adjust some of the scripts to account for different secret names and adjust the LoadBalancer to expose the etcd pods.
- Nodes in all clusters must have IP connectivity between each other. This requirement is typically met by establishing peering or VPN tunnels between the networks of the nodes of each cluster.
- All nodes must have a unique IP address assigned them. Node IPs of clusters being connected together may not conflict with each other.
- Cilium must be configured to use etcd as the kvstore, along with the identity allocation mode (). With the identity allocation mode set to
kvstore
, this allows direct etcd connections, identity propagation across the clusters, and enables cross-cluster policy functionality. Consul is not currently supported with cluster mesh. - It is highly recommended to use a TLS protected etcd cluster with Cilium. The server certificate of etcd must whitelist the host name
*.mesh.cilium.io
. If you are using thecilium-etcd-operator
as set up in the instructions then this is automatically taken care of. - The network between clusters must allow the inter-cluster communication. The exact ports are documented in the Firewall Rules section.
Prepare the clusters
Each cluster must be assigned a unique human-readable name. The name will be used to group nodes of a cluster together. The cluster name is specified with the --cluster-name=NAME
argument or cluster-name
ConfigMap option.
To ensure scalability of identity allocation and policy enforcement, each cluster continues to manage its own security identity allocation. In order to guarantee compatibility with identities across clusters, each cluster is configured with a unique cluster ID configured with the --cluster-id=ID
argument or cluster-id
ConfigMap option. The value must be between 1 and 255.
Repeat this step for each cluster.
Note
This can also be done by passing --set global.cluster.id=<id>
and --set global.cluster.name=<name>
to helm install
when installing or updating Cilium.
Expose the Cilium etcd to other clusters
The Cilium etcd must be exposed to other clusters. There are many ways to achieve this. The method documented in this guide will work with cloud providers that implement the Kubernetes LoadBalancer
service type, as well as with services of type NodePort
(assuming that nodes can reach each other using their internal IPs):
GCP
AWS
NodePort
apiVersion: v1
kind: Service
metadata:
name: cilium-etcd-external
annotations:
cloud.google.com/load-balancer-type: "Internal"
# if all the clusters are in the same region you can comment out this annotation
networking.gke.io/internal-load-balancer-allow-global-access: "true"
spec:
type: LoadBalancer
ports:
- port: 2379
selector:
app: etcd
etcd_cluster: cilium-etcd
io.cilium/app: etcd-operator
apiVersion: v1
kind: Service
metadata:
name: cilium-etcd-external
annotations:
service.beta.kubernetes.io/aws-load-balancer-internal: 0.0.0.0/0
spec:
type: LoadBalancer
ports:
- port: 2379
selector:
app: etcd
etcd_cluster: cilium-etcd
io.cilium/app: etcd-operator
apiVersion: v1
kind: Service
name: cilium-etcd-external
spec:
type: NodePort
ports:
- port: 2379
selector:
app: etcd
etcd_cluster: cilium-etcd
io.cilium/app: etcd-operator
The example used here exposes the etcd cluster as managed by cilium-etcd-operator
installed by the standard installation instructions as an internal service which means that it is only exposed inside of a VPC and not publicly accessible outside of the VPC. It is recommended to use a static IP for the ServiceIP to avoid requiring to update the IP mapping as done in one of the later steps.
If you are running the cilium-etcd-operator you can simply apply the following service to expose etcd:
GCP
AWS
NodePort
kubectl apply -f https://raw.githubusercontent.com/cilium/cilium/v1.8/examples/kubernetes/clustermesh/cilium-etcd-external-service/cilium-etcd-external-gke.yaml
kubectl apply -f https://raw.githubusercontent.com/cilium/cilium/v1.8/examples/kubernetes/clustermesh/cilium-etcd-external-service/cilium-etcd-external-eks.yaml
kubectl apply -f https://raw.githubusercontent.com/cilium/cilium/v1.8/examples/kubernetes/clustermesh/cilium-etcd-external-service/cilium-etcd-external-nodeport.yaml
Note
Make sure that you create the service in namespace in which cilium and/or etcd is running. Depending on which installation method you chose, this could be or cilium
.
Extract the TLS keys and generate the etcd configuration
The cluster mesh control plane performs TLS based authentication and encryption. For this purpose, the TLS keys and certificates of each etcd need to be made available to all clusters that wish to connect.
Clone the
cilium/clustermesh-tools
repository. It contains scripts to extracts the secrets and generate a Kubernetes secret in form of a YAML file:git clone https://github.com/cilium/clustermesh-tools.git
cd clustermesh-tools
Ensure that the kubectl context is pointing to the cluster you want to extract the secret from.
Extract the TLS certificate, key and root CA authority.
./extract-etcd-secrets.sh
This will extract the keys that Cilium is using to connect to the etcd in the local cluster. The key files are written to
config/<cluster-name>.*.{key|crt|-ca.crt}
Repeat this step for all clusters you want to connect with each other.
Generate a single Kubernetes secret from all the keys and certificates extracted. The secret will contain the etcd configuration with the service IP or host name of the etcd including the keys and certificates to access it.
The key files in config/
and the secret represented as YAML are sensitive. Anyone gaining access to these files is able to connect to the etcd instances in the local cluster. Delete the files after the you are done setting up the cluster mesh.
For TLS authentication to work properly, agents will connect to etcd in remote clusters using a pre-defined naming schema {clustername}.mesh.cilium.io
. In order for DNS resolution to work on these virtual host name, the names are statically mapped to the service IP via the /etc/hosts
file.
The following script will generate the required segment which has to be inserted into the
cilium
DaemonSet:Apply the patch to all DaemonSets in all clusters:
kubectl -n kube-system patch ds cilium -p "$(cat ds.patch)"
1. Import the cilium-clustermesh
secret that you generated in the last chapter into all of your clusters:
kubectl -n kube-system apply -f clustermesh.yaml
- Restart the cilium-agent in all clusters so it picks up the new cluster name, cluster id and mounts the
cilium-clustermesh
secret. Cilium will automatically establish connectivity between the clusters.
kubectl -n kube-system delete pod -l k8s-app=cilium
- For global services to work (see below), also restart the cilium-operator:
kubectl -n kube-system delete pod -l name=cilium-operator
Test pod connectivity between clusters
Run cilium node list
to see the full list of nodes discovered. You can run this command inside any Cilium pod in any cluster:
$ kubectl -n kube-system exec -ti cilium-g6btl cilium node list
Name IPv4 Address Endpoint CIDR IPv6 Address Endpoint CIDR
cluster5/ip-172-0-117-60.us-west-2.compute.internal 172.0.117.60 10.2.2.0/24 <nil> f00d::a02:200:0:0/112
cluster5/ip-172-0-186-231.us-west-2.compute.internal 172.0.186.231 10.2.3.0/24 <nil> f00d::a02:300:0:0/112
cluster5/ip-172-0-50-227.us-west-2.compute.internal 172.0.50.227 10.2.0.0/24 <nil> f00d::a02:0:0:0/112
cluster5/ip-172-0-51-175.us-west-2.compute.internal 172.0.51.175 10.2.1.0/24 <nil> f00d::a02:100:0:0/112
cluster7/ip-172-0-121-242.us-west-2.compute.internal 172.0.121.242 10.4.2.0/24 <nil> f00d::a04:200:0:0/112
cluster7/ip-172-0-58-194.us-west-2.compute.internal 172.0.58.194 10.4.1.0/24 <nil> f00d::a04:100:0:0/112
cluster7/ip-172-0-60-118.us-west-2.compute.internal 172.0.60.118 10.4.0.0/24 <nil> f00d::a04:0:0:0/112
$ kubectl exec -ti pod-cluster5-xxx curl <pod-ip-cluster7>
[...]
Load-balancing with Global Services
Establishing load-balancing between clusters is achieved by defining a Kubernetes service with identical name and namespace in each cluster and adding the annotation io.cilium/global-service: "true"`
to declare it global. Cilium will automatically perform load-balancing to pods in both clusters.
Deploying a simple example service
In cluster 1, deploy:
kubectl apply -f https://raw.githubusercontent.com/cilium/cilium/v1.8/examples/kubernetes/clustermesh/global-service-example/cluster1.yaml
In cluster 2, deploy:
kubectl apply -f https://raw.githubusercontent.com/cilium/cilium/v1.8/examples/kubernetes/clustermesh/global-service-example/cluster2.yaml
From either cluster, access the global service:
kubectl exec -ti xwing-xxx -- curl rebel-base
You will see replies from pods in both clusters.
As addressing and network security is decoupled, network security enforcement automatically spans across clusters. Note that Kubernetes security policies are not automatically distributed across clusters, it is your responsibility to apply CiliumNetworkPolicy
or NetworkPolicy
in all clusters.
The following policy illustrates how to allow particular pods to allow communicate between two clusters. The cluster name refers to the name given via the --cluster-name
agent option or cluster-name
ConfigMap option.
apiVersion: "cilium.io/v2"
metadata:
name: "allow-cross-cluster"
description: "Allow x-wing in cluster1 to contact rebel-base in cluster2"
spec:
endpointSelector:
matchLabels:
name: x-wing
io.cilium.k8s.policy.cluster: cluster1
- toEndpoints:
- matchLabels:
name: rebel-base
io.cilium.k8s.policy.cluster: cluster2
Troubleshooting
Use the following list of steps to troubleshoot issues with ClusterMesh:
Generic
Validate that the
cilium-xxx
as well as thecilium-operator-xxx
pods are healthy and ready. It is important that thecilium-operator
is healthy as well as it is responsible for synchronizing state from the local cluster into the kvstore. If this fails, check the logs of these pods to track the reason for failure.Validate that the ClusterMesh subsystem is initialized by looking for a
cilium-agent
log message like this:
level=info msg="Initializing ClusterMesh routing" path=/var/lib/cilium/clustermesh/ subsys=daemon
Control Plane Connectivity
Run
cilium node list
in one of the Cilium pods and validate that it lists both local nodes and nodes from remote clusters. If this discovery does not work, validate the following:
In each cluster, check that the kvstore contains information about local nodes by running:
Note
The kvstore will only contain nodes of the local cluster. It will not contain nodes of remote clusters. The state in the kvstore is used for other clusters to discover all nodes so it is important that local nodes are listed.
Validate the connectivity health matrix across clusters by running
cilium-health status
inside any Cilium pod. It will list the status of the connectivity health check to each remote node.If this fails:
- Make sure that the network allows the health checking traffic as specified in the section Firewall Rules.
Validate that identities are synchronized correctly by running
cilium identity list
in one of the Cilium pods. It must list identities from all clusters. You can determine what cluster an identity belongs to by looking at the labelio.cilium.k8s.policy.cluster
.If this fails:
Is the identity information available in the kvstore of each cluster? You can confirm this by running
cilium kvstore get --recursive cilium/state/identities/v1/
.Note
The kvstore will only contain identities of the local cluster. It will not contain identities of remote clusters. The state in the kvstore is used for other clusters to discover all identities so it is important that local identities are listed.
Validate that the IP cache is synchronized correctly by running
cilium bpf ipcache list
orcilium map get cilium_ipcache
. The output must contain pod IPs from local and remote clusters.If this fails:
Is the IP cache information available in the kvstore of each cluster? You can confirm this by running
cilium kvstore get --recursive cilium/state/ip/v1/
.Note
The kvstore will only contain IPs of the local cluster. It will not contain IPs of remote clusters. The state in the kvstore is used for other clusters to discover all pod IPs so it is important that local identities are listed.
When using global services, ensure that global services are configured with endpoints from all clusters. Run
cilium service list
in any Cilium pod and validate that the backend IPs consist of pod IPs from all clusters running relevant backends. You can further validate the correct datapath plumbing by runningcilium bpf lb list
to inspect the state of the BPF maps.If this fails:
Are services available in the kvstore of each cluster? You can confirm this by running
cilium kvstore get --recursive cilium/state/services/v1/
.Run
cilium debuginfo
and look for the section “k8s-service-cache”. In that section, you will find the contents of the service correlation cache. it will list the Kubernetes services and endpoints of the local cluster. It will also have a sectionexternalEndpoints
which must list all endpoints of remote clusters.
#### k8s-service-cache
(*k8s.ServiceCache)(0xc00000c500)({
[...]
services: (map[k8s.ServiceID]*k8s.Service) (len=2) {
(k8s.ServiceID) default/kubernetes: (*k8s.Service)(0xc000cd11d0)(frontend:172.20.0.1/ports=[https]/selector=map[]),
(k8s.ServiceID) kube-system/kube-dns: (*k8s.Service)(0xc000cd1220)(frontend:172.20.0.10/ports=[metrics dns dns-tcp]/selector=map[k8s-app:kube-dns])
},
endpoints: (map[k8s.ServiceID]*k8s.Endpoints) (len=2) {
(k8s.ServiceID) kube-system/kube-dns: (*k8s.Endpoints)(0xc0000103c0)(10.16.127.105:53/TCP,10.16.127.105:53/UDP,10.16.127.105:9153/TCP),
(k8s.ServiceID) default/kubernetes: (*k8s.Endpoints)(0xc0000103f8)(192.168.33.11:6443/TCP)
},
externalEndpoints: (map[k8s.ServiceID]k8s.externalEndpoints) {
}
})
The sections
services
andendpoints
represent the services of the local cluster, the sectionexternalEndpoints
lists all remote services and will be correlated with services matching the same .
Roadmap Ahead
- Introduction of IPsec and use of ESP or utilization of the traffic class field in the IPv6 header will allow to use more than 8 bits for the cluster-id and thus support more than 256 clusters.