Image
How to "build once, run anywhere" at the edge with containers
Use Podman, systemd, OSBuild, and Kubernetes to package your entire application and all its dependencies in a container and run it anywhere.
The "build once, run anywhere" approach has gained momentum alongside the growing use of containers. It has mainly targeted cloud-native development, but by using Podman, systemd, and OSBuild, it can now also apply to developing edge devices.
[ Getting started with containers? Check out Deploying containerized applications: A technical overview. ]
What does "build once, run anywhere" mean?
"Build once, run anywhere" was coined by Sun Microsystems long ago to describe the ability to write Java code once and run it anywhere. Recently, this phrase has expanded to include containers, meaning that developers can package their entire application and all its dependencies in a container and run it anywhere.
What does "running a container" really mean?
To run a container, you need the container image and its running instructions. Running instructions can take different forms. You can run commands in a terminal, a docker-compose file, or a Kubernetes YAML file. Even though you can reuse the same container in each scenario, the instruction's format changes. But what if you could use the same format everywhere?
By using podman kube play
, you can pass Kubernetes manifests to Podman, which exports the objects into Podman objects.
Using this mechanism, you can employ the "build once, run anywhere" principle not just for containers but also for the instructions about how to run these containers.
Using systemd to monitor containers
Although you can run Podman as a daemon, in most cases, it runs without one. As a result, the containers are not monitored, so failed containers remain down with no service to restart them.
In Linux systems, systemd monitors processes. When you use a daemon to monitor a container, systemd monitors the daemon as well. You can skip added complexity by allowing systemd to monitor the container directly.
[ Get the Systemd commands cheat sheet ]
You can use systemd to run and manage containerized applications. That means systemd starts a containerized application and manages its entire lifecycle. Podman simplifies this process with the podman generate systemd
command, which generates a systemd unit file for a specified container or pod. Alternatively, you can use the systemd template unit file for Kubernetes YAML files.
As of version 4.4, Podman includes built-in support for Quadlet, a service that allows users to manage containerized services using service-like .container
or .kube
files, but that's enough content for a separate article.
Containers on the edge
OSBuild is an open source build pipeline that allows you "to create images of your Linux operating system in a reliable fashion, isolating the image creation from your host operating system, and producing a reliable, well-defined image ready to be deployed." Using OSBuild, you can embed the containers, Kubernetes YAML files, and systemd unit files to create a device image that runs the workloads.
Bring it together with a demo
This demo uses a sample automotive application implemented in the sample automotive applications repository, along with a containerized version of vsomeip, which implements SOME/IP (Scalable service-Oriented MiddlewarE over IP), a communications protocol for automotive applications. Of note:
- A prebuilt version of these containers for AArch64 and x86-64 is available in the automotive container registry.
- All Kubernetes YAML files are located in the Git repository.
- Although the instructions get the files directly from GitLab, you may get the code by cloning the repo:
git clone https://gitlab.com/CentOS/automotive/sample-images.git
- This demo uses Kubernetes and the command
kubectl
, but you can achieve the same thing using OpenShift and theoc
command.
[ Download the Podman basics cheat sheet ]
1. Deploy on Kubernetes
a. Create the namespace:
kubectl create namespace build-once-run-anywhere
b. Deploy vsomeip:
kubectl apply -n build-once-run-anywhere -f https://gitlab.com/CentOS/automotive/sample-images/-/raw/main/osbuild-manifests/files/ocp/vsomeip.yml
c. Deploy the engine application:
kubectl apply -n build-once-run-anywhere -f https://gitlab.com/CentOS/automotive/sample-images/-/raw/main/osbuild-manifests/files/ocp/engine.yml
d. Deploy the radio application:
kubectl apply -n build-once-run-anywhere -f https://gitlab.com/CentOS/automotive/sample-images/-/raw/main/osbuild-manifests/files/ocp/radio.yml
e. After the deployments are running, you can monitor changes in the radio service's volume based on the notification from the engine service:
kubectl get pods -n build-once-run-anywhere -l app=radio -o jsonpath={.items[0].metadata.name} | xargs kubectl logs -n build-once-run-anywhere -f
RADIO: Started main thread
RADIO: Started playing
Engine Service is NOT available.
Engine Service is available.
RADIO: Playing song "R-Cali" by A$AP Rocky feat. Aston Matthews & Joey Fatts (on Radio Los Santos) 50% volume
RADIO: Lowering volume due to reverse
RADIO: Playing song "R-Cali" by A$AP Rocky feat. Aston Matthews & Joey Fatts (on Radio Los Santos) 30% volume
RADIO: Playing song "Swimming Pools, Drank" by Kendrick Lamar (on Radio Los Santos) 30% volume
RADIO: Restoring volume due to cancelled reverse
RADIO: Playing song "Swimming Pools, Drank" by Kendrick Lamar (on Radio Los Santos) 50% volume
2. Deploy locally using Podman
a. Deploy vsomeip:
podman kube play https://gitlab.com/CentOS/automotive/sample-images/-/raw/main/osbuild-manifests/files/ocp/vsomeip.yml
b. Deploy the engine application:
podman kube play https://gitlab.com/CentOS/automotive/sample-images/-/raw/main/osbuild-manifests/files/ocp/engine.yml
c. Deploy the radio application:
podman kube play https://gitlab.com/CentOS/automotive/sample-images/-/raw/main/osbuild-manifests/files/ocp/radio.yml
d. After the deployments are running, you can monitor changes in the radio service's volume based on the notification from the engine service:
podman logs -f radio-pod-0-radio
RADIO: Started main thread
RADIO: Started playing
RADIO: Playing song "How It Was" by DJ Esco feat. Future (on Radio Los Santos) 50% volume
Engine Service is NOT available.
Engine Service is available.
RADIO: Playing song "Swimming Pools, Drank" by Kendrick Lamar (on Radio Los Santos) 50% volume
RADIO: Lowering volume due to reverse
RADIO: Playing song "Swimming Pools, Drank" by Kendrick Lamar (on Radio Los Santos) 30% volume
RADIO: Playing song "Hood Gone Love It" by Jay Rock feat. Kendrick Lamar (on Radio Los Santos) 30% volume
RADIO: Restoring volume due to cancelled reverse
RADIO: Playing song "Hood Gone Love It" by Jay Rock feat. Kendrick Lamar (on Radio Los Santos) 50% volume
3. Embed inside an image
For more information about how to build an image, see Building images on the Automotive SIG documentation site. The image manifest of this demo is the ocp.mpp
file.
Contents of the manifest file
The manifest describes the different steps required to create the image.
Embed the container image
- type: org.osbuild.skopeo
inputs:
images:
type: org.osbuild.containers
origin: org.osbuild.source
mpp-resolve-images:
images:
- source: registry.gitlab.com/centos/automotive/sample-images/demo/auto-apps
tag: latest
- source: registry.gitlab.com/centos/automotive/sample-images/demo/vsomeip
tag: v0.1
options:
destination:
type: containers-storage
storage-path: /usr/share/containers/storage
Copy the unit and Kubernetes YAML files
- type: org.osbuild.copy
inputs:
ocp-vsomeip:
type: org.osbuild.files
origin: org.osbuild.source
mpp-embed:
id: vsomeip.yml
path: ../files/ocp/vsomeip.yml
unit-vsomeip:
type: org.osbuild.files
origin: org.osbuild.source
mpp-embed:
id: vsomeip.service
path: ../files/ocp/vsomeip.service
options:
paths:
- from:
mpp-format-string: input://ocp-vsomeip/{embedded['vsomeip.yml']}
to: tree:///demo/ocp/vsomeip.yml
- from:
mpp-format-string: input://unit-vsomeip/{embedded['vsomeip.service']}
to: tree:///usr/lib/systemd/system/vsomeip.service
Notice the similar copy operations for the engine and radio services.
Enable the radio service
- type: org.osbuild.systemd
options:
enabled_services:
- radio.service
Build the image
Now it's time to build the image.
For the demo
To run the demo image on an ARM machine on AWS, make the target with:
cd osbuild-manifests
make cs9-aws-ocp-regular.aarch64.img
Building other images
Alternatively, you can build the demo on x86 hardware and run it with QEMU:
cd osbuild-manifests
make cs9-aws-ocp-regular.x86_64.qcow2
c. You can find a convenient tool to simplify running the qcow2 image in the repository at osbuild-manifests/runvm. This can make it easier for you to run the image with QEMU virtualization.
[ Try the Getting started with Red Hat OpenShift Service on AWS (ROSA) learning path. ]
Convert the image into an AWS AMI
To convert the image you created into an Amazon Machine Image (AMI), follow the instructions below.
Prerequisites
- Create an S3 bucket to upload the image to.
- Access to the AWS command-line interface (CLI).
- AWS credentials:
- If you are running it on your own machine, configure the AWS CLI with your credentials.
- If you are running on an EC2 instance, attach an identity and access management (IAM) instance profile with the VMImporter policy to your instance.
Create the AMI
a. From the osbuild-manifests
directory, run the export-image-aws.sh
tool:
./tools/export-image-aws.sh cs9-aws-ocp-regular.aarch64.img <The name of the S3 Bucket> 8
b. Once completed, you can find the new AMI with the name cs9-aws-ocp-regular.aarch64
.
Use the newly created AMI
Once you've created the AMI, you can start using it.
Start an EC2 instance
- Locate the AMI you created in the previous step in your AWS console (its name is
cs9-aws-soafee-regular.aarch64
). - Click Launch instance from AMI.
- Complete the instance-launching flow:
- Give your instance a name.
- Choose the instance type. The recommended one is c6g.large.
- Set (or create) your key pair to Secure Shell (SSH) into the instance.
- Set (or create) the Security Group that allows SSH access to the machine from your IP.
- Click Launch instance.
Verify the service operation
SSH into your instance and check the logs coming from the radio service:
journalctl -u radio -f
RADIO: Started main thread
RADIO: Started playing
RADIO: Playing song "How It Was" by DJ Esco feat. Future (on Radio Los Santos) 50% volume
Engine Service is NOT available.
Engine Service is available.
RADIO: Playing song "R-Cali" by A$AP Rocky feat. Aston Matthews & Joey Fatts (on Radio Los Santos) 50% volume
RADIO: Playing song "Swimming Pools, Drank" by Kendrick Lamar (on Radio Los Santos) 50% volume
RADIO: Lowering volume due to reverse
RADIO: Playing song "Swimming Pools, Drank" by Kendrick Lamar (on Radio Los Santos) 30% volume
RADIO: Restoring volume due to cancelled reverse
RADIO: Playing song "Swimming Pools, Drank" by Kendrick Lamar (on Radio Los Santos) 50% volume
Build once, run anywhere
Container images are the manifestation of the "build once, run anywhere" approach. Using Kubernetes, Podman, and OSBuild (and, in the future, Quadlet), "anywhere" is now bigger than ever.
[ Boost security, flexibility, and scale at the edge with Red Hat Enterprise Linux. ]
Image
Both projects let you run multiple Podman containers on a single machine. But their differences might make one more appealing than the other.
Image
The Enable Sysadmin community continues to answer key questions about OpenShift and Kubernetes.
Image
More information about Podman in containers; specifically with regard to Kubernetes.
Ygal Blum
Ygal Blum is a Principal Software Engineer who is also an experienced manager and tech lead. More about me