From 2444a852f886b67d3bd8594c3df126aa867423ef Mon Sep 17 00:00:00 2001 From: Michael Vogt Date: Wed, 23 Oct 2024 16:36:02 +0200 Subject: [PATCH] test: extract a common osbuild_fixture() helper We currently have three copies of the osbuild_fixture() fixture that are slightly different. This commit extract it into a single place in `test.py` and shares it accross the various callers. Note that it would be even nicer to put it into `conftest.py` but there is a circular import right now when trying this. We should still do it eventually it just requires a bit more preperation work. Split out from https://github.com/osbuild/osbuild/pull/1878 --- test/run/test_assemblers.py | 261 +++++++++++++++++------------------- test/run/test_exports.py | 8 +- test/run/test_noop.py | 6 +- test/test.py | 20 +++ 4 files changed, 148 insertions(+), 147 deletions(-) diff --git a/test/run/test_assemblers.py b/test/run/test_assemblers.py index 17ca20f79e..87ba9c313d 100644 --- a/test/run/test_assemblers.py +++ b/test/run/test_assemblers.py @@ -15,15 +15,10 @@ from osbuild import loop from .. import test - -MEBIBYTE = 1024 * 1024 +from ..test import osbuild_fixture # noqa: F401, pylint: disable=unused-import -@pytest.fixture(name="osbuild") -def osbuild_fixture(): - store = os.getenv("OSBUILD_TEST_STORE") - osb = test.OSBuild(cache_from=store) - yield osb +MEBIBYTE = 1024 * 1024 def assertImageFile(filename, fmt, expected_size): @@ -83,7 +78,7 @@ def read_partition_table(device): @pytest.mark.skipif(not test.TestBase.have_test_data(), reason="no test-data access") @pytest.mark.skipif(not test.TestBase.can_bind_mount(), reason="root-only") @pytest.mark.parametrize("fs_type", ["ext4", "xfs", "btrfs"]) -def test_rawfs(osbuild, fs_type): +def test_rawfs(osb, fs_type): if not test.TestBase.has_filesystem_support(fs_type): pytest.skip(f"The {fs_type} was explicitly marked as unsupported on this platform.") options = { @@ -92,65 +87,63 @@ def test_rawfs(osbuild, fs_type): "size": 1024 * MEBIBYTE, "fs_type": fs_type, } - with osbuild as osb: - with run_assembler(osb, "org.osbuild.rawfs", options, "image.raw") as (tree, image): - assertImageFile(image, "raw", options["size"]) - assertFilesystem(image, options["root_fs_uuid"], fs_type, tree) + with run_assembler(osb, "org.osbuild.rawfs", options, "image.raw") as (tree, image): + assertImageFile(image, "raw", options["size"]) + assertFilesystem(image, options["root_fs_uuid"], fs_type, tree) @pytest.mark.skipif(not test.TestBase.have_tree_diff(), reason="tree-diff missing") @pytest.mark.skipif(not test.TestBase.have_test_data(), reason="no test-data access") @pytest.mark.skipif(not test.TestBase.can_bind_mount(), reason="root-only") @pytest.mark.skipif(not test.TestBase.have_rpm_ostree(), reason="rpm-ostree missing") -def test_ostree(osbuild): - with osbuild as osb: - with open(os.path.join(test.TestBase.locate_test_data(), - "manifests/fedora-ostree-commit.json"), - encoding="utf8") as f: - manifest = json.load(f) - - data = json.dumps(manifest) - with tempfile.TemporaryDirectory(dir="/var/tmp") as output_dir: - result = osb.compile(data, output_dir=output_dir, exports=["ostree-commit"]) - compose_file = os.path.join(output_dir, "ostree-commit", "compose.json") - repo = os.path.join(output_dir, "ostree-commit", "repo") - - with open(compose_file, encoding="utf8") as f: - compose = json.load(f) - commit_id = compose["ostree-commit"] - ref = compose["ref"] - rpmostree_inputhash = compose["rpm-ostree-inputhash"] - os_version = compose["ostree-version"] - assert commit_id - assert ref - assert rpmostree_inputhash - assert os_version - assert "metadata" in result - metadata = result["metadata"] - commit = metadata["ostree-commit"] - info = commit["org.osbuild.ostree.commit"] - assert "compose" in info - assert info["compose"] == compose - - md = subprocess.check_output( - [ - "ostree", - "show", - "--repo", repo, - "--print-metadata-key=rpmostree.inputhash", - commit_id - ], encoding="utf8").strip() - assert md == f"'{rpmostree_inputhash}'" - - md = subprocess.check_output( - [ - "ostree", - "show", - "--repo", repo, - "--print-metadata-key=version", - commit_id - ], encoding="utf8").strip() - assert md == f"'{os_version}'" +def test_ostree(osb): + with open(os.path.join(test.TestBase.locate_test_data(), + "manifests/fedora-ostree-commit.json"), + encoding="utf8") as f: + manifest = json.load(f) + + data = json.dumps(manifest) + with tempfile.TemporaryDirectory(dir="/var/tmp") as output_dir: + result = osb.compile(data, output_dir=output_dir, exports=["ostree-commit"]) + compose_file = os.path.join(output_dir, "ostree-commit", "compose.json") + repo = os.path.join(output_dir, "ostree-commit", "repo") + + with open(compose_file, encoding="utf8") as f: + compose = json.load(f) + commit_id = compose["ostree-commit"] + ref = compose["ref"] + rpmostree_inputhash = compose["rpm-ostree-inputhash"] + os_version = compose["ostree-version"] + assert commit_id + assert ref + assert rpmostree_inputhash + assert os_version + assert "metadata" in result + metadata = result["metadata"] + commit = metadata["ostree-commit"] + info = commit["org.osbuild.ostree.commit"] + assert "compose" in info + assert info["compose"] == compose + + md = subprocess.check_output( + [ + "ostree", + "show", + "--repo", repo, + "--print-metadata-key=rpmostree.inputhash", + commit_id + ], encoding="utf8").strip() + assert md == f"'{rpmostree_inputhash}'" + + md = subprocess.check_output( + [ + "ostree", + "show", + "--repo", repo, + "--print-metadata-key=version", + commit_id + ], encoding="utf8").strip() + assert md == f"'{os_version}'" @pytest.mark.skipif(not test.TestBase.have_tree_diff(), reason="tree-diff missing") @@ -158,51 +151,50 @@ def test_ostree(osbuild): @pytest.mark.skipif(not test.TestBase.can_bind_mount(), reason="root-only") @pytest.mark.parametrize("fmt,", ["raw", "raw.xz", "qcow2", "vmdk", "vdi"]) @pytest.mark.parametrize("fs_type", ["ext4", "xfs", "btrfs"]) -def test_qemu(osbuild, fmt, fs_type): +def test_qemu(osb, fmt, fs_type): loctl = loop.LoopControl() - with osbuild as osb: - if not test.TestBase.has_filesystem_support(fs_type): - pytest.skip(f"The {fs_type} was explicitly marked as unsupported on this platform.") - options = { - "format": fmt, - "filename": f"image.{fmt}", - "ptuuid": str(uuid4()), - "root_fs_uuid": str(uuid4()), - "size": 1024 * MEBIBYTE, - "root_fs_type": fs_type, - } - with run_assembler(osb, - "org.osbuild.qemu", - options, - f"image.{fmt}") as (tree, image): - if fmt == "raw.xz": - subprocess.run(["unxz", "--keep", "--force", image], check=True) - image = image[:-3] - fmt = "raw" - assertImageFile(image, fmt, options["size"]) - with open_image(loctl, image, fmt) as (target, device): - ptable = read_partition_table(device) - assertPartitionTable(ptable, - "dos", - options["ptuuid"], - 1, - boot_partition=1) - if fs_type == "btrfs": - l2hash = "dfea8604cdcdc2cf0dc3e816929d302f837f03c98f6bb19b366fda4ee26957f1" - elif fs_type == "xfs": - l2hash = "18d145b699e0ff4a815868f0b761b4bd6ece391885830ddb56dac557eff8b732" - else: - l2hash = "84792ea01e63bd60c4018d686d48b6207c56d5bc844f2791634b276ce82c9116" - assertGRUB2(device, - "b8cea7475422d35cd6f85ad099fb4f921557fd1b25db62cd2a92709ace21cf0f", - l2hash, - 1024 * 1024) - - p1 = ptable["partitions"][0] - ssize = ptable.get("sectorsize", 512) - start, size = p1["start"] * ssize, p1["size"] * ssize - with loop_open(loctl, target, offset=start, size=size) as dev: - assertFilesystem(dev, options["root_fs_uuid"], fs_type, tree) + if not test.TestBase.has_filesystem_support(fs_type): + pytest.skip(f"The {fs_type} was explicitly marked as unsupported on this platform.") + options = { + "format": fmt, + "filename": f"image.{fmt}", + "ptuuid": str(uuid4()), + "root_fs_uuid": str(uuid4()), + "size": 1024 * MEBIBYTE, + "root_fs_type": fs_type, + } + with run_assembler(osb, + "org.osbuild.qemu", + options, + f"image.{fmt}") as (tree, image): + if fmt == "raw.xz": + subprocess.run(["unxz", "--keep", "--force", image], check=True) + image = image[:-3] + fmt = "raw" + assertImageFile(image, fmt, options["size"]) + with open_image(loctl, image, fmt) as (target, device): + ptable = read_partition_table(device) + assertPartitionTable(ptable, + "dos", + options["ptuuid"], + 1, + boot_partition=1) + if fs_type == "btrfs": + l2hash = "dfea8604cdcdc2cf0dc3e816929d302f837f03c98f6bb19b366fda4ee26957f1" + elif fs_type == "xfs": + l2hash = "18d145b699e0ff4a815868f0b761b4bd6ece391885830ddb56dac557eff8b732" + else: + l2hash = "84792ea01e63bd60c4018d686d48b6207c56d5bc844f2791634b276ce82c9116" + assertGRUB2(device, + "b8cea7475422d35cd6f85ad099fb4f921557fd1b25db62cd2a92709ace21cf0f", + l2hash, + 1024 * 1024) + + p1 = ptable["partitions"][0] + ssize = ptable.get("sectorsize", 512) + start, size = p1["start"] * ssize, p1["size"] * ssize + with loop_open(loctl, target, offset=start, size=size) as dev: + assertFilesystem(dev, options["root_fs_uuid"], fs_type, tree) @pytest.mark.skipif(not test.TestBase.have_tree_diff(), reason="tree-diff missing") @@ -213,37 +205,36 @@ def test_qemu(osbuild, fmt, fs_type): [("tree.tar.gz", None, ["application/x-tar"]), ("tree.tar.gz", "gzip", ["application/x-gzip", "application/gzip"])] ) -def test_tar(osbuild, filename, compression, expected_mimetypes): - with osbuild as osb: - options = {"filename": filename} +def test_tar(osb, filename, compression, expected_mimetypes): + options = {"filename": filename} + if compression: + options["compression"] = compression + with run_assembler(osb, + "org.osbuild.tar", + options, + filename) as (tree, image): + output = subprocess.check_output(["file", "--mime-type", image], encoding="utf8") + _, mimetype = output.strip().split(": ") # "filename: mimetype" + assert mimetype in expected_mimetypes + if compression: - options["compression"] = compression - with run_assembler(osb, - "org.osbuild.tar", - options, - filename) as (tree, image): - output = subprocess.check_output(["file", "--mime-type", image], encoding="utf8") - _, mimetype = output.strip().split(": ") # "filename: mimetype" - assert mimetype in expected_mimetypes - - if compression: - return - - # In the non-compression case, we verify the tree's content - with tempfile.TemporaryDirectory(dir="/var/tmp") as tmp: - args = [ - "tar", - "--numeric-owner", - "--selinux", - "--acls", - "--xattrs", "--xattrs-include", "*", - "-xaf", image, - "-C", tmp] - subprocess.check_output(args, encoding="utf8") - diff = test.TestBase.tree_diff(tree, tmp) - assert diff["added_files"] == [] - assert diff["deleted_files"] == [] - assert diff["differences"] == {} + return + + # In the non-compression case, we verify the tree's content + with tempfile.TemporaryDirectory(dir="/var/tmp") as tmp: + args = [ + "tar", + "--numeric-owner", + "--selinux", + "--acls", + "--xattrs", "--xattrs-include", "*", + "-xaf", image, + "-C", tmp] + subprocess.check_output(args, encoding="utf8") + diff = test.TestBase.tree_diff(tree, tmp) + assert diff["added_files"] == [] + assert diff["deleted_files"] == [] + assert diff["differences"] == {} @contextlib.contextmanager diff --git a/test/run/test_exports.py b/test/run/test_exports.py index c2c1fab825..52d290c9cf 100644 --- a/test/run/test_exports.py +++ b/test/run/test_exports.py @@ -7,7 +7,7 @@ import pytest -from .. import test +from ..test import osbuild_fixture # noqa: F401, pylint: disable=unused-import @pytest.fixture(name="jsondata", scope="module") @@ -42,12 +42,6 @@ def jsondata_fixture(): }) -@pytest.fixture(name="osb", scope="module") -def osbuild_fixture(): - with test.OSBuild() as osb: - yield osb - - @pytest.fixture(name="testing_libdir", scope="module") def testing_libdir_fixture(tmpdir_factory): tests_path = pathlib.Path(__file__).parent.parent diff --git a/test/run/test_noop.py b/test/run/test_noop.py index 57af1e88d0..dfbf4e01c9 100644 --- a/test/run/test_noop.py +++ b/test/run/test_noop.py @@ -10,6 +10,7 @@ import osbuild.main_cli from .. import test +from ..test import osbuild_fixture # noqa: F401, pylint: disable=unused-import @pytest.fixture(name="jsondata", scope="module") @@ -39,11 +40,6 @@ def jsondata_fixture(): }) -@pytest.fixture(name="osb", scope="module") -def osbuild_fixture(): - with test.OSBuild() as osb: - yield osb - # # Run a noop Pipeline. Run twice to verify the cache does not affect # the operation (we do not have checkpoints, nor any stages that could diff --git a/test/test.py b/test/test.py index 018bc4e009..564e252e45 100644 --- a/test/test.py +++ b/test/test.py @@ -6,11 +6,14 @@ import errno import json import os +import shutil import subprocess import sys import tempfile import unittest +import pytest + import osbuild.meta from osbuild.objectstore import ObjectStore from osbuild.util import linux @@ -296,6 +299,7 @@ class OSBuild(contextlib.AbstractContextManager): def __init__(self, *, cache_from=None): self._cache_from = cache_from + self.store = None def __enter__(self): self._exitstack = contextlib.ExitStack() @@ -496,3 +500,19 @@ def copy_source_data(self, target, source): "cp", "--reflink=auto", "-a", os.path.join(from_path, "."), to_path ], check=True) + + +@pytest.fixture(name="osb") +def osbuild_fixture(): + cleanup_dir = None + store = os.getenv("OSBUILD_TEST_STORE") + if not store: + # we cannot use tmp_path_factory here as there is no easy way + # to put it under /var/tmp + store = tempfile.mkdtemp(prefix="osbuild-test-", dir="/var/tmp") + cleanup_dir = store + with OSBuild(cache_from=store) as osb: + osb.store = store + yield osb + if cleanup_dir: + shutil.rmtree(cleanup_dir)