Files
mongo/buildscripts/resmokelib/testing/hooks.py
2016-07-27 18:02:44 -04:00

250 lines
8.7 KiB
Python

"""
Customize the behavior of a fixture by allowing special code to be
executed before or after each test, and before or after each suite.
"""
from __future__ import absolute_import
import os
import sys
import bson
import pymongo
from . import fixtures
from . import testcases
from .. import errors
from .. import logging
from .. import utils
def make_custom_behavior(class_name, *args, **kwargs):
"""
Factory function for creating CustomBehavior instances.
"""
if class_name not in _CUSTOM_BEHAVIORS:
raise ValueError("Unknown custom behavior class '%s'" % (class_name))
return _CUSTOM_BEHAVIORS[class_name](*args, **kwargs)
class CustomBehavior(object):
"""
The common interface all CustomBehaviors will inherit from.
"""
@staticmethod
def start_dynamic_test(hook_test_case, test_report):
"""
If a CustomBehavior wants to add a test case that will show up
in the test report, it should use this method to add it to the
report, since we will need to count it as a dynamic test to get
the stats in the summary information right.
"""
test_report.startTest(hook_test_case, dynamic=True)
def __init__(self, logger, fixture, description):
"""
Initializes the CustomBehavior with the specified fixture.
"""
if not isinstance(logger, logging.Logger):
raise TypeError("logger must be a Logger instance")
self.logger = logger
self.fixture = fixture
self.hook_test_case = None
self.logger_name = self.__class__.__name__
self.description = description
def before_suite(self, test_report):
"""
The test runner calls this exactly once before they start
running the suite.
"""
pass
def after_suite(self, test_report):
"""
The test runner calls this exactly once after all tests have
finished executing. Be sure to reset the behavior back to its
original state so that it can be run again.
"""
pass
def before_test(self, test, test_report):
"""
Each test will call this before it executes.
Raises a TestFailure if the test should be marked as a failure,
or a ServerFailure if the fixture exits uncleanly or
unexpectedly.
"""
pass
def after_test(self, test, test_report):
"""
Each test will call this after it executes.
Raises a TestFailure if the test should be marked as a failure,
or a ServerFailure if the fixture exits uncleanly or
unexpectedly.
"""
pass
class CleanEveryN(CustomBehavior):
"""
Restarts the fixture after it has ran 'n' tests.
On mongod-related fixtures, this will clear the dbpath.
"""
DEFAULT_N = 20
def __init__(self, logger, fixture, n=DEFAULT_N):
description = "CleanEveryN (restarts the fixture after running `n` tests)"
CustomBehavior.__init__(self, logger, fixture, description)
# Try to isolate what test triggers the leak by restarting the fixture each time.
if "detect_leaks=1" in os.getenv("ASAN_OPTIONS", ""):
self.logger.info("ASAN_OPTIONS environment variable set to detect leaks, so restarting"
" the fixture after each test instead of after every %d.", n)
n = 1
self.n = n
self.tests_run = 0
def after_test(self, test, test_report):
self.tests_run += 1
if self.tests_run >= self.n:
self.logger.info("%d tests have been run against the fixture, stopping it...",
self.tests_run)
self.tests_run = 0
teardown_success = self.fixture.teardown()
self.logger.info("Starting the fixture back up again...")
self.fixture.setup()
self.fixture.await_ready()
# Raise this after calling setup in case --continueOnFailure was specified.
if not teardown_success:
raise errors.TestFailure("%s did not exit cleanly" % (self.fixture))
class JsCustomBehavior(CustomBehavior):
def __init__(self, logger, fixture, js_filename, description, shell_options=None):
CustomBehavior.__init__(self, logger, fixture, description)
self.hook_test_case = testcases.JSTestCase(logger,
js_filename,
shell_options=shell_options,
test_kind="Hook")
def before_suite(self, test_report):
# Configure the test case after the fixture has been set up.
self.hook_test_case.configure(self.fixture)
def after_test(self, test, test_report):
description = "{0} after running '{1}'".format(self.description, test.short_name())
try:
# Change test_name and description to be more descriptive.
self.hook_test_case.test_name = test.short_name() + ":" + self.logger_name
CustomBehavior.start_dynamic_test(self.hook_test_case, test_report)
self.hook_test_case.run_test()
self.hook_test_case.return_code = 0
test_report.addSuccess(self.hook_test_case)
except self.hook_test_case.failureException as err:
self.hook_test_case.logger.exception("{0} failed".format(description))
test_report.addFailure(self.hook_test_case, sys.exc_info())
raise errors.TestFailure(err.args[0])
finally:
test_report.stopTest(self.hook_test_case)
class ValidateCollections(JsCustomBehavior):
"""
Runs full validation on all collections in all databases on every stand-alone
node, primary replica-set node, or primary shard node.
"""
DEFAULT_N = 1
def __init__(self, logger, fixture, n=DEFAULT_N, improvedV33Validate=False):
description = "Collection validation"
js_filename = os.path.join("jstests", "hooks", "run_validate_collections.js")
JsCustomBehavior.__init__(self, logger, fixture, js_filename, description)
self.n = n
self.tests_run = 0
self.improved_validate = improvedV33Validate
def after_test(self, test, test_report):
self.tests_run += 1
if self.tests_run >= self.n:
self.tests_run = 0
if not self.improved_validate:
JsCustomBehavior.after_test(self, test, test_report)
return
elif not isinstance(self.fixture, fixtures.MongoDFixture):
raise errors.TestFailure(("ValidateCollections with the improved validate command"
" can only be run against a stand-alone mongod"))
# Need to remember the dbpath and port of the original mongod.
mongod_options = self.fixture.mongod_options.copy()
mongod_options["queryableBackupMode"] = ""
mongod_executable = "/data/multiversion/mongod-3.3.10"
if sys.platform == "win32":
mongod_executable += ".exe"
self.fixture.reconfigure(mongod_executable, mongod_options)
JsCustomBehavior.after_test(self, test, test_report)
self.fixture.reset_configuration()
class CheckReplDBHash(JsCustomBehavior):
"""
Checks that the dbhashes of all non-local databases and non-replicated system collections
match on the primary and secondaries.
"""
def __init__(self, logger, fixture):
description = "Check dbhashes of all replica set or master/slave members"
js_filename = os.path.join("jstests", "hooks", "run_check_repl_dbhash.js")
JsCustomBehavior.__init__(self, logger, fixture, js_filename, description)
class TypeSensitiveSON(bson.SON):
"""
Extends bson.SON to perform additional type-checking of document values
to differentiate BSON types.
"""
def items_with_types(self):
"""
Returns a list of triples. Each triple consists of a field name, a
field value, and a field type for each field in the document.
"""
return [(key, self[key], type(self[key])) for key in self]
def __eq__(self, other):
"""
Comparison to another TypeSensitiveSON is order-sensitive and
type-sensitive while comparison to a regular dictionary ignores order
and type mismatches.
"""
if isinstance(other, TypeSensitiveSON):
return (len(self) == len(other) and
self.items_with_types() == other.items_with_types())
raise TypeError("TypeSensitiveSON objects cannot be compared to other types")
_CUSTOM_BEHAVIORS = {
"CleanEveryN": CleanEveryN,
"CheckReplDBHash": CheckReplDBHash,
"ValidateCollections": ValidateCollections,
}