dc7d3b36ab
Fixes: #521 Signed-off-by: Lubomír Sedlář <lsedlar@redhat.com>
135 lines
4.8 KiB
Python
135 lines
4.8 KiB
Python
# -*- coding: utf-8 -*-
|
|
|
|
|
|
# This program is free software; you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation; version 2 of the License.
|
|
#
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU Library General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with this program; if not, see <https://gnu.org/licenses/>.
|
|
|
|
|
|
import os
|
|
|
|
from kobo.shortcuts import run
|
|
|
|
from pungi.wrappers import repoclosure
|
|
from pungi.arch import get_valid_arches
|
|
from pungi.phases.base import PhaseBase
|
|
from pungi.phases.gather import get_lookaside_repos
|
|
from pungi.util import rmtree, is_arch_multilib, failable
|
|
|
|
|
|
class TestPhase(PhaseBase):
|
|
name = "test"
|
|
|
|
def run(self):
|
|
run_repoclosure(self.compose)
|
|
check_image_sanity(self.compose)
|
|
|
|
|
|
def run_repoclosure(compose):
|
|
msg = "Running repoclosure"
|
|
compose.log_info("[BEGIN] %s" % msg)
|
|
|
|
# Variant repos
|
|
for arch in compose.get_arches():
|
|
is_multilib = is_arch_multilib(compose.conf, arch)
|
|
arches = get_valid_arches(arch, is_multilib)
|
|
for variant in compose.get_variants(arch=arch):
|
|
if variant.is_empty:
|
|
continue
|
|
lookaside = {}
|
|
if variant.parent:
|
|
repo_id = "repoclosure-%s.%s" % (variant.parent.uid, arch)
|
|
repo_dir = compose.paths.compose.repository(arch=arch, variant=variant.parent)
|
|
lookaside[repo_id] = repo_dir
|
|
|
|
repos = {}
|
|
repo_id = "repoclosure-%s.%s" % (variant.uid, arch)
|
|
repo_dir = compose.paths.compose.repository(arch=arch, variant=variant)
|
|
repos[repo_id] = repo_dir
|
|
|
|
if compose.conf["release_is_layered"]:
|
|
for i, lookaside_url in enumerate(get_lookaside_repos(compose, arch, variant)):
|
|
lookaside["lookaside-%s.%s-%s" % (variant.uid, arch, i)] = lookaside_url
|
|
|
|
cmd = repoclosure.get_repoclosure_cmd(backend=compose.conf['repoclosure_backend'],
|
|
repos=repos, lookaside=lookaside, arch=arches)
|
|
# Use temp working directory directory as workaround for
|
|
# https://bugzilla.redhat.com/show_bug.cgi?id=795137
|
|
tmp_dir = compose.mkdtemp(prefix="repoclosure_")
|
|
try:
|
|
run(cmd, logfile=compose.paths.log.log_file(arch, "repoclosure-%s" % variant),
|
|
show_cmd=True, can_fail=True, workdir=tmp_dir)
|
|
finally:
|
|
rmtree(tmp_dir)
|
|
|
|
compose.log_info("[DONE ] %s" % msg)
|
|
|
|
|
|
def check_image_sanity(compose):
|
|
"""
|
|
Go through all images in manifest and make basic sanity tests on them. If
|
|
any check fails for a failable deliverable, a message will be printed and
|
|
logged. Otherwise the compose will be aborted.
|
|
"""
|
|
im = compose.im
|
|
for variant in compose.get_variants():
|
|
if variant.uid not in im.images:
|
|
continue
|
|
for arch in variant.arches:
|
|
if arch not in im.images[variant.uid]:
|
|
continue
|
|
for img in im.images[variant.uid][arch]:
|
|
check(compose, variant, arch, img)
|
|
|
|
|
|
def check(compose, variant, arch, image):
|
|
path = os.path.join(compose.paths.compose.topdir(), image.path)
|
|
deliverable = getattr(image, 'deliverable')
|
|
can_fail = getattr(image, 'can_fail', False)
|
|
with failable(compose, can_fail, variant, arch, deliverable,
|
|
subvariant=image.subvariant):
|
|
with open(path) as f:
|
|
iso = is_iso(f)
|
|
if image.format == 'iso' and not iso:
|
|
raise RuntimeError('%s does not look like an ISO file' % path)
|
|
if (image.arch in ('x86_64', 'i386') and
|
|
image.bootable and
|
|
not has_mbr(f) and
|
|
not has_gpt(f) and
|
|
not (iso and has_eltorito(f))):
|
|
raise RuntimeError(
|
|
'%s is supposed to be bootable, but does not have MBR nor '
|
|
'GPT nor is it a bootable ISO' % path)
|
|
# If exception is raised above, failable may catch it, in which case
|
|
# nothing else will happen.
|
|
|
|
|
|
def _check_magic(f, offset, bytes):
|
|
"""Check that the file has correct magic number at correct offset."""
|
|
f.seek(offset)
|
|
return f.read(len(bytes)) == bytes
|
|
|
|
|
|
def is_iso(f):
|
|
return _check_magic(f, 0x8001, 'CD001')
|
|
|
|
|
|
def has_mbr(f):
|
|
return _check_magic(f, 0x1fe, '\x55\xAA')
|
|
|
|
|
|
def has_gpt(f):
|
|
return _check_magic(f, 0x200, 'EFI PART')
|
|
|
|
|
|
def has_eltorito(f):
|
|
return _check_magic(f, 0x8801, 'CD001\1EL TORITO SPECIFICATION')
|