#!/usr/bin/python2 # # Copyright (C) 2014 Red Hat, Inc. # # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU Lesser General Public License as published # by the Free Software Foundation; either version 2.1 of the License, or # (at your option) any later version. # # 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 Lesser General Public License for more details. # # You should have received a copy of the GNU Lesser General Public License # along with this program. If not, see . # # Author: Chris Lumens __all__ = ["Creator", "OutsideMixin"] from blivet.size import MiB from contextlib import contextmanager from nose.plugins.multiprocess import TimedOutException import os import shutil import subprocess import tempfile import errno # Copied from python's subprocess.py def eintr_retry_call(func, *args): """Retry an interruptible system call if interrupted.""" while True: try: return func(*args) except (OSError, IOError) as e: if e.errno == errno.EINTR: continue raise class Creator(object): """A Creator subclass defines all the parameters for making a VM to run a test against as well as handles creating and running that VM, inspecting results, and managing temporary data. Most Creator subclasses will only need to define the following four attributes: drives -- A list of tuples describing disk images to create. Each tuple is the name of the drive and its size as a blivet.Size. environ -- A dictionary of environment variables that should be added to the environment the test suite will run under. name -- A unique string that names a Creator. This name will be used in creating the results directory (and perhaps other places in the future) so make sure it doesn't conflict with another object. tests -- A list of tuples describing which test cases make up this test. Each tuple is the name of the module containing the test case (minus the leading "inside." and the name of the test case class. Tests will be run in the order provided. """ drives = [] environ = {} name = "Creator" tests = [] def __init__(self): self._drivePaths = {} self._mountpoint = None self._proc = None self._tempdir = None self._reqMemory = 1536 def _call(self, args): subprocess.call(args, stdout=open("/dev/null", "w"), stderr=open("/dev/null", "w")) def archive(self): """Copy all log files and other test results to a subdirectory of the given resultsdir. If logs are no longer available, this method does nothing. It is up to the caller to make sure logs are available beforehand and clean up afterwards. """ from testconfig import config if not os.path.ismount(self.mountpoint): return shutil.copytree(self.mountpoint + "/result", config["resultsdir"] + "/" + self.name) def cleanup(self): """Remove all disk images used during this test case and the temporary directory they were stored in. """ shutil.rmtree(self.tempdir, ignore_errors=True) def die(self): """Kill any running qemu process previously started by this test.""" if self._proc: self._proc.kill() self._proc = None def makeDrives(self): """Create all hard drive images associated with this test. Images must be listed in Creator.drives and will be stored in a temporary directory this method creates. It is up to the caller to remove everything later by calling Creator.cleanup. """ for (drive, size) in self.drives: (fd, diskimage) = tempfile.mkstemp(dir=self.tempdir) eintr_retry_call(os.close, fd) # For now we are using qemu-img to create these files but specifying # sizes in blivet Size objects. Unfortunately, qemu-img wants sizes # as xM or xG, not xMB or xGB. That's what the conversion here is for. self._call(["/usr/bin/qemu-img", "create", "-f", "raw", diskimage, "%sM" % size.convertTo(MiB)]) self._drivePaths[drive] = diskimage @property def template(self): with open("outside/template.py", "r") as f: return f.read() def makeSuite(self): """The suite is a small disk image attached to every test VM automatically by the test framework. It includes all the inside/ stuff, a special suite.py file that will be automatically run by the live CD (and is what actually runs the test), and a directory structure for reporting results. It is mounted under Creator.mountpoint as needed. This method creates the suite image and adds it to the internal list of images associated with this test. Note that because this image is attached to the VM, anaconda will always see two hard drives and thus will never automatically select disks. Note also that this means tests must be careful to not select this disk. """ from testconfig import config self._call(["/usr/bin/qemu-img", "create", "-f", "raw", self.suitepath, "10M"]) self._call(["/sbin/mkfs.ext4", "-F", self.suitepath, "-L", "ANACTEST"]) self._call(["/usr/bin/mount", "-o", "loop", self.suitepath, self.mountpoint]) # Create the directory structure needed for storing results. os.makedirs(self.mountpoint + "/result/anaconda") # Copy all the inside stuff into the mountpoint. shutil.copytree("inside", self.mountpoint + "/inside") # Create the suite file, which contains all the test cases to run and is how # the VM will figure out what to run. with open(self.mountpoint + "/suite.py", "w") as f: imports = map(lambda (path, cls): " from inside.%s import %s" % (path, cls), self.tests) addtests = map(lambda (path, cls): " s.addTest(%s())" % cls, self.tests) f.write(self.template % {"environ": " os.environ.update(%s)" % self.environ, "imports": "\n".join(imports), "addtests": "\n".join(addtests), "anacondaArgs": config.get("anacondaArgs", "").strip('"')}) self._call(["/usr/bin/umount", self.mountpoint]) # This ensures it gets passed to qemu-kvm as a disk arg. self._drivePaths[self.suitename] = self.suitepath @contextmanager def suiteMounted(self): """This context manager allows for wrapping code that needs to access the suite. It mounts the disk image beforehand and unmounts it afterwards. """ if self._drivePaths.get(self.suitename, "") == "": return self._call(["/usr/bin/mount", "-o", "loop", self.suitepath, self.mountpoint]) try: yield except: raise finally: self._call(["/usr/bin/umount", self.mountpoint]) def run(self): """Given disk images previously created by Creator.makeDrives and Creator.makeSuite, start qemu and wait for it to terminate. """ from testconfig import config args = ["/usr/bin/qemu-kvm", "-vnc", "none", "-m", str(self._reqMemory), "-boot", "d", "-drive", "file=%s,media=cdrom,readonly" % config["liveImage"]] for drive in self._drivePaths.values(): args += ["-drive", "file=%s,media=disk" % drive] # Save a reference to the running qemu process so we can later kill # it if necessary. For now, the only reason we'd want to kill it is # an expired timer. self._proc = subprocess.Popen(args) try: self._proc.wait() except TimedOutException: self.die() self.cleanup() raise finally: self._proc = None @property def mountpoint(self): """The directory where the suite is mounted. This is a subdirectory of Creator.tempdir, and it is assumed the mountpoint directory (though not the mount itself) exists throughout this test. """ if not self._mountpoint: self._mountpoint = tempfile.mkdtemp(dir=self.tempdir) return self._mountpoint @property def tempdir(self): """The temporary directory used to store disk images and other data this test requires. This directory will be removed by Creator.cleanup. It is up to the caller to call that method, though. """ if not self._tempdir: self._tempdir = tempfile.mkdtemp(prefix="%s-" % self.name, dir="/var/tmp") return self._tempdir @property def suitename(self): return self.name + "_suite" @property def suitepath(self): return self.tempdir + "/" + self.suitename class OutsideMixin(object): """A BaseOutsideTestCase subclass is the interface between the unittest framework and a running VM. It interfaces with an associated Creator object to create devices and fire up a VM, and also handles actually reporting a result that unittest knows how to process. Each subclass will likely only want to define a single attribute: creatorClass -- A Creator subclass that goes with this test. """ creatorClass = None def archive(self): self.creator.archive() def runTest(self): self.creator.run() with self.creator.suiteMounted(): self.assertTrue(os.path.exists(self.creator.mountpoint + "/result"), msg="results directory does not exist") self.archive() self.assertFalse(os.path.exists(self.creator.mountpoint + "/result/unittest-failures"), msg="automated UI test %s failed" % self.creator.name) def setUp(self): # pylint: disable=not-callable self.creator = self.creatorClass() self.creator.makeDrives() self.creator.makeSuite() def tearDown(self): self.creator.cleanup()