Configuring CI/CD on Kubernetes

Last modified: December 1, 2024

Introduction

This document explains the configuration options available when configuring a Continuous Integration and Delivery (CI/CD) solution for Private Mendix Platform on a Kubernetes Cluster.

Prerequisites

To configure the CI/CD pipeline, prepare the following:

  • A namespace where you want to deploy the Mendix app.
  • An S3-compatible endpoint or Azure storage account where you can store an MDA file.

Configuring the CI/CD Pipeline

If you have a Kubernetes cluster, you can set Kubernetes as your CI System in Switch to Admin Mode > Settings > Build settings > Build Utility. You need to first obtain and configure a CA certificate, and then configure the followings settings:

Finally, you must also register your Kubernetes cluster.

Obtaining and Configuring the CA Certificate

Most Kubernetes cluster API servers use self-signed certificates. In order to access the API server from Private Mendix Platform, you must add its CA certificate to the operator configuration of the namespace where Private Mendix Platform is installed. For more information, see Creating a Private Cloud Cluster: Custom TLS.

You can obtain the CA certificate by running the following commands:

export context=`kubectl config current-context`
export cluster=`kubectl config view -o jsonpath="{.contexts[?(@.name == \"$context\")].context.cluster}"`
kubectl config view --raw -o jsonpath="{.clusters[?(@.name == \"$cluster\")].cluster.certificate-authority-data}" | base64 -d > custom.crt

If you are configuring a custom CA certificate for the first time, you must also update the Mendix Operator configuration for Private Mendix Platform by running the following command:

# update operator configuration
# please switch kubeconfig file if PMP cluster is different with app cluster.
export namespace=YOUR_PMP_NAMESPACE
kubectl -n ${namespace} create secret generic mendix-custom-ca --from-file=custom.crt=custom.crt
echo -e "spec:\n trust:\n    customCASecretName: mendix-custom-ca" > patchfile
kubectl -n ${namespace} patch operatorconfiguration mendix-operator-configuration --type merge --patch-file patchfile

If you have already configured a custom CA certificate, you must only add your new CA certificate to the secret and restart Private Mendix Platform.

Configuring Build Cluster Setting

The settings in this section configure the Kubernetes cluster.

  • API Server - Your Kubernetes API server.
  • Namespace - The namespace used to create the Kubernetes pod.
  • Token - You must create a service account, role, and role binding in the above namespace, and then get the service account’s token. For reference, see the following shell script:
# create ServiceAccount, Role, and RoleBinding
export NAMESPACE=default
kubectl create namespace $NAMESPACE || true
kubectl apply -f << EOF -
apiVersion: v1
kind: ServiceAccount
metadata:
  name: mxplatform-cicd
  namespace: $NAMESPACE
---
apiVersion: v1
kind: Secret
metadata:
  name: mxplatform-cicd
  namespace: $NAMESPACE
  annotations:
    kubernetes.io/service-account.name: mxplatform-cicd
type: kubernetes.io/service-account-token
---
apiVersion: rbac.authorization.k8s.io/v1
kind: Role
metadata:
  name: mxplatform-cicd
  namespace: $NAMESPACE
rules:
- apiGroups:
  - ""
  resources:
  - pods
  - pods/log
  verbs:
  - create
  - get
  - delete
---
apiVersion: rbac.authorization.k8s.io/v1
kind: RoleBinding
metadata:
  name: mxplatform-cicd
  namespace: $NAMESPACE
subjects:
- kind: ServiceAccount
  name: mxplatform-cicd
  namespace: $NAMESPACE
roleRef:
  kind: Role
  name: mxplatform-cicd
  apiGroup: rbac.authorization.k8s.io
EOF

# get service account token:
kubectl get secret mxplatform-cicd -n$NAMESPACE -o jsonpath='{.data.token}'|base64 -d
# for openshift cluster
kubectl get secret mxplatform-cicd -n$NAMESPACE -o jsonpath='{.metadata.annotations.openshift\.io/token-secret\.value}'

Configuring Build Images Setting

The settings in this section configure the images.

  • Keep Build Pod - Select this checkbox to keep the build pod after the build is completed. This is useful for troubleshooting if the build failed due to pod creation failure or build failure. You can describe the pod to see its status, or check the logs of the pod.

  • Run As User - The default value is 1001110000. For an OpenShift cluster, you must check the user ID range by using the below command, and check the annotations. For example, for openshift.io/sa.scc.uid-range: 1001190000/10000, you can choose one ID from the 1001190000 - 1001199999 range.

    export NAMESPACE=default
    kubectl get ns $NAMESPACE -oyaml
    
  • Build Image - The default value is private-cloud.registry.mendix.com/privateplatform/pmp-pipeline-tools. This image is used to build MDA package and OCI image.

  • Build Package Source - Select the download source for the mxbuild package. The following values are supported:

    • File Server - This option allows anonymous access without any authentication.

    • S3 Bucket - This option requires an access key ID and secret access key for authentication.

    • Azure Blob - This option requires the Azure Workload identity authentication. The default service account is used in the build pod for downloading the mxbuild package. To configure the managed identity and service account, perform the following steps:

      1. Create a managed identity in the Azure portal

      2. Configure federated credentials for the Kubernetes service account.

      3. In the Managed Identity section, add a role assignment with the Storage Blob Data Reader role scoped to the storage account.

      4. Add an annotation to the service account, as in the following example:

        kind: ServiceAccount
        metadata:
          name: default
          namespace: default # The same as the one in Configuring Build Cluster Setting
          annotations:
            azure.workload.identity/client-id: {client-id-build}
        
  • Build Package Path - This setting is required for the File Server build package source. The default value is https://cdn.mendix.com/runtime. If you have your own file server, you must download the package from the Mendix Content Delivery Network, and then upload it to your file server. The file name format is mxbuild-9.24.1.4658.tar.gz.

  • S3 Endpoint, S3 Bucket Name, Region, Access Key ID, Secret Access Key - These settings are required for the S3 Bucket build package source.

  • Storage Account, Container - These settings are required for the Azure Blob build package source.

  • Build OCI Image - Select this check box to build the OCI image besides the MDA file. Only OCI image can be used for deployment if this is checked. This option can be used to avoid configuring anonymous access to your S3 bucket or Azure Blob container.

  • Registry Type - This setting is only applicable if you selected the Build OCI Image check box. The following values are supported:

    • Generic - Supports user and password authentication

    • AWS ECR - Supports IRSA authentication; add the following policy to the IAM role:

      {
          "Version": "2012-10-17",
          "Statement": [
              {
                  "Sid": "AllowImageBuilds",
                  "Effect": "Allow",
                  "Action": [
                      "ecr:BatchGetImage",
                      "ecr:BatchCheckLayerAvailability",
                      "ecr:CompleteLayerUpload",
                      "ecr:GetDownloadUrlForLayer",
                      "ecr:InitiateLayerUpload",
                      "ecr:PutImage",
                      "ecr:ListImages",
                      "ecr:UploadLayerPart",
                      "ecr:DescribeRepositories",
                      "ecr:CreateRepository"
                  ],
                  "Resource": "arn:aws:ecr:*:<account_id>:repository/*"
              },
              {
                  "Sid": "AllowAuthentication",
                  "Effect": "Allow",
                  "Action": "ecr:GetAuthorizationToken",
                  "Resource": "*"
              }
          ]
      }
      
    • Azure ACR - Supports Workload Identity authentication; add the AcrPush role to the build managed identity.

  • Runtime Base Image - This setting is only applicable if you selected the Build OCI Image check box. The default value is private-cloud.registry.mendix.com/app-building-blocks. If you are in an air gap environment, sync tag ubi9-1-jre{XX}-entrypoint and runtime-{YYYYY}, where {XX} is java version in your app, and {YYYYY} is your app runtime version. For example: app-building-blocks:ubi9-1-jre21-entrypoint and app-building-blocks:runtime-10.12.1.39914.

  • Allow Anonymous Access - Select this checkbox if above Runtime Base Image is accessible without authentication.

  • Runtime Base Registry User - This setting is only applicable if you did not select the Allow Anonymous Access check box. User name for the registry authentication.

  • Runtime Base Registry Password - This setting is only applicable if you did not select the Allow Anonymous Access check box. Password for the registry authentication.

  • OCI Registry - This setting is only applicable if you selected the Build OCI Image check box. This registry is used to store OCI image, also repository name should be appended to the registry, quay.io/pmp as an example.

  • OCI Registry User - This setting is only applicable if you selected the Build OCI Image check box. User name for the registry authentication.

  • OCI Registry Password - This setting is only applicable if you selected the Build OCI Image check box. Password for the registry authentication.

Configuring MDA Storage Setting

The settings in this section configure the storage for build output artifacts.

  • Mda Storage Option - Configure where to store the build output artifacts. The supported values are S3 Bucket and Azure Blob. This option requires the Azure Workload identity authentication. The default service account is used in the build pod for uploading the build artifacts. To configure the managed identity and service account, perform the following steps:

    1. Create or reuse a managed identity on Azure portal, and configure federated credentials for the Kubernetes service account.

    2. In the Managed Identity, add a role assignment with the Storage Blob Data Contributor role scoped to the storage account.

    3. Add the correct annotation to the Service Account for build pod and PMP.

    4. Add annotations for the build pod and Private Mendix Platform to the service account, as in the following example:

      kind: ServiceAccount
      metadata:
          name: default
          namespace: default # The same as the one in Configuring Build Cluster Setting
          annotations:
              azure.workload.identity/client-id: {client-id-build}
      
    5. Add a role assignment with the Storage Blob Data Reader role scoped to the storage account to ensure that Private Mendix Platform can access the build metadata after the build is completed. If you are already using Azure Blob Storage (Azure managed identity authentication) for Private Mendix Platform, you can reuse the managed identity which was created by the Mendix Operator.

      kind: ServiceAccount
      metadata:
          name: {pmp-serviceaccount-name}
          namespace: {pmp-namespace} # The namespace where Private Mendix Platform is installed
          annotations:
              azure.workload.identity/client-id: {client-id-pmp}
      
    6. Add customPodLabels to the Mendix Operator to label the Private Mendix Platform pod with the proper configuration. This configuration allows Private Mendix Platform to get build artifacts from Azure Storage Blob.

      kind: OperatorConfiguration
      metadata:
        name: mendix-operator-configuration
        namespace: {pmp-namespace}
      spec:
        customPodLabels:
          general:
              azure.workload.identity/use: "true"
      
    7. Restart Private Mendix Platform to ensure that the label is applied.

  • S3 Endpoint - For example, https://s3.ap-southeast-1.amazonaws.com.

  • No Verify SSL - Select this checkbox if you use your own bucket server, and its certificate is self-signed. Selecting this option adds –no-verify-ssl to the AWS CLI command to avoid failure.

  • S3 Bucket Name - Your S3 bucket name, for example, mybucket.

  • Mda Location - Your S3 bucket name’s domain, for example, https://mybucket.s3.ap-southeast-1.amazonaws.com. This URL is used to access build artifacts, the whole path is: Mda Location + Appid + Mda/Meta file. Make sure that the S3 bucket is configured to allow anonymous access.

  • Region - For example, ap-southeast-1.

  • Use K8S Secret - Select whether you want to input the Access Key ID and Secret Access Key, or set them in a Kubernetes secret. Enable this setting to avoid showing sensitive credentials in a build pod.

  • Secret Name - This setting is only applicable if you selected the Use K8S Secret check box. This is the secret name where you want to store the Access Key ID and Secret Access Key. Use the following command to create this secret, where your-namespace is the namespace that you specified in Build Cluster Setting > Namespace.

    kubectl create secret generic mxplatform-awssecret -n your-namespace --from-literal=aws_access_key_id=your-aws-access-key-id --from-literal=aws_secret_access_key=your-aws-secret-access-key
    
  • Access Key ID - This setting is only applicable if you did not select the Use K8S Secret check box. This value is used to access the S3 bucket.

  • Secret Access Key - This setting is only applicable if you did not select the Use K8S Secret check box. This value is used to access the S3 bucket.

  • Storage Account, Container - This setting is used to configure Azure Blob when the Mda Storage Option is set to Azure Blob.

  • Mda Location - Your Azure Blob container’s domain, for example, https://your-storage-account.blob.core.windows.net/your-container. This setting is only applicable if Build OCI Image is not selected. Make sure that the container’s anonymous access level is set to Blob.

Registering a Kubernetes Cluster

Before creating any environments, you must register your Kubernetes clusters by doing the following steps:

  1. Click Register New Cluster.

  2. Configure the following values:

    • Cluster Name - Specify a name for the cluster.

    • API Server - Specify your Kubernetes API server.

    • Token - You must first create a service account, cluster role, and cluster role binding in the cluster, and then get the service account’s token. For reference, see the following shell script:

      # create ServiceAccount, ClusterRole, and ClusterRoleBinding
      kubectl apply -f << EOF -
      apiVersion: v1
      kind: ServiceAccount
      metadata:
        name: mxplatform-cicd
        namespace: kube-system
      ---
      apiVersion: v1
      kind: Secret
      metadata:
        name: mxplatform-cicd
        namespace: kube-system
        annotations:
          kubernetes.io/service-account.name: mxplatform-cicd
      type: kubernetes.io/service-account-token
      ---
      apiVersion: rbac.authorization.k8s.io/v1
      kind: ClusterRole
      metadata:
        name: mxplatform-cicd
      rules:
      - apiGroups:
        - ""
        resources:
        - namespaces
        verbs:
        - list
      - apiGroups:
        - privatecloud.mendix.com
        resources:
        - storageplans
        verbs:
        - list
      - apiGroups:
        - privatecloud.mendix.com
        resources:
        - mendixapps
        verbs:
        - '*'
      ---
      apiVersion: rbac.authorization.k8s.io/v1
      kind: ClusterRoleBinding
      metadata:
        name: mxplatform-cicd
      subjects:
      - kind: ServiceAccount
        name: mxplatform-cicd
        namespace: kube-system
      roleRef:
        kind: ClusterRole
        name: mxplatform-cicd
        apiGroup: rbac.authorization.k8s.io
      EOF
      
      # get service account token:
      kubectl get secret mxplatform-cicd -nkube-system -o jsonpath='{.data.token}'|base64 -d
      # for openshift cluster
      kubectl get secret mxplatform-cicd -nkube-system -o jsonpath='{.metadata.annotations.openshift\.io/token-secret\.value}'
      
  3. Optionally, enable the Help Me feature. For reference, see the following shell script:

    # create ServiceAccount, ClusterRole, and ClusterRoleBinding
    kubectl apply -f << EOF -
    apiVersion: v1
    kind: ServiceAccount
    metadata:
      name: mxplatform-cicd
      namespace: kube-system
    ---
    apiVersion: v1
    kind: Secret
    metadata:
      name: mxplatform-cicd
      namespace: kube-system
      annotations:
        kubernetes.io/service-account.name: mxplatform-cicd
    type: kubernetes.io/service-account-token
    ---
    apiVersion: rbac.authorization.k8s.io/v1
    kind: ClusterRole
    metadata:
        name: mxplatform-cicd
    rules:
    - apiGroups:
        - ""
        resources:
        - namespaces
        verbs:
        - list
    - apiGroups:
        - ""
        resources:
        - deployments
        verbs:
        - get
        - list
        - watch
    - apiGroups:
        - ""
        resources:
        - pods
        verbs:
        - get
        - list
    - apiGroups:
        - ""
        resources:
        - pods/log
        verbs:
        - get
    - apiGroups:
        - ""
        resources:
        - events
        verbs:
        - get
        - list
    - apiGroups:
        - privatecloud.mendix.com
        resources:
        - storageplans
        verbs:
        - list
    - apiGroups:
        - privatecloud.mendix.com
        resources:
        - mendixapps
        verbs:
        - '*'
    ---
    apiVersion: rbac.authorization.k8s.io/v1
    kind: ClusterRoleBinding
    metadata:
        name: mxplatform-cicd
    subjects:
    - kind: ServiceAccount
        name: mxplatform-cicd
        namespace: kube-system
    roleRef:
        kind: ClusterRole
        name: mxplatform-cicd
        apiGroup: rbac.authorization.k8s.io
    EOF
    
    # get service account token:
    kubectl get secret mxplatform-cicd -nkube-system -o jsonpath='{.data.token}'|base64 -d
    # for openshift cluster
    kubectl get secret mxplatform-cicd -nkube-system -o jsonpath='{.metadata.annotations.openshift\.io/token-secret\.value}'
    
  4. Click Save.

  5. Click the newly created cluster and expand it, and then click Retrieve Namespace(s) to retrieve all the namespace and storage plans.

    Namespaces without any storage plan are skipped. This step requires the Mendix Operator to be installed and configured. You can repeat this step as required to retrieve additional namespaces.

  6. After the cluster is registered, create environments with the cluster, namespace and plans.

Architecture of the CI/CD Pipeline

The diagram in this section presents the architecture and components of the pipeline. For more information, see Build Images Setting above.

Auto Detect Mx Runtime Version