Gloo Edge Federation allows you to administer multiple instances of Gloo Edge across multiple Kubernetes clusters. One Gloo Edge Federation object might modify configuration across many instances of Gloo Edge across many Kubernetes clusters. Multicluster role-based access control is a feature of Gloo Edge Federation that controls access and actions on Gloo Edge Federation APIs that might reconfigure many Gloo Edge instances. The feature ensures that users are only allowed to modify Gloo Edge Federation resources that configure Gloo Edge resources in clusters and namespaces that they have explicitly been granted access to in order to facilitate multitenancy in the Gloo Edge Federation control plane.
To successfully follow this Multicluster RBAC guide, you will need the following software available and configured on your system.
- Kubectl - Used to execute commands against the clusters
- Glooctl - Used to register the Kubernetes clusters with Gloo Edge Federation
- Kind - Required if using the
glooctlfederation demo environment
- Docker - Required if using the
glooctlfederation demo environment
In this guide we are going to use the Gloo Edge Federation environment available from the
glooctl demo federation command. You can follow the directions in the Getting Started guide to set up the demonstration environment. Otherwise, you will need at least one Kubernetes cluster running Gloo Edge Enterprise and Gloo Edge Federation.
For the purposes of this example, we have two clusters
remote. The local cluster is also running Gloo Edge Federation in addition to Gloo Edge Enterprise. The kubectl context for the local cluster is
kind-local and the remote cluster is
Enable Multicluster RBAC
Multicluster RBAC can be enabled during Gloo Edge Federation installation by overriding the
enableMultiClusterRbac value in the helm chart to
true. To do so, you would run the following
echo "enableMultiClusterRbac: true" > values.yaml glooctl install federation --values values.yaml --license-key LICENSE_KEY
On an existing installation, the
enableMultiClusterRbac setting can be updated by running
helm upgrade and overriding the value. Enabling the Multicluster RBAC feature creates an RBAC webhook and pod enforcing permissions on the Gloo Edge Federation API groups.
In the demonstration environment, we will upgrade the existing Gloo Edge Federation installation by running the following:
# Add the gloo-fed helm repo helm repo add gloo-fed https://storage.googleapis.com/gloo-fed-helm # Update your repos helm repo update # Upgrade your Gloo Edge Federation deployment # Assumes your install is called gloo-fed in the gloo-fed namspace helm upgrade -n gloo-fed gloo-fed gloo-fed/gloo-fed --set enableMultiClusterRbac=true
Once the installation or upgrade is complete, you can verify by running the following:
kubectl get deployment -n gloo-fed rbac-validating-webhook-gloo-fed
You should see the following output:
NAME READY UP-TO-DATE AVAILABLE AGE rbac-validating-webhook-gloo-fed 1/1 1 1 33m
Examine the Custom Resources
The installation of Multicluster RBAC also creates two MultiClusterRole Custom Resources and two MultiClusterRoleBinding Custom Resources.
kubectl get multiclusterrole -n gloo-fed
NAME AGE gloo-fed 35m gloo-fed-console 35m
kubectl get multiclusterrolebinding -n gloo-fed
NAME AGE gloo-fed 35m gloo-fed-console 35m
gloo-fed MutliClusterRole defines a role with permissions to take any actions across all Gloo Edge instances in all clusters. The relevant portion of the spec is shown below:
spec: rules: - apiGroup: fed.solo.io placements: - clusters: - '*' namespaces: - '*' - apiGroup: fed.gloo.solo.io placements: - clusters: - '*' namespaces: - '*' - apiGroup: fed.gateway.solo.io placements: - clusters: - '*' namespaces: - '*' - apiGroup: fed.enterprise.gloo.solo.io placements: - clusters: - '*' namespaces: - '*'
gloo-fed MultiClusterRoleBinding associates the MultiClusterRole with the
gloo-fed service account. Without the binding, the gloo-fed pod wouldn’t be able to update the status of Gloo Edge Federation API objects.
spec: roleRef: name: gloo-fed namespace: gloo-fed subjects: - kind: User name: system:serviceaccount:gloo-fed:gloo-fed
gloo-fed-console MultiClusterRole and MultiClusterRoleBinding grant the same set of permissions to the
gloo-fed-console service account.
Create Roles and Bindings
In the previous section we installed Multicluster RBAC. The process created a MultiClusterRole and MultiClusterRoleBinding for both the
gloo-fed-console service account. It did not create any role or binding for the default kind user account, kubernetes-admin, which means that we cannot make any changes to the Gloo Edge Federation custom resources.
Let’s try and create a FederatedUpstream on the local cluster:
kubectl apply -f - <<EOF apiVersion: fed.gloo.solo.io/v1 kind: FederatedUpstream metadata: name: my-federated-upstream namespace: gloo-fed spec: placement: clusters: - local namespaces: - gloo-system template: spec: static: hosts: - addr: solo.io port: 80 metadata: name: fed-upstream EOF
We get the following error:
Error from server (User kubernetes-admin does not have the permissions necessary to perform this action.): error when creating "STDIN": admission webhook "rbac-validating-webhook-gloo-fed.gloo-fed.svc" denied the request: User kubernetes-admin does not have the permissions necessary to perform this action.
Great! That is what we should expect and shows that Multicluster RBAC is doing its job. Now let’s create a new MultiClusterRoleBinding for the
kubernetes-admin account binding it to the gloo-fed MultiClusterRole.
kubectl apply -f - <<EOF apiVersion: multicluster.solo.io/v1alpha1 kind: MultiClusterRoleBinding metadata: name: kind-admin namespace: gloo-fed spec: roleRef: name: gloo-fed namespace: gloo-fed subjects: - kind: User name: kubernetes-admin EOF
Now if we try and create the same FederatedUpstream again, we’ll get the following response:
Create a New MultiClusterRole
We may want to restrict an account to perform specific actions on a specific cluster and namespace. We can do this by creating a new MultiClusterRole and then binding it to a service account or user. First, let’s create a new role.
kubectl apply -f - <<EOF apiVersion: multicluster.solo.io/v1alpha1 kind: MultiClusterRole metadata: name: remote-admin namespace: gloo-fed spec: rules: - apiGroup: fed.solo.io action: CREATE placements: - clusters: - remote namespaces: - gloo-system - apiGroup: fed.gloo.solo.io action: CREATE placements: - clusters: - remote namespaces: - gloo-system - apiGroup: fed.enterprise.gloo.solo.io action: CREATE placements: - clusters: - remote namespaces: - gloo-system - apiGroup: fed.gateway.solo.io action: CREATE placements: - clusters: - remote namespaces: - gloo-system EOF
The MultiClusterRole allows the
CREATE action for all Gloo Edge Federation API groups on the remote cluster and namspace gloo-system. Any account bound to this role would have no permissions on the local cluster, and would only be able to create new items in the remote cluster. The
action key of the spec can be set to CREATE, UPDATE, or DELETE.
The next step is to create a service account and bind it to the role. We’ll create an account by running the following:
kubectl create serviceaccount remote-admin -n gloo-fed
Now we will create the binding between the MultiClusterRole and the service account:
kubectl apply -f - <<EOF apiVersion: multicluster.solo.io/v1alpha1 kind: MultiClusterRoleBinding metadata: name: remote-admin namespace: gloo-fed spec: roleRef: name: remote-admin namespace: gloo-fed subjects: - kind: User name: system:serviceaccount:gloo-fed:remote-admin EOF
It is possible to set multiple subjects in the binding if necessary. We can check on the binding by running:
kubectl get multiclusterrolebinding -n gloo-fed remote-admin -oyaml
You can customize both the MultiClusterRole and MultiClusterRoleBindings to match your unique requirements. Since the RBAC model is deny by default, any access not explicitly granted will be denied.
To get deeper into Federated Configuration, you can check out our guide on the topic next, or learn more about the concepts behind Gloo Edge Federation.