2016-05-03 14:31:20 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
2022-11-11 13:32:00 +00:00
|
|
|
from unittest import mock
|
2016-05-03 14:31:20 +00:00
|
|
|
import os
|
|
|
|
|
|
|
|
import pungi.phases.test as test_phase
|
2021-11-04 08:01:28 +00:00
|
|
|
from tests.helpers import DummyCompose, PungiTestCase, touch, FIXTURE_DIR
|
2016-05-03 14:31:20 +00:00
|
|
|
|
2019-08-13 12:15:35 +00:00
|
|
|
try:
|
2020-02-06 02:21:54 +00:00
|
|
|
import dnf # noqa: F401
|
2020-01-22 10:02:22 +00:00
|
|
|
|
2019-08-13 12:15:35 +00:00
|
|
|
HAS_DNF = True
|
|
|
|
except ImportError:
|
|
|
|
HAS_DNF = False
|
|
|
|
|
|
|
|
try:
|
2020-02-06 02:21:54 +00:00
|
|
|
import yum # noqa: F401
|
2020-01-22 10:02:22 +00:00
|
|
|
|
2019-08-13 12:15:35 +00:00
|
|
|
HAS_YUM = True
|
|
|
|
except ImportError:
|
|
|
|
HAS_YUM = False
|
|
|
|
|
2016-05-03 14:31:20 +00:00
|
|
|
|
2020-01-22 10:02:22 +00:00
|
|
|
PAD = b"\0" * 100
|
|
|
|
UNBOOTABLE_ISO = (b"\0" * 0x8001) + b"CD001" + PAD
|
|
|
|
ISO_WITH_MBR = (b"\0" * 0x1FE) + b"\x55\xAA" + (b"\0" * 0x7E01) + b"CD001" + PAD
|
|
|
|
ISO_WITH_GPT = (b"\0" * 0x200) + b"EFI PART" + (b"\0" * 0x7DF9) + b"CD001" + PAD
|
|
|
|
ISO_WITH_MBR_AND_GPT = (
|
|
|
|
(b"\0" * 0x1FE) + b"\x55\xAAEFI PART" + (b"\0" * 0x7DF9) + b"CD001" + PAD
|
|
|
|
)
|
|
|
|
ISO_WITH_TORITO = (
|
|
|
|
(b"\0" * 0x8001)
|
|
|
|
+ b"CD001"
|
|
|
|
+ (b"\0" * 0x7FA)
|
|
|
|
+ b"\0CD001\1EL TORITO SPECIFICATION"
|
|
|
|
+ PAD
|
|
|
|
)
|
2016-05-03 14:31:20 +00:00
|
|
|
|
|
|
|
|
|
|
|
class TestCheckImageSanity(PungiTestCase):
|
|
|
|
def test_missing_file_reports_error(self):
|
|
|
|
compose = DummyCompose(self.topdir, {})
|
|
|
|
|
|
|
|
with self.assertRaises(IOError):
|
|
|
|
test_phase.check_image_sanity(compose)
|
|
|
|
|
|
|
|
def test_missing_file_doesnt_report_if_failable(self):
|
2016-06-24 07:44:40 +00:00
|
|
|
compose = DummyCompose(self.topdir, {})
|
2020-01-22 10:02:22 +00:00
|
|
|
compose.image.deliverable = "iso"
|
2016-06-24 07:44:40 +00:00
|
|
|
compose.image.can_fail = True
|
2016-05-03 14:31:20 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
test_phase.check_image_sanity(compose)
|
2017-01-26 08:44:45 +00:00
|
|
|
except Exception:
|
2020-01-22 10:02:22 +00:00
|
|
|
self.fail("Failable deliverable must not raise")
|
2016-05-03 14:31:20 +00:00
|
|
|
|
|
|
|
def test_correct_iso_does_not_raise(self):
|
|
|
|
compose = DummyCompose(self.topdir, {})
|
2020-01-22 10:02:22 +00:00
|
|
|
compose.image.format = "iso"
|
2016-05-03 14:31:20 +00:00
|
|
|
compose.image.bootable = False
|
2020-01-22 10:02:22 +00:00
|
|
|
touch(os.path.join(self.topdir, "compose", compose.image.path), UNBOOTABLE_ISO)
|
2016-05-03 14:31:20 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
test_phase.check_image_sanity(compose)
|
2017-01-26 08:44:45 +00:00
|
|
|
except Exception:
|
2020-01-22 10:02:22 +00:00
|
|
|
self.fail("Correct unbootable image must not raise")
|
2016-05-03 14:31:20 +00:00
|
|
|
|
|
|
|
def test_incorrect_iso_raises(self):
|
|
|
|
compose = DummyCompose(self.topdir, {})
|
2020-01-22 10:02:22 +00:00
|
|
|
compose.image.format = "iso"
|
2016-05-03 14:31:20 +00:00
|
|
|
compose.image.bootable = False
|
2020-01-22 10:02:22 +00:00
|
|
|
touch(os.path.join(self.topdir, "compose", compose.image.path), "Hey there")
|
2016-05-03 14:31:20 +00:00
|
|
|
|
|
|
|
with self.assertRaises(RuntimeError) as ctx:
|
|
|
|
test_phase.check_image_sanity(compose)
|
|
|
|
|
2020-01-22 10:02:22 +00:00
|
|
|
self.assertIn("does not look like an ISO file", str(ctx.exception))
|
2016-05-03 14:31:20 +00:00
|
|
|
|
2016-08-10 11:02:56 +00:00
|
|
|
def test_bootable_iso_without_mbr_or_gpt_raises_on_x86_64(self):
|
2016-05-03 14:31:20 +00:00
|
|
|
compose = DummyCompose(self.topdir, {})
|
2020-01-22 10:02:22 +00:00
|
|
|
compose.image.arch = "x86_64"
|
|
|
|
compose.image.format = "iso"
|
2016-05-03 14:31:20 +00:00
|
|
|
compose.image.bootable = True
|
2020-01-22 10:02:22 +00:00
|
|
|
touch(os.path.join(self.topdir, "compose", compose.image.path), UNBOOTABLE_ISO)
|
2016-05-03 14:31:20 +00:00
|
|
|
|
|
|
|
with self.assertRaises(RuntimeError) as ctx:
|
|
|
|
test_phase.check_image_sanity(compose)
|
|
|
|
|
2020-01-22 10:02:22 +00:00
|
|
|
self.assertIn(
|
|
|
|
"is supposed to be bootable, but does not have MBR nor GPT",
|
|
|
|
str(ctx.exception),
|
|
|
|
)
|
2016-05-03 14:31:20 +00:00
|
|
|
|
2016-08-10 11:02:56 +00:00
|
|
|
def test_bootable_iso_without_mbr_or_gpt_doesnt_raise_on_arm(self):
|
|
|
|
compose = DummyCompose(self.topdir, {})
|
2020-01-22 10:02:22 +00:00
|
|
|
compose.image.arch = "armhfp"
|
|
|
|
compose.image.format = "iso"
|
2016-08-10 11:02:56 +00:00
|
|
|
compose.image.bootable = True
|
2020-01-22 10:02:22 +00:00
|
|
|
touch(os.path.join(self.topdir, "compose", compose.image.path), UNBOOTABLE_ISO)
|
2016-08-10 11:02:56 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
test_phase.check_image_sanity(compose)
|
2017-01-26 08:44:45 +00:00
|
|
|
except Exception:
|
2020-01-22 10:02:22 +00:00
|
|
|
self.fail("Failable deliverable must not raise")
|
2016-08-10 11:02:56 +00:00
|
|
|
|
2016-05-03 14:31:20 +00:00
|
|
|
def test_failable_bootable_iso_without_mbr_gpt_doesnt_raise(self):
|
2016-06-24 07:44:40 +00:00
|
|
|
compose = DummyCompose(self.topdir, {})
|
2020-01-22 10:02:22 +00:00
|
|
|
compose.image.format = "iso"
|
2016-05-03 14:31:20 +00:00
|
|
|
compose.image.bootable = True
|
2020-01-22 10:02:22 +00:00
|
|
|
compose.image.deliverable = "iso"
|
2016-06-24 07:44:40 +00:00
|
|
|
compose.image.can_fail = True
|
2020-01-22 10:02:22 +00:00
|
|
|
touch(os.path.join(self.topdir, "compose", compose.image.path), UNBOOTABLE_ISO)
|
2016-05-03 14:31:20 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
test_phase.check_image_sanity(compose)
|
2017-01-26 08:44:45 +00:00
|
|
|
except Exception:
|
2020-01-22 10:02:22 +00:00
|
|
|
self.fail("Failable deliverable must not raise")
|
2016-05-03 14:31:20 +00:00
|
|
|
|
|
|
|
def test_bootable_iso_with_mbr_does_not_raise(self):
|
|
|
|
compose = DummyCompose(self.topdir, {})
|
2020-01-22 10:02:22 +00:00
|
|
|
compose.image.format = "iso"
|
2016-05-03 14:31:20 +00:00
|
|
|
compose.image.bootable = True
|
2020-01-22 10:02:22 +00:00
|
|
|
touch(os.path.join(self.topdir, "compose", compose.image.path), ISO_WITH_MBR)
|
2016-05-03 14:31:20 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
test_phase.check_image_sanity(compose)
|
2017-01-26 08:44:45 +00:00
|
|
|
except Exception:
|
2020-01-22 10:02:22 +00:00
|
|
|
self.fail("Bootable image with MBR must not raise")
|
2016-05-03 14:31:20 +00:00
|
|
|
|
|
|
|
def test_bootable_iso_with_gpt_does_not_raise(self):
|
|
|
|
compose = DummyCompose(self.topdir, {})
|
2020-01-22 10:02:22 +00:00
|
|
|
compose.image.format = "iso"
|
2016-05-03 14:31:20 +00:00
|
|
|
compose.image.bootable = True
|
2020-01-22 10:02:22 +00:00
|
|
|
touch(os.path.join(self.topdir, "compose", compose.image.path), ISO_WITH_GPT)
|
2016-05-03 14:31:20 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
test_phase.check_image_sanity(compose)
|
2017-01-26 08:44:45 +00:00
|
|
|
except Exception:
|
2020-01-22 10:02:22 +00:00
|
|
|
self.fail("Bootable image with GPT must not raise")
|
2016-05-03 14:31:20 +00:00
|
|
|
|
|
|
|
def test_bootable_iso_with_mbr_and_gpt_does_not_raise(self):
|
|
|
|
compose = DummyCompose(self.topdir, {})
|
2020-01-22 10:02:22 +00:00
|
|
|
compose.image.format = "iso"
|
2016-05-03 14:31:20 +00:00
|
|
|
compose.image.bootable = True
|
2020-01-22 10:02:22 +00:00
|
|
|
touch(
|
|
|
|
os.path.join(self.topdir, "compose", compose.image.path),
|
|
|
|
ISO_WITH_MBR_AND_GPT,
|
|
|
|
)
|
2016-05-03 14:31:20 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
test_phase.check_image_sanity(compose)
|
2017-01-26 08:44:45 +00:00
|
|
|
except Exception:
|
2020-01-22 10:02:22 +00:00
|
|
|
self.fail("Bootable image with MBR and GPT must not raise")
|
2016-05-03 14:31:20 +00:00
|
|
|
|
2016-07-22 11:58:29 +00:00
|
|
|
def test_bootable_iso_with_el_torito_does_not_raise(self):
|
|
|
|
compose = DummyCompose(self.topdir, {})
|
2020-01-22 10:02:22 +00:00
|
|
|
compose.image.format = "iso"
|
2016-07-22 11:58:29 +00:00
|
|
|
compose.image.bootable = True
|
2020-01-22 10:02:22 +00:00
|
|
|
touch(os.path.join(self.topdir, "compose", compose.image.path), ISO_WITH_TORITO)
|
2016-07-22 11:58:29 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
test_phase.check_image_sanity(compose)
|
2017-01-26 08:44:45 +00:00
|
|
|
except Exception:
|
2020-01-22 10:02:22 +00:00
|
|
|
self.fail("Bootable image with El Torito must not raise")
|
2016-07-22 11:58:29 +00:00
|
|
|
|
2016-06-06 08:29:40 +00:00
|
|
|
def test_checks_with_optional_variant(self):
|
|
|
|
compose = DummyCompose(self.topdir, {})
|
2020-01-22 10:02:22 +00:00
|
|
|
compose.variants["Server"].variants = {
|
|
|
|
"optional": mock.Mock(
|
|
|
|
uid="Server-optional",
|
|
|
|
arches=["x86_64"],
|
|
|
|
type="optional",
|
|
|
|
is_empty=False,
|
|
|
|
)
|
2016-06-06 08:29:40 +00:00
|
|
|
}
|
2020-01-22 10:02:22 +00:00
|
|
|
compose.image.format = "iso"
|
2016-06-06 08:29:40 +00:00
|
|
|
compose.image.bootable = True
|
2020-01-22 10:02:22 +00:00
|
|
|
touch(
|
|
|
|
os.path.join(self.topdir, "compose", compose.image.path),
|
|
|
|
ISO_WITH_MBR_AND_GPT,
|
|
|
|
)
|
2016-06-06 08:29:40 +00:00
|
|
|
|
2020-01-22 10:02:22 +00:00
|
|
|
image = mock.Mock(
|
|
|
|
path="Server/i386/optional/iso/image.iso", format="iso", bootable=False
|
|
|
|
)
|
|
|
|
compose.im.images["Server-optional"] = {"i386": [image]}
|
2016-06-06 08:29:40 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
test_phase.check_image_sanity(compose)
|
2017-01-26 08:44:45 +00:00
|
|
|
except Exception:
|
2020-01-22 10:02:22 +00:00
|
|
|
self.fail("Checking optional variant must not raise")
|
2016-06-06 08:29:40 +00:00
|
|
|
|
2019-02-12 11:09:08 +00:00
|
|
|
@mock.patch("pungi.phases.test.check_sanity", new=mock.Mock())
|
|
|
|
def test_too_big_iso(self):
|
|
|
|
compose = DummyCompose(self.topdir, {"createiso_max_size": [(".*", {"*": 10})]})
|
2020-01-22 10:02:22 +00:00
|
|
|
compose.image.format = "iso"
|
2019-02-12 11:09:08 +00:00
|
|
|
compose.image.bootable = False
|
|
|
|
compose.image.size = 20
|
|
|
|
|
|
|
|
test_phase.check_image_sanity(compose)
|
|
|
|
|
|
|
|
warnings = [call[0][0] for call in compose.log_warning.call_args_list]
|
|
|
|
self.assertIn(
|
|
|
|
"ISO Client/i386/iso/image.iso is too big. Expected max 10B, got 20B",
|
|
|
|
warnings,
|
|
|
|
)
|
|
|
|
|
2019-07-15 07:25:26 +00:00
|
|
|
@mock.patch("pungi.phases.test.check_sanity", new=mock.Mock())
|
|
|
|
def test_too_big_iso_strict(self):
|
|
|
|
compose = DummyCompose(
|
|
|
|
self.topdir,
|
|
|
|
{
|
|
|
|
"createiso_max_size": [(".*", {"*": 10})],
|
|
|
|
"createiso_max_size_is_strict": [(".*", {"*": True})],
|
|
|
|
},
|
|
|
|
)
|
2020-01-22 10:02:22 +00:00
|
|
|
compose.image.format = "iso"
|
2019-07-15 07:25:26 +00:00
|
|
|
compose.image.bootable = False
|
|
|
|
compose.image.size = 20
|
|
|
|
|
|
|
|
with self.assertRaises(RuntimeError) as ctx:
|
|
|
|
test_phase.check_image_sanity(compose)
|
|
|
|
|
|
|
|
self.assertEqual(
|
|
|
|
str(ctx.exception),
|
|
|
|
"ISO Client/i386/iso/image.iso is too big. Expected max 10B, got 20B",
|
|
|
|
)
|
|
|
|
|
2019-07-16 06:22:28 +00:00
|
|
|
@mock.patch("pungi.phases.test.check_sanity", new=mock.Mock())
|
|
|
|
def test_too_big_iso_not_strict(self):
|
|
|
|
compose = DummyCompose(
|
|
|
|
self.topdir,
|
|
|
|
{
|
|
|
|
"createiso_max_size": [(".*", {"*": 10})],
|
|
|
|
"createiso_max_size_is_strict": [(".*", {"*": False})],
|
|
|
|
},
|
|
|
|
)
|
2020-01-22 10:02:22 +00:00
|
|
|
compose.image.format = "iso"
|
2019-07-16 06:22:28 +00:00
|
|
|
compose.image.bootable = False
|
|
|
|
compose.image.size = 20
|
|
|
|
|
|
|
|
test_phase.check_image_sanity(compose)
|
|
|
|
|
|
|
|
warnings = [call[0][0] for call in compose.log_warning.call_args_list]
|
|
|
|
self.assertIn(
|
|
|
|
"ISO Client/i386/iso/image.iso is too big. Expected max 10B, got 20B",
|
|
|
|
warnings,
|
|
|
|
)
|
|
|
|
|
2019-02-12 11:09:08 +00:00
|
|
|
@mock.patch("pungi.phases.test.check_sanity", new=mock.Mock())
|
|
|
|
def test_too_big_unified(self):
|
|
|
|
compose = DummyCompose(self.topdir, {})
|
2020-01-22 10:02:22 +00:00
|
|
|
compose.image.format = "iso"
|
2019-02-12 11:09:08 +00:00
|
|
|
compose.image.bootable = False
|
|
|
|
compose.image.size = 20
|
|
|
|
compose.image.unified = True
|
|
|
|
setattr(compose.image, "_max_size", 10)
|
|
|
|
|
|
|
|
test_phase.check_image_sanity(compose)
|
|
|
|
|
|
|
|
warnings = [call[0][0] for call in compose.log_warning.call_args_list]
|
|
|
|
self.assertIn(
|
|
|
|
"ISO Client/i386/iso/image.iso is too big. Expected max 10B, got 20B",
|
|
|
|
warnings,
|
|
|
|
)
|
|
|
|
|
2019-07-15 07:25:26 +00:00
|
|
|
@mock.patch("pungi.phases.test.check_sanity", new=mock.Mock())
|
|
|
|
def test_too_big_unified_strict(self):
|
|
|
|
compose = DummyCompose(
|
2021-03-02 10:19:05 +00:00
|
|
|
self.topdir,
|
|
|
|
{"createiso_max_size_is_strict": [(".*", {"*": True})]},
|
2019-07-15 07:25:26 +00:00
|
|
|
)
|
2020-01-22 10:02:22 +00:00
|
|
|
compose.image.format = "iso"
|
2019-07-15 07:25:26 +00:00
|
|
|
compose.image.bootable = False
|
|
|
|
compose.image.size = 20
|
|
|
|
compose.image.unified = True
|
|
|
|
setattr(compose.image, "_max_size", 10)
|
|
|
|
|
|
|
|
with self.assertRaises(RuntimeError) as ctx:
|
|
|
|
test_phase.check_image_sanity(compose)
|
|
|
|
|
|
|
|
self.assertEqual(
|
|
|
|
str(ctx.exception),
|
|
|
|
"ISO Client/i386/iso/image.iso is too big. Expected max 10B, got 20B",
|
|
|
|
)
|
|
|
|
|
2019-02-12 11:09:08 +00:00
|
|
|
@mock.patch("pungi.phases.test.check_sanity", new=mock.Mock())
|
|
|
|
def test_fits_in_limit(self):
|
|
|
|
compose = DummyCompose(self.topdir, {"createiso_max_size": [(".*", {"*": 20})]})
|
2020-01-22 10:02:22 +00:00
|
|
|
compose.image.format = "iso"
|
2019-02-12 11:09:08 +00:00
|
|
|
compose.image.bootable = False
|
|
|
|
compose.image.size = 5
|
|
|
|
|
|
|
|
test_phase.check_image_sanity(compose)
|
|
|
|
|
|
|
|
self.assertEqual(compose.log_warning.call_args_list, [])
|
|
|
|
|
|
|
|
@mock.patch("pungi.phases.test.check_sanity", new=mock.Mock())
|
|
|
|
def test_non_iso(self):
|
|
|
|
compose = DummyCompose(self.topdir, {"createiso_max_size": [(".*", {"*": 10})]})
|
2020-01-22 10:02:22 +00:00
|
|
|
compose.image.format = "qcow2"
|
2019-02-12 11:09:08 +00:00
|
|
|
compose.image.bootable = False
|
|
|
|
compose.image.size = 20
|
|
|
|
|
|
|
|
test_phase.check_image_sanity(compose)
|
|
|
|
|
|
|
|
self.assertEqual(compose.log_warning.call_args_list, [])
|
2021-11-04 08:01:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
class TestImageMetadataValidation(PungiTestCase):
|
|
|
|
def test_valid_metadata(self):
|
|
|
|
compose = mock.Mock()
|
|
|
|
compose.im.images = {"Server": mock.Mock()}
|
|
|
|
compose.paths.compose.topdir = lambda: os.path.join(
|
|
|
|
FIXTURE_DIR, "basic-metadata"
|
|
|
|
)
|
|
|
|
|
|
|
|
test_phase.check_image_metadata(compose)
|
|
|
|
|
|
|
|
def test_missing_metadata(self):
|
|
|
|
compose = mock.Mock()
|
|
|
|
compose.im.images = {}
|
|
|
|
compose.paths.compose.topdir = lambda: self.topdir
|
|
|
|
|
|
|
|
test_phase.check_image_metadata(compose)
|
|
|
|
|
|
|
|
def test_invalid_metadata(self):
|
|
|
|
compose = mock.Mock()
|
|
|
|
compose.im.images = {"Server": mock.Mock()}
|
|
|
|
compose.paths.compose.topdir = lambda: os.path.join(
|
|
|
|
FIXTURE_DIR, "invalid-image-metadata"
|
|
|
|
)
|
|
|
|
|
|
|
|
with self.assertRaises(RuntimeError):
|
|
|
|
test_phase.check_image_metadata(compose)
|