Using Helm Dashboard and Intents-Based Access Control for Pain-Free Network Segmentation

Helm Dashboard is an open-source project which graphically shows installed Helm charts, revisions, and changes to their Kubernetes resources.

The intents operator is an open-source Kubernetes operator which makes it possible to roll out network policies in a Kubernetes cluster, chart by chart, and gradually achieve zero trust or network segmentation.

Modern engineering organizations may want to achieve network segmentation as part of adhering to the OWASP Kubernetes Top 10 recommendations, in particular, K07: Missing Network Segmentation Controls, where Kubernetes network policies and service meshes are the recommended way to achieve segmentation controls.

Network policies are notoriously hard to manage, requiring you to label all client services and server services, and then create an ingress network policy for each server.

This means that each server’s chart would need to reference labels for services that live in other charts – this can be difficult to achieve and requires coordination across multiple teams, since different charts may be owned by different teams.

For example, here’s a network policy that allows access to multiple services, owned by two teams, team2, and team1, to a server owned by team3. Notice how it refers to pod labels in the `from` and the `podSelector` section.

This network policy would live in the Helm chart for the `ledger` service, owned by team3. The labels for the other two services, which call the `ledger`, would need to be in the other charts that deploy those services. This means that the 3 teams would need to coordinate the deployment of all three services, so that the labels appear on all services first, then the network policy is added. If the network policy is added before all the labels are verified to be there, then whoever is misconfigured will be blocked.

apiVersion: networking.k8s.io/v1

kind: NetworkPolicy

metadata:

  name: allow-access-to-ledger

  namespace: team3

spec:

  ingress:

    - from:

        - namespaceSelector:

            matchLabels:

              namespace-name: team1

          podSelector:

            matchLabels:

              how-team1-labels-their-pods: transaction-service

        - namespaceSelector:

            matchLabels:

              namespace-name: team2

          podSelector:

            matchLabels:

              how-team2-labels-their-pods: some-other-transaction-service

  podSelector:

    matchLabels:

      how-team3-labels-their-pods: ledger

  policyTypes:

    - Ingress

Managing so many charts can be taxing and time-consuming even for the most seasoned engineers, but Helm-Dashboard’s visualization of charts makes it easy to correlate them with Kubernetes resources, compare different versions, and quickly rollback or upgrade. In a use case similar to the example above, you could leverage Helm-Dashboard to update the charts and enforce standardization. If something breaks, you can be confident that you’re one click away from rolling back to a working version of the charts. 

What if you’ve done all that hard work of coordinating the different charts and different teams, and now due to an unrelated problem, you want to roll back the server? This would result in also rolling back the network policy for the server, since that’s part of the same Helm chart, blocking any clients who were allowed by the newer network policy. But you may not want that: the rollback will often happen due to an unrelated reason, such as a bug.

With the intents operator, you can declare a separate resource for each client, and the operator then collects those into a network policy per server. This is more in line with how people manage Helm charts – each client is managed on its own, without affecting other Helm charts.

Instead of the network policy we saw before, where the server declares which clients can access, we’ll now have two ClientIntents resources that declare which servers the clients want to access.

For transaction-service in namespace team1, we will declare we want to access the ledger in namespace team3:

apiVersion: k8s.otterize.com/v1alpha2

kind: ClientIntents

metadata:

  name: transaction-service-intents

  namespace: team1

spec:

  service:

    name: transaction-service

  calls:

    - name: ledger.team3

For some-other-transaction-service in namespace team2, we will also declare we want to access ledger in namespace team3:

apiVersion: k8s.otterize.com/v1alpha2

kind: ClientIntents

metadata:

  name: some-other-transaction-service-intents

  namespace: team2

spec:

  service:

    name: transaction-service

  calls:

    - name: ledger.team3

Once you apply these ClientIntents, both the client and server pods are labeled, and a network policy is created for the server. The server’s chart did not have to change! How cool is that?

For each client namespace, team1 and team2, the following network policies were created in namespace team3:

apiVersion: networking.k8s.io/v1

kind: NetworkPolicy

metadata:

  labels:

    intents.otterize.com/network-policy: ledger-team3-7e16db

  name: access-to-ledger-from-team1

  namespace: team3

spec:

  ingress:

    - from:

        - namespaceSelector:

            matchLabels:

              intents.otterize.com/namespace-name: team1

          podSelector:

            matchLabels:

              intents.otterize.com/access-ledger-team3-7e16db: "true"

  podSelector:

    matchLabels:

      intents.otterize.com/server: ledger-team3-7e16db

  policyTypes:

    - Ingress

And the client pods, transaction-service and some-other-transaction-service, had this label applied to them automatically:

intents.otterize.com/access-ledger-team3-7e16db: "true"

While the server, ledger, had this label applied to it automatically:

intents.otterize.com/server: ledger-team3-7e16db

Now that each chart contains only the intended access for the services managed within it, you can diff the chart and track changes over time. If you see that a service is being blocked, you can check its chart and see what’s changed.

In this example, let’s look how we can determine what happened when we see that the transaction-service is no longer able to connect to ledger. We need only look at transaction-service’s chart. Let’s do that using Helm-Dashboard:

Oh no! It looks like somebody accidentally removed the ledger service from the ClientIntents while adding the telemetry service. Let’s correct it:

With Komodor, you’ll be able to see that a service is unable to call another service, and then see whether its ClientIntents changed – which could be the cause for it being blocked, or if its code has changed and it now requires different access, you could use Otterize to map existing traffic, and autogenerate the new client intents:

~ ❯ otterize mapper export -n team1 -n team2                                                                                                                                                                                             

apiVersion: k8s.otterize.com/v1alpha2

kind: ClientIntents

metadata:

  name: some-other-transaction-service

  namespace: team2

spec:

  service:

    name: some-other-transaction-service

  calls:

    - name: ledger.team3

---

apiVersion: k8s.otterize.com/v1alpha2

kind: ClientIntents

metadata:

  name: transaction-service

  namespace: team1

spec:

  service:

    name: transaction-service

  calls:

    - name: ledger.team3

Try out Otterize with Helm Dashboard – both are completely open source and super simple to get started with. Use the mapper export feature of Otterize to generate ClientIntents for one of your charts, and get to zero trust in no time.

Stay tuned for more news from Komodor and Otterize! We have an exciting announcement coming up soon 🤫 In the meantime check out this post by Otterize: why Network Policies are Hard for Developers: https://otterize.com/blog/network-policies-are-not-the-right-abstraction

How useful was this post?

Click on a star to rate it!

Average rating 5 / 5. Vote count: 7

No votes so far! Be the first to rate this post.