[Guide] RBAC on VKS

Overview

Intro

This Blogpost explains RBAC on vSphere Kubernetes Service (VKS), formely known as Tanzu Kubernetes Grid Service (TKGS). It also shows how to give access via kubectl with granular permissions, leveraging OIDC Auth with pinniped on VKS-Clusters.

Naming

I use VKS-, TKGS- and K8s Clusters in this Blog Post. All have the same meaning - simply a Kubernetes Cluster :)

RBAC on vSphere Kubernetes Service

Role-based Access Control on vSphere Kubernetes Cluster has a few Keypoint which should be though about. The main question you have to ask yourself: Which Level of access do I want to grant to which people?

Option 1: vSphere Namespace Permissions

With vSphere Namespace Permissions (Select Permissions > Add Permissions) you grant access directly on the vSphere Namespace via vCenter.

vSphere Namespace Permissions

Option Description
Can View Can read TKG cluster objects in the vSphere Namespace. No permissions mapped to Kubernetes roles. See Role Permissions and Bindings.
Can Edit Can create, read, update, and delete TKG cluster objects in the vSphere Namespace. Can operate TKG clusters provisioned in the vSphere Namespace as the Kubernetes cluster-admin. See Role Permissions and Bindings.
Owner Same permissions as Can Edit, with the additional permission to create and manage vSphere Namespaces using kubectl. Only available with vCenter SSO. See Role Permissions and Bindings.

This means - depending on the Permission you give:

  • User has Access to all Ressources within that vSphere Namespace:
    • all TKGS/VKS Guest Clusers (kubectl)
    • all VMs wihtin that Namespace
  • Create new TKGS/VKS Clusters
  • Create new VMs

But I only want to give "some" permissions to the Kubernetes Cluster for my devs - now Kubernetes RBAC Concept comes into play.

Option 2: K8s RBAC

With Kubernetes RBAC you have way more granular control over the permission:

Core Components of Kubernetes RBAC

  • Roles: Define a set of permissions for accessing Kubernetes resources within a single namespace.
  • ClusterRoles: Similar to Roles but cluster-scoped, allowing permissions across all namespaces and for cluster-wide resources.
  • RoleBindings: Grant the permissions defined in a Role to users or service accounts within a specific namespace.
  • ClusterRoleBindings: Cluster-scoped version of RoleBindings, granting permissions defined in a ClusterRole across the entire cluster.
 1                 +-------------+
 2                 |    User    |
 3                 +------+------+
 4                        |
 5                        v
 6              +-------------------+
 7              | Authentication    |
 8              +--------+----------+
 9                       |
10                       v
11              +-------------------+
12              |   Authorization   |
13              +--------+----------+
14                       |
15                       v
16         +-------------+-------------+
17         |                           |
18         v                           v
19  +-------------+            +----------------+
20  |    Roles    |            |  ClusterRoles  |
21  +------+------+            +--------+-------+
22         |                            |
23         |              +-------------+
24         |              |
25         v              v
26  +-------------+  +-----------------+
27  | RoleBinding |  | ClusterRoleBinding |
28  +------+------+  +-----------------+
29         |                   |
30         |    +--------------+
31         |    |
32         v    v
33    +----+----+----+
34    |  Resources   |
35    | (Namespaced) |
36    +-------------+

1.) The process starts with a user attempting to access the cluster.

2.) Authentication verifies the user's identity.

3.) Authorization (RBAC) determines what actions the user can perform.

4.) Roles and ClusterRoles define permissions for resources.

5.) RoleBindings and ClusterRoleBindings associate users with roles.

6.) Finally, access is granted or denied to the requested resources based on the RBAC configuration.

When to choose what

Choosing between vSphere Namespace Permissions and K8s RBAC can be simplified to the following:

  • Full Access to the all VKS/TKGS Cluster within a vSphere Namespace -> vSphere Namespace Permissions
  • Granular Access to VKS/TKGS Cluster or even just K8s Namespace -> K8s RBAC

K8s RBAC on VKS with pinniped

Scenario: Developer Access to a K8s Namespace

Scenario Intro:

  • Your companies Dev-Team needs kubectl access to a K8s Namespace on a shared VKS/TKGS Cluster.
  • You (Infra Admin) have vSphere Namespace Permissions of "Edit" or "Owner", also admin rights to the vSphere Supervisor
  • You have a OIDC Provider (GitLab, WorkspaceONE Access, etc) already in place

We will configure the infrastructure that the following is possible:

  • Devs Team will have a kubeconfig File, which they can use to logon via kubectl
  • GitLab will act as a SSO Provider
  • Role and RoleBindings are applied to the VKS-Cluster to grant permissions to the K8s-Namespace.
 1+------------------+
 2|      User        |
 3+--------+---------+
 4         |
 5         v
 6+------------------+    +--------------------------------------------------------+
 7| Identity Provider|<-->|Pinniped Supervisor - running on the vSphere Supervisor |
 8+------------------+    +--------------------------------------------------------+
 9                               ^
10                               |
11                        +------+------+
12                        |   Pinniped  |
13                        |  Concierge  |
14                        +------+------+
15                               |
16                               v
17+------------------+    +------------------+
18|    vSphere       |    | Tanzu Kubernetes |
19|    Namespace     |    |     Cluster      |
20+------------------+    +------------------+

Configure SSO with vSphere Supervisor

First we will configure the vSphere Supervisor to use GitLab as a Identity Provider. For that we need to create a OIDC Application in GitLab:

  • Go to your GitLab Admin Panel (eg. https://gitlab.yourdomain.tld/admin/applications )
    • Create a Application
      • Example:
        • Name: my-supervisor.yourdomain.tld
        • Redirect URI: https://supervisor-fqdn/wcp/pinniped/callback
        • Scopes:
          • read_user
          • openid
          • profile
          • email

Futher Information: GitLab Docs

Then configure GitLab as a Identity Provider in the vSphere Supervisor:

vSphere Supervisor Identity Provider

Example:

  • Provider Name: gitlab-oidc
  • Issuer URL: https://gitlab.mydomain.local
  • Username Claim: nickname
  • Groups Claim: groups
  • Client ID: the ID from your GitLab Application
  • Client Secret: the Secret from your GitLab Application
  • Additional Scopes: openid
  • Certififacte Authority Data: PEM Formatted Cert

The configuration of a Idendity Provider will install pinniped supervisor on the vSphere Namespace. Pinniped is used for OIDC.

Role and RoleBinding

GitLab Configuration

Be careful with the "name: my-gitlab-group" within your Kubernetes Role. It really depends on how the GitLab Group was created. I > personally had problems with nested groups or groups created on projects. I always created the GitLab Groups/Membership with a Admin, didn't use the Group from a GitLab project.

Further we need a Role and a RoleBinding. Apply those on the actual Kubernetes Cluster.

RoleBinding

 1apiVersion: rbac.authorization.k8s.io/v1
 2kind: RoleBinding
 3metadata:
 4  name: rbac-default-developer-rolebinding
 5  namespace: NAMESPACE_NAME
 6roleRef:
 7  apiGroup: rbac.authorization.k8s.io
 8  kind: Role
 9  name: rbac-default-developer-role
10subjects:
11- apiGroup: rbac.authorization.k8s.io
12  kind: Group
13  name: my-gitlab-group

Role

Adjust the permissions as needed

 1apiVersion: rbac.authorization.k8s.io/v1
 2kind: Role
 3metadata:
 4  name: rbac-default-developer-role
 5  namespace: my-k8s-namespace
 6rules:
 7- apiGroups: ["*"]
 8  resources: ["*"]
 9  verbs: ["get", "list", "watch"]
10- apiGroups: ["", "apps", "batch"]
11  resources: ["pods", "services", "deployments", "replicasets", "statefulsets", "jobs", "cronjobs", "pods/exec"]
12  verbs: ["create", "get", "list", "watch", "update", "patch", "delete"]
13- apiGroups: [""]
14  resources: ["configmaps", "secrets", "persistentvolumeclaims"]
15  verbs: ["get", "list", "watch", "patch", "delete", "create","update"]
16- apiGroups: ["networking.k8s.io"]
17  resources: ["ingresses"]
18  verbs: ["create", "get", "list", "watch", "update", "patch", "delete"]
19- apiGroups: ["*"]
20  resources: ["*"]
21  verbs: ["get", "list", "watch"]
22- apiGroups: ["cluster.x-k8s.io"]
23  resources: ["*"]
24  verbs: ["get", "list", "watch"]
25- apiGroups: [""]
26  resources: ["namespaces"]
27  verbs: ["get", "list", "watch"]

Create the kubeconfig File

Permissions

Use your Account that has "Edit" or "Owner" Permissions on the vSphere Namespace, where the VKS-Cluster is residing in.

To create a kubeconfig file run the following commands:

1tanzu context create my-kubernetes-cluster --endpoint https://SupervisorVIP
1tanzu cluster kubeconfig get my-kubernetes-cluster -n my-vSphere-Namespace --export-file my-kubeconfig

Note: The kubeconfig File does not store any personalized credentials, so can safely be shared with your devs. The kubeconfig also isn't specific to a K8s-Namespace. The kubeconfig is specific to a Kubernetes Cluster (kubeapi-server IP).

You could also use the Local Consumption Interface to download the kubeconfig:

Local Consumption Interface - LCI

Test the Access

Either let the user test or you can use the following command with your administrator account on the k8s cluster:

1kubectl auth can-i get pods -n namespace --as=user

From a Dev's point of view, do the following:

Install Tanzu CLI and the pinniped auth Plugin:

1tanzu plugin install pinniped-auth

Run kubectl with a the created kubeconfig file.

1kubectl --kubeconfig ./my-kubeconfig get ns

Now you will be given a link (to the pinniped Service running on the vSphere Supervisor). Copy this link into your browser, GitLab will open (if not already logged in) and authenticate you. Afterwards you will be presented with a Token. Copy this token back in to your terminal to get access to your kubernetes cluster or namespace.

 1$ kubectl --kubeconfig ./my-kubeconfig get pods
 2
 3Optionally, paste your authorization code: G2TcS145Q4e6A1YKf743n3BJlfQAQ_UdjXy38TtEEIo.ju4QV3PTsUvOigVUtQllZ7AJFU0YnjuLHTRVoNxvdZc
 4
 5✔ successfully logged in to management cluster using the kubeconfig 
 6Checking for required plugins...
 7All required plugins are already installed and up-to-date
 8
 9NAME                           READY   STATUS    RESTARTS   AGE
10nginx-deployment-66b6c48dd5-7tzxb   1/1     Running   0          3d
11nginx-deployment-66b6c48dd5-8p4vk   1/1     Running   0          3d
12nginx-deployment-66b6c48dd5-f8t5r   1/1     Running   0          3d

Optional: Overwrite or create the default kubeconfig file:

1cp my-kubeconfig ~/.kube/config

Et voila, now your Developer has access via kubectl.

Scenario: Developer Access to a K8s Cluster, but not will full permission

Basically the same as above, but now you will use a K8s ClusterRole and a ClusterRoleBinding instead of a Role & RoleBinding.

ClusterRole

Adjust the permissions as needed

 1apiVersion: rbac.authorization.k8s.io/v1
 2kind: ClusterRole
 3metadata:
 4  name: rbac-default-cluster-operator-clusterrole
 5aggregationRule:
 6  clusterRoleSelectors:
 7  - matchLabels:
 8rules:
 9- apiGroups: ["*"]
10  resources: ["*"]
11  verbs: ["get", "list", "watch"]
12- apiGroups: [""]
13  resources: ["namespaces", "configmaps", "pods", "secrets", "persistentvolumeclaims", "persistentvolumes", "services"]
14  verbs: ["create", "patch", "delete"]
15- apiGroups: ["apps"]
16  resources: ["deployments", "replicasets", "statefulsets"]
17  verbs: ["create", "patch", "delete"]
18- apiGroups: ["networking.k8s.io"]
19  resources: ["ingressclasses", "ingresses","networkpolicies"]
20  verbs: ["create", "patch", "delete"]

ClusterRoleBinding

 1apiVersion: rbac.authorization.k8s.io/v1
 2kind: ClusterRoleBinding
 3metadata:
 4  name: rbac-default-cluster-operator-clusterrolebinding
 5roleRef:
 6  apiGroup: rbac.authorization.k8s.io
 7  kind: ClusterRole
 8  name: rbac-default-cluster-operator-clusterrole
 9subjects:
10- apiGroup: rbac.authorization.k8s.io
11  kind: Group
12  name: my-gitlab-group

Further Thoughts

GitOps your K8s RBAC!

  • Define a default ClusterRole (for K8s-Cluster-Access)
  • Define a default Role (for K8s-Namespace Access)
  • Use ArgoCD or FluxCD to roll-out your RBAC Config

*Token Lifetime

Depending on your IdP, the Token TTL defines how many times the Dev have to re-authenticate. GitLab isn't really configurabe as a "real" IdP. Look at WorkspaceONE Access, Zitadel for further configurations.