diff options
| -rw-r--r-- | meta/classes/testimage.bbclass | 5 | ||||
| -rw-r--r-- | meta/lib/oeqa/oetest.py | 52 | ||||
| -rw-r--r-- | meta/lib/oeqa/targetcontrol.py | 6 | ||||
| -rw-r--r-- | meta/lib/oeqa/utils/dump.py | 77 |
4 files changed, 91 insertions, 49 deletions
diff --git a/meta/classes/testimage.bbclass b/meta/classes/testimage.bbclass index 1d9464f5e9..824b47f5dd 100644 --- a/meta/classes/testimage.bbclass +++ b/meta/classes/testimage.bbclass | |||
| @@ -231,6 +231,7 @@ def testimage_main(d): | |||
| 231 | import time | 231 | import time |
| 232 | from oeqa.oetest import loadTests, runTests | 232 | from oeqa.oetest import loadTests, runTests |
| 233 | from oeqa.targetcontrol import get_target_controller | 233 | from oeqa.targetcontrol import get_target_controller |
| 234 | from oeqa.utils.dump import get_host_dumper | ||
| 234 | 235 | ||
| 235 | pn = d.getVar("PN", True) | 236 | pn = d.getVar("PN", True) |
| 236 | export = oe.utils.conditional("TEST_EXPORT_ONLY", "1", True, False, d) | 237 | export = oe.utils.conditional("TEST_EXPORT_ONLY", "1", True, False, d) |
| @@ -245,6 +246,9 @@ def testimage_main(d): | |||
| 245 | testslist = get_tests_list(d) | 246 | testslist = get_tests_list(d) |
| 246 | testsrequired = [t for t in d.getVar("TEST_SUITES", True).split() if t != "auto"] | 247 | testsrequired = [t for t in d.getVar("TEST_SUITES", True).split() if t != "auto"] |
| 247 | 248 | ||
| 249 | # we need the host dumper in test context | ||
| 250 | host_dumper = get_host_dumper(d) | ||
| 251 | |||
| 248 | # the robot dance | 252 | # the robot dance |
| 249 | target = get_target_controller(d) | 253 | target = get_target_controller(d) |
| 250 | 254 | ||
| @@ -255,6 +259,7 @@ def testimage_main(d): | |||
| 255 | self.testsrequired = testsrequired | 259 | self.testsrequired = testsrequired |
| 256 | self.filesdir = os.path.join(os.path.dirname(os.path.abspath(oeqa.runtime.__file__)),"files") | 260 | self.filesdir = os.path.join(os.path.dirname(os.path.abspath(oeqa.runtime.__file__)),"files") |
| 257 | self.target = target | 261 | self.target = target |
| 262 | self.host_dumper = host_dumper | ||
| 258 | self.imagefeatures = d.getVar("IMAGE_FEATURES", True).split() | 263 | self.imagefeatures = d.getVar("IMAGE_FEATURES", True).split() |
| 259 | self.distrofeatures = d.getVar("DISTRO_FEATURES", True).split() | 264 | self.distrofeatures = d.getVar("DISTRO_FEATURES", True).split() |
| 260 | manifest = os.path.join(d.getVar("DEPLOY_DIR_IMAGE", True), d.getVar("IMAGE_LINK_NAME", True) + ".manifest") | 265 | manifest = os.path.join(d.getVar("DEPLOY_DIR_IMAGE", True), d.getVar("IMAGE_LINK_NAME", True) + ".manifest") |
diff --git a/meta/lib/oeqa/oetest.py b/meta/lib/oeqa/oetest.py index fbf6c56376..9cb8a53795 100644 --- a/meta/lib/oeqa/oetest.py +++ b/meta/lib/oeqa/oetest.py | |||
| @@ -11,8 +11,6 @@ import os, re, mmap | |||
| 11 | import unittest | 11 | import unittest |
| 12 | import inspect | 12 | import inspect |
| 13 | import subprocess | 13 | import subprocess |
| 14 | import datetime | ||
| 15 | import commands | ||
| 16 | import bb | 14 | import bb |
| 17 | from oeqa.utils.decorators import LogResults | 15 | from oeqa.utils.decorators import LogResults |
| 18 | from sys import exc_info, exc_clear | 16 | from sys import exc_info, exc_clear |
| @@ -124,51 +122,13 @@ class oeRuntimeTest(oeTest): | |||
| 124 | # If a test fails or there is an exception | 122 | # If a test fails or there is an exception |
| 125 | if not exc_info() == (None, None, None): | 123 | if not exc_info() == (None, None, None): |
| 126 | exc_clear() | 124 | exc_clear() |
| 127 | dump_dir = self.create_dump_dir() | 125 | self.tc.host_dumper.create_dir(self._testMethodName) |
| 126 | self.target.target_dumper.dump_target( | ||
| 127 | self.tc.host_dumper.dump_dir) | ||
| 128 | self.tc.host_dumper.dump_host() | ||
| 128 | print ("%s dump data from host and target " | 129 | print ("%s dump data from host and target " |
| 129 | "stored in %s" % (self._testMethodName, dump_dir)) | 130 | "stored in %s" % (self._testMethodName, |
| 130 | self.dump_host_logs(dump_dir) | 131 | self.target.target_dumper.dump_dir)) |
| 131 | self.dump_target_logs(dump_dir) | ||
| 132 | |||
| 133 | def create_dump_dir(self): | ||
| 134 | dump_sub_dir = ("%s_%s" % ( | ||
| 135 | datetime.datetime.now().strftime('%Y%m%d%H%M'), | ||
| 136 | self._testMethodName)) | ||
| 137 | dump_dir = os.path.join(self.target.dump_dir, dump_sub_dir) | ||
| 138 | os.makedirs(dump_dir) | ||
| 139 | return dump_dir | ||
| 140 | |||
| 141 | def dump_host_logs(self, dump_dir): | ||
| 142 | for cmd in self.target.dump_host.split('\n'): | ||
| 143 | cmd = cmd.lstrip() | ||
| 144 | if not cmd: | ||
| 145 | continue | ||
| 146 | output = commands.getoutput(cmd) | ||
| 147 | filename = "host_%s" % cmd.split()[0] | ||
| 148 | with open(os.path.join(dump_dir, filename), 'w') as f: | ||
| 149 | f.write(output) | ||
| 150 | |||
| 151 | def dump_target_logs(self, dump_dir): | ||
| 152 | for cmd in self.target.dump_target.split('\n'): | ||
| 153 | cmd = cmd.lstrip() | ||
| 154 | if not cmd: | ||
| 155 | continue | ||
| 156 | # This will ping the host from target | ||
| 157 | if cmd == "_ping": | ||
| 158 | comm = "ping -c3 %s" % self.target.server_ip | ||
| 159 | # This will get all the logs from /var/log/ | ||
| 160 | elif cmd == "_logs": | ||
| 161 | comm = 'find /var/log/ -type f 2>/dev/null ' | ||
| 162 | comm = '%s-exec echo "%s" \\; ' % (comm, '='*20) | ||
| 163 | comm = '%s-exec echo {} \\; ' % comm | ||
| 164 | comm = '%s-exec echo "%s" \\; ' % (comm, '='*20) | ||
| 165 | comm = '%s-exec cat {} \\; -exec echo "" \\;' % comm | ||
| 166 | else: | ||
| 167 | comm = cmd | ||
| 168 | (status, output) = self.target.run_serial(comm) | ||
| 169 | filename = "target_%s" % cmd.split()[0] | ||
| 170 | with open(os.path.join(dump_dir, filename), 'w') as f: | ||
| 171 | f.write(output) | ||
| 172 | 132 | ||
| 173 | #TODO: use package_manager.py to install packages on any type of image | 133 | #TODO: use package_manager.py to install packages on any type of image |
| 174 | def install_packages(self, packagelist): | 134 | def install_packages(self, packagelist): |
diff --git a/meta/lib/oeqa/targetcontrol.py b/meta/lib/oeqa/targetcontrol.py index 59cae2eff1..2d58f17ddb 100644 --- a/meta/lib/oeqa/targetcontrol.py +++ b/meta/lib/oeqa/targetcontrol.py | |||
| @@ -14,6 +14,7 @@ import logging | |||
| 14 | from oeqa.utils.sshcontrol import SSHControl | 14 | from oeqa.utils.sshcontrol import SSHControl |
| 15 | from oeqa.utils.qemurunner import QemuRunner | 15 | from oeqa.utils.qemurunner import QemuRunner |
| 16 | from oeqa.utils.qemutinyrunner import QemuTinyRunner | 16 | from oeqa.utils.qemutinyrunner import QemuTinyRunner |
| 17 | from oeqa.utils.dump import TargetDumper | ||
| 17 | from oeqa.controllers.testtargetloader import TestTargetLoader | 18 | from oeqa.controllers.testtargetloader import TestTargetLoader |
| 18 | from abc import ABCMeta, abstractmethod | 19 | from abc import ABCMeta, abstractmethod |
| 19 | 20 | ||
| @@ -123,9 +124,6 @@ class QemuTarget(BaseTarget): | |||
| 123 | self.origrootfs = os.path.join(d.getVar("DEPLOY_DIR_IMAGE", True), d.getVar("IMAGE_LINK_NAME", True) + '.' + self.image_fstype) | 124 | self.origrootfs = os.path.join(d.getVar("DEPLOY_DIR_IMAGE", True), d.getVar("IMAGE_LINK_NAME", True) + '.' + self.image_fstype) |
| 124 | self.rootfs = os.path.join(self.testdir, d.getVar("IMAGE_LINK_NAME", True) + '-testimage.' + self.image_fstype) | 125 | self.rootfs = os.path.join(self.testdir, d.getVar("IMAGE_LINK_NAME", True) + '-testimage.' + self.image_fstype) |
| 125 | self.kernel = os.path.join(d.getVar("DEPLOY_DIR_IMAGE", True), d.getVar("KERNEL_IMAGETYPE", False) + '-' + d.getVar('MACHINE', False) + '.bin') | 126 | self.kernel = os.path.join(d.getVar("DEPLOY_DIR_IMAGE", True), d.getVar("KERNEL_IMAGETYPE", False) + '-' + d.getVar('MACHINE', False) + '.bin') |
| 126 | self.dump_target = d.getVar("testimage_dump_target", True) | ||
| 127 | self.dump_host = d.getVar("testimage_dump_host", True) | ||
| 128 | self.dump_dir = d.getVar("TESTIMAGE_DUMP_DIR", True) | ||
| 129 | 127 | ||
| 130 | # Log QemuRunner log output to a file | 128 | # Log QemuRunner log output to a file |
| 131 | import oe.path | 129 | import oe.path |
| @@ -155,6 +153,8 @@ class QemuTarget(BaseTarget): | |||
| 155 | logfile = self.qemulog, | 153 | logfile = self.qemulog, |
| 156 | boottime = int(d.getVar("TEST_QEMUBOOT_TIMEOUT", True))) | 154 | boottime = int(d.getVar("TEST_QEMUBOOT_TIMEOUT", True))) |
| 157 | 155 | ||
| 156 | self.target_dumper = TargetDumper(d, self.runner) | ||
| 157 | |||
| 158 | def deploy(self): | 158 | def deploy(self): |
| 159 | try: | 159 | try: |
| 160 | bb.utils.mkdirhier(self.testdir) | 160 | bb.utils.mkdirhier(self.testdir) |
diff --git a/meta/lib/oeqa/utils/dump.py b/meta/lib/oeqa/utils/dump.py new file mode 100644 index 0000000000..a0fa699a27 --- /dev/null +++ b/meta/lib/oeqa/utils/dump.py | |||
| @@ -0,0 +1,77 @@ | |||
| 1 | import os | ||
| 2 | import sys | ||
| 3 | import errno | ||
| 4 | import datetime | ||
| 5 | import itertools | ||
| 6 | from commands import runCmd | ||
| 7 | |||
| 8 | def get_host_dumper(d): | ||
| 9 | return HostDumper(d) | ||
| 10 | |||
| 11 | |||
| 12 | class BaseDumper(object): | ||
| 13 | |||
| 14 | def __init__(self, d): | ||
| 15 | self.parent_dir = d.getVar("TESTIMAGE_DUMP_DIR", True) | ||
| 16 | |||
| 17 | def create_dir(self, dir_suffix): | ||
| 18 | dump_subdir = ("%s_%s" % ( | ||
| 19 | datetime.datetime.now().strftime('%Y%m%d%H%M'), | ||
| 20 | dir_suffix)) | ||
| 21 | dump_dir = os.path.join(self.parent_dir, dump_subdir) | ||
| 22 | try: | ||
| 23 | os.makedirs(dump_dir) | ||
| 24 | except OSError as err: | ||
| 25 | if err.errno != errno.EEXIST: | ||
| 26 | raise err | ||
| 27 | self.dump_dir = dump_dir | ||
| 28 | |||
| 29 | def write_dump(self, command, output): | ||
| 30 | if isinstance(self, HostDumper): | ||
| 31 | prefix = "host" | ||
| 32 | elif isinstance(self, TargetDumper): | ||
| 33 | prefix = "target" | ||
| 34 | else: | ||
| 35 | prefix = "unknown" | ||
| 36 | for i in itertools.count(): | ||
| 37 | filename = "%s_%02d_%s" % (prefix, i, command) | ||
| 38 | fullname = os.path.join(self.dump_dir, filename) | ||
| 39 | if not os.path.exists(fullname): | ||
| 40 | break | ||
| 41 | with open(fullname, 'w') as dump_file: | ||
| 42 | dump_file.write(output) | ||
| 43 | |||
| 44 | |||
| 45 | class HostDumper(BaseDumper): | ||
| 46 | |||
| 47 | def __init__(self, d): | ||
| 48 | super(HostDumper, self).__init__(d) | ||
| 49 | self.host_cmds = d.getVar("testimage_dump_host", True) | ||
| 50 | |||
| 51 | def dump_host(self, dump_dir=""): | ||
| 52 | if dump_dir: | ||
| 53 | self.dump_dir = dump_dir | ||
| 54 | for cmd in self.host_cmds.split('\n'): | ||
| 55 | cmd = cmd.lstrip() | ||
| 56 | if not cmd or cmd[0] == '#': | ||
| 57 | continue | ||
| 58 | result = runCmd(cmd, ignore_status=True) | ||
| 59 | self.write_dump(cmd.split()[0], result.output) | ||
| 60 | |||
| 61 | |||
| 62 | class TargetDumper(BaseDumper): | ||
| 63 | |||
| 64 | def __init__(self, d, qemurunner): | ||
| 65 | super(TargetDumper, self).__init__(d) | ||
| 66 | self.target_cmds = d.getVar("testimage_dump_target", True) | ||
| 67 | self.runner = qemurunner | ||
| 68 | |||
| 69 | def dump_target(self, dump_dir=""): | ||
| 70 | if dump_dir: | ||
| 71 | self.dump_dir = dump_dir | ||
| 72 | for cmd in self.target_cmds.split('\n'): | ||
| 73 | cmd = cmd.lstrip() | ||
| 74 | if not cmd or cmd[0] == '#': | ||
| 75 | continue | ||
| 76 | (status, output) = self.runner.run_serial(cmd) | ||
| 77 | self.write_dump(cmd.split()[0], output) | ||
