2015-05-30 11:20:59 +00:00
|
|
|
#!/usr/bin/python2
|
2015-03-23 11:36:12 +00:00
|
|
|
#
|
|
|
|
# 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 <http://www.gnu.org/licenses/>.
|
|
|
|
#
|
|
|
|
# Author: Chris Lumens <clumens@redhat.com>
|
|
|
|
|
|
|
|
__all__ = ["Creator", "OutsideMixin"]
|
|
|
|
|
2015-05-30 11:20:59 +00:00
|
|
|
from blivet.size import MiB
|
2015-03-23 11:36:12 +00:00
|
|
|
|
|
|
|
from contextlib import contextmanager
|
2015-05-30 11:20:59 +00:00
|
|
|
from nose.plugins.multiprocess import TimedOutException
|
2015-03-23 11:36:12 +00:00
|
|
|
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
|
2015-05-30 11:20:59 +00:00
|
|
|
tuple is the name of the drive and its size as a blivet.Size.
|
2015-03-23 11:36:12 +00:00
|
|
|
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
|
|
|
|
|
2015-05-30 11:20:59 +00:00
|
|
|
self._reqMemory = 1536
|
|
|
|
|
|
|
|
def _call(self, args):
|
|
|
|
subprocess.call(args, stdout=open("/dev/null", "w"), stderr=open("/dev/null", "w"))
|
|
|
|
|
2015-03-23 11:36:12 +00:00
|
|
|
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)
|
|
|
|
|
2015-05-30 11:20:59 +00:00
|
|
|
# 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)])
|
2015-03-23 11:36:12 +00:00
|
|
|
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
|
|
|
|
|
2015-05-30 11:20:59 +00:00
|
|
|
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])
|
2015-03-23 11:36:12 +00:00
|
|
|
|
2015-05-30 11:20:59 +00:00
|
|
|
# Create the directory structure needed for storing results.
|
|
|
|
os.makedirs(self.mountpoint + "/result/anaconda")
|
2015-03-23 11:36:12 +00:00
|
|
|
|
2015-05-30 11:20:59 +00:00
|
|
|
# Copy all the inside stuff into the mountpoint.
|
|
|
|
shutil.copytree("inside", self.mountpoint + "/inside")
|
2015-03-23 11:36:12 +00:00
|
|
|
|
2015-05-30 11:20:59 +00:00
|
|
|
# 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)
|
2015-03-23 11:36:12 +00:00
|
|
|
|
2015-05-30 11:20:59 +00:00
|
|
|
f.write(self.template % {"environ": " os.environ.update(%s)" % self.environ,
|
|
|
|
"imports": "\n".join(imports),
|
|
|
|
"addtests": "\n".join(addtests),
|
|
|
|
"anacondaArgs": config.get("anacondaArgs", "").strip('"')})
|
2015-03-23 11:36:12 +00:00
|
|
|
|
2015-05-30 11:20:59 +00:00
|
|
|
self._call(["/usr/bin/umount", self.mountpoint])
|
2015-03-23 11:36:12 +00:00
|
|
|
|
|
|
|
# This ensures it gets passed to qemu-kvm as a disk arg.
|
2015-05-30 11:20:59 +00:00
|
|
|
self._drivePaths[self.suitename] = self.suitepath
|
2015-03-23 11:36:12 +00:00
|
|
|
|
|
|
|
@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.
|
|
|
|
"""
|
2015-05-30 11:20:59 +00:00
|
|
|
if self._drivePaths.get(self.suitename, "") == "":
|
2015-03-23 11:36:12 +00:00
|
|
|
return
|
|
|
|
|
2015-05-30 11:20:59 +00:00
|
|
|
self._call(["/usr/bin/mount", "-o", "loop", self.suitepath, self.mountpoint])
|
2015-03-23 11:36:12 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
yield
|
|
|
|
except:
|
|
|
|
raise
|
|
|
|
finally:
|
2015-05-30 11:20:59 +00:00
|
|
|
self._call(["/usr/bin/umount", self.mountpoint])
|
2015-03-23 11:36:12 +00:00
|
|
|
|
|
|
|
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",
|
2015-05-30 11:20:59 +00:00
|
|
|
"-vnc", "none",
|
|
|
|
"-m", str(self._reqMemory),
|
2015-03-23 11:36:12 +00:00
|
|
|
"-boot", "d",
|
2015-05-30 11:20:59 +00:00
|
|
|
"-drive", "file=%s,media=cdrom,readonly" % config["liveImage"]]
|
2015-03-23 11:36:12 +00:00
|
|
|
|
|
|
|
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)
|
2015-05-30 11:20:59 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
self._proc.wait()
|
|
|
|
except TimedOutException:
|
|
|
|
self.die()
|
|
|
|
self.cleanup()
|
|
|
|
raise
|
|
|
|
finally:
|
|
|
|
self._proc = None
|
2015-03-23 11:36:12 +00:00
|
|
|
|
|
|
|
@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
|
|
|
|
|
2015-05-30 11:20:59 +00:00
|
|
|
@property
|
|
|
|
def suitename(self):
|
|
|
|
return self.name + "_suite"
|
|
|
|
|
|
|
|
@property
|
|
|
|
def suitepath(self):
|
|
|
|
return self.tempdir + "/" + self.suitename
|
|
|
|
|
2015-03-23 11:36:12 +00:00
|
|
|
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()
|