| # Copyright Martin J. Bligh, Andy Whitcroft, 2007 |
| # |
| # Define the server-side test class |
| # |
| |
| import os, tempfile |
| |
| from autotest_lib.client.common_lib import log, utils, test as common_test |
| |
| |
| class test(common_test.base_test): |
| pass |
| |
| |
| _sysinfo_before_test_script = """\ |
| import pickle |
| from autotest_lib.client.bin import test |
| mytest = test.test(job, '', %r) |
| job.sysinfo.log_before_each_test(mytest) |
| sysinfo_pickle = os.path.join(mytest.outputdir, 'sysinfo.pickle') |
| pickle.dump(job.sysinfo, open(sysinfo_pickle, 'w')) |
| job.record('GOOD', '', 'sysinfo.before') |
| """ |
| |
| _sysinfo_after_test_script = """\ |
| import pickle |
| from autotest_lib.client.bin import test |
| mytest = test.test(job, '', %r) |
| sysinfo_pickle = os.path.join(mytest.outputdir, 'sysinfo.pickle') |
| if os.path.exists(sysinfo_pickle): |
| job.sysinfo = pickle.load(open(sysinfo_pickle)) |
| job.sysinfo.__init__(job.resultdir) |
| job.sysinfo.log_after_each_test(mytest) |
| job.record('GOOD', '', 'sysinfo.after') |
| """ |
| |
| # this script is ran after _sysinfo_before_test_script and before |
| # _sysinfo_after_test_script which means the pickle file exists |
| # already and should be dumped with updated state for |
| # _sysinfo_after_test_script to pick it up later |
| _sysinfo_iteration_script = """\ |
| import pickle |
| from autotest_lib.client.bin import test |
| mytest = test.test(job, '', %r) |
| sysinfo_pickle = os.path.join(mytest.outputdir, 'sysinfo.pickle') |
| if os.path.exists(sysinfo_pickle): |
| job.sysinfo = pickle.load(open(sysinfo_pickle)) |
| job.sysinfo.__init__(job.resultdir) |
| job.sysinfo.%s(mytest, iteration=%d) |
| pickle.dump(job.sysinfo, open(sysinfo_pickle, 'w')) |
| job.record('GOOD', '', 'sysinfo.iteration.%s') |
| """ |
| |
| |
| def install_autotest_and_run(func): |
| def wrapper(self, mytest): |
| host, at = self._install() |
| try: |
| outputdir = host.get_tmp_dir() |
| try: |
| func(self, mytest, host, at, outputdir) |
| finally: |
| host.delete_tmp_dir(outputdir) |
| finally: |
| at.uninstall() |
| return wrapper |
| |
| |
| class _sysinfo_logger(object): |
| def __init__(self, job): |
| self.job = job |
| self.pickle = None |
| if len(job.machines) != 1: |
| # disable logging on multi-machine tests |
| self.before_hook = self.after_hook = None |
| |
| |
| def _install(self): |
| from autotest_lib.server import hosts, autotest |
| host = hosts.create_host(self.job.machines[0], auto_monitor=False) |
| tmp_dir = host.get_tmp_dir(parent="/tmp/sysinfo") |
| at = autotest.Autotest(host) |
| at.install_base(autodir=tmp_dir) |
| return host, at |
| |
| |
| def _pull_pickle(self, host, outputdir): |
| """Pulls from the client the pickle file with the saved sysinfo state. |
| """ |
| fd, path = tempfile.mkstemp(dir=self.job.tmpdir) |
| os.close(fd) |
| host.get_file(os.path.join(outputdir, "sysinfo.pickle"), path) |
| self.pickle = path |
| |
| |
| def _push_pickle(self, host, outputdir): |
| """Pushes the server saved sysinfo pickle file to the client. |
| """ |
| if self.pickle: |
| host.send_file(self.pickle, |
| os.path.join(outputdir, "sysinfo.pickle")) |
| os.remove(self.pickle) |
| self.pickle = None |
| |
| |
| def _pull_sysinfo_keyval(self, host, outputdir, mytest): |
| """Pulls sysinfo and keyval data from the client. |
| """ |
| # pull the sysinfo data back on to the server |
| host.get_file(os.path.join(outputdir, "sysinfo"), mytest.outputdir) |
| |
| # pull the keyval data back into the local one |
| fd, path = tempfile.mkstemp(dir=self.job.tmpdir) |
| os.close(fd) |
| host.get_file(os.path.join(outputdir, "keyval"), path) |
| keyval = utils.read_keyval(path) |
| os.remove(path) |
| mytest.write_test_keyval(keyval) |
| |
| |
| @log.log_and_ignore_errors("pre-test server sysinfo error:") |
| @install_autotest_and_run |
| def before_hook(self, mytest, host, at, outputdir): |
| # run the pre-test sysinfo script |
| at.run(_sysinfo_before_test_script % outputdir, |
| results_dir=self.job.resultdir) |
| |
| self._pull_pickle(host, outputdir) |
| |
| |
| @log.log_and_ignore_errors("pre-test iteration server sysinfo error:") |
| @install_autotest_and_run |
| def before_iteration_hook(self, mytest, host, at, outputdir): |
| # this function is called after before_hook() se we have sysinfo state |
| # to push to the server |
| self._push_pickle(host, outputdir); |
| # run the pre-test iteration sysinfo script |
| at.run(_sysinfo_iteration_script % |
| (outputdir, 'log_before_each_iteration', mytest.iteration, |
| 'before'), |
| results_dir=self.job.resultdir) |
| |
| # get the new sysinfo state from the client |
| self._pull_pickle(host, outputdir) |
| |
| |
| @log.log_and_ignore_errors("post-test iteration server sysinfo error:") |
| @install_autotest_and_run |
| def after_iteration_hook(self, mytest, host, at, outputdir): |
| # push latest sysinfo state to the client |
| self._push_pickle(host, outputdir); |
| # run the post-test iteration sysinfo script |
| at.run(_sysinfo_iteration_script % |
| (outputdir, 'log_after_each_iteration', mytest.iteration, |
| 'after'), |
| results_dir=self.job.resultdir) |
| |
| # get the new sysinfo state from the client |
| self._pull_pickle(host, outputdir) |
| self._pull_sysinfo_keyval(host, outputdir, mytest) |
| |
| |
| @log.log_and_ignore_errors("post-test server sysinfo error:") |
| @install_autotest_and_run |
| def after_hook(self, mytest, host, at, outputdir): |
| self._push_pickle(host, outputdir); |
| # run the post-test sysinfo script |
| at.run(_sysinfo_after_test_script % outputdir, |
| results_dir=self.job.resultdir) |
| |
| self._pull_sysinfo_keyval(host, outputdir, mytest) |
| |
| |
| def runtest(job, url, tag, args, dargs): |
| if not dargs.pop('disable_sysinfo', False): |
| logger = _sysinfo_logger(job) |
| logging_args = [logger.before_hook, logger.after_hook, |
| logger.before_iteration_hook, |
| logger.after_iteration_hook] |
| else: |
| logging_args = [None, None] |
| common_test.runtest(job, url, tag, args, dargs, locals(), globals(), |
| *logging_args) |