Containerized Deployment
WSO2 Integrator leverages the Ballerina Code to Cloud feature to generate containerized deployment artifacts directly from your source code. You can target Docker, Kubernetes, or Red Hat OpenShift without writing deployment descriptors by hand. The compiler derives images and manifests from your code and the optional Cloud.toml configuration file.
Supported platforms
The Code to Cloud feature supports the following containerized deployment platforms:
- Docker — Build and run containerized applications locally or on any Docker-compatible runtime
- Kubernetes — Deploy to any Kubernetes cluster with auto-generated manifests, services, and autoscaling configurations
- Red Hat OpenShift — Deploy to OpenShift using the
ocCLI with platform-specific manifests
- Docker installed and running on your build machine
- A WSO2 Integrator project based on Ballerina
- For Kubernetes: kubectl installed and configured against a Kubernetes cluster
- For OpenShift: OpenShift CLI (
oc) installed and logged in to your cluster
How Code to Cloud works
When you build a Ballerina project with a cloud target, the compiler extension generates deployment artifacts alongside the executable JAR. The artifacts land in the target/ directory:
├── Cloud.toml
├── Ballerina.toml
├── Config.toml
└── target/
├── bin/
│ └── <module>.jar
├── docker/
│ └── Dockerfile
├── kubernetes/ # when cloud = "k8s"
│ └── <module>-0.0.1.yaml
└── openshift/ # when cloud = "openshift"
└── <module>-0.0.1.yaml
Cloud.toml overrides defaults that the compiler infers from your code. Every field is optional. The compiler provides sensible defaults when the file is absent or when a field is omitted. See the Cloud.toml reference for the full field list.
Config.toml is intentionally excluded from the container image because it can contain sensitive values. For Docker, you supply it at runtime via a volume mount. For Kubernetes and OpenShift, use the [[cloud.config.files]] entry in Cloud.toml to mount it as a ConfigMap.
Docker deployment
The Docker deployment path generates a Dockerfile and optionally builds the Docker image locally. This is the simplest containerized deployment option and serves as the foundation for Kubernetes and OpenShift deployments.
Step 1: Set the cloud target
Open Ballerina.toml and add the cloud build option:
[build-options]
cloud = "docker"
Alternatively, pass the flag inline at build time without modifying Ballerina.toml:
bal build --cloud=docker
Step 2: Configure the image
Create a Cloud.toml file in the project root. The [container.image] section controls the generated image name and tag:
[container.image]
repository = "myorg"
name = "my-integration"
tag = "v1.0.0"
[settings]
buildImage = true
Set buildImage = false if you only need the generated Dockerfile without building the image locally.
Cloud.toml is optional. If you skip it, the compiler falls back to the package metadata in Ballerina.toml (the org, name, and version fields) to name the image and set the tag.
Step 3: Create a Config.toml
Create a Config.toml with values for any configurable variables in your code:
# Config.toml - provided at runtime, not packed into the image
greeting = "Hello"
Step 4: Build
bal build
The output confirms the image was built and shows the docker run command:
Compiling source
myorg/my-integration:1.0.0
Generating executable
Generating artifacts...
@kubernetes:Docker - complete 2/2
Execute the below command to run the generated Docker image:
docker run -d -p 9090:9090 myorg/my-integration:v1.0.0
target/bin/my-integration.jar
Step 5: Run the container
Mount your Config.toml into the container at runtime to supply the configurable values:
docker run -d \
-v /absolute/path/to/Config.toml:/home/ballerina/Config.toml \
-p 9090:9090 \
myorg/my-integration:v1.0.0
If you encounter a "Is a directory" error when mounting Config.toml, mount the parent directory instead:
docker run -d \
-v /absolute/path/to/project:/home/ballerina/config \
-p 9090:9090 \
--env BAL_CONFIG_FILES=/home/ballerina/config/Config.toml \
myorg/my-integration:v1.0.0
Verify the integration is running by calling your service endpoint:
curl http://localhost:9090/<your-service-path>
Kubernetes deployment
The Kubernetes deployment path generates a complete set of Kubernetes manifests (Deployment, Service, ConfigMap, HorizontalPodAutoscaler) alongside the Docker image. This enables you to deploy your integration to any Kubernetes cluster with a single kubectl apply command.
Step 1: Set the cloud target
Open Ballerina.toml and add the cloud build option:
[build-options]
cloud = "k8s"
Alternatively, pass the flag inline at build time:
bal build --cloud=k8s
Step 2: Configure the deployment
Create a Cloud.toml with container, resource, autoscaling, ConfigMap, and probe settings:
[container.image]
repository = "myorg"
name = "my-integration"
tag = "v1.0.0"
[cloud.deployment]
min_memory = "100Mi"
max_memory = "256Mi"
min_cpu = "500m"
max_cpu = "500m"
[cloud.deployment.autoscaling]
min_replicas = 2
max_replicas = 5
cpu = 60
[[cloud.config.files]]
file = "./Config.toml"
[cloud.deployment.probes.liveness]
port = 9091
path = "/probes/healthz"
[cloud.deployment.probes.readiness]
port = 9091
path = "/probes/readyz"
The [[cloud.config.files]] entry mounts your Config.toml as a Kubernetes ConfigMap, which is the recommended way to supply configuration to Kubernetes workloads.
The [cloud.deployment.probes.liveness] and [cloud.deployment.probes.readiness] sections only apply to long-running service workloads. Omit them if your integration is an Automation. For services, add a dedicated probe listener in your Ballerina code to back these endpoints:
import ballerina/http;
listener http:Listener probeEndpoint = new (9091);
service /probes on probeEndpoint {
resource function get healthz() returns boolean {
return true;
}
resource function get readyz() returns boolean {
return true;
}
}
Step 3: Build
bal build
The compiler generates all Kubernetes manifests and prints the kubectl apply command:
Generating artifacts...
@kubernetes:Service - complete 1/2
@kubernetes:Service - complete 2/2
@kubernetes:ConfigMap - complete 1/1
@kubernetes:Deployment - complete 1/1
@kubernetes:HPA - complete 1/1
@kubernetes:Docker - complete 2/2
Execute the below command to deploy the Kubernetes artifacts:
kubectl apply -f /path/to/project/target/kubernetes/my-integration
For a service-type workload, the generated manifest includes a Deployment, Service, ConfigMap, and HorizontalPodAutoscaler. The HorizontalPodAutoscaler is only generated when [cloud.deployment.autoscaling] is configured.
For Automations, the compiler generates a Job or CronJob resource instead of a Deployment, with no Service or HorizontalPodAutoscaler.
Step 4: Push the image
Push the built image to your container registry before applying the manifests:
docker push myorg/my-integration:v1.0.0
If you are using Minikube, run eval $(minikube docker-env) before bal build to build the image directly into the Minikube Docker daemon. You can then skip the push step.
Step 5: Deploy
kubectl apply -f target/kubernetes/my-integration/
Expected output for a service-type workload:
service/my-integration-svc created
configmap/config-config-map created
deployment.apps/my-integration-deployment created
horizontalpodautoscaler.autoscaling/my-integration-hpa created
Automations generate Job or CronJob resources instead of a Deployment, and do not produce a Service or HorizontalPodAutoscaler.
Step 6: Verify
kubectl get pods
kubectl get services
kubectl logs -f deployment/my-integration-deployment
Step 7: Expose and test
This step applies to service-type workloads that expose an HTTP endpoint. If your integration is an Automation or Event Listener, skip this step.
Expose the deployment via NodePort to access it in a development cluster:
kubectl expose deployment my-integration-deployment \
--type=NodePort \
--name=my-integration-svc-local
Get the assigned port:
kubectl get svc my-integration-svc-local
If you are using Minikube, get the cluster IP:
minikube ip
Then call the service:
curl http://<cluster-ip>:<node-port>/<your-service-path>
Code to Cloud does not expose every Kubernetes configuration option. For changes beyond what Cloud.toml supports, use Kustomize to patch the generated YAML without modifying it directly. This keeps generated files untouched and makes upgrades easier when you rebuild.
Red Hat OpenShift deployment
The OpenShift deployment path generates manifests that are structurally identical to the Kubernetes output but land in target/openshift/ and are applied using the oc CLI. This makes deploying to Red Hat OpenShift as straightforward as deploying to any Kubernetes cluster.
Step 1: Set the cloud target
Open Ballerina.toml and add the cloud build option:
[build-options]
cloud = "openshift"
Alternatively, pass the flag inline at build time:
bal build --cloud=openshift
Step 2: Configure the deployment
Create a Cloud.toml with container, resource, autoscaling, ConfigMap, and probe settings:
[container.image]
repository = "myorg"
name = "my-integration"
tag = "v1.0.0"
[cloud.deployment]
min_memory = "100Mi"
max_memory = "256Mi"
min_cpu = "500m"
max_cpu = "500m"
[cloud.deployment.autoscaling]
min_replicas = 2
max_replicas = 5
cpu = 60
[[cloud.config.files]]
file = "./Config.toml"
[cloud.deployment.probes.liveness]
port = 9091
path = "/probes/healthz"
[cloud.deployment.probes.readiness]
port = 9091
path = "/probes/readyz"
The [[cloud.config.files]] entry mounts your Config.toml as a ConfigMap, which is the recommended way to supply configuration to OpenShift workloads.
The [cloud.deployment.probes.liveness] and [cloud.deployment.probes.readiness] sections only apply to long-running service workloads. Omit them if your integration is an Automation. For services, add a dedicated probe listener in your Ballerina code to back these endpoints:
import ballerina/http;
listener http:Listener probeEndpoint = new (9091);
service /probes on probeEndpoint {
resource function get healthz() returns boolean {
return true;
}
resource function get readyz() returns boolean {
return true;
}
}
Step 3: Build
bal build
The compiler generates all OpenShift manifests and prints the oc apply command:
Generating artifacts...
@kubernetes:Service - complete 1/2
@kubernetes:Service - complete 2/2
@kubernetes:ConfigMap - complete 1/1
@kubernetes:Deployment - complete 1/1
@kubernetes:HPA - complete 1/1
@kubernetes:Docker - complete 2/2
Execute the below command to deploy the Kubernetes artifacts:
oc apply -f /path/to/project/target/openshift/my-integration
target/bin/my-integration.jar
For a service-type workload, the generated manifest includes a Deployment, Service, ConfigMap, and HorizontalPodAutoscaler. The HorizontalPodAutoscaler is only generated when [cloud.deployment.autoscaling] is configured.
For Automations, the compiler generates a Job or CronJob resource instead of a Deployment, with no Service or HorizontalPodAutoscaler.
Step 4: Push the image
Push the built image to your container registry before applying the manifests:
docker push myorg/my-integration:v1.0.0
Step 5: Deploy
oc apply -f target/openshift/my-integration/
Expected output:
service/my-integration-svc created
configmap/config-config-map created
deployment.apps/my-integration-deployment created
horizontalpodautoscaler.autoscaling/my-integration-hpa created
Automations generate Job or CronJob resources instead of a Deployment, and do not produce a Service or HorizontalPodAutoscaler.
Step 6: Verify
oc get pods
oc get services
oc logs -f deployment/my-integration-deployment
Step 7: Expose and test
This step applies to service-type workloads that expose an HTTP endpoint. If your integration is an Automation or Event Listener, skip this step.
Expose the deployment via NodePort to access it in a development cluster:
oc expose deployment my-integration-deployment \
--type=NodePort \
--name=my-integration-svc-local
Get the assigned port:
oc get svc my-integration-svc-local
Then call the service:
curl http://<cluster-ip>:<node-port>/<your-service-path>
Code to Cloud does not expose every OpenShift configuration option. For changes beyond what Cloud.toml supports, use Kustomize to patch the generated YAML without modifying it directly. This keeps generated files untouched and makes upgrades easier when you rebuild.