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:
Adam Williamson 2016-05-23 06:55:17 -07:00
parent fc57ccdecc
commit 6ee0e2217d

View file

@ -206,10 +206,12 @@ 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."""
tmpfile = "{0}.tmp".format(self.filename)
try:
# Basic creation command with standard params # Basic creation command with standard params
args = ["virt-builder", "fedora-{0}".format(str(self.release)), "-o", self.filename, args = ["virt-builder", "fedora-{0}".format(str(self.release)), "-o", tmpfile,
"--arch", self.arch] "--arch", self.arch]
if self.size: if self.size:
args.extend(["--size", "{0}b".format(str(int(self.size)))]) args.extend(["--size", "{0}b".format(str(int(self.size)))])
@ -218,14 +220,34 @@ class VirtBuilderImage(object):
# SCRIPTDIR; if this file exists, we pass it to virt-builder. # SCRIPTDIR; if this file exists, we pass it to virt-builder.
if os.path.isfile("{0}/{1}.commands".format(SCRIPTDIR, self.name)): if os.path.isfile("{0}/{1}.commands".format(SCRIPTDIR, self.name)):
args.extend(["--commands-from-file", "{0}/{1}.commands".format(SCRIPTDIR, self.name)]) args.extend(["--commands-from-file", "{0}/{1}.commands".format(SCRIPTDIR, self.name)])
ret = subprocess.call(args) # run the command, timing out after 1 hour; sometimes creation
# seems to just get mysteriously stuck, we need to bail and
# retry in this case
try:
ret = subprocess.call(args, timeout=3600)
except subprocess.TimeoutExpired:
logger.warning("Image creation timed out!")
if os.path.isfile(tmpfile):
os.remove(tmpfile)
if retries:
logger.info("Retrying: %s retries remain after this", str(retries))
return self.create(retries=retries-1)
else:
sys.exit("Image creation timed out too many times!")
if ret > 0: 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))) sys.exit("virt-builder command {0} failed!".format(' '.join(args)))
# We have to boot the disk to make SELinux relabelling happen; # We have to boot the disk to make SELinux relabelling happen;
# virt-builder can't do it unless the policy version on the host # 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 # is the same as the guest(?) There are lots of bad ways to do
# this, expect is the one we're using. # this, expect is the one we're using. This can also get stuck
child = pexpect.spawnu("qemu-kvm -m 2G -nographic {0}".format(self.filename), timeout=None) # 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.expect(u"localhost login:")
child.sendline(u"root") child.sendline(u"root")
child.expect(u"Password:") child.expect(u"Password:")
@ -234,6 +256,18 @@ class VirtBuilderImage(object):
child.sendline(u"poweroff") child.sendline(u"poweroff")
child.expect(u"reboot: Power down") child.expect(u"reboot: Power down")
child.close() 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)