Please note the code snippets in this section reuses the environment variables you set up when progressing through the tutorial, including the optional step of installing the Artifact Storage. Please make sure you correctly define them or adapt the snippet to your specific use case.
You must configure an Ingress or a Load Balancer to expose the Mender Server outside the Kubernetes cluster.
Required configuration:
/ui
${MENDER_SERVER_DOMAIN}
must exists
as a secret named mender-ingress-tls
, or you already have configured the
Cert-manager
resourceYou can refer to your Infrastructure Provider's documentation for creating an Ingress resource. Starting from the Mender Helm Chart version 5.1.0, a sample Ingress resource is included. Starting from the Mender Helm Chart version 6.0.0, the Ingress feature is enabled by default.
Following some Ingress examples. Please refer to your Infrastructure Provider's documentation for creating an Ingress resource and for the required annotations.
For example, here's an Ingress for the Traefik Provider, included in K3s:
export MENDER_SERVER_DOMAIN="mender.example.com"
cat <<-EOF > mender-values.yml
ingress:
enabled: true
annotations:
cert-manager.io/issuer: "letsencrypt"
ingressClassName: traefik
path: /
hosts:
- ${MENDER_SERVER_DOMAIN}
tls:
# this secret must exists or it can be created from a working cert-manager instance
- secretName: mender-ingress-tls
hosts:
- ${MENDER_SERVER_DOMAIN}
EOF
For example, here's an Ingress for the AWS EKS Provider:
export MENDER_SERVER_DOMAIN="mender.example.com"
cat <<-EOF > mender-values.yml
ingress:
enabled: true
annotations:
cert-manager.io/issuer: "letsencrypt"
alb.ingress.kubernetes.io/healthcheck-path: /ui/
alb.ingress.kubernetes.io/actions.ssl-redirect: '{"Type": "redirect", "RedirectConfig":{ "Protocol": "HTTPS", "Port": "443", "StatusCode": "HTTP_301"}}'
alb.ingress.kubernetes.io/backend-protocol: HTTP
alb.ingress.kubernetes.io/listen-ports: '[{"HTTP": 80}, {"HTTPS":443}]'
alb.ingress.kubernetes.io/load-balancer-attributes: routing.http2.enabled=true,idle_timeout.timeout_seconds=600
alb.ingress.kubernetes.io/scheme: internet-facing
alb.ingress.kubernetes.io/ssl-policy: ELBSecurityPolicy-TLS-1-2-2017-01
alb.ingress.kubernetes.io/target-type: ip
ingressClassName: alb
path: /
extraPaths:
- path: /
backend:
serviceName: ssl-redirect
servicePort: use-annotation
hosts:
- ${MENDER_SERVER_DOMAIN}
tls:
# this secret must exists or it can be created from a working cert-manager instance
- secretName: mender-ingress-tls
hosts:
- ${MENDER_SERVER_DOMAIN}
EOF
For example, here's an Ingress for Azure AKS Provider:
export MENDER_SERVER_DOMAIN="mender.example.com"
cat <<-EOF > mender-values.yml
ingress:
enabled: true
ingress:
enabled: true
annotations:
appgw.ingress.kubernetes.io/backend-protocol: http
appgw.ingress.kubernetes.io/health-probe-path: /ui/
appgw.ingress.kubernetes.io/request-timeout: "600"
appgw.ingress.kubernetes.io/ssl-redirect: "true"
ingressClassName: azure/application-gateway
path: /
hosts:
- ${MENDER_SERVER_DOMAIN}
tls:
# this secret must exists or it can be created from a working cert-manager instance
- secretName: mender-ingress-tls
hosts:
- ${MENDER_SERVER_DOMAIN}
EOF
For example, here's an Ingress for the Ingress-NGinx Provider:
export MENDER_SERVER_DOMAIN="mender.example.com"
cat <<-EOF > mender-values.yml
ingress:
enabled: true
ingress:
enabled: true
annotations:
nginx.ingress.kubernetes.io/proxy-body-size: "0"
nginx.ingress.kubernetes.io/proxy-buffering: "off"
nginx.ingress.kubernetes.io/proxy-read-timeout: "600"
nginx.ingress.kubernetes.io/proxy-send-timeout: "600"
path: /
ingressClassName: nginx
hosts:
- ${MENDER_SERVER_DOMAIN}
tls:
# this secret must exists or it can be created from a working cert-manager instance
- secretName: mender-ingress-tls
hosts:
- ${MENDER_SERVER_DOMAIN}
EOF
Some users have reported that when using the nginx ingress controller, the troubleshoot add-on may not work properly. If you encounter this issue, you can try using the following annotation:
metadata:
annotations:
nginx.org/client-max-body-size: 128m
nginx.org/websocket-services: mender-api-gateway
Alternatively, you could use a NodePort service to expose the Mender Server:
cat <<-EOF >> mender-values.yml
ingress:
enabled: false
api_gateway:
service:
type: NodePort
api_gateway:
httpNodePort: 30080
httpsNodePort: 30443
You then have to configure a custom Load Balancer to forward the traffic to the NodePort service.
Alternatively, you can create your own Ingress resource. Some references:
The Mender Server requires the following services to operate:
The Mender Helm chart already provides as sub-charts the following services:
Using these packages is fine for test or PoC setups. For production setups, however, it's recommended to use external dedicated services. Keep reading for instructions on how to use production-grade services.
Please refer to the Requirements section for more information on the supported versions of the external services.
To use an external MongoDB service, create a secret with the MongoDB connection string:
kubectl create secret generic mender-mongo \
--from-literal=MONGO="mongodb://mymongouser:mymongopassword@my-mongo-host:27017/mender" \
--from-literal=MONGO_URL="mongodb://mymongouser:mymongopassword@my-mongo-host:27017/mender"
Then, configure the Mender Server to use the external MongoDB service:
global:
mongodb:
existingSecret: "mender-mongo"
# Disable the integrated MongoDB subchart:
mongodb:
enabled: false
If you want to use the Service Provider Tenants feature, you need to create
a database user with the atlasAdmin@admin
role.
If you want to keep
this user separate to the rest of the application, you have to provide
dedicated connection to the Deployments and the Inventory services:
kubectl create secret generic mender-mongo-admin \
--from-literal=MONGO="mongodb://mymongoadmin:mymongopassword@my-mongo-host:27017/mender" \
--from-literal=MONGO_URL="mongodb://mymongoadmin:mymongopassword@my-mongo-host:27017/mender"
And then configure the Mender Server to use the external MongoDB service:
deployments:
mongodbExistingSecret: "mender-mongo-admin"
inventory:
mongodbExistingSecret: "mender-mongo-admin"
To use an external NATS service, create a secret with the NATS connection string:
kubectl create secret generic mender-nats-url \
--from-literal=NATS_URL="nats://my-nats-host"
Then, configure the Mender Server to use the external NATS service:
global:
nats:
existingSecret: "mender-nats-url"
# Disable the integrated NATS subchart:
nats:
enabled: false
To use an external Redis service, create a secret with the Redis connection string:
kubectl create secret generic mender-redis-url \
--from-literal=REDIS_CONNECTION_STRING="redis://my-redis-user:my-redis-password@my-redis-host:6379"
Specific example for the Redis Cluster:
kubectl create secret generic mender-redis-url \
--from-literal=REDIS_CONNECTION_STRING="redis://my-redis-user:my-redis-password@my-redis-host:6379?addr=my-redis-host:6379"
Then, configure the Mender Server to use the external Redis service:
global:
redis:
existingSecret: "mender-redis-url"
# Disable the integrated Redis subchart:
redis:
enabled: false
Please note the code snippets in this section reuses the environment variables you set up when progressing through the tutorial, including the optional step of installing the Artifact Storage. Please make sure you correctly define them or adapt the snippet to your specific use case.
Before installing the Mender Server on the Kubernetes cluster using the Mender Helm chart, add the Mender Helm Chart repository:
helm repo add mender https://charts.mender.io
helm repo update
You can now install the Mender Server.
export MENDER_SERVER_DOMAIN="mender.example.com"
export MENDER_SERVER_URL="https://${MENDER_SERVER_DOMAIN}"
cat <<-EOF >> mender-values.yml
global:
s3:
AWS_URI: "${MENDER_SERVER_URL}"
AWS_BUCKET: "${STORAGE_BUCKET}"
AWS_ACCESS_KEY_ID: "${AWS_ACCESS_KEY_ID}"
AWS_SECRET_ACCESS_KEY: "${AWS_SECRET_ACCESS_KEY}"
url: "${MENDER_SERVER_URL}"
api_gateway:
storage_proxy:
enabled: true
url: "${STORAGE_ENDPOINT}"
customRule: "PathRegexp(\`^/${STORAGE_BUCKET}\`)"
EOF
Finally, install the Mender Server:
helm upgrade --install mender mender/mender -f mender-values.yml
If you want to test a release candidate, you can run the following command:
helm upgrade --install mender mender/mender --set default.image.tag=v4.0.0-rc.6 -f mender-values.yml --devel
The following deployment requires access to the Mender Enterprise Container Registry. Please email contact@mender.io to receive an evaluation account.
Create the Docker registry secret:
export MENDER_REGISTRY_USERNAME="replace-with-your-username"
export MENDER_REGISTRY_PASSWORD="replace-with-your-password"
export MENDER_REGISTRY_EMAIL="replace-with-your-email"
kubectl create secret docker-registry my-mender-pull-secret \
--docker-username=${MENDER_REGISTRY_USERNAME} \
--docker-password=${MENDER_REGISTRY_PASSWORD} \
--docker-email=${MENDER_REGISTRY_EMAIL} \
--docker-server=registry.mender.io
and reference it in the mender-values.yaml
file:
cat <<-EOF >> mender-values.yml
default:
imagePullSecrets:
- name: my-mender-pull-secret
EOF
If you provide your own MongoDB, create a secret with
with the keys: MONGO
and MONGO_URL
both with the
MongoDB connection string format, for example:
kubectl create secret generic mender-mongo \
--from-literal=MONGO="mongodb://mymongouser:mymongopassword@my-mongo-host:27017/mender" \
--from-literal=MONGO_URL="mongodb://mymongouser:mymongopassword@my-mongo-host:27017/mender"
Now you can configure the Mender Server:
export MENDER_SERVER_DOMAIN="mender.example.com"
export MENDER_SERVER_URL="https://${MENDER_SERVER_DOMAIN}"
cat <<-EOF >> mender-values.yml
global:
enterprise: true
mongodb:
existingSecret: mender-mongo
s3:
AWS_URI: "${STORAGE_ENDPOINT}"
AWS_BUCKET: "${STORAGE_BUCKET}"
AWS_REGION: "${AWS_REGION}"
AWS_ACCESS_KEY_ID: "${AWS_ACCESS_KEY_ID}"
AWS_SECRET_ACCESS_KEY: "${AWS_SECRET_ACCESS_KEY}"
url: "${MENDER_SERVER_URL}"
# Disable the MongoDB subchart:
mongodb:
enabled: false
# Enable Redis
redis:
enabled: true
api_gateway:
storage_proxy:
enabled: true
url: "${STORAGE_ENDPOINT}"
customRule: "PathRegexp(\`^/${STORAGE_BUCKET}\`)"
EOF
Alternatively, if you don't want to use the Storage Proxy feature, and get the artifacts directly from the Artifact Storage, you can disable it:
export MENDER_SERVER_DOMAIN="mender.example.com"
export MENDER_SERVER_URL="https://${MENDER_SERVER_DOMAIN}"
cat <<-EOF >> mender-values.yml
global:
enterprise: true
mongodb:
existingSecret: mender-mongo
s3:
AWS_URI: "${STORAGE_ENDPOINT}"
AWS_BUCKET: "${STORAGE_BUCKET}"
AWS_REGION: "${AWS_REGION}"
AWS_ACCESS_KEY_ID: "${AWS_ACCESS_KEY_ID}"
AWS_SECRET_ACCESS_KEY: "${AWS_SECRET_ACCESS_KEY}"
url: "${MENDER_SERVER_URL}"
# Disable the MongoDB subchart:
mongodb:
enabled: false
# Enable Redis
redis:
enabled: true
api_gateway:
storage_proxy:
enabled: false
deployments:
customEnvs: []
EOF
Finally, install the Mender Server:
helm upgrade --install mender mender/mender -f mender-values.yml
If you want to test a release candidate, you can run the following command:
helm upgrade --install mender mender/mender --set default.image.tag=v4.0.0-rc.6 -f mender-values.yml --devel
We provide the following Docker image tags:
vX.Y.Z
(recommended)vX.Y
vX
latest
main
For production setup that require high availablility, we recommend enabling replication for the NATS Jetstream work queues. By default, NATS deploys 3 replicas, but the stream created by the workflows service does not have replication enabled and therefore has no fault tolerance. The following snippet increases the number of replicas to 3.
Please replace NATS_URL
in the following snippet with a URL that resolves to any of the NATS pods.
NATS_URL="nats://mender-nats" # REPLACE
kubectl run --env=NATS_URL=$NATS_URL --image=natsio/nats-box:0.14.5-nonroot \
nats-increase-replicas -- \
nats stream edit WORKFLOWS --force --replicas=3
Create the initial user using the useradm
pod:
USERADM_POD=$(kubectl get pod -l 'app.kubernetes.io/component=useradm' -o name | head -1)
kubectl exec $USERADM_POD -- useradm create-user --username "demo@mender.io" --password "demodemo"
Create the administrator user using the tenantadm
pod:
TENANTADM_POD=$(kubectl get pod -l 'app.kubernetes.io/component=tenantadm' -o name | head -1)
TENANT_ID=$(kubectl exec $TENANTADM_POD -- tenantadm create-org --name demo --username "admin@mender.io" --password "adminadmin" --plan enterprise)
You can create additional users from the command line of the useradm
pod:
USERADM_POD=$(kubectl get pod -l 'app.kubernetes.io/component=useradm' -o name | head -1)
kubectl exec $USERADM_POD -- useradm create-user --username "demo@mender.io" --password "demodemo" --tenant-id $TENANT_ID
To use a pre-release (master) version of the backend, please refer to the Development section of the documentation.
© 2025 Northern.tech AS