diff --git a/trove/tests/scenario/groups/replication_group.py b/trove/tests/scenario/groups/replication_group.py index 015d83fb0a..de8f28c4bd 100644 --- a/trove/tests/scenario/groups/replication_group.py +++ b/trove/tests/scenario/groups/replication_group.py @@ -21,6 +21,12 @@ from trove.tests.scenario.runners import test_runners GROUP = "scenario.replication_group" +GROUP_REPL_CREATE = "scenario.repl_create_group" +GROUP_REPL_CREATE_WAIT = "scenario.repl_create_wait_group" +GROUP_REPL_MULTI_CREATE = "scenario.repl_multi_create_group" +GROUP_REPL_MULTI_CREATE_WAIT = "scenario.repl_multi_create_wait_group" +GROUP_REPL_DELETE = "scenario.repl_delete_group" +GROUP_REPL_DELETE_WAIT = "scenario.repl_delete_wait_group" class ReplicationRunnerFactory(test_runners.RunnerFactory): @@ -29,12 +35,13 @@ class ReplicationRunnerFactory(test_runners.RunnerFactory): _runner_cls = 'ReplicationRunner' -@test(depends_on_groups=[instance_create_group.GROUP], groups=[GROUP]) -class ReplicationGroup(TestGroup): - """Test Replication functionality.""" +@test(depends_on_groups=[instance_create_group.GROUP], + groups=[GROUP, GROUP_REPL_CREATE]) +class ReplicationCreateGroup(TestGroup): + """Test Replication Create functionality.""" def __init__(self): - super(ReplicationGroup, self).__init__( + super(ReplicationCreateGroup, self).__init__( ReplicationRunnerFactory.instance()) @test @@ -57,32 +64,27 @@ class ReplicationGroup(TestGroup): """Test creating a single replica.""" self.test_runner.run_create_single_replica() - @test(runs_after=[create_single_replica]) - def add_data_after_replica(self): - """Add data to master after initial replica is setup""" - self.test_runner.run_add_data_after_replica() - @test(runs_after=[add_data_after_replica]) - def verify_replica_data_after_single(self): - """Verify data exists on single replica""" - self.test_runner.run_verify_replica_data_after_single() +@test(depends_on_groups=[GROUP_REPL_CREATE], + groups=[GROUP, GROUP_REPL_CREATE_WAIT]) +class ReplicationCreateWaitGroup(TestGroup): + """Wait for Replication Create to complete.""" - @test(runs_after=[verify_replica_data_after_single]) + def __init__(self): + super(ReplicationCreateWaitGroup, self).__init__( + ReplicationRunnerFactory.instance()) + + @test def wait_for_non_affinity_master(self): """Wait for non-affinity master to complete.""" self.test_runner.run_wait_for_non_affinity_master() - @test(runs_after=[wait_for_non_affinity_master]) + @test(depends_on=[wait_for_non_affinity_master]) def create_non_affinity_replica(self): """Test creating a non-affinity replica.""" self.test_runner.run_create_non_affinity_replica() - @test(runs_after=[create_non_affinity_replica]) - def create_multiple_replicas(self): - """Test creating multiple replicas.""" - self.test_runner.run_create_multiple_replicas() - - @test(runs_after=[create_multiple_replicas]) + @test(depends_on=[create_non_affinity_replica]) def wait_for_non_affinity_replica_fail(self): """Wait for non-affinity replica to fail.""" self.test_runner.run_wait_for_non_affinity_replica_fail() @@ -93,17 +95,71 @@ class ReplicationGroup(TestGroup): self.test_runner.run_delete_non_affinity_repl() @test(runs_after=[delete_non_affinity_repl]) + def wait_for_single_replica(self): + """Wait for single replica to complete.""" + self.test_runner.run_wait_for_single_replica() + + @test(depends_on=[wait_for_single_replica]) + def add_data_after_replica(self): + """Add data to master after initial replica is setup""" + self.test_runner.run_add_data_after_replica() + + @test(depends_on=[add_data_after_replica]) + def verify_replica_data_after_single(self): + """Verify data exists on single replica""" + self.test_runner.run_verify_replica_data_after_single() + + +@test(depends_on_groups=[GROUP_REPL_CREATE_WAIT], + groups=[GROUP, GROUP_REPL_MULTI_CREATE]) +class ReplicationMultiCreateGroup(TestGroup): + """Test Replication Multi-Create functionality.""" + + def __init__(self): + super(ReplicationMultiCreateGroup, self).__init__( + ReplicationRunnerFactory.instance()) + + @test + def create_multiple_replicas(self): + """Test creating multiple replicas.""" + self.test_runner.run_create_multiple_replicas() + + @test(runs_after=[create_multiple_replicas]) + def wait_for_delete_non_affinity_repl(self): + """Wait for the non-affinity replica to delete.""" + self.test_runner.run_wait_for_delete_non_affinity_repl() + + @test(depends_on=[wait_for_delete_non_affinity_repl]) def delete_non_affinity_master(self): """Test deleting non-affinity master.""" self.test_runner.run_delete_non_affinity_master() - @test(depends_on=[create_single_replica, create_multiple_replicas], - runs_after=[delete_non_affinity_master]) + +@test(depends_on_groups=[GROUP_REPL_MULTI_CREATE], + groups=[GROUP, GROUP_REPL_MULTI_CREATE_WAIT]) +class ReplicationMultiCreateWaitGroup(TestGroup): + """Wait for Replication Multi-Create to complete.""" + + def __init__(self): + super(ReplicationMultiCreateWaitGroup, self).__init__( + ReplicationRunnerFactory.instance()) + + @test + def wait_for_delete_non_affinity_master(self): + """Wait for the non-affinity master to delete.""" + self.test_runner.run_wait_for_delete_non_affinity_master() + + @test(runs_after=[wait_for_delete_non_affinity_master]) + def wait_for_multiple_replicas(self): + """Wait for multiple replicas to complete.""" + self.test_runner.run_wait_for_multiple_replicas() + + @test(depends_on=[wait_for_multiple_replicas]) def verify_replica_data_orig(self): """Verify original data was transferred to replicas.""" self.test_runner.run_verify_replica_data_orig() - @test(depends_on=[create_single_replica, create_multiple_replicas], + @test(depends_on=[wait_for_multiple_replicas], runs_after=[verify_replica_data_orig]) def add_data_to_replicate(self): """Add new data to master to verify replication.""" @@ -114,45 +170,43 @@ class ReplicationGroup(TestGroup): """Verify new data exists on master.""" self.test_runner.run_verify_data_to_replicate() - @test(depends_on=[create_single_replica, create_multiple_replicas, - add_data_to_replicate], + @test(depends_on=[add_data_to_replicate], runs_after=[verify_data_to_replicate]) - def wait_for_data_to_replicate(self): - """Wait to ensure that the data is replicated.""" - self.test_runner.run_wait_for_data_to_replicate() + def verify_replica_data_orig(self): + """Verify original data was transferred to replicas.""" + self.test_runner.run_verify_replica_data_orig() - @test(depends_on=[create_single_replica, create_multiple_replicas, - add_data_to_replicate], - runs_after=[wait_for_data_to_replicate]) + @test(depends_on=[add_data_to_replicate], + runs_after=[verify_replica_data_orig]) def verify_replica_data_new(self): """Verify new data was transferred to replicas.""" self.test_runner.run_verify_replica_data_new() - @test(depends_on=[create_single_replica, create_multiple_replicas], + @test(depends_on=[wait_for_multiple_replicas], runs_after=[verify_replica_data_new]) def promote_master(self): """Ensure promoting master fails.""" self.test_runner.run_promote_master() - @test(depends_on=[create_single_replica, create_multiple_replicas], + @test(depends_on=[wait_for_multiple_replicas], runs_after=[promote_master]) def eject_replica(self): """Ensure ejecting non master fails.""" self.test_runner.run_eject_replica() - @test(depends_on=[create_single_replica, create_multiple_replicas], + @test(depends_on=[wait_for_multiple_replicas], runs_after=[eject_replica]) def eject_valid_master(self): """Ensure ejecting valid master fails.""" self.test_runner.run_eject_valid_master() - @test(depends_on=[create_single_replica, create_multiple_replicas], + @test(depends_on=[wait_for_multiple_replicas], runs_after=[eject_valid_master]) def delete_valid_master(self): """Ensure deleting valid master fails.""" self.test_runner.run_delete_valid_master() - @test(depends_on=[create_single_replica, create_multiple_replicas], + @test(depends_on=[wait_for_multiple_replicas], runs_after=[delete_valid_master]) def promote_to_replica_source(self): """Test promoting a replica to replica source (master).""" @@ -163,7 +217,7 @@ class ReplicationGroup(TestGroup): """Verify data is still on new master.""" self.test_runner.run_verify_replica_data_new_master() - @test(depends_on=[create_single_replica, create_multiple_replicas, + @test(depends_on=[wait_for_multiple_replicas, promote_to_replica_source], runs_after=[verify_replica_data_new_master]) def add_data_to_replicate2(self): @@ -175,15 +229,9 @@ class ReplicationGroup(TestGroup): """Verify data exists on new master.""" self.test_runner.run_verify_data_to_replicate2() - @test(depends_on=[add_data_to_replicate2], - runs_after=[verify_data_to_replicate2]) - def wait_for_data_to_replicate2(self): - """Wait to ensure that the new data was replicated.""" - self.test_runner.run_wait_for_data_to_replicate() - - @test(depends_on=[create_single_replica, create_multiple_replicas, + @test(depends_on=[wait_for_multiple_replicas, add_data_to_replicate2], - runs_after=[wait_for_data_to_replicate2]) + runs_after=[verify_data_to_replicate2]) def verify_replica_data_new2(self): """Verify data was transferred to new replicas.""" self.test_runner.run_verify_replica_data_new2() @@ -195,6 +243,22 @@ class ReplicationGroup(TestGroup): self.test_runner.run_promote_original_source() @test(depends_on=[promote_original_source]) + def add_final_data_to_replicate(self): + """Add final data to original master to verify switch.""" + self.test_runner.run_add_final_data_to_replicate() + + @test(depends_on=[add_final_data_to_replicate]) + def verify_data_to_replicate_final(self): + """Verify final data exists on master.""" + self.test_runner.run_verify_data_to_replicate_final() + + @test(depends_on=[verify_data_to_replicate_final]) + def verify_final_data_replicated(self): + """Verify final data was transferred to all replicas.""" + self.test_runner.run_verify_final_data_replicated() + + @test(depends_on=[promote_original_source], + runs_after=[verify_final_data_replicated]) def remove_replicated_data(self): """Remove replication data.""" self.test_runner.run_remove_replicated_data() @@ -205,8 +269,17 @@ class ReplicationGroup(TestGroup): """Test detaching a replica from the master.""" self.test_runner.run_detach_replica_from_source() - @test(depends_on=[promote_original_source], - runs_after=[detach_replica_from_source]) + +@test(depends_on_groups=[GROUP_REPL_MULTI_CREATE_WAIT], + groups=[GROUP, GROUP_REPL_DELETE]) +class ReplicationDeleteGroup(TestGroup): + """Test Replication Delete functionality.""" + + def __init__(self): + super(ReplicationDeleteGroup, self).__init__( + ReplicationRunnerFactory.instance()) + + @test def delete_detached_replica(self): """Test deleting the detached replica.""" self.test_runner.run_delete_detached_replica() @@ -216,7 +289,22 @@ class ReplicationGroup(TestGroup): """Test deleting all the remaining replicas.""" self.test_runner.run_delete_all_replicas() - @test(runs_after=[delete_all_replicas]) + +@test(depends_on_groups=[GROUP_REPL_DELETE], + groups=[GROUP, GROUP_REPL_DELETE_WAIT]) +class ReplicationDeleteWaitGroup(TestGroup): + """Wait for Replication Delete to complete.""" + + def __init__(self): + super(ReplicationDeleteWaitGroup, self).__init__( + ReplicationRunnerFactory.instance()) + + @test + def wait_for_delete_replicas(self): + """Wait for all the replicas to delete.""" + self.test_runner.run_wait_for_delete_replicas() + + @test(runs_after=[wait_for_delete_replicas]) def test_backup_deleted(self): """Test that the created backup is now gone.""" self.test_runner.run_test_backup_deleted() diff --git a/trove/tests/scenario/helpers/cassandra_helper.py b/trove/tests/scenario/helpers/cassandra_helper.py index 05de19ae6e..74fa4690cc 100644 --- a/trove/tests/scenario/helpers/cassandra_helper.py +++ b/trove/tests/scenario/helpers/cassandra_helper.py @@ -58,8 +58,8 @@ class CassandraHelper(TestHelper): DATA_COLUMN_NAME = 'value' - def __init__(self, expected_override_name): - super(CassandraHelper, self).__init__(expected_override_name) + def __init__(self, expected_override_name, report): + super(CassandraHelper, self).__init__(expected_override_name, report) self._data_cache = dict() diff --git a/trove/tests/scenario/helpers/couchdb_helper.py b/trove/tests/scenario/helpers/couchdb_helper.py index 6b00e306cc..7c424a792d 100644 --- a/trove/tests/scenario/helpers/couchdb_helper.py +++ b/trove/tests/scenario/helpers/couchdb_helper.py @@ -21,8 +21,8 @@ from trove.tests.scenario.runners.test_runners import TestRunner class CouchdbHelper(TestHelper): - def __init__(self, expected_override_name): - super(CouchdbHelper, self).__init__(expected_override_name) + def __init__(self, expected_override_name, report): + super(CouchdbHelper, self).__init__(expected_override_name, report) self._data_cache = dict() self.field_name = 'ff-%s' self.database = 'firstdb' diff --git a/trove/tests/scenario/helpers/mariadb_helper.py b/trove/tests/scenario/helpers/mariadb_helper.py index 1a4b94598b..6ea4841d36 100644 --- a/trove/tests/scenario/helpers/mariadb_helper.py +++ b/trove/tests/scenario/helpers/mariadb_helper.py @@ -18,5 +18,5 @@ from trove.tests.scenario.helpers.mysql_helper import MysqlHelper class MariadbHelper(MysqlHelper): - def __init__(self, expected_override_name): - super(MariadbHelper, self).__init__(expected_override_name) + def __init__(self, expected_override_name, report): + super(MariadbHelper, self).__init__(expected_override_name, report) diff --git a/trove/tests/scenario/helpers/mongodb_helper.py b/trove/tests/scenario/helpers/mongodb_helper.py index 0fbb7de74b..60e7901438 100644 --- a/trove/tests/scenario/helpers/mongodb_helper.py +++ b/trove/tests/scenario/helpers/mongodb_helper.py @@ -18,8 +18,8 @@ from trove.tests.scenario.helpers.test_helper import TestHelper class MongodbHelper(TestHelper): - def __init__(self, expected_override_name): - super(MongodbHelper, self).__init__(expected_override_name) + def __init__(self, expected_override_name, report): + super(MongodbHelper, self).__init__(expected_override_name, report) def get_valid_database_definitions(self): return [{"name": 'db1'}, {"name": 'db2'}, {'name': 'db3'}] diff --git a/trove/tests/scenario/helpers/mysql_helper.py b/trove/tests/scenario/helpers/mysql_helper.py index 032c2a8562..fcd0ba96b3 100644 --- a/trove/tests/scenario/helpers/mysql_helper.py +++ b/trove/tests/scenario/helpers/mysql_helper.py @@ -18,8 +18,9 @@ from trove.tests.scenario.helpers.sql_helper import SqlHelper class MysqlHelper(SqlHelper): - def __init__(self, expected_override_name): - super(MysqlHelper, self).__init__(expected_override_name, 'mysql') + def __init__(self, expected_override_name, report): + super(MysqlHelper, self).__init__(expected_override_name, report, + 'mysql') def get_helper_credentials(self): return {'name': 'lite', 'password': 'litepass', 'database': 'firstdb'} diff --git a/trove/tests/scenario/helpers/percona_helper.py b/trove/tests/scenario/helpers/percona_helper.py index c9e43ee915..ce379c3618 100644 --- a/trove/tests/scenario/helpers/percona_helper.py +++ b/trove/tests/scenario/helpers/percona_helper.py @@ -18,5 +18,5 @@ from trove.tests.scenario.helpers.mysql_helper import MysqlHelper class PerconaHelper(MysqlHelper): - def __init__(self, expected_override_name): - super(PerconaHelper, self).__init__(expected_override_name) + def __init__(self, expected_override_name, report): + super(PerconaHelper, self).__init__(expected_override_name, report) diff --git a/trove/tests/scenario/helpers/postgresql_helper.py b/trove/tests/scenario/helpers/postgresql_helper.py index 26632cd425..56b8593243 100644 --- a/trove/tests/scenario/helpers/postgresql_helper.py +++ b/trove/tests/scenario/helpers/postgresql_helper.py @@ -18,8 +18,8 @@ from trove.tests.scenario.helpers.sql_helper import SqlHelper class PostgresqlHelper(SqlHelper): - def __init__(self, expected_override_name): - super(PostgresqlHelper, self).__init__(expected_override_name, + def __init__(self, expected_override_name, report): + super(PostgresqlHelper, self).__init__(expected_override_name, report, 'postgresql') @property diff --git a/trove/tests/scenario/helpers/pxc_helper.py b/trove/tests/scenario/helpers/pxc_helper.py index 7bd58c358c..dd52391a4f 100644 --- a/trove/tests/scenario/helpers/pxc_helper.py +++ b/trove/tests/scenario/helpers/pxc_helper.py @@ -18,5 +18,5 @@ from trove.tests.scenario.helpers.mysql_helper import MysqlHelper class PxcHelper(MysqlHelper): - def __init__(self, expected_override_name): - super(PxcHelper, self).__init__(expected_override_name) + def __init__(self, expected_override_name, report): + super(PxcHelper, self).__init__(expected_override_name, report) diff --git a/trove/tests/scenario/helpers/redis_helper.py b/trove/tests/scenario/helpers/redis_helper.py index 0c0fcbc01e..e8d4738d36 100644 --- a/trove/tests/scenario/helpers/redis_helper.py +++ b/trove/tests/scenario/helpers/redis_helper.py @@ -22,8 +22,8 @@ from trove.tests.scenario.runners.test_runners import TestRunner class RedisHelper(TestHelper): - def __init__(self, expected_override_name): - super(RedisHelper, self).__init__(expected_override_name) + def __init__(self, expected_override_name, report): + super(RedisHelper, self).__init__(expected_override_name, report) self.key_patterns = ['user_a:%s', 'user_b:%s'] self.value_pattern = 'id:%s' diff --git a/trove/tests/scenario/helpers/sql_helper.py b/trove/tests/scenario/helpers/sql_helper.py index 03bc43af09..325fb03e77 100644 --- a/trove/tests/scenario/helpers/sql_helper.py +++ b/trove/tests/scenario/helpers/sql_helper.py @@ -27,8 +27,8 @@ class SqlHelper(TestHelper): DATA_COLUMN_NAME = 'value' - def __init__(self, expected_override_name, protocol, port=None): - super(SqlHelper, self).__init__(expected_override_name) + def __init__(self, expected_override_name, report, protocol, port=None): + super(SqlHelper, self).__init__(expected_override_name, report) self.protocol = protocol self.port = port diff --git a/trove/tests/scenario/helpers/test_helper.py b/trove/tests/scenario/helpers/test_helper.py index bcf45b2597..01b1d1a419 100644 --- a/trove/tests/scenario/helpers/test_helper.py +++ b/trove/tests/scenario/helpers/test_helper.py @@ -38,11 +38,13 @@ class DataType(Enum): tiny = 3 # another tiny dataset (also for replication propagation) tiny2 = 4 + # a third tiny dataset (also for replication propagation) + tiny3 = 5 # small amount of data (this can be added to each instance # after creation, for example). - small = 5 + small = 6 # large data, enough to make creating a backup take 20s or more. - large = 6 + large = 7 class TestHelper(object): @@ -67,7 +69,7 @@ class TestHelper(object): # actual data manipulation work. DT_ACTUAL = 'actual' - def __init__(self, expected_override_name): + def __init__(self, expected_override_name, report): """Initialize the helper class by creating a number of stub functions that each datastore specific class can chose to override. Basically, the functions are of the form: @@ -86,6 +88,7 @@ class TestHelper(object): super(TestHelper, self).__init__() self._expected_override_name = expected_override_name + self.report = report # For building data access functions # name/fn pairs for each action @@ -114,6 +117,9 @@ class TestHelper(object): DataType.tiny2.name: { self.DATA_START: 2000, self.DATA_SIZE: 100}, + DataType.tiny3.name: { + self.DATA_START: 3000, + self.DATA_SIZE: 100}, DataType.small.name: { self.DATA_START: 10000, self.DATA_SIZE: 1000}, @@ -180,13 +186,25 @@ class TestHelper(object): ############## def add_data(self, data_type, host, *args, **kwargs): """Adds data of type 'data_type' to the database. Descendant - classes should implement a function for each DataType value - of the form 'add_{DataType.name}_data' - for example: - 'add_tiny_data' - 'add_small_data' - ... - Since this method may be called multiple times, the implemented - 'add_*_data' functions should be idempotent. + classes should implement a function 'add_actual_data' that has the + following signature: + def add_actual_data( + self, # standard self reference + data_label, # label used to identify the 'type' to add + data_start, # a start count + data_size, # a size to use + host, # the host to add the data to + *args, # for possible future expansion + **kwargs # for possible future expansion + ): + The data_label could be used to create a database or a table if the + datastore supports that. The data_start and data_size values are + designed not to overlap, such that all the data could be stored + in a single namespace (for example, creating ids from data_start + to data_start + data_size). + + Since this method may be called multiple times, the + 'add_actual_data' function should be idempotent. """ self._perform_data_action(self.FN_ADD, data_type.name, host, *args, **kwargs) @@ -203,9 +221,27 @@ class TestHelper(object): datastore. This can be done by testing edge cases, and possibly some random elements within the set. See instructions for 'add_data' for implementation guidance. + By default, the verification is attempted 10 times, sleeping for 3 + seconds between each attempt. This can be controlled by the + retry_count and retry_sleep kwarg values. """ - self._perform_data_action(self.FN_VERIFY, data_type.name, host, - *args, **kwargs) + retry_count = kwargs.pop('retry_count', 10) or 0 + retry_sleep = kwargs.pop('retry_sleep', 3) or 0 + attempts = -1 + while True: + attempts += 1 + try: + self._perform_data_action(self.FN_VERIFY, data_type.name, host, + *args, **kwargs) + break + except Exception as ex: + self.report.log("Attempt %d to verify data type %s failed\n%s" + % (attempts, data_type.name, ex)) + if attempts > retry_count: + raise + self.report.log("Trying again (after %d second sleep)" % + retry_sleep) + sleep(retry_sleep) def _perform_data_action(self, fn_type, fn_name, host, *args, **kwargs): fns = self._data_fns[fn_type] @@ -285,15 +321,6 @@ class TestHelper(object): if name in fns: fns[name] = fn - ##################### - # Replication related - ##################### - def wait_for_replicas(self): - """Wait for data to propagate to all the replicas. Datastore - specific overrides could increase (or decrease) this delay. - """ - sleep(30) - ####################### # Database/User related ####################### diff --git a/trove/tests/scenario/helpers/vertica_helper.py b/trove/tests/scenario/helpers/vertica_helper.py index 7245ced170..ef9ecb411b 100644 --- a/trove/tests/scenario/helpers/vertica_helper.py +++ b/trove/tests/scenario/helpers/vertica_helper.py @@ -20,8 +20,9 @@ from trove.tests.scenario.helpers.sql_helper import SqlHelper class VerticaHelper(SqlHelper): - def __init__(self, expected_override_name): - super(VerticaHelper, self).__init__(expected_override_name, 'vertica') + def __init__(self, expected_override_name, report): + super(VerticaHelper, self).__init__(expected_override_name, report, + 'vertica') def get_helper_credentials(self): return {'name': 'lite', 'password': 'litepass', 'database': 'lite'} diff --git a/trove/tests/scenario/runners/instance_create_runners.py b/trove/tests/scenario/runners/instance_create_runners.py index 4d4c9fb32d..e944077f6a 100644 --- a/trove/tests/scenario/runners/instance_create_runners.py +++ b/trove/tests/scenario/runners/instance_create_runners.py @@ -190,6 +190,7 @@ class InstanceCreateRunner(TestRunner): self.report.log("Using an existing instance: %s" % instance.id) self.assert_equal(expected_states[-1], instance.status, "Given instance is in a bad state.") + instance_info.name = instance.name else: self.report.log("Creating a new instance.") instance = self.auth_client.instances.create( diff --git a/trove/tests/scenario/runners/replication_runners.py b/trove/tests/scenario/runners/replication_runners.py index 9f341893e2..13afeb0c9b 100644 --- a/trove/tests/scenario/runners/replication_runners.py +++ b/trove/tests/scenario/runners/replication_runners.py @@ -42,7 +42,7 @@ class ReplicationRunner(TestRunner): def assert_add_replication_data(self, data_type, host): """In order for this to work, the corresponding datastore - 'helper' class should implement the 'add__data' method. + 'helper' class should implement the 'add_actual_data' method. """ self.test_helper.add_data(data_type, host) self.used_data_sets.add(data_type) @@ -55,7 +55,7 @@ class ReplicationRunner(TestRunner): def assert_verify_replication_data(self, data_type, host): """In order for this to work, the corresponding datastore - 'helper' class should implement the 'verify__data' method. + 'helper' class should implement the 'verify_actual_data' method. """ self.test_helper.verify_data(data_type, host) @@ -69,18 +69,14 @@ class ReplicationRunner(TestRunner): locality='anti-affinity').id self.assert_client_code(expected_http_code) - def run_create_single_replica(self, expected_states=['BUILD', 'ACTIVE'], - expected_http_code=200): - master_id = self.instance_info.id + def run_create_single_replica(self, expected_http_code=200): self.master_backup_count = len( - self.auth_client.instances.backups(master_id)) + self.auth_client.instances.backups(self.master_id)) self.replica_1_id = self.assert_replica_create( - master_id, 'replica1', 1, expected_states, expected_http_code) - self.replica_1_host = self.get_instance_host(self.replica_1_id) + self.master_id, 'replica1', 1, expected_http_code) def assert_replica_create( - self, master_id, replica_name, replica_count, - expected_states, expected_http_code): + self, master_id, replica_name, replica_count, expected_http_code): replica = self.auth_client.instances.create( self.instance_info.name + replica_name, self.instance_info.dbaas_flavor_href, @@ -89,14 +85,15 @@ class ReplicationRunner(TestRunner): datastore_version=self.instance_info.dbaas_datastore_version, nics=self.instance_info.nics, replica_count=replica_count) - replica_id = replica.id + self.assert_client_code(expected_http_code) + return replica.id - self.assert_instance_action(replica_id, expected_states, - expected_http_code) - self._assert_is_master(master_id, [replica_id]) - self._assert_is_replica(replica_id, master_id) - self._assert_locality(master_id) - return replica_id + def run_wait_for_single_replica(self, expected_states=['BUILD', 'ACTIVE']): + self.assert_instance_action(self.replica_1_id, expected_states) + self._assert_is_master(self.master_id, [self.replica_1_id]) + self._assert_is_replica(self.replica_1_id, self.master_id) + self._assert_locality(self.master_id) + self.replica_1_host = self.get_instance_host(self.replica_1_id) def _assert_is_master(self, instance_id, replica_ids): instance = self.get_instance(instance_id) @@ -148,43 +145,49 @@ class ReplicationRunner(TestRunner): replica_count=1).id self.assert_client_code(expected_http_code) - def run_create_multiple_replicas(self, expected_states=['BUILD', 'ACTIVE'], - expected_http_code=200): - master_id = self.instance_info.id + def run_create_multiple_replicas(self, expected_http_code=200): self.replica_2_id = self.assert_replica_create( - master_id, 'replica2', 2, expected_states, expected_http_code) + self.master_id, 'replica2', 2, expected_http_code) + + def run_wait_for_multiple_replicas( + self, expected_states=['BUILD', 'ACTIVE']): + replica_ids = self._get_replica_set(self.master_id) + self.assert_instance_action(replica_ids, expected_states) + self._assert_is_master(self.master_id, replica_ids) + for replica_id in replica_ids: + self._assert_is_replica(replica_id, self.master_id) + self._assert_locality(self.master_id) def run_wait_for_non_affinity_replica_fail( - self, expected_states=['BUILD', 'FAILED']): + self, expected_states=['BUILD', 'ERROR']): self._assert_instance_states(self.non_affinity_repl_id, expected_states, fast_fail_status=['ACTIVE']) - def run_delete_non_affinity_repl(self, - expected_last_state=['SHUTDOWN'], - expected_http_code=202): + def run_delete_non_affinity_repl(self, expected_http_code=202): self.assert_delete_instances( - self.non_affinity_repl_id, - expected_last_state=expected_last_state, - expected_http_code=expected_http_code) + self.non_affinity_repl_id, expected_http_code=expected_http_code) - def assert_delete_instances( - self, instance_ids, expected_last_state, expected_http_code): + def assert_delete_instances(self, instance_ids, expected_http_code): instance_ids = (instance_ids if utils.is_collection(instance_ids) else [instance_ids]) for instance_id in instance_ids: self.auth_client.instances.delete(instance_id) self.assert_client_code(expected_http_code) - self.assert_all_gone(instance_ids, expected_last_state) + def run_wait_for_delete_non_affinity_repl( + self, expected_last_status=['SHUTDOWN']): + self.assert_all_gone([self.non_affinity_repl_id], + expected_last_status=expected_last_status) - def run_delete_non_affinity_master(self, - expected_last_state=['SHUTDOWN'], - expected_http_code=202): + def run_delete_non_affinity_master(self, expected_http_code=202): self.assert_delete_instances( - self.non_affinity_master_id, - expected_last_state=expected_last_state, - expected_http_code=expected_http_code) + self.non_affinity_master_id, expected_http_code=expected_http_code) + + def run_wait_for_delete_non_affinity_master( + self, expected_last_status=['SHUTDOWN']): + self.assert_all_gone([self.non_affinity_master_id], + expected_last_status=expected_last_status) self.assert_server_group_gone(self.non_affinity_srv_grp_id) def run_add_data_to_replicate(self): @@ -193,9 +196,6 @@ class ReplicationRunner(TestRunner): def run_verify_data_to_replicate(self): self.assert_verify_replication_data(DataType.tiny, self.master_host) - def run_wait_for_data_to_replicate(self): - self.test_helper.wait_for_replicas() - def run_verify_replica_data_orig(self): self.assert_verify_replica_data(self.instance_info.id, DataType.small) @@ -292,6 +292,15 @@ class ReplicationRunner(TestRunner): self.instance_info.id, self.replica_1_id, expected_states, expected_http_code) + def run_add_final_data_to_replicate(self): + self.assert_add_replication_data(DataType.tiny3, self.master_host) + + def run_verify_data_to_replicate_final(self): + self.assert_verify_replication_data(DataType.tiny3, self.master_host) + + def run_verify_final_data_replicated(self): + self.assert_verify_replica_data(self.master_id, DataType.tiny3) + def run_remove_replicated_data(self): self.assert_remove_replicated_data(self.master_host) @@ -343,25 +352,26 @@ class ReplicationRunner(TestRunner): else: self.fail("Unexpected replica_of ID.") - def run_delete_detached_replica(self, - expected_last_state=['SHUTDOWN'], - expected_http_code=202): + def run_delete_detached_replica(self, expected_http_code=202): self.assert_delete_instances( - self.replica_1_id, expected_last_state=expected_last_state, - expected_http_code=expected_http_code) + self.replica_1_id, expected_http_code=expected_http_code) - def run_delete_all_replicas(self, expected_last_state=['SHUTDOWN'], - expected_http_code=202): + def run_delete_all_replicas(self, expected_http_code=202): self.assert_delete_all_replicas( - self.instance_info.id, expected_last_state, - expected_http_code) + self.instance_info.id, expected_http_code) def assert_delete_all_replicas( - self, master_id, expected_last_state, expected_http_code): + self, master_id, expected_http_code): self.report.log("Deleting a replication set: %s" % master_id) replica_ids = self._get_replica_set(master_id) - self.assert_delete_instances(replica_ids, expected_last_state, - expected_http_code) + self.assert_delete_instances(replica_ids, expected_http_code) + + def run_wait_for_delete_replicas( + self, expected_last_status=['SHUTDOWN']): + replica_ids = self._get_replica_set(self.master_id) + replica_ids.update(self.replica_1_id) + self.assert_all_gone(replica_ids, + expected_last_status=expected_last_status) def run_test_backup_deleted(self): backup = self.auth_client.instances.backups(self.master_id) diff --git a/trove/tests/scenario/runners/test_runners.py b/trove/tests/scenario/runners/test_runners.py index 757dc48277..fc35fdbaa3 100644 --- a/trove/tests/scenario/runners/test_runners.py +++ b/trove/tests/scenario/runners/test_runners.py @@ -72,17 +72,19 @@ class RunnerFactory(object): runner_module_name, class_prefix, runner_base_name, TEST_RUNNERS_NS) runner = runner_cls(*args, **kwargs) - runner._test_helper = cls._get_helper() + runner._test_helper = cls._get_helper(runner.report) return runner @classmethod - def _get_helper(cls): + def _get_helper(cls, report): class_prefix = cls._get_test_datastore() helper_cls = cls._load_dynamic_class( TEST_HELPER_MODULE_NAME, class_prefix, TEST_HELPER_BASE_NAME, TEST_HELPERS_NS) - return helper_cls(cls._build_class_name( - class_prefix, TEST_HELPER_BASE_NAME, strip_test=True)) + return helper_cls( + cls._build_class_name(class_prefix, + TEST_HELPER_BASE_NAME, strip_test=True), + report) @classmethod def _get_test_datastore(cls): @@ -224,6 +226,13 @@ class TestRunner(object): @classmethod def assert_is_sublist(cls, sub_list, full_list, message=None): + if not message: + message = 'Unexpected sublist' + try: + message += ": sub_list '%s' (full_list '%s')." % ( + sub_list, full_list) + except TypeError: + pass return cls.assert_true(set(sub_list).issubset(full_list), message) @classmethod @@ -396,7 +405,7 @@ class TestRunner(object): return self.has_env_flag(self.DO_NOT_DELETE_INSTANCE_FLAG) def assert_instance_action( - self, instance_ids, expected_states, expected_http_code): + self, instance_ids, expected_states, expected_http_code=None): self.assert_client_code(expected_http_code) if expected_states: self.assert_all_instance_states(