## Codefresh Runner {{ template "chart.versionBadge" . }}{{ template "chart.typeBadge" . }}{{ template "chart.appVersionBadge" . }} Helm chart for deploying [Codefresh Runner](https://codefresh.io/docs/docs/installation/codefresh-runner/) to Kubernetes. ## Table of Content - [Prerequisites](#prerequisites) - [Get Chart Info](#get-chart-info) - [Install Chart](#install-chart) - [Chart Configuration](#chart-configuration) - [Upgrade Chart](#upgrade-chart) - [To 2.x](#to-2-x) - [To 3.x](#to-3-x) - [To 4.x](#to-4-x) - [To 5.x](#to-5-x) - [To 6.x](#to-6-x) - [Architecture](#architecture) - [Configuration](#configuration) - [EBS backend volume configuration in AWS](#ebs-backend-volume-configuration) - [Azure Disks backend volume configuration in AKS](#azure-disks-backend-volume-configuration) - [GCE Disks backend volume configuration in GKE](#gce-disks-backend-volume-configuration-in-gke) - [Custom volume mounts](#custom-volume-mounts) - [Custom global environment variables](#custom-global-environment-variables) - [Volume reuse policy](#volume-reuse-policy) - [Volume cleaners](#volume-cleaners) - [Rootless DinD](#rootless-dind) - [ARM](#arm) - [Openshift](#openshift) - [On-premise](#on-premise) ## Prerequisites - Kubernetes **1.19+** - Helm **3.8.0+** ⚠️⚠️⚠️ > Since version 6.2.x chart is pushed **only** to OCI registry at `oci://quay.io/codefresh/cf-runtime` > Versions prior to 6.2.x are still available in ChartMuseum at `http://chartmuseum.codefresh.io/cf-runtime` ## Get Chart Info ```console helm show all oci://quay.io/codefresh/cf-runtime ``` See [Use OCI-based registries](https://helm.sh/docs/topics/registries/) ## Install Chart **Important:** only helm3 is supported - Specify the following mandatory values `values.yaml` ```yaml # -- Global parameters # @default -- See below global: # -- User token in plain text (required if `global.codefreshTokenSecretKeyRef` is omitted!) # Ref: https://g.codefresh.io/user/settings (see API Keys) # Minimal API key scopes: Runner-Installation(read+write), Agent(read+write), Agents(read+write) codefreshToken: "" # -- User token that references an existing secret containing API key (required if `global.codefreshToken` is omitted!) codefreshTokenSecretKeyRef: {} # E.g. # codefreshTokenSecretKeyRef: # name: my-codefresh-api-token # key: codefresh-api-token # -- Account ID (required!) # Can be obtained here https://g.codefresh.io/2.0/account-settings/account-information accountId: "" # -- K8s context name (required!) context: "" # E.g. # context: prod-ue1-runtime-1 # -- Agent Name (optional!) # If omitted, the following format will be used '{{ `{{ .Values.global.context }}_{{ .Release.Namespace }}` }}' agentName: "" # E.g. # agentName: prod-ue1-runtime-1 # -- Runtime name (optional!) # If omitted, the following format will be used '{{ `{{ .Values.global.context }}/{{ .Release.Namespace }}` }}' runtimeName: "" # E.g. # runtimeName: prod-ue1-runtime-1/namespace ``` - Install chart ```console helm upgrade --install cf-runtime oci://quay.io/codefresh/cf-runtime -f values.yaml --create-namespace --namespace codefresh ``` ## Chart Configuration See [Customizing the Chart Before Installing](https://helm.sh/docs/intro/using_helm/#customizing-the-chart-before-installing). ## Upgrade Chart ### To 2.x This major release renames and deprecated several values in the chart. Most of the workload templates have been refactored. Affected values: - `dockerRegistry` is deprecated. Replaced with `global.imageRegistry` - `re` is renamed to `runtime` - `storage.localVolumeMonitor` is replaced with `volumeProvisioner.dind-lv-monitor` - `volumeProvisioner.volume-cleanup` is replaced with `volumeProvisioner.dind-volume-cleanup` - `image` values structure has been updated. Split to `image.registry` `image.repository` `image.tag` - pod's `annotations` is renamed to `podAnnotations` ### To 3.x ⚠️⚠️⚠️ ### READ this before the upgrade! This major release adds [runtime-environment](https://codefresh.io/docs/docs/installation/codefresh-runner/#runtime-environment-specification) spec into chart templates. That means it is possible to set parametes for `dind` and `engine` pods via [values.yaml](./values.yaml). **If you had any overrides (i.e. tolerations/nodeSelector/environment variables/etc) added in runtime spec via [codefresh CLI](https://codefresh-io.github.io/cli/) (for example, you did use [get](https://codefresh-io.github.io/cli/runtime-environments/get-runtime-environments/) and [patch](https://codefresh-io.github.io/cli/runtime-environments/apply-runtime-environments/) commands to modify the runtime-environment), you MUST add these into chart's [values.yaml](./values.yaml) for `.Values.runtime.dind` or(and) .`Values.runtime.engine`** **For backward compatibility, you can disable updating runtime-environment spec via** `.Values.runtime.patch.enabled=false` Affected values: - added **mandatory** `global.codefreshToken`/`global.codefreshTokenSecretKeyRef` **You must specify it before the upgrade!** - `runtime.engine` is added - `runtime.dind` is added - `global.existingAgentToken` is replaced with `global.agentTokenSecretKeyRef` - `global.existingDindCertsSecret` is replaced with `global.dindCertsSecretRef` ### To 4.x This major release adds **agentless inCluster** runtime mode (relevant only for [Codefresh On-Premises](#on-premise) users) Affected values: - `runtime.agent` / `runtime.inCluster` / `runtime.accounts` / `runtime.description` are added ### To 5.x This major release converts `.runtime.dind.pvcs` from **list** to **dict** > 4.x chart's values example: ```yaml runtime: dind: pvcs: - name: dind storageClassName: my-storage-class-name volumeSize: 32Gi reuseVolumeSelector: 'codefresh-app,io.codefresh.accountName' reuseVolumeSortOrder: pipeline_id ``` > 5.x chart's values example: ```yaml runtime: dind: pvcs: dind: name: dind storageClassName: my-storage-class-name volumeSize: 32Gi reuseVolumeSelector: 'codefresh-app,io.codefresh.accountName' reuseVolumeSortOrder: pipeline_id ``` Affected values: - `.runtime.dind.pvcs` converted from **list** to **dict** ### To 6.x ⚠️⚠️⚠️ ### READ this before the upgrade! This major release deprecates previously required `codefresh runner init --generate-helm-values-file`. Affected values: - **Replaced** `.monitor.clusterId` with `.global.context` as **mandatory** value! - **Deprecated** `.global.agentToken` / `.global.agentTokenSecretKeyRef` - **Removed** `.global.agentId` - **Removed** `.global.keys` / `.global.dindCertsSecretRef` - **Removed** `.global.existingAgentToken` / `existingDindCertsSecret` - **Removed** `.monitor.clusterId` / `.monitor.token` / `.monitor.existingMonitorToken` #### Migrate the Helm chart from version 5.x to 6.x Given this is the legacy `generated_values.yaml` values: > legacy `generated_values.yaml` ```yaml { "appProxy": { "enabled": false, }, "monitor": { "enabled": false, "clusterId": "my-cluster-name", "token": "1234567890" }, "global": { "namespace": "namespace", "codefreshHost": "https://g.codefresh.io", "agentToken": "0987654321", "agentId": "agent-id-here", "agentName": "my-cluster-name_my-namespace", "accountId": "my-account-id", "runtimeName": "my-cluster-name/my-namespace", "codefreshToken": "1234567890", "keys": { "key": "-----BEGIN RSA PRIVATE KEY-----...", "csr": "-----BEGIN CERTIFICATE REQUEST-----...", "ca": "-----BEGIN CERTIFICATE-----...", "serverCert": "-----BEGIN CERTIFICATE-----..." } } } ``` Update `values.yaml` for new chart version: > For existing installation for backward compatibility `.Values.global.agentToken/agentTokenSecretKeyRef` **must be provided!** For installation from scratch this value is no longer required. > updated `values.yaml` ```yaml global: codefreshToken: "1234567890" accountId: "my-account-id" context: "my-cluster-name" agentToken: "0987654321" # MANDATORY when migrating from < 6.x chart version ! agentName: "my-cluster-name_my-namespace" # optional runtimeName: "my-cluster-name/my-namespace" # optional ``` > **Note!** Though it's still possible to update runtime-environment via [get](https://codefresh-io.github.io/cli/runtime-environments/get-runtime-environments/) and [patch](https://codefresh-io.github.io/cli/runtime-environments/apply-runtime-environments/) commands, it's recommended to enable sidecar container to pull runtime spec from Codefresh API to detect any drift in configuration. ```yaml runner: # -- Sidecar container # Reconciles runtime spec from Codefresh API for drift detection sidecar: enabled: true ``` ## Architecture [Codefresh Runner architecture](https://codefresh.io/docs/docs/installation/codefresh-runner/#codefresh-runner-architecture) ## Configuration See [Customizing the Chart Before Installing](https://helm.sh/docs/intro/using_helm/#customizing-the-chart-before-installing). ### EBS backend volume configuration `dind-volume-provisioner` should have permissions to create/attach/detach/delete/get EBS volumes Minimal IAM policy for `dind-volume-provisioner` ```json { "Version": "2012-10-17", "Statement": [ { "Effect": "Allow", "Action": [ "ec2:AttachVolume", "ec2:CreateSnapshot", "ec2:CreateTags", "ec2:CreateVolume", "ec2:DeleteSnapshot", "ec2:DeleteTags", "ec2:DeleteVolume", "ec2:DescribeInstances", "ec2:DescribeSnapshots", "ec2:DescribeTags", "ec2:DescribeVolumes", "ec2:DetachVolume" ], "Resource": "*" } ] } ``` There are three options: 1. Run `dind-volume-provisioner` pod on the node/node-group with IAM role ```yaml storage: # -- Set backend volume type (`local`/`ebs`/`ebs-csi`/`gcedisk`/`azuredisk`) backend: ebs-csi ebs: availabilityZone: "us-east-1a" volumeProvisioner: # -- Set node selector nodeSelector: {} # -- Set tolerations tolerations: [] ``` 2. Pass static credentials in `.Values.storage.ebs.accessKeyId/accessKeyIdSecretKeyRef` and `.Values.storage.ebs.secretAccessKey/secretAccessKeySecretKeyRef` ```yaml storage: # -- Set backend volume type (`local`/`ebs`/`ebs-csi`/`gcedisk`/`azuredisk`) backend: ebs-csi ebs: availabilityZone: "us-east-1a" # -- Set AWS_ACCESS_KEY_ID for volume-provisioner (optional) accessKeyId: "" # -- Existing secret containing AWS_ACCESS_KEY_ID. accessKeyIdSecretKeyRef: {} # E.g. # accessKeyIdSecretKeyRef: # name: # key: # -- Set AWS_SECRET_ACCESS_KEY for volume-provisioner (optional) secretAccessKey: "" # -- Existing secret containing AWS_SECRET_ACCESS_KEY secretAccessKeySecretKeyRef: {} # E.g. # secretAccessKeySecretKeyRef: # name: # key: ``` 3. Assign IAM role to `dind-volume-provisioner` service account ```yaml storage: # -- Set backend volume type (`local`/`ebs`/`ebs-csi`/`gcedisk`/`azuredisk`) backend: ebs-csi ebs: availabilityZone: "us-east-1a" volumeProvisioner: # -- Service Account parameters serviceAccount: # -- Create service account create: true # -- Additional service account annotations annotations: eks.amazonaws.com/role-arn: "arn:aws:iam:::role/" ``` ### Custom volume mounts You can add your own volumes and volume mounts in the runtime environment, so that all pipeline steps will have access to the same set of external files. ```yaml runtime: dind: userVolumes: regctl-docker-registry: name: regctl-docker-registry secret: items: - key: .dockerconfigjson path: config.json secretName: regctl-docker-registry optional: true userVolumeMounts: regctl-docker-registry: name: regctl-docker-registry mountPath: /home/appuser/.docker/ readOnly: true ``` ### Azure Disks backend volume configuration `dind-volume-provisioner` should have permissions to create/delete/get Azure Disks Role definition for `dind-volume-provisioner` `dind-volume-provisioner-role.json` ```json { "Name": "CodefreshDindVolumeProvisioner", "Description": "Perform create/delete/get disks", "IsCustom": true, "Actions": [ "Microsoft.Compute/disks/read", "Microsoft.Compute/disks/write", "Microsoft.Compute/disks/delete" ], "AssignableScopes": ["/subscriptions/"] } ``` When creating an AKS cluster in Azure there is the option to use a [managed identity](https://learn.microsoft.com/en-us/azure/aks/use-managed-identity) that is assigned to the kubelet. This identity is assigned to the underlying node pool in the AKS cluster and can then be used by the dind-volume-provisioner. ```console export ROLE_DEFINITIN_FILE=dind-volume-provisioner-role.json export SUBSCRIPTION_ID=$(az account show --query "id" | xargs echo ) export RESOURCE_GROUP= export AKS_NAME= export LOCATION=$(az aks show -g $RESOURCE_GROUP -n $AKS_NAME --query location | xargs echo) export NODES_RESOURCE_GROUP=MC_${RESOURCE_GROUP}_${AKS_NAME}_${LOCATION} export NODE_SERVICE_PRINCIPAL=$(az aks show -g $RESOURCE_GROUP -n $AKS_NAME --query identityProfile.kubeletidentity.objectId | xargs echo) az role definition create --role-definition @${ROLE_DEFINITIN_FILE} az role assignment create --assignee $NODE_SERVICE_PRINCIPAL --scope /subscriptions/$SUBSCRIPTION_ID/resourceGroups/$NODES_RESOURCE_GROUP --role CodefreshDindVolumeProvisioner ``` Deploy Helm chart with the following values: `values.yaml` ```yaml volumeProvisioner: podSecurityContext: enabled: true runAsUser: 0 runAsGroup: 0 fsGroup: 0 storage: backend: azuredisk azuredisk: availabilityZone: northeurope-1 # replace with your zone resourceGroup: my-resource-group-name mountAzureJson: true runtime: dind: nodeSelector: topology.kubernetes.io/zone: northeurope-1 ``` ### GCE Disks backend volume configuration in GKE `dind-volume-provisioner` should have `ComputeEngine.StorageAdmin` permissions There are three options: 1. Run `dind-volume-provisioner` pod on the node/node-group with IAM Service Account ```yaml storage: # -- Set backend volume type (`local`/`ebs`/`ebs-csi`/`gcedisk`/`azuredisk`) backend: gcedisk gcedisk: # -- Set GCP volume backend type (`pd-ssd`/`pd-standard`) volumeType: "pd-standard" # -- Set GCP volume availability zone availabilityZone: "us-central1-c" volumeProvisioner: # -- Set node selector nodeSelector: {} # -- Set tolerations tolerations: [] # -- Set runtime parameters runtime: # -- Parameters for DinD (docker-in-docker) pod dind: # -- Set node selector. nodeSelector: topology.kubernetes.io/zone: us-central1-c ``` 2. Pass static credentials in `.Values.storage.gcedisk.serviceAccountJson` (inline) or `.Values.storage.gcedisk.serviceAccountJsonSecretKeyRef` (from your own secret) ```yaml storage: # -- Set backend volume type (`local`/`ebs`/`ebs-csi`/`gcedisk`/`azuredisk`) backend: gcedisk gcedisk: # -- Set GCP volume backend type (`pd-ssd`/`pd-standard`) volumeType: "`pd-standard" # -- Set GCP volume availability zone availabilityZone: "us-central1-c" # -- Set Google SA JSON key for volume-provisioner (optional) serviceAccountJson: | { "type": "service_account", "project_id": "...", "private_key_id": "...", "private_key": "...", "client_email": "...", "client_id": "...", "auth_uri": "...", "token_uri": "...", "auth_provider_x509_cert_url": "...", "client_x509_cert_url": "..." } # -- Existing secret containing containing Google SA JSON key for volume-provisioner (optional) serviceAccountJsonSecretKeyRef: {} # E.g.: # serviceAccountJsonSecretKeyRef: # name: gce-service-account # key: service-account.json # -- Set runtime parameters runtime: # -- Parameters for DinD (docker-in-docker) pod dind: # -- Set node selector. nodeSelector: topology.kubernetes.io/zone: us-central1-c ``` 3. Assign IAM role to `dind-volume-provisioner` service account ```yaml storage: # -- Set backend volume type (`local`/`ebs`/`ebs-csi`/`gcedisk`/`azuredisk`) backend: gcedisk gcedisk: # -- Set GCP volume backend type (`pd-ssd`/`pd-standard`) volumeType: "`pd-standard" # -- Set GCP volume availability zone availabilityZone: "us-central1-c" volumeProvisioner: # -- Service Account parameters serviceAccount: # -- Create service account create: true # -- Additional service account annotations annotations: iam.gke.io/gcp-service-account: @.iam.gserviceaccount.com # -- Set runtime parameters runtime: # -- Parameters for DinD (docker-in-docker) pod dind: # -- Set node selector. nodeSelector: topology.kubernetes.io/zone: us-central1-c ``` ### Custom global environment variables You can add your own environment variables to the runtime environment. All pipeline steps have access to the global variables. ```yaml runtime: engine: userEnvVars: - name: GITHUB_TOKEN valueFrom: secretKeyRef: name: github-token key: token ``` ### Volume reuse policy Volume reuse behavior depends on the configuration for `reuseVolumeSelector` in the runtime environment spec. ```yaml runtime: dind: pvcs: - name: dind ... reuseVolumeSelector: 'codefresh-app,io.codefresh.accountName' reuseVolumeSortOrder: pipeline_id ``` The following options are available: - `reuseVolumeSelector: 'codefresh-app,io.codefresh.accountName'` - PV can be used by ANY pipeline in the specified account (default). Benefit: Fewer PVs, resulting in lower costs. Since any PV can be used by any pipeline, the cluster needs to maintain/reserve fewer PVs in its PV pool for Codefresh. Downside: Since the PV can be used by any pipeline, the PVs could have assets and info from different pipelines, reducing the probability of cache. - `reuseVolumeSelector: 'codefresh-app,io.codefresh.accountName,project_id'` - PV can be used by ALL pipelines in your account, assigned to the same project. - `reuseVolumeSelector: 'codefresh-app,io.codefresh.accountName,pipeline_id'` - PV can be used only by a single pipeline. Benefit: More probability of cache without “spam” from other pipelines. Downside: More PVs to maintain and therefore higher costs. - `reuseVolumeSelector: 'codefresh-app,io.codefresh.accountName,pipeline_id,io.codefresh.branch_name'` - PV can be used only by single pipeline AND single branch. - `reuseVolumeSelector: 'codefresh-app,io.codefresh.accountName,pipeline_id,trigger'` - PV can be used only by single pipeline AND single trigger. ### Volume cleaners Codefresh pipelines require disk space for: * [Pipeline Shared Volume](https://codefresh.io/docs/docs/pipelines/introduction-to-codefresh-pipelines/#sharing-the-workspace-between-build-steps) (`/codefresh/volume`, implemented as [docker volume](https://docs.docker.com/storage/volumes/)) * Docker containers, both running and stopped * Docker images and cached layers Codefresh offers two options to manage disk space and prevent out-of-space errors: * Use runtime cleaners on Docker images and volumes * [Set the minimum disk space per pipeline build volume](https://codefresh.io/docs/docs/pipelines/pipelines/#set-minimum-disk-space-for-a-pipeline-build) To improve performance by using Docker cache, Codefresh `volume-provisioner` can provision previously used disks with Docker images and pipeline volumes from previously run builds. ### Types of runtime volume cleaners Docker images and volumes must be cleaned on a regular basis. * [IN-DIND cleaner](https://github.com/codefresh-io/dind/tree/master/cleaner): Deletes extra Docker containers, volumes, and images in **DIND pod**. * [External volume cleaner](https://github.com/codefresh-io/dind-volume-cleanup): Deletes unused **external** PVs (EBS, GCE/Azure disks). * [Local volume cleaner](https://github.com/codefresh-io/dind-volume-utils/blob/master/local-volumes/lv-cleaner.sh): Deletes **local** volumes if node disk space is close to the threshold. ### IN-DIND cleaner **Purpose:** Removes unneeded *docker containers, images, volumes* inside Kubernetes volume mounted on the DIND pod **How it runs:** Inside each DIND pod as script **Triggered by:** SIGTERM and also during the run when disk usage > 90% (configurable) **Configured by:** Environment Variables which can be set in Runtime Environment spec **Configuration/Logic:** [README.md](https://github.com/codefresh-io/dind/tree/master/cleaner#readme) Override `.Values.runtime.dind.env` if necessary (the following are **defaults**): ```yaml runtime: dind: env: CLEAN_PERIOD_SECONDS: '21600' # launch clean if last clean was more than CLEAN_PERIOD_SECONDS seconds ago CLEAN_PERIOD_BUILDS: '5' # launch clean if last clean was more CLEAN_PERIOD_BUILDS builds since last build IMAGE_RETAIN_PERIOD: '14400' # do not delete docker images if they have events since current_timestamp - IMAGE_RETAIN_PERIOD VOLUMES_RETAIN_PERIOD: '14400' # do not delete docker volumes if they have events since current_timestamp - VOLUMES_RETAIN_PERIOD DISK_USAGE_THRESHOLD: '0.8' # launch clean based on current disk usage DISK_USAGE_THRESHOLD INODES_USAGE_THRESHOLD: '0.8' # launch clean based on current inodes usage INODES_USAGE_THRESHOLD ``` ### External volumes cleaner **Purpose:** Removes unused *kubernetes volumes and related backend volumes* **How it runs:** Runs as `dind-volume-cleanup` CronJob. Installed in case the Runner uses non-local volumes `.Values.storage.backend != local` **Triggered by:** CronJob every 10min (configurable) **Configuration:** Set `codefresh.io/volume-retention` for dinds' PVCs: ```yaml runtime: dind: pvcs: dind: ... annotations: codefresh.io/volume-retention: 7d ``` Or override environment variables for `dind-volume-cleanup` cronjob: ```yaml volumeProvisioner: dind-volume-cleanup: env: RETENTION_DAYS: 7 # clean volumes that were last used more than `RETENTION_DAYS` (default is 4) ago ``` ### Local volumes cleaner **Purpose:** Deletes local volumes when node disk space is close to the threshold **How it runs:** Runs as `dind-lv-monitor` DaemonSet. Installed in case the Runner uses local volumes `.Values.storage.backend == local` **Triggered by:** Disk space usage or inode usage that exceeds thresholds (configurable) **Configuration:** Override environment variables for `dind-lv-monitor` daemonset: ```yaml volumeProvisioner: dind-lv-monitor: env: KB_USAGE_THRESHOLD: 60 # default 80 (percentage) INODE_USAGE_THRESHOLD: 60 # default 80 ``` ### Rootless DinD DinD pod runs a `priviliged` container with **rootfull** docker. To run the docker daemon as non-root user (**rootless** mode), change dind image tag: `values.yaml` ```yaml runtime: dind: image: tag: rootless ``` ### ARM With the Codefresh Runner, you can run native ARM64v8 builds. > **Note!** > You cannot run both amd64 and arm64 images within the same pipeline. As one pipeline can map only to one runtime, you can run either amd64 or arm64 within the same pipeline. Provide `nodeSelector` and(or) `tolerations` for dind pods: `values.yaml` ```yaml runtime: dind: nodeSelector: arch: arm64 tolerations: - key: arch operator: Equal value: arm64 effect: NoSchedule ``` ### Openshift To install Codefresh Runner on OpenShift use the following `values.yaml` example ```yaml runner: podSecurityContext: enabled: false volumeProvisioner: podSecurityContext: enabled: false env: PRIVILEGED_CONTAINER: true dind-lv-monitor: containerSecurityContext: enabled: true privileged: true volumePermissions: enabled: true securityContext: privileged: true runAsUser: auto ``` Grant `privileged` SCC to `cf-runtime-runner` and `cf-runtime-volume-provisioner` service accounts. ```console oc adm policy add-scc-to-user privileged system:serviceaccount:codefresh:cf-runtime-runner oc adm policy add-scc-to-user privileged system:serviceaccount:codefresh:cf-runtime-volume-provisioner ``` ### On-premise If you have [Codefresh On-Premises](https://artifacthub.io/packages/helm/codefresh-onprem/codefresh) deployed, you can install Codefresh Runner in **agentless** mode. **What is agentless mode?** Agent (aka venona) is Runner component which responsible for calling Codefresh API to run builds and create dind/engine pods and pvc objects. Agent can only be assigned to a single account, thus you can't share one runtime across multiple accounts. However, with **agentless** mode it's possible to register the runtime as **system**-type runtime so it's registered on the platform level and can be assigned/shared across multiple accounts. **What are the prerequisites?** - You have a running [Codefresh On-Premises](https://artifacthub.io/packages/helm/codefresh-onprem/codefresh) control-plane environment - You have a Codefresh API token with platform **Admin** permissions scope ### How to deploy agentless runtime when it's on the SAME k8s cluster as On-Premises control-plane environment? - Enable cluster-level permissions for cf-api (On-Premises control-plane component) > `values.yaml` for [Codefresh On-Premises](https://artifacthub.io/packages/helm/codefresh-onprem/codefresh) Helm chart ```yaml cfapi: ... # -- Enable ClusterRole/ClusterRoleBinding rbac: namespaced: false ``` - Set the following values for Runner Helm chart `.Values.global.codefreshHost=...` \ `.Values.global.codefreshToken=...` \ `.Values.global.runtimeName=system/...` \ `.Values.runtime.agent=false` \ `.Values.runtime.inCluster=true` > `values.yaml` for [Codefresh Runner](https://artifacthub.io/packages/helm/codefresh-runner/cf-runtime) helm chart ```yaml global: # -- URL of Codefresh On-Premises Platform codefreshHost: "https://myonprem.somedomain.com" # -- User token in plain text with Admin permission scope codefreshToken: "" # -- User token that references an existing secret containing API key. codefreshTokenSecretKeyRef: {} # E.g. # codefreshTokenSecretKeyRef: # name: my-codefresh-api-token # key: codefresh-api-token # -- Distinguished runtime name # (for On-Premise only; mandatory!) Must be prefixed with "system/..." runtimeName: "system/prod-ue1-some-cluster-name" # -- Set runtime parameters runtime: # -- (for On-Premise only; mandatory!) Disable agent agent: false # -- (for On-Premise only; optional) Set inCluster runtime (default: `true`) # `inCluster=true` flag is set when Runtime and On-Premises control-plane are run on the same cluster # `inCluster=false` flag is set when Runtime and On-Premises control-plane are on different clusters inCluster: true # -- (for On-Premise only; optional) Assign accounts to runtime (list of account ids; default is empty) # Accounts can be assigned to the runtime in Codefresh UI later so you can kepp it empty. accounts: [] # -- Set parent runtime to inherit. runtimeExtends: [] ``` - Install the chart ```console helm upgrade --install cf-runtime oci://quay.io/codefresh/cf-runtime -f values.yaml --create-namespace --namespace cf-runtime ``` - Verify the runtime and run test pipeline Go to [https:///admin/runtime-environments/system](https:///admin/runtime-environments/system) to check the runtime. Assign it to the required account(s). Run test pipeline on it. ### How to deploy agentless runtime when it's on the DIFFERENT k8s cluster than On-Premises control-plane environment? In this case, it's required to mount runtime cluster's `KUBECONFIG` into On-Premises `cf-api` deployment - Create the neccessary RBAC resources > `values.yaml` for [Codefresh Runner](https://artifacthub.io/packages/helm/codefresh-runner/cf-runtime) helm chart ```yaml extraResources: - apiVersion: rbac.authorization.k8s.io/v1 kind: Role metadata: name: codefresh-role namespace: '{{ "{{ .Release.Namespace }}" }}' rules: - apiGroups: [""] resources: ["pods", "persistentvolumeclaims", "persistentvolumes"] verbs: ["list", "watch", "get", "create", "patch", "delete"] - apiGroups: ["snapshot.storage.k8s.io"] resources: ["volumesnapshots"] verbs: ["list", "watch", "get", "create", "patch", "delete"] - apiVersion: v1 kind: ServiceAccount metadata: name: codefresh-runtime-user namespace: '{{ "{{ .Release.Namespace }}" }}' - apiVersion: rbac.authorization.k8s.io/v1 kind: RoleBinding metadata: name: codefresh-runtime-user namespace: '{{ "{{ .Release.Namespace }}" }}' roleRef: apiGroup: rbac.authorization.k8s.io kind: Role name: codefresh-role subjects: - kind: ServiceAccount name: codefresh-runtime-user namespace: '{{ "{{ .Release.Namespace }}" }}' - apiVersion: v1 kind: Secret metadata: name: codefresh-runtime-user-token namespace: '{{ "{{ .Release.Namespace }}" }}' annotations: kubernetes.io/service-account.name: codefresh-runtime-user type: kubernetes.io/service-account-token ``` - Set up the following environment variables to create a `KUBECONFIG` file ```shell NAMESPACE=cf-runtime CLUSTER_NAME=prod-ue1-some-cluster-name CURRENT_CONTEXT=$(kubectl config current-context) USER_TOKEN_VALUE=$(kubectl -n cf-runtime get secret/codefresh-runtime-user-token -o=go-template='{{ `{{.data.token}}` }}' | base64 --decode) CURRENT_CLUSTER=$(kubectl config view --raw -o=go-template='{{ `{{range .contexts}}{{if eq .name "'''${CURRENT_CONTEXT}'''"}}{{ index .context "cluster" }}{{end}}{{end}}` }}') CLUSTER_CA=$(kubectl config view --raw -o=go-template='{{ `{{range .clusters}}{{if eq .name "'''${CURRENT_CLUSTER}'''"}}"{{with index .cluster "certificate-authority-data" }}{{.}}{{end}}"{{ end }}{{ end }}` }}') CLUSTER_SERVER=$(kubectl config view --raw -o=go-template='{{ `{{range .clusters}}{{if eq .name "'''${CURRENT_CLUSTER}'''"}}{{ .cluster.server }}{{end}}{{ end }}` }}') export -p USER_TOKEN_VALUE CURRENT_CONTEXT CURRENT_CLUSTER CLUSTER_CA CLUSTER_SERVER CLUSTER_NAME ``` - Create a kubeconfig file ```console cat << EOF > $CLUSTER_NAME-kubeconfig apiVersion: v1 kind: Config current-context: ${CLUSTER_NAME} contexts: - name: ${CLUSTER_NAME} context: cluster: ${CLUSTER_NAME} user: codefresh-runtime-user namespace: ${NAMESPACE} clusters: - name: ${CLUSTER_NAME} cluster: certificate-authority-data: ${CLUSTER_CA} server: ${CLUSTER_SERVER} users: - name: ${CLUSTER_NAME} user: token: ${USER_TOKEN_VALUE} EOF ``` - **Switch context to On-Premises control-plane cluster**. Create k8s secret (via any tool like [ESO](https://external-secrets.io/v0.4.4/), `kubectl`, etc ) containing runtime cluster's `KUBECONFG` created in previous step. ```shell NAMESPACE=codefresh kubectl create secret generic dind-runtime-clusters --from-file=$CLUSTER_NAME=$CLUSTER_NAME-kubeconfig -n $NAMESPACE ``` - Mount secret containing runtime cluster's `KUBECONFG` into cf-api in On-Premises control-plane cluster > `values.yaml` for [Codefresh On-Premises](https://artifacthub.io/packages/helm/codefresh-onprem/codefresh) helm chart ```yaml cf-api: ... volumes: dind-clusters: enabled: true type: secret nameOverride: dind-runtime-clusters optional: true ``` > volumeMount `/etc/kubeconfig` is already configured in cf-api Helm chart template. No need to specify it. - Set the following values for Runner helm chart > `values.yaml` for [Codefresh Runner](https://artifacthub.io/packages/helm/codefresh-runner/cf-runtime) helm chart `.Values.global.codefreshHost=...` \ `.Values.global.codefreshToken=...` \ `.Values.global.runtimeName=system/...` \ `.Values.runtime.agent=false` \ `.Values.runtime.inCluster=false` **Important!** `.Values.global.name` ("system/" prefix is ignored!) should match the cluster name (key in `dind-runtime-clusters` secret created previously) ```yaml global: # -- URL of Codefresh On-Premises Platform codefreshHost: "https://myonprem.somedomain.com" # -- User token in plain text with Admin permission scope codefreshToken: "" # -- User token that references an existing secret containing API key. codefreshTokenSecretKeyRef: {} # E.g. # codefreshTokenSecretKeyRef: # name: my-codefresh-api-token # key: codefresh-api-token # -- Distinguished runtime name # (for On-Premise only; mandatory!) Must be prefixed with "system/..." name: "system/prod-ue1-some-cluster-name" # -- Set runtime parameters runtime: # -- (for On-Premise only; mandatory!) Disable agent agent: false # -- (for On-Premise only; optional) Set inCluster runtime (default: `true`) # `inCluster=true` flag is set when Runtime and On-Premises control-plane are run on the same cluster # `inCluster=false` flag is set when Runtime and On-Premises control-plane are on different clusters inCluster: false # -- (for On-Premise only; optional) Assign accounts to runtime (list of account ids; default is empty) # Accounts can be assigned to the runtime in Codefresh UI later so you can kepp it empty. accounts: [] # -- (optional) Set parent runtime to inherit. runtimeExtends: [] ``` - Install the chart ```console helm upgrade --install cf-runtime oci://quay.io/codefresh/cf-runtime -f values.yaml --create-namespace --namespace cf-runtime ``` - Verify the runtime and run test pipeline Go to [https:///admin/runtime-environments/system](https:///admin/runtime-environments/system) to see the runtime. Assign it to the required account(s). {{ template "chart.requirementsSection" . }} {{ template "chart.valuesSection" . }}