1"""Base support for parser scenario testing. 2""" 3 4from os import path 5import ConfigParser, os, shelve, shutil, sys, tarfile, time 6import difflib, itertools 7import common 8from autotest_lib.client.common_lib import utils, autotemp 9from autotest_lib.tko import status_lib 10from autotest_lib.tko.parsers.test import templates 11from autotest_lib.tko.parsers.test import unittest_hotfix 12 13TEMPLATES_DIRPATH = templates.__path__[0] 14# Set TZ used to UTC 15os.environ['TZ'] = 'UTC' 16time.tzset() 17 18KEYVAL = 'keyval' 19STATUS_VERSION = 'status_version' 20PARSER_RESULT_STORE = 'parser_result.store' 21RESULTS_DIR_TARBALL = 'results_dir.tgz' 22CONFIG_FILENAME = 'scenario.cfg' 23TEST = 'test' 24PARSER_RESULT_TAG = 'parser_result_tag' 25 26 27class Error(Exception): 28 pass 29 30 31class BadResultsDirectoryError(Error): 32 pass 33 34 35class UnsupportedParserResultError(Error): 36 pass 37 38 39class UnsupportedTemplateTypeError(Error): 40 pass 41 42 43 44class ParserException(object): 45 """Abstract representation of exception raised from parser execution. 46 47 We will want to persist exceptions raised from the parser but also change 48 the objects that make them up during refactor. For this reason 49 we can't merely pickle the original. 50 """ 51 52 def __init__(self, orig): 53 """ 54 Args: 55 orig: Exception; To copy 56 """ 57 self.classname = orig.__class__.__name__ 58 print "Copying exception:", self.classname 59 for key, val in orig.__dict__.iteritems(): 60 setattr(self, key, val) 61 62 63 def __eq__(self, other): 64 """Test if equal to another ParserException.""" 65 return self.__dict__ == other.__dict__ 66 67 68 def __ne__(self, other): 69 """Test if not equal to another ParserException.""" 70 return self.__dict__ != other.__dict__ 71 72 73 def __str__(self): 74 sd = self.__dict__ 75 pairs = ['%s="%s"' % (k, sd[k]) for k in sorted(sd.keys())] 76 return "<%s: %s>" % (self.classname, ', '.join(pairs)) 77 78 79class ParserTestResult(object): 80 """Abstract representation of test result parser state. 81 82 We will want to persist test results but also change the 83 objects that make them up during refactor. For this reason 84 we can't merely pickle the originals. 85 """ 86 87 def __init__(self, orig): 88 """ 89 Tracking all the attributes as they change over time is 90 not desirable. Instead we populate the instance's __dict__ 91 by introspecting orig. 92 93 Args: 94 orig: testobj; Framework test result instance to copy. 95 """ 96 for key, val in orig.__dict__.iteritems(): 97 if key == 'kernel': 98 setattr(self, key, dict(val.__dict__)) 99 elif key == 'iterations': 100 setattr(self, key, [dict(it.__dict__) for it in val]) 101 else: 102 setattr(self, key, val) 103 104 105 def __eq__(self, other): 106 """Test if equal to another ParserTestResult.""" 107 return self.__dict__ == other.__dict__ 108 109 110 def __ne__(self, other): 111 """Test if not equal to another ParserTestResult.""" 112 return self.__dict__ != other.__dict__ 113 114 115 def __str__(self): 116 sd = self.__dict__ 117 pairs = ['%s="%s"' % (k, sd[k]) for k in sorted(sd.keys())] 118 return "<%s: %s>" % (self.__class__.__name__, ', '.join(pairs)) 119 120 121def copy_parser_result(parser_result): 122 """Copy parser_result into ParserTestResult instances. 123 124 Args: 125 parser_result: 126 list; [testobj, ...] 127 - Or - 128 Exception 129 130 Returns: 131 list; [ParserTestResult, ...] 132 - Or - 133 ParserException 134 135 Raises: 136 UnsupportedParserResultError; If parser_result type is not supported 137 """ 138 if type(parser_result) is list: 139 return [ParserTestResult(test) for test in parser_result] 140 elif isinstance(parser_result, Exception): 141 return ParserException(parser_result) 142 else: 143 raise UnsupportedParserResultError 144 145 146def compare_parser_results(left, right): 147 """Generates a textual report (for now) on the differences between. 148 149 Args: 150 left: list of ParserTestResults or a single ParserException 151 right: list of ParserTestResults or a single ParserException 152 153 Returns: Generator returned from difflib.Differ().compare() 154 """ 155 def to_los(obj): 156 """Generate a list of strings representation of object.""" 157 if type(obj) is list: 158 return [ 159 '%d) %s' % pair 160 for pair in itertools.izip(itertools.count(), obj)] 161 else: 162 return ['i) %s' % obj] 163 164 return difflib.Differ().compare(to_los(left), to_los(right)) 165 166 167class ParserHarness(object): 168 """Harness for objects related to the parser. 169 170 This can exercise a parser on specific result data in various ways. 171 """ 172 173 def __init__( 174 self, parser, job, job_keyval, status_version, status_log_filepath): 175 """ 176 Args: 177 parser: tko.parsers.base.parser; Subclass instance of base parser. 178 job: job implementation; Returned from parser.make_job() 179 job_keyval: dict; Result of parsing job keyval file. 180 status_version: str; Status log format version 181 status_log_filepath: str; Path to result data status.log file 182 """ 183 self.parser = parser 184 self.job = job 185 self.job_keyval = job_keyval 186 self.status_version = status_version 187 self.status_log_filepath = status_log_filepath 188 189 190 def execute(self): 191 """Basic exercise, pass entire log data into .end() 192 193 Returns: list; [testobj, ...] 194 """ 195 status_lines = open(self.status_log_filepath).readlines() 196 self.parser.start(self.job) 197 return self.parser.end(status_lines) 198 199 200class BaseScenarioTestCase(unittest_hotfix.TestCase): 201 """Base class for all Scenario TestCase implementations. 202 203 This will load up all resources from scenario package directory upon 204 instantiation, and initialize a new ParserHarness before each test 205 method execution. 206 """ 207 def __init__(self, methodName='runTest'): 208 unittest_hotfix.TestCase.__init__(self, methodName) 209 self.package_dirpath = path.dirname( 210 sys.modules[self.__module__].__file__) 211 self.tmp_dirpath, self.results_dirpath = load_results_dir( 212 self.package_dirpath) 213 self.parser_result_store = load_parser_result_store( 214 self.package_dirpath) 215 self.config = load_config(self.package_dirpath) 216 self.parser_result_tag = self.config.get( 217 TEST, PARSER_RESULT_TAG) 218 self.expected_status_version = self.config.getint( 219 TEST, STATUS_VERSION) 220 self.harness = None 221 222 223 def setUp(self): 224 if self.results_dirpath: 225 self.harness = new_parser_harness(self.results_dirpath) 226 227 228 def tearDown(self): 229 if self.tmp_dirpath: 230 self.tmp_dirpath.clean() 231 232 233 def test_status_version(self): 234 """Ensure basic sanity.""" 235 self.skipIf(not self.harness) 236 self.assertEquals( 237 self.harness.status_version, self.expected_status_version) 238 239 240def shelve_open(filename, flag='c', protocol=None, writeback=False): 241 """A more system-portable wrapper around shelve.open, with the exact 242 same arguments and interpretation.""" 243 import dumbdbm 244 return shelve.Shelf(dumbdbm.open(filename, flag), protocol, writeback) 245 246 247def new_parser_harness(results_dirpath): 248 """Ensure sane environment and create new parser with wrapper. 249 250 Args: 251 results_dirpath: str; Path to job results directory 252 253 Returns: 254 ParserHarness; 255 256 Raises: 257 BadResultsDirectoryError; If results dir does not exist or is malformed. 258 """ 259 if not path.exists(results_dirpath): 260 raise BadResultsDirectoryError 261 262 keyval_path = path.join(results_dirpath, KEYVAL) 263 job_keyval = utils.read_keyval(keyval_path) 264 status_version = job_keyval[STATUS_VERSION] 265 parser = status_lib.parser(status_version) 266 job = parser.make_job(results_dirpath) 267 status_log_filepath = path.join(results_dirpath, 'status.log') 268 if not path.exists(status_log_filepath): 269 raise BadResultsDirectoryError 270 271 return ParserHarness( 272 parser, job, job_keyval, status_version, status_log_filepath) 273 274 275def store_parser_result(package_dirpath, parser_result, tag): 276 """Persist parser result to specified scenario package, keyed by tag. 277 278 Args: 279 package_dirpath: str; Path to scenario package directory. 280 parser_result: list or Exception; Result from ParserHarness.execute 281 tag: str; Tag to use as shelve key for persisted parser_result 282 """ 283 copy = copy_parser_result(parser_result) 284 sto_filepath = path.join(package_dirpath, PARSER_RESULT_STORE) 285 sto = shelve_open(sto_filepath) 286 sto[tag] = copy 287 sto.close() 288 289 290def load_parser_result_store(package_dirpath, open_for_write=False): 291 """Load parser result store from specified scenario package. 292 293 Args: 294 package_dirpath: str; Path to scenario package directory. 295 open_for_write: bool; Open store for writing. 296 297 Returns: 298 shelve.DbfilenameShelf; Looks and acts like a dict 299 """ 300 open_flag = open_for_write and 'c' or 'r' 301 sto_filepath = path.join(package_dirpath, PARSER_RESULT_STORE) 302 return shelve_open(sto_filepath, flag=open_flag) 303 304 305def store_results_dir(package_dirpath, results_dirpath): 306 """Make tarball of results_dirpath in package_dirpath. 307 308 Args: 309 package_dirpath: str; Path to scenario package directory. 310 results_dirpath: str; Path to job results directory 311 """ 312 tgz_filepath = path.join(package_dirpath, RESULTS_DIR_TARBALL) 313 tgz = tarfile.open(tgz_filepath, 'w:gz') 314 results_dirname = path.basename(results_dirpath) 315 tgz.add(results_dirpath, results_dirname) 316 tgz.close() 317 318 319def load_results_dir(package_dirpath): 320 """Unpack results tarball in package_dirpath to temp dir. 321 322 Args: 323 package_dirpath: str; Path to scenario package directory. 324 325 Returns: 326 str; New temp path for extracted results directory. 327 - Or - 328 None; If tarball does not exist 329 """ 330 tgz_filepath = path.join(package_dirpath, RESULTS_DIR_TARBALL) 331 if not path.exists(tgz_filepath): 332 return None, None 333 334 tgz = tarfile.open(tgz_filepath, 'r:gz') 335 tmp_dirpath = autotemp.tempdir(unique_id='scenario_base') 336 results_dirname = tgz.next().name 337 tgz.extract(results_dirname, tmp_dirpath.name) 338 for info in tgz: 339 tgz.extract(info.name, tmp_dirpath.name) 340 return tmp_dirpath, path.join(tmp_dirpath.name, results_dirname) 341 342 343def write_config(package_dirpath, **properties): 344 """Write test configuration file to package_dirpath. 345 346 Args: 347 package_dirpath: str; Path to scenario package directory. 348 properties: dict; Key value entries to write to to config file. 349 """ 350 config = ConfigParser.RawConfigParser() 351 config.add_section(TEST) 352 for key, val in properties.iteritems(): 353 config.set(TEST, key, val) 354 355 config_filepath = path.join(package_dirpath, CONFIG_FILENAME) 356 fi = open(config_filepath, 'w') 357 config.write(fi) 358 fi.close() 359 360 361def load_config(package_dirpath): 362 """Load config from package_dirpath. 363 364 Args: 365 package_dirpath: str; Path to scenario package directory. 366 367 Returns: 368 ConfigParser.RawConfigParser; 369 """ 370 config = ConfigParser.RawConfigParser() 371 config_filepath = path.join(package_dirpath, CONFIG_FILENAME) 372 config.read(config_filepath) 373 return config 374 375 376def install_unittest_module(package_dirpath, template_type): 377 """Install specified unittest template module to package_dirpath. 378 379 Template modules are stored in tko/parsers/test/templates. 380 Installation includes: 381 Copying to package_dirpath/template_type_unittest.py 382 Copying scenario package common.py to package_dirpath 383 Touching package_dirpath/__init__.py 384 385 Args: 386 package_dirpath: str; Path to scenario package directory. 387 template_type: str; Name of template module to install. 388 389 Raises: 390 UnsupportedTemplateTypeError; If there is no module in 391 templates package called template_type. 392 """ 393 from_filepath = path.join( 394 TEMPLATES_DIRPATH, '%s.py' % template_type) 395 if not path.exists(from_filepath): 396 raise UnsupportedTemplateTypeError 397 398 to_filepath = path.join( 399 package_dirpath, '%s_unittest.py' % template_type) 400 shutil.copy(from_filepath, to_filepath) 401 402 # For convenience we must copy the common.py hack file too :-( 403 from_common_filepath = path.join( 404 TEMPLATES_DIRPATH, 'scenario_package_common.py') 405 to_common_filepath = path.join(package_dirpath, 'common.py') 406 shutil.copy(from_common_filepath, to_common_filepath) 407 408 # And last but not least, touch an __init__ file 409 os.mknod(path.join(package_dirpath, '__init__.py')) 410 411 412def fix_package_dirname(package_dirname): 413 """Convert package_dirname to a valid package name string, if necessary. 414 415 Args: 416 package_dirname: str; Name of scenario package directory. 417 418 Returns: 419 str; Possibly fixed package_dirname 420 """ 421 # Really stupid atm, just enough to handle results dirnames 422 package_dirname = package_dirname.replace('-', '_') 423 pre = '' 424 if package_dirname[0].isdigit(): 425 pre = 'p' 426 return pre + package_dirname 427 428 429def sanitize_results_data(results_dirpath): 430 """Replace or remove any data that would possibly contain IP 431 432 Args: 433 results_dirpath: str; Path to job results directory 434 """ 435 raise NotImplementedError 436