This directory contains the sources to build the podvm image (qcow2 file) for various Linux distributions and cloud providers. So use the instructions in the next sections if you need to build your own image with changes to meet your requirements. Otherwise you can find here information on how to consume pre-built images.
In order to build locally it requires the source trees and softwares mentioned in the developer's guide to build this project binaries. It will also need packer (to build the qcow2), rust (to build the Kata Containers's agent), as well as the following packages:
-
On Ubuntu:
$ apt-get install -y qemu-kvm cloud-utils qemu-utils protobuf-compiler pkg-config libdevmapper-dev libgpgme-dev
You may need to link the agent with the musl C library. In this case, you should install the musl-tools (Ubuntu) package and setup the Rust toolchain as explained here.
Finally run the following commands to build the qcow2 image:
$ export CLOUD_PROVIDER=[aws|azure|ibmcloud|libvirt|vsphere|generic]
$ make image
NOTE: "generic" is a best-effort provider agnostic image creation
This directory contains dockerfiles to build the podvm image entirely within container so that it only requires docker or podman installed on the host.
In general it is needed to follow these steps:
- Build a builder container image for a given Linux distribution
- Build an image containing all the required podvm binaries for a given Linux distribution
- Build an image containing the podvm qcow2 image for a given cloud provider
- Extract the podvm image (qcow2 file) from the container image
The next sections describe that process in details. Note that although the following examples use docker, it can be carried out with podman too.
The builder image packages the cloud-api-adaptor and Kata Containers sources as well as the softwares to build the binaries (e.g. kata-agent and agent-protocol-forwarder) that should be installed in the podvm image.
The builder image is agnostic to cloud providers in the sense that one can be used to build for multiple providers, however it is dependent on the Linux distribution the image is built for. Therefore, in this directory you will find dockerfiles for each supported distributions, which are currently Ubuntu 20.04 (Dockerfile.podvm_builder), CentOS Stream 9 (Dockerfile.podvm_builder.centos), and RHEL 9 (Dockerfile.podvm_builder.rhel).
As an example, to build the builder image for Ubuntu, run:
$ docker build -t podvm_builder \
-f Dockerfile.podvm_builder .
Use --build-arg
to pass build arguments to docker to overwrite default values if needed. Following are the arguments
currently accepted:
Argument | Default value | Description |
---|---|---|
CAA_SRC | https://github.com/confidential-containers/cloud-api-adaptor | The cloud-api-adaptor source repository |
CAA_SRC_REF | main | cloud-api-adaptor repository branch or commit |
KATA_SRC | https://github.com/kata-containers/kata-containers | The Kata Containers source repository |
KATA_SRC_BRANCH | CCv0 | The Kata Containers repository branch |
GO_VERSION | 1.20.10 | Go version |
PROTOC_VERSION | 3.11.4 | Protobuf version |
RUST_VERSION | 1.72.0 | Rust version |
As it can be noted in the table above the cloud-api-adaptor repository is cloned within the builder image, so rather than copying the local source tree, it will be using the upstream source. But if you want to test local changes then you should:
- Push the changes to your fork in github (e.g. https://github.com/$USER/cloud-api-adaptor/tree/my-changes-in-a-branch).
- Overwrite the CAA_SRC and CAA_SRC_REF arguments as shown below:
$ docker build -t podvm_builder \
--build-arg CAA_SRC=https://github.com/$USER/cloud-api-adaptor \
--build-arg CAA_SRC_REF=my-changes-in-a-branch \
-f Dockerfile.podvm_builder .
In order to build the binaries you should be using the corresponding dockerfile of the Linux distro for which the builder image was built. For example, if the builder image was built with Dockerfile.podvm_builder.DISTRO then you should use the Dockerfile.podvm_binaries.DISTRO to build the image with podvm binaries.
Assuming you have built the podvm_builder image for Ubuntu as explained in the previous step, running the following command will build the image with the podvm binaries.
$ docker build -t podvm_binaries \
--build-arg BUILDER_IMG=podvm_builder \
-f Dockerfile.podvm_binaries .
The build process can take significant time.
The binaries can be built for other architectures than x86_64
by passing the ARCH
build
argument to docker. Currently this is only supported for Ubuntu s390x
as shown below:
$ docker build -t podvm_binaries_s390x \
--build-arg BUILDER_IMG=podvm_builder \
--build-arg ARCH=s390x \
-f Dockerfile.podvm_binaries .
In order to build the podvm image you should be using the corresponding dockerfile of the Linux distro for which the builder and binaries image were built. For example, if the builder image was built with Dockerfile.podvm_builder.DISTRO then you should use the Dockerfile.podvm.DISTRO to build the podvm image.
The builder image has to be indicated via BUILDER_IMG
build argument and
binaries image has to be indicated via BINARIES_IMG
build argument to docker.
Below command will build the qcow2 image that can be used for all cloud providers based on Ubuntu distro.
$ docker build -t podvm \
--build-arg BUILDER_IMG=podvm_builder \
--build-arg BINARIES_IMG=podvm_binaries \
-f Dockerfile.podvm .
This step will take several minutes to complete, mostly because packer
will
use the QEMU builder in emulation mode when running within container.
Tip: If you are using podman then you can speed up QEMU by enabling native virtualization, by passing the
--device=/dev/kvm
argument to enable KVM inside the container.
Note: Beware that the process consume a bunch of memory and disk from the host. If the build fails at the point QEMU was launched but packer couldn't connect via ssh, with an error similar to:
Build 'qemu.ubuntu' errored after 5 minutes 57 seconds: Timeout waiting for SSH.
then it might indicate lack of memory, so try to increase the amount of memory if running on VM.
The podvm image can be built for other architectures than x86_64
by passing
the ARCH
build argument to docker. Currently this is only supported for
Ubuntu s390x
, which also needs the UBUNTU_IMAGE_URL
and
UBUNTU_IMAGE_CHECKSUM
to be overridden with build arguments as shown below:
$ docker build -t podvm_s390x \
--build-arg ARCH=s390x \
--build-arg BUILDER_IMG=podvm_builder \
--build-arg BINARIES_IMG=podvm_binaries_s390x \
--build-arg UBUNTU_IMAGE_URL="" \
--build-arg UBUNTU_IMAGE_CHECKSUM="" \
-f Dockerfile.podvm .
The Secure Execution enabled podvm image can be built by passing the SE_BOOT
build argument to docker. Currently this is only supported for Ubutu s390x
, which also needs put the HOST KEY documents
to the files folder, please follow the Download host key document from Resource Link
section at this document to download HOST KEY documents
.
$ tree -L 1 files
files
├── HKD-8562-1234567.crt
├── etc
└── usr
Running below command will build the Secure Execution enabled qcow2 image:
$ docker build -t se_podvm_s390x \
--build-arg ARCH=s390x \
--build-arg SE_BOOT=1 \
--build-arg BUILDER_IMG=podvm_builder \
--build-arg BINARIES_IMG=podvm_binaries_s390x \
--build-arg UBUNTU_IMAGE_URL="" \
--build-arg UBUNTU_IMAGE_CHECKSUM="" \
-f Dockerfile.podvm .
The podvm image can also be built using UEFI based images. For example if you want to build a RHEL podvm image using UEFI based qcow2 image, then run the build using as shown below:
$ docker build -t podvm-uefi \
--build-arg BUILDER_IMG=podvm_builder \
--build-arg BINARIES_IMG=podvm_binaries \
--build-arg RHEL_IMAGE_URL="_url_to_uefi_based_qcow2_" \
--build-arg RHEL_IMAGE_CHECKSUM="_qcow2_image_checksum" \
--build-arg UEFI=true \
-f Dockerfile.podvm.rhel .
The final podvm image, i.e. the qcow2 file, is stored on the root of the podvm container image.
There are a couple of ways to extract files from a container image using docker or podman. However, to ease that task we have the hack/download-image.sh script, which copy the qcow2 file out of the podvm container image.
Running the below command will extract the qcow2 image built in the previous step.
$ ./hack/download-image.sh podvm:latest . -o podvm.qcow2
Running the below command will extract the Secure Execution enabled qcow2 image built in the previous step.
$ ./hack/download-image.sh se_podvm_s390x:latest . -o se_podvm.qcow2
In order to add a new Linux distribution essentially it is needed to create some dockerfiles and add the packer configuration files.
Follow the steps below, replacing DISTRO
with the name of the distribution being added:
- Create the builder dockerfile by copying
Dockerfile.podvm_builder
toDockerfile.podvm_builder.DISTRO
and adjusting the file properly (e.g. replaceFROM ubuntu:20.04
withFROM DISTRO
). Try to keep the same software versions (e.g. Golang, Rust) as much as possible. - Create the podvm image dockerfile by copying
Dockerfile.podvm
toDockerfile.podvm.DISTRO
and adjusting the file properly likewise. In particular, the PODVM_DISTRO and BUILDER_IMG arguments should be changed. - Create the podvm binaries dockerfile by copying
Dockerfile.podvm_binaries
toDockerfile.podvm_binaries.DISTRO
and adjusting the file as needed. - Create the packer directory (
mkdir qcow2/DISTRO
) where theqemu-DISTRO.pkr.hcl
andvariables.pkr.hcl
files should be placed. Also on this step you can also use an existing configuration (e.g.qcow2/ubuntu
) as a template. Ensure that common scripts and files like theqcow2/misc-settings.sh
are adjusted to support the DISTRO if needed. - Define the base image URL and checksum in the
Makefile
file. - Update this README.md properly in case that there are specific instructions and/or constraints for the DISTRO.