# Manage Helm Releases The [helm-controller](../components/helm/controller.md) allows you to declaratively manage Helm chart releases with Kubernetes manifests. It makes use of the artifacts produced by the [source-controller](../components/source/controller.md) from `HelmRepository` and `HelmChart` resources. The helm-controller is part of the default toolkit installation. ## Prerequisites To follow this guide you'll need a Kubernetes cluster with the GitOps toolkit controllers installed on it. Please see the [get started guide](../get-started/index.md) or the [installation guide](installation.md). ## Define a Helm repository To be able to deploy a Helm chart, the Helm chart repository has to be known first to the source-controller, so that the `HelmRelease` can reference to it. A cluster administrator should register trusted sources by creating `HelmRepository` resources in the `gitops-system` namespace. By default, the source-controller watches for sources only in the `gitops-system` namespace, this way cluster admins can prevent untrusted sources from being registered by users. ```yaml apiVersion: source.toolkit.fluxcd.io/v1alpha1 kind: HelmRepository metadata: name: podinfo namespace: gitops-system spec: interval: 1m url: https://stefanprodan.github.io/podinfo ``` The `interval` defines at which interval the Helm repository index is fetched, and should be at least `1m`. Setting this to a higher value means newer chart versions will be detected at a slower pace, a push-based fetch can be introduced using [webhook receivers](webhook-receivers.md) The `url` can be any HTTP/S Helm repository URL. !!! hint "Authentication" HTTP/S basic and TLS authentication can be configured for private Helm repositories. See the [`HelmRepository` CRD docs](../components/source/helmrepositories.md) for more details. ## Define a Helm release With the `HelmRepository` created, define a new `HelmRelease` to deploy the Helm chart from the repository: ```yaml apiVersion: helm.toolkit.fluxcd.io/v2alpha1 kind: HelmRelease metadata: name: podinfo namespace: default spec: interval: 5m chart: name: podinfo version: '^4.0.0' sourceRef: kind: HelmRepository name: podinfo namespace: gitops-system interval: 1m values: replicaCount: 2 ``` The `chart.name` is the name of the chart as made available by the Helm repository, and may not include any aliases. The `chart.version` can be a fixed semver, or any semver range (i.e. `>=4.0.0 <4.0.2`). The `chart` values are used by the helm-controller as a template to create a new `HelmChart` resource in the same namespace as the `sourceRef`. The source-controller will then lookup the chart in the artifact of the referenced `HelmRepository`, fetch the chart, and make it available as a `HelmChart` artifact to be used by the helm-controller. !!! hint "Advanced configuration" The `HelmRelease` offers an extensive set of configurable flags for finer grain control over how Helm actions are performed. See the [`HelmRelease` CRD docs](../components/helm/helmreleases.md) for more details. ## Refer to values in `ConfigMap` and `Secret` resources It is possible to define a list of `ConfigMap` and `Secret` resources from which to take values. The values are merged in the order given, with the later values overwriting earlier. These values always have a lower priority than the values inlined in the `HelmRelease` via the `spec.values` parameter. ```yaml spec: valuesFrom: - kind: ConfigMap name: prod-env-values valuesKey: values-prod.yaml - kind: Secret name: prod-tls-values valuesKey: crt targetPath: tls.crt ``` The definition of the listed keys is as follows: - `kind`: Kind of the values referent (`ConfigMap` or `Secret`). - `name`: Name of the values referent, in the same namespace as the `HelmRelease`. - `valuesKey` _(Optional)_: The data key where the values.yaml or a specific value can be found. Defaults to `values.yaml` when omitted. - `targetPath` _(Optional)_: The YAML dot notation path at which the value should be merged. When set, the `valuesKey` is expected to be a single flat value. Defaults to `None` when omitted, which results in the values getting merged at the root. !!! hint "Note" The `targetPath` supports the same formatting as you would supply as an argument to the `helm` binary using `--set [path]=[value]`. In addition to this, the referred value can contain the same value formats (e.g. `{a,b,c}` for a list). You can read more about the available formats and limitations in the [Helm documentation](https://helm.sh/docs/intro/using_helm/#the-format-and-limitations-of---set). ## Configure notifications The default toolkit installation configures the helm-controller to broadcast events to the [notification-controller](../components/notification/controller.md). To receive the events as notifications, a `Provider` needs to be setup first as described in the [notifications guide](notifications.md#define-a-provider). Once you have set up the `Provider`, create a new `Alert` resource in the `gitops-system` to start receiving notifications about the Helm release: ```yaml apiVersion: notification.toolkit.fluxcd.io/v1alpha1 kind: Alert metadata: generation: 2 name: helm-podinfo namespace: gitops-system spec: providerRef: name: slack eventSeverity: info eventSources: - kind: HelmRepository name: podinfo - kind: HelmChart name: default-podinfo - kind: HelmRelease name: podinfo namespace: default ``` ![helm-controller alerts](../diagrams/helm-controller-alerts.png) ## Configure webhook receivers When using semver ranges for Helm releases, you may want to trigger an update as soon as a new chart version is published to your Helm repository. In order to notify source-controller about a chart update, you can [setup webhook receivers](webhook-receivers.md). First generate a random string and create a secret with a `token` field: ```sh TOKEN=$(head -c 12 /dev/urandom | shasum | cut -d ' ' -f1) echo $TOKEN kubectl -n gitops-system create secret generic webhook-token \ --from-literal=token=$TOKEN ``` When using [Harbor](https://goharbor.io/) as your Helm repository, you can define a receiver with: ```yaml apiVersion: notification.toolkit.fluxcd.io/v1alpha1 kind: Receiver metadata: name: helm-podinfo namespace: gitops-system spec: type: harbor secretRef: name: webhook-token resources: - kind: HelmRepository name: podinfo ``` The notification-controller generates a unique URL using the provided token and the receiver name/namespace. Find the URL with: ```console $ kubectl -n gitops-system get receiver/helm-podinfo NAME READY STATUS helm-podinfo True Receiver initialised with URL: /hook/bed6d00b5555b1603e1f59b94d7fdbca58089cb5663633fb83f2815dc626d92b ``` Log in to the Harbor interface, go to Projects, select a project, and select Webhooks. Fill the form with: * Endpoint URL: compose the address using the receiver LB and the generated URL `http:///` * Auth Header: use the `token` string With the above settings, when you upload a chart, the following happens: * Harbor sends the chart push event to the receiver address * Notification controller validates the authenticity of the payload using the auth header * Source controller is notified about the changes * Source controller pulls the changes into the cluster and updates the `HelmChart` version * Helm controller is notified about the version change and upgrades the release !!! hint "Note" Besides Harbor, you can define receivers for **GitHub**, **GitLab**, **Bitbucket** and any other system that supports webhooks e.g. Jenkins, CircleCI, etc. See the [Receiver CRD docs](../components/notification/receiver.md) for more details.