1 #!/usr/bin/python 2 # pylint: disable=missing-docstring 3 4 import logging 5 import os 6 import shutil 7 import StringIO 8 import sys 9 import tempfile 10 import unittest 11 12 import common 13 from autotest_lib.client.bin import job, sysinfo, harness 14 from autotest_lib.client.bin import utils 15 from autotest_lib.client.common_lib import error 16 from autotest_lib.client.common_lib import logging_manager, logging_config 17 from autotest_lib.client.common_lib import base_job_unittest 18 from autotest_lib.client.common_lib.test_utils import mock 19 20 21 class job_test_case(unittest.TestCase): 22 """Generic job TestCase class that defines a standard job setUp and 23 tearDown, with some standard stubs.""" 24 25 job_class = job.base_client_job 26 27 def setUp(self): 28 self.god = mock.mock_god(ut=self) 29 self.god.stub_with(job.base_client_job, '_get_environ_autodir', 30 classmethod(lambda cls: '/adir')) 31 self.job = self.job_class.__new__(self.job_class) 32 self.job._job_directory = base_job_unittest.stub_job_directory 33 34 _, self.control_file = tempfile.mkstemp() 35 36 37 def tearDown(self): 38 self.god.unstub_all() 39 os.remove(self.control_file) 40 41 42 class test_find_base_directories( 43 base_job_unittest.test_find_base_directories.generic_tests, 44 job_test_case): 45 46 def test_autodir_equals_clientdir(self): 47 autodir, clientdir, _ = self.job._find_base_directories() 48 self.assertEqual(autodir, '/adir') 49 self.assertEqual(clientdir, '/adir') 50 51 52 def test_serverdir_is_none(self): 53 _, _, serverdir = self.job._find_base_directories() 54 self.assertEqual(serverdir, None) 55 56 57 class abstract_test_init(base_job_unittest.test_init.generic_tests): 58 """Generic client job mixin used when defining variations on the 59 job.__init__ generic tests.""" 60 OPTIONAL_ATTRIBUTES = ( 61 base_job_unittest.test_init.generic_tests.OPTIONAL_ATTRIBUTES 62 - set(['control', 'harness'])) 63 64 65 class test_init_minimal_options(abstract_test_init, job_test_case): 66 67 def call_init(self): 68 # TODO(jadmanski): refactor more of the __init__ code to not need to 69 # stub out countless random APIs 70 self.god.stub_function_to_return(job.os, 'mkdir', None) 71 self.god.stub_function_to_return(job.os.path, 'exists', True) 72 self.god.stub_function_to_return(self.job, '_load_state', None) 73 self.god.stub_function_to_return(self.job, 'record', None) 74 self.god.stub_function_to_return(job.shutil, 'copyfile', None) 75 self.god.stub_function_to_return(job.logging_manager, 76 'configure_logging', None) 77 class manager: 78 def start_logging(self): 79 return None 80 self.god.stub_function_to_return(job.logging_manager, 81 'get_logging_manager', manager()) 82 class stub_sysinfo: 83 def log_per_reboot_data(self): 84 return None 85 self.god.stub_function_to_return(job.sysinfo, 'sysinfo', 86 stub_sysinfo()) 87 class stub_harness: 88 run_start = lambda self: None 89 self.god.stub_function_to_return(job.harness, 'select', stub_harness()) 90 class options: 91 tag = '' 92 verbose = False 93 cont = False 94 harness = 'stub' 95 harness_args = None 96 hostname = None 97 user = None 98 log = False 99 args = '' 100 output_dir = '' 101 self.god.stub_function_to_return(job.utils, 'drop_caches', None) 102 103 self.job._job_state = base_job_unittest.stub_job_state 104 self.job.__init__(self.control_file, options) 105 106 107 class dummy(object): 108 """A simple placeholder for attributes""" 109 pass 110 111 112 class first_line_comparator(mock.argument_comparator): 113 def __init__(self, first_line): 114 self.first_line = first_line 115 116 117 def is_satisfied_by(self, parameter): 118 return self.first_line == parameter.splitlines()[0] 119 120 121 class test_base_job(unittest.TestCase): 122 def setUp(self): 123 # make god 124 self.god = mock.mock_god(ut=self) 125 126 # need to set some environ variables 127 self.autodir = "autodir" 128 os.environ['AUTODIR'] = self.autodir 129 130 # set up some variables 131 _, self.control = tempfile.mkstemp() 132 self.jobtag = "jobtag" 133 134 # get rid of stdout and logging 135 sys.stdout = StringIO.StringIO() 136 logging_manager.configure_logging(logging_config.TestingConfig()) 137 logging.disable(logging.CRITICAL) 138 def dummy_configure_logging(*args, **kwargs): 139 pass 140 self.god.stub_with(logging_manager, 'configure_logging', 141 dummy_configure_logging) 142 real_get_logging_manager = logging_manager.get_logging_manager 143 def get_logging_manager_no_fds(manage_stdout_and_stderr=False, 144 redirect_fds=False): 145 return real_get_logging_manager(manage_stdout_and_stderr, False) 146 self.god.stub_with(logging_manager, 'get_logging_manager', 147 get_logging_manager_no_fds) 148 149 # stub out some stuff 150 self.god.stub_function(os.path, 'exists') 151 self.god.stub_function(os.path, 'isdir') 152 self.god.stub_function(os, 'makedirs') 153 self.god.stub_function(os, 'mkdir') 154 self.god.stub_function(os, 'remove') 155 self.god.stub_function(shutil, 'rmtree') 156 self.god.stub_function(shutil, 'copyfile') 157 self.god.stub_function(job, 'open') 158 self.god.stub_function(utils, 'system') 159 self.god.stub_function(utils, 'drop_caches') 160 self.god.stub_function(harness, 'select') 161 self.god.stub_function(sysinfo, 'log_per_reboot_data') 162 163 self.god.stub_class(job.local_host, 'LocalHost') 164 self.god.stub_class(sysinfo, 'sysinfo') 165 166 self.god.stub_class_method(job.base_client_job, 167 '_cleanup_debugdir_files') 168 self.god.stub_class_method(job.base_client_job, '_cleanup_results_dir') 169 170 self.god.stub_with(job.base_job.job_directory, '_ensure_valid', 171 lambda *_: None) 172 173 174 def tearDown(self): 175 sys.stdout = sys.__stdout__ 176 self.god.unstub_all() 177 os.remove(self.control) 178 179 180 def _setup_pre_record_init(self, cont): 181 self.god.stub_function(self.job, '_load_state') 182 183 resultdir = os.path.join(self.autodir, 'results', self.jobtag) 184 tmpdir = os.path.join(self.autodir, 'tmp') 185 if not cont: 186 job.base_client_job._cleanup_debugdir_files.expect_call() 187 job.base_client_job._cleanup_results_dir.expect_call() 188 189 self.job._load_state.expect_call() 190 191 my_harness = self.god.create_mock_class(harness.harness, 192 'my_harness') 193 harness.select.expect_call(None, 194 self.job, 195 None).and_return(my_harness) 196 197 return resultdir, my_harness 198 199 200 def _setup_post_record_init(self, cont, resultdir, my_harness): 201 # now some specific stubs 202 self.god.stub_function(self.job, 'config_get') 203 self.god.stub_function(self.job, 'config_set') 204 self.god.stub_function(self.job, 'record') 205 206 # other setup 207 results = os.path.join(self.autodir, 'results') 208 download = os.path.join(self.autodir, 'tests', 'download') 209 pkgdir = os.path.join(self.autodir, 'packages') 210 211 utils.drop_caches.expect_call() 212 job_sysinfo = sysinfo.sysinfo.expect_new(resultdir) 213 if not cont: 214 os.path.exists.expect_call(download).and_return(False) 215 os.mkdir.expect_call(download) 216 shutil.copyfile.expect_call(mock.is_string_comparator(), 217 os.path.join(resultdir, 'control')) 218 219 job.local_host.LocalHost.expect_new(hostname='localhost') 220 job_sysinfo.log_per_reboot_data.expect_call() 221 if not cont: 222 self.job.record.expect_call('START', None, None) 223 224 my_harness.run_start.expect_call() 225 226 227 def construct_job(self, cont): 228 # will construct class instance using __new__ 229 self.job = job.base_client_job.__new__(job.base_client_job) 230 231 # record 232 resultdir, my_harness = self._setup_pre_record_init(cont) 233 self._setup_post_record_init(cont, resultdir, my_harness) 234 235 # finish constructor 236 options = dummy() 237 options.tag = self.jobtag 238 options.cont = cont 239 options.harness = None 240 options.harness_args = None 241 options.log = False 242 options.verbose = False 243 options.hostname = 'localhost' 244 options.user = 'my_user' 245 options.args = '' 246 options.output_dir = '' 247 self.job.__init__(self.control, options) 248 249 # check 250 self.god.check_playback() 251 252 253 def get_partition_mock(self, devname): 254 """ 255 Create a mock of a partition object and return it. 256 """ 257 class mock(object): 258 device = devname 259 get_mountpoint = self.god.create_mock_function('get_mountpoint') 260 return mock 261 262 263 def test_constructor_first_run(self): 264 self.construct_job(False) 265 266 267 def test_constructor_continuation(self): 268 self.construct_job(True) 269 270 271 def test_constructor_post_record_failure(self): 272 """ 273 Test post record initialization failure. 274 """ 275 self.job = job.base_client_job.__new__(job.base_client_job) 276 options = dummy() 277 options.tag = self.jobtag 278 options.cont = False 279 options.harness = None 280 options.harness_args = None 281 options.log = False 282 options.verbose = False 283 options.hostname = 'localhost' 284 options.user = 'my_user' 285 options.args = '' 286 options.output_dir = '' 287 error = Exception('fail') 288 289 self.god.stub_function(self.job, '_post_record_init') 290 self.god.stub_function(self.job, 'record') 291 292 self._setup_pre_record_init(False) 293 self.job._post_record_init.expect_call( 294 self.control, options, True).and_raises(error) 295 self.job.record.expect_call( 296 'ABORT', None, None,'client.bin.job.__init__ failed: %s' % 297 str(error)) 298 299 self.assertRaises( 300 Exception, self.job.__init__, self.control, options, 301 drop_caches=True) 302 303 # check 304 self.god.check_playback() 305 306 307 def test_control_functions(self): 308 self.construct_job(True) 309 control_file = "blah" 310 self.job.control_set(control_file) 311 self.assertEquals(self.job.control_get(), os.path.abspath(control_file)) 312 313 314 def test_harness_select(self): 315 self.construct_job(True) 316 317 # record 318 which = "which" 319 harness_args = '' 320 harness.select.expect_call(which, self.job, 321 harness_args).and_return(None) 322 323 # run and test 324 self.job.harness_select(which, harness_args) 325 self.god.check_playback() 326 327 328 def test_setup_dirs_raise(self): 329 self.construct_job(True) 330 331 # setup 332 results_dir = 'foo' 333 tmp_dir = 'bar' 334 335 # record 336 os.path.exists.expect_call(tmp_dir).and_return(True) 337 os.path.isdir.expect_call(tmp_dir).and_return(False) 338 339 # test 340 self.assertRaises(ValueError, self.job.setup_dirs, results_dir, tmp_dir) 341 self.god.check_playback() 342 343 344 def test_setup_dirs(self): 345 self.construct_job(True) 346 347 # setup 348 results_dir1 = os.path.join(self.job.resultdir, 'build') 349 results_dir2 = os.path.join(self.job.resultdir, 'build.2') 350 results_dir3 = os.path.join(self.job.resultdir, 'build.3') 351 tmp_dir = 'bar' 352 353 # record 354 os.path.exists.expect_call(tmp_dir).and_return(False) 355 os.mkdir.expect_call(tmp_dir) 356 os.path.isdir.expect_call(tmp_dir).and_return(True) 357 os.path.exists.expect_call(results_dir1).and_return(True) 358 os.path.exists.expect_call(results_dir2).and_return(True) 359 os.path.exists.expect_call(results_dir3).and_return(False) 360 os.path.exists.expect_call(results_dir3).and_return(False) 361 os.mkdir.expect_call(results_dir3) 362 363 # test 364 self.assertEqual(self.job.setup_dirs(None, tmp_dir), 365 (results_dir3, tmp_dir)) 366 self.god.check_playback() 367 368 369 def test_run_test_logs_test_error_from_unhandled_error(self): 370 self.construct_job(True) 371 372 # set up stubs 373 self.god.stub_function(self.job.pkgmgr, 'get_package_name') 374 self.god.stub_function(self.job, "_runtest") 375 376 # create an unhandled error object 377 class MyError(error.TestError): 378 pass 379 real_error = MyError("this is the real error message") 380 unhandled_error = error.UnhandledTestError(real_error) 381 382 # set up the recording 383 testname = "error_test" 384 outputdir = os.path.join(self.job.resultdir, testname) 385 self.job.pkgmgr.get_package_name.expect_call( 386 testname, 'test').and_return(("", testname)) 387 os.path.exists.expect_call(outputdir).and_return(False) 388 self.job.record.expect_call("START", testname, testname, 389 optional_fields=None) 390 self.job._runtest.expect_call(testname, "", None, (), {}).and_raises( 391 unhandled_error) 392 self.job.record.expect_call("ERROR", testname, testname, 393 first_line_comparator(str(real_error))) 394 self.job.record.expect_call("END ERROR", testname, testname) 395 self.job.harness.run_test_complete.expect_call() 396 utils.drop_caches.expect_call() 397 398 # run and check 399 self.job.run_test(testname) 400 self.god.check_playback() 401 402 403 def test_run_test_logs_non_test_error_from_unhandled_error(self): 404 self.construct_job(True) 405 406 # set up stubs 407 self.god.stub_function(self.job.pkgmgr, 'get_package_name') 408 self.god.stub_function(self.job, "_runtest") 409 410 # create an unhandled error object 411 class MyError(Exception): 412 pass 413 real_error = MyError("this is the real error message") 414 unhandled_error = error.UnhandledTestError(real_error) 415 reason = first_line_comparator("Unhandled MyError: %s" % real_error) 416 417 # set up the recording 418 testname = "error_test" 419 outputdir = os.path.join(self.job.resultdir, testname) 420 self.job.pkgmgr.get_package_name.expect_call( 421 testname, 'test').and_return(("", testname)) 422 os.path.exists.expect_call(outputdir).and_return(False) 423 self.job.record.expect_call("START", testname, testname, 424 optional_fields=None) 425 self.job._runtest.expect_call(testname, "", None, (), {}).and_raises( 426 unhandled_error) 427 self.job.record.expect_call("ERROR", testname, testname, reason) 428 self.job.record.expect_call("END ERROR", testname, testname) 429 self.job.harness.run_test_complete.expect_call() 430 utils.drop_caches.expect_call() 431 432 # run and check 433 self.job.run_test(testname) 434 self.god.check_playback() 435 436 437 def test_report_reboot_failure(self): 438 self.construct_job(True) 439 440 # record 441 self.job.record.expect_call("ABORT", "sub", "reboot.verify", 442 "boot failure") 443 self.job.record.expect_call("END ABORT", "sub", "reboot", 444 optional_fields={"kernel": "2.6.15-smp"}) 445 446 # playback 447 self.job._record_reboot_failure("sub", "reboot.verify", "boot failure", 448 running_id="2.6.15-smp") 449 self.god.check_playback() 450 451 452 def _setup_check_post_reboot(self, mount_info, cpu_count): 453 # setup 454 self.god.stub_function(job.partition_lib, "get_partition_list") 455 self.god.stub_function(utils, "count_cpus") 456 457 part_list = [self.get_partition_mock("/dev/hda1"), 458 self.get_partition_mock("/dev/hdb1")] 459 mount_list = ["/mnt/hda1", "/mnt/hdb1"] 460 461 # record 462 job.partition_lib.get_partition_list.expect_call( 463 self.job, exclude_swap=False).and_return(part_list) 464 for i in xrange(len(part_list)): 465 part_list[i].get_mountpoint.expect_call().and_return(mount_list[i]) 466 if cpu_count is not None: 467 utils.count_cpus.expect_call().and_return(cpu_count) 468 self.job._state.set('client', 'mount_info', mount_info) 469 self.job._state.set('client', 'cpu_count', 8) 470 471 472 def test_check_post_reboot_success(self): 473 self.construct_job(True) 474 475 mount_info = set([("/dev/hda1", "/mnt/hda1"), 476 ("/dev/hdb1", "/mnt/hdb1")]) 477 self._setup_check_post_reboot(mount_info, 8) 478 479 # playback 480 self.job._check_post_reboot("sub") 481 self.god.check_playback() 482 483 484 def test_check_post_reboot_mounts_failure(self): 485 self.construct_job(True) 486 487 mount_info = set([("/dev/hda1", "/mnt/hda1")]) 488 self._setup_check_post_reboot(mount_info, None) 489 490 self.god.stub_function(self.job, "_record_reboot_failure") 491 self.job._record_reboot_failure.expect_call("sub", 492 "reboot.verify_config", "mounted partitions are different after" 493 " reboot (old entries: set([]), new entries: set([('/dev/hdb1'," 494 " '/mnt/hdb1')]))", running_id=None) 495 496 # playback 497 self.assertRaises(error.JobError, self.job._check_post_reboot, "sub") 498 self.god.check_playback() 499 500 501 def test_check_post_reboot_cpu_failure(self): 502 self.construct_job(True) 503 504 mount_info = set([("/dev/hda1", "/mnt/hda1"), 505 ("/dev/hdb1", "/mnt/hdb1")]) 506 self._setup_check_post_reboot(mount_info, 4) 507 508 self.god.stub_function(self.job, "_record_reboot_failure") 509 self.job._record_reboot_failure.expect_call( 510 'sub', 'reboot.verify_config', 511 'Number of CPUs changed after reboot (old count: 8, new count: 4)', 512 running_id=None) 513 514 # playback 515 self.assertRaises(error.JobError, self.job._check_post_reboot, "sub") 516 self.god.check_playback() 517 518 519 def test_parse_args(self): 520 test_set = {"a='foo bar baz' b='moo apt'": 521 ["a='foo bar baz'", "b='moo apt'"], 522 "a='foo bar baz' only=gah": 523 ["a='foo bar baz'", "only=gah"], 524 "a='b c d' no=argh": 525 ["a='b c d'", "no=argh"]} 526 for t in test_set: 527 parsed_args = job.base_client_job._parse_args(t) 528 expected_args = test_set[t] 529 self.assertEqual(parsed_args, expected_args) 530 531 532 def test_run_test_timeout_parameter_is_propagated(self): 533 self.construct_job(True) 534 535 # set up stubs 536 self.god.stub_function(self.job.pkgmgr, 'get_package_name') 537 self.god.stub_function(self.job, "_runtest") 538 539 # create an unhandled error object 540 #class MyError(error.TestError): 541 # pass 542 #real_error = MyError("this is the real error message") 543 #unhandled_error = error.UnhandledTestError(real_error) 544 545 # set up the recording 546 testname = "test" 547 outputdir = os.path.join(self.job.resultdir, testname) 548 self.job.pkgmgr.get_package_name.expect_call( 549 testname, 'test').and_return(("", testname)) 550 os.path.exists.expect_call(outputdir).and_return(False) 551 timeout = 60 552 optional_fields = {} 553 optional_fields['timeout'] = timeout 554 self.job.record.expect_call("START", testname, testname, 555 optional_fields=optional_fields) 556 self.job._runtest.expect_call(testname, "", timeout, (), {}) 557 self.job.record.expect_call("GOOD", testname, testname, 558 "completed successfully") 559 self.job.record.expect_call("END GOOD", testname, testname) 560 self.job.harness.run_test_complete.expect_call() 561 utils.drop_caches.expect_call() 562 563 # run and check 564 self.job.run_test(testname, timeout=timeout) 565 self.god.check_playback() 566 567 568 if __name__ == "__main__": 569 unittest.main() 570