Skip to content

Commit

Permalink
Use layers digests for comparing podman images
Browse files Browse the repository at this point in the history
Previous method checked default digest provided by podman. This digest
is "local" and changed every time image is saved/load or at any other
point manifest is modified. This doesn't mean that it is a different
image.

Viable way for our purposes is to compare that all layers are identical
and in the same order. Simple way to distill this into one value is to
concatenate individual layers' digests in order of appearance in RootFS.

Related to discussion at
containers/podman#24818
  • Loading branch information
tkopecek committed Dec 18, 2024
1 parent 9f0247c commit 7d5cd17
Show file tree
Hide file tree
Showing 6 changed files with 59 additions and 19 deletions.
2 changes: 1 addition & 1 deletion mock/docs/buildroot-lock-schema-1.0.0.json
Original file line number Diff line number Diff line change
Expand Up @@ -83,7 +83,7 @@
"additionalProperties": false,
"properties": {
"image_digest": {
"description": "Digest got by the 'podman image inspect --format {{ .Digest }}' command, sha256 string",
"description": "SHA256 digest concatenated RootFS layer digests and Config section from 'podman image inspect' command, sha256 string",
"type": "string"
}
}
Expand Down
6 changes: 3 additions & 3 deletions mock/py/mockbuild/buildroot.py
Original file line number Diff line number Diff line change
Expand Up @@ -281,13 +281,13 @@ def _fallback(message):
self.chroot_image, podman.image_id)
podman.tag_image()

digest_expected = self.image_assert_digest
digest_expected = self.config.get("image_assert_digest")
if digest_expected:
getLog().info("Checking image digest: %s",
digest_expected)
digest = podman.get_image_digest()
digest = podman.get_oci_digest()
if digest != digest_expected:
getLog().warning(
raise BootstrapError(
f"Expected digest for image {podman.image} is"
f"{digest_expected}, but {digest} found.")

Expand Down
2 changes: 1 addition & 1 deletion mock/py/mockbuild/plugins/buildroot_lock.py
Original file line number Diff line number Diff line change
Expand Up @@ -104,7 +104,7 @@ def _executor(cmd):
try:
podman = Podman(self.buildroot,
data["config"]["bootstrap_image"])
digest = podman.get_image_digest()
digest = podman.get_oci_digest()
except PodmanError:
digest = "unknown"
data["bootstrap"] = {
Expand Down
64 changes: 51 additions & 13 deletions mock/py/mockbuild/podman.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
# -*- coding: utf-8 -*-
# vim: noai:ts=4:sw=4:expandtab

import hashlib
import json
import os
import logging
import subprocess
Expand All @@ -16,6 +18,47 @@ class PodmanError(Exception):
"""


def podman_get_oci_digest(image, logger=None, podman_binary=None):
"""
Get sha256 digest of RootFS layers. This must be identical for
all images containing same order of layers, thus it can be used
as the check that we've loaded same image.
"""
logger = logger or logging.getLogger()
podman = podman_binary or "/usr/bin/podman"
logger.info("Calculating %s image OCI digest", image)
check = [podman, "image", "inspect", image]
result = subprocess.run(check, stdout=subprocess.PIPE,
stderr=subprocess.PIPE, check=False,
encoding="utf8")
if result.returncode:
logger.error(f"Can't get {image} podman image digest: {result.stderr}")

Check warning

Code scanning / vcs-diff-lint

podman_get_oci_digest: Use lazy % formatting in logging functions Warning

podman_get_oci_digest: Use lazy % formatting in logging functions
return None
result = result.stdout.strip()

try:
data = json.loads(result)[0]
except json.JSONDecodeError:
logger.error(f"The manifest data of {image} are not json-formatted.")
return None

if 'RootFS' not in data:
logger.error(f"RootFS section of {image} is missing.")

Check warning

Code scanning / vcs-diff-lint

podman_get_oci_digest: Use lazy % formatting in logging functions Warning

podman_get_oci_digest: Use lazy % formatting in logging functions
return None
if data['RootFS']['Type'] != 'layers':
logger.error(f"Unexpected format for RootFS in {image}.")

Check warning

Code scanning / vcs-diff-lint

podman_get_oci_digest: Use lazy % formatting in logging functions Warning

podman_get_oci_digest: Use lazy % formatting in logging functions
return None

# data which should be sufficient to confirm the image
data = {
'RootFS': data['RootFS'],
'Config': data['Config'],
}
# convert to json string with ordered dicts and create hash
data = json.dumps(data, sort_keys=True)
return hashlib.sha256(data.encode()).hexdigest()


def podman_check_native_image_architecture(image, logger=None, podman_binary=None):
"""
Return True if image's architecture is "native" for this host.
Expand Down Expand Up @@ -132,26 +175,21 @@ def mounted_image(self):
subprocess.run(cmd_umount, stdout=subprocess.PIPE,
stderr=subprocess.PIPE, check=True)

def get_image_digest(self):
def get_oci_digest(self):
"""
Get the "sha256:..." string for the image we work with.
Get sha256 digest of RootFS layers. This must be identical for
all images containing same order of layers, thus it can be used
as the check that we've loaded same image.
"""
the_image = self.image
if the_image.startswith("oci-archive:"):
# We can't query digest from tarball directly, but note
# the image needs to be tagged first!
the_image = self._tagged_id
check = [self.podman_binary, "image", "inspect", the_image,
"--format", "{{ .Digest }}"]
result = subprocess.run(check, stdout=subprocess.PIPE,
stderr=subprocess.PIPE, check=False,
encoding="utf8")
if result.returncode:
raise PodmanError(f"Can't get {the_image} podman image digest: {result.stderr}")
result = result.stdout.strip()
if len(result.splitlines()) != 1:
raise PodmanError(f"The digest of {the_image} image is not a single-line string")
return result
digest = podman_get_oci_digest(the_image, logger=getLog())
if digest is None:
raise PodmanError(f"Getting OCI digest for image {self.image} failed")
return digest

def check_native_image_architecture(self):
"""
Expand Down
2 changes: 1 addition & 1 deletion mock/tests/test_buildroot_lock.py
Original file line number Diff line number Diff line change
Expand Up @@ -99,7 +99,7 @@ def _call_method(plugins, buildroot):
_, method = plugins.add_hook.call_args[0]

podman_obj = MagicMock()
podman_obj.get_image_digest.return_value = EXPECTED_OUTPUT["bootstrap"]["image_digest"]
podman_obj.get_oci_digest.return_value = EXPECTED_OUTPUT["bootstrap"]["image_digest"]
podman_cls = MagicMock(return_value=podman_obj)
with patch("mockbuild.plugins.buildroot_lock.Podman", side_effect=podman_cls):
method()
Expand Down
2 changes: 2 additions & 0 deletions releng/release-notes-next/podman-digests.feature
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
Hermetic build process is enhanced by adding used imaged digests into the
metadata and confirming that exactly same image is used in the next step.

0 comments on commit 7d5cd17

Please sign in to comment.