mirror of
https://github.com/rocky-linux/createhdds.git
synced 2025-01-05 08:00:55 +00:00
createhdds: use a temp file for virt-builder image creation
Summary: Creating the virt-builder images directly with their final file name means that if we're rebuilding them for age and the build fails, we'll lose the existing image: it seems better to keep it than have no image at all, when this happens. It also means that while the rebuild is in process, the file might exist but be useless and cause any tests that happen to be running at the time to fail. So just like the guestfs images, create the file with a .tmp extension initially and rename it after a successful build. Test Plan: Do some image builds and check they work, check that temp file is cleaned up on failure and ctrl-c if you can... Reviewers: jskladan, garretraziel Reviewed By: garretraziel Subscribers: tflink Differential Revision: https://phab.qadevel.cloud.fedoraproject.org/D858
This commit is contained in:
parent
fc57ccdecc
commit
6ee0e2217d
1 changed files with 62 additions and 28 deletions
|
@ -206,34 +206,68 @@ class VirtBuilderImage(object):
|
||||||
self.arch = arch
|
self.arch = arch
|
||||||
self.maxage = maxage
|
self.maxage = maxage
|
||||||
|
|
||||||
def create(self):
|
def create(self, retries=3):
|
||||||
"""Create the image."""
|
"""Create the image."""
|
||||||
# Basic creation command with standard params
|
tmpfile = "{0}.tmp".format(self.filename)
|
||||||
args = ["virt-builder", "fedora-{0}".format(str(self.release)), "-o", self.filename,
|
try:
|
||||||
"--arch", self.arch]
|
# Basic creation command with standard params
|
||||||
if self.size:
|
args = ["virt-builder", "fedora-{0}".format(str(self.release)), "-o", tmpfile,
|
||||||
args.extend(["--size", "{0}b".format(str(int(self.size)))])
|
"--arch", self.arch]
|
||||||
# We use guestfs's ability to read customization commands from
|
if self.size:
|
||||||
# a file. The convention is to have a file 'name.commands' in
|
args.extend(["--size", "{0}b".format(str(int(self.size)))])
|
||||||
# SCRIPTDIR; if this file exists, we pass it to virt-builder.
|
# We use guestfs's ability to read customization commands from
|
||||||
if os.path.isfile("{0}/{1}.commands".format(SCRIPTDIR, self.name)):
|
# a file. The convention is to have a file 'name.commands' in
|
||||||
args.extend(["--commands-from-file", "{0}/{1}.commands".format(SCRIPTDIR, self.name)])
|
# SCRIPTDIR; if this file exists, we pass it to virt-builder.
|
||||||
ret = subprocess.call(args)
|
if os.path.isfile("{0}/{1}.commands".format(SCRIPTDIR, self.name)):
|
||||||
if ret > 0:
|
args.extend(["--commands-from-file", "{0}/{1}.commands".format(SCRIPTDIR, self.name)])
|
||||||
sys.exit("virt-builder command {0} failed!".format(' '.join(args)))
|
# run the command, timing out after 1 hour; sometimes creation
|
||||||
# We have to boot the disk to make SELinux relabelling happen;
|
# seems to just get mysteriously stuck, we need to bail and
|
||||||
# virt-builder can't do it unless the policy version on the host
|
# retry in this case
|
||||||
# is the same as the guest(?) There are lots of bad ways to do
|
try:
|
||||||
# this, expect is the one we're using.
|
ret = subprocess.call(args, timeout=3600)
|
||||||
child = pexpect.spawnu("qemu-kvm -m 2G -nographic {0}".format(self.filename), timeout=None)
|
except subprocess.TimeoutExpired:
|
||||||
child.expect(u"localhost login:")
|
logger.warning("Image creation timed out!")
|
||||||
child.sendline(u"root")
|
if os.path.isfile(tmpfile):
|
||||||
child.expect(u"Password:")
|
os.remove(tmpfile)
|
||||||
child.sendline(u"weakpassword")
|
if retries:
|
||||||
child.expect(u"~]#")
|
logger.info("Retrying: %s retries remain after this", str(retries))
|
||||||
child.sendline(u"poweroff")
|
return self.create(retries=retries-1)
|
||||||
child.expect(u"reboot: Power down")
|
else:
|
||||||
child.close()
|
sys.exit("Image creation timed out too many times!")
|
||||||
|
if ret > 0:
|
||||||
|
# wipe the temp file
|
||||||
|
if os.path.isfile(tmpfile):
|
||||||
|
os.remove(tmpfile)
|
||||||
|
sys.exit("virt-builder command {0} failed!".format(' '.join(args)))
|
||||||
|
# We have to boot the disk to make SELinux relabelling happen;
|
||||||
|
# virt-builder can't do it unless the policy version on the host
|
||||||
|
# is the same as the guest(?) There are lots of bad ways to do
|
||||||
|
# this, expect is the one we're using. This can also get stuck
|
||||||
|
# apparently, so let's set a 300 sec timeout for each expect
|
||||||
|
# and bail if it's hit.
|
||||||
|
try:
|
||||||
|
logger.info("Booting image to trigger SELinux relabel...")
|
||||||
|
child = pexpect.spawnu("qemu-kvm -m 2G -nographic {0}".format(tmpfile), timeout=300)
|
||||||
|
child.expect(u"localhost login:")
|
||||||
|
child.sendline(u"root")
|
||||||
|
child.expect(u"Password:")
|
||||||
|
child.sendline(u"weakpassword")
|
||||||
|
child.expect(u"~]#")
|
||||||
|
child.sendline(u"poweroff")
|
||||||
|
child.expect(u"reboot: Power down")
|
||||||
|
child.close()
|
||||||
|
except pexpect.TIMEOUT:
|
||||||
|
if os.path.isfile(tmpfile):
|
||||||
|
os.remove(tmpfile)
|
||||||
|
sys.exit("Timed out booting image!")
|
||||||
|
# we're all done! rename to the correct name
|
||||||
|
os.rename(tmpfile, self.filename)
|
||||||
|
except:
|
||||||
|
# if anything went wrong, we want to wipe the temp file
|
||||||
|
# then raise
|
||||||
|
if os.path.isfile(tmpfile):
|
||||||
|
os.remove(tmpfile)
|
||||||
|
raise
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def outdated(self):
|
def outdated(self):
|
||||||
|
@ -653,7 +687,7 @@ def main():
|
||||||
args.func(args, hdds)
|
args.func(args, hdds)
|
||||||
except KeyboardInterrupt:
|
except KeyboardInterrupt:
|
||||||
sys.stderr.write("Interrupted, exiting...\n")
|
sys.stderr.write("Interrupted, exiting...\n")
|
||||||
# there may be a guestfs temp image file lying around...
|
# there may be temp image files lying around...
|
||||||
tmps = [fl for fl in os.listdir('.') if fl.startswith('disk') and fl.endswith('.tmp')]
|
tmps = [fl for fl in os.listdir('.') if fl.startswith('disk') and fl.endswith('.tmp')]
|
||||||
for tmp in tmps:
|
for tmp in tmps:
|
||||||
os.remove(tmp)
|
os.remove(tmp)
|
||||||
|
|
Loading…
Reference in a new issue