Fix an error in archiving 'migrations' table
Add soft deleting 'migrations' table when the VM instance is deleted. And add soft deleting 'migrations' table when archiving deleted rows for the case to upgrade. Change-Id: Ica35ce2628dfcf412eb097c2c61fdde8828e9d90 Closes-Bug: #1584702
This commit is contained in:
parent
581262101a
commit
1dfd79495e
@ -1847,6 +1847,9 @@ def instance_destroy(context, instance_uuid, constraint=None):
|
|||||||
model_query(context, models.BlockDeviceMapping).\
|
model_query(context, models.BlockDeviceMapping).\
|
||||||
filter_by(instance_uuid=instance_uuid).\
|
filter_by(instance_uuid=instance_uuid).\
|
||||||
soft_delete()
|
soft_delete()
|
||||||
|
model_query(context, models.Migration).\
|
||||||
|
filter_by(instance_uuid=instance_uuid).\
|
||||||
|
soft_delete()
|
||||||
# NOTE(snikitin): We can't use model_query here, because there is no
|
# NOTE(snikitin): We can't use model_query here, because there is no
|
||||||
# column 'deleted' in 'tags' table.
|
# column 'deleted' in 'tags' table.
|
||||||
context.session.query(models.Tag).filter_by(
|
context.session.query(models.Tag).filter_by(
|
||||||
@ -6314,7 +6317,10 @@ def _archive_deleted_rows_for_table(tablename, max_rows):
|
|||||||
# NOTE(clecomte): Tables instance_actions and instances_actions_events
|
# NOTE(clecomte): Tables instance_actions and instances_actions_events
|
||||||
# have to be manage differently so we soft-delete them here to let
|
# have to be manage differently so we soft-delete them here to let
|
||||||
# the archive work the same for all tables
|
# the archive work the same for all tables
|
||||||
if tablename == "instance_actions":
|
# NOTE(takashin): The record in table migrations should be
|
||||||
|
# soft deleted when the instance is deleted.
|
||||||
|
# This is just for upgrading.
|
||||||
|
if tablename in ("instance_actions", "migrations"):
|
||||||
instances = models.BASE.metadata.tables["instances"]
|
instances = models.BASE.metadata.tables["instances"]
|
||||||
deleted_instances = sql.select([instances.c.uuid]).\
|
deleted_instances = sql.select([instances.c.uuid]).\
|
||||||
where(instances.c.deleted != instances.c.deleted.default.arg)
|
where(instances.c.deleted != instances.c.deleted.default.arg)
|
||||||
|
@ -2881,6 +2881,27 @@ class InstanceTestCase(test.TestCase, ModelsObjectComparatorMixin):
|
|||||||
db.instance_group_members_get(ctxt,
|
db.instance_group_members_get(ctxt,
|
||||||
group['uuid']))
|
group['uuid']))
|
||||||
|
|
||||||
|
def test_delete_migrations_on_instance_destroy(self):
|
||||||
|
ctxt = context.get_admin_context()
|
||||||
|
uuid = uuidsentinel.uuid1
|
||||||
|
db.instance_create(ctxt, {'uuid': uuid})
|
||||||
|
|
||||||
|
migrations_values = {'instance_uuid': uuid}
|
||||||
|
migration = db.migration_create(ctxt, migrations_values)
|
||||||
|
|
||||||
|
migrations = db.migration_get_all_by_filters(
|
||||||
|
ctxt, {'instance_uuid': uuid})
|
||||||
|
|
||||||
|
self.assertEqual(1, len(migrations))
|
||||||
|
self._assertEqualObjects(migration, migrations[0])
|
||||||
|
|
||||||
|
instance = db.instance_destroy(ctxt, uuid)
|
||||||
|
migrations = db.migration_get_all_by_filters(
|
||||||
|
ctxt, {'instance_uuid': uuid})
|
||||||
|
|
||||||
|
self.assertTrue(instance.deleted)
|
||||||
|
self.assertEqual(0, len(migrations))
|
||||||
|
|
||||||
def test_instance_update_and_get_original(self):
|
def test_instance_update_and_get_original(self):
|
||||||
instance = self.create_instance_with_args(vm_state='building')
|
instance = self.create_instance_with_args(vm_state='building')
|
||||||
(old_ref, new_ref) = db.instance_update_and_get_original(self.ctxt,
|
(old_ref, new_ref) = db.instance_update_and_get_original(self.ctxt,
|
||||||
@ -8842,6 +8863,9 @@ class ArchiveTestCase(test.TestCase, ModelsObjectComparatorMixin):
|
|||||||
self.instances = models.Instance.__table__
|
self.instances = models.Instance.__table__
|
||||||
self.shadow_instances = sqlalchemyutils.get_table(
|
self.shadow_instances = sqlalchemyutils.get_table(
|
||||||
self.engine, "shadow_instances")
|
self.engine, "shadow_instances")
|
||||||
|
self.migrations = models.Migration.__table__
|
||||||
|
self.shadow_migrations = sqlalchemyutils.get_table(
|
||||||
|
self.engine, "shadow_migrations")
|
||||||
|
|
||||||
self.uuidstrs = []
|
self.uuidstrs = []
|
||||||
for _ in range(6):
|
for _ in range(6):
|
||||||
@ -9044,8 +9068,7 @@ class ArchiveTestCase(test.TestCase, ModelsObjectComparatorMixin):
|
|||||||
'shadow_dns_domains',
|
'shadow_dns_domains',
|
||||||
)
|
)
|
||||||
|
|
||||||
def test_archive_deleted_rows_fk_constraint(self):
|
def _check_sqlite_version_less_than_3_7(self):
|
||||||
# consoles.pool_id depends on console_pools.id
|
|
||||||
# SQLite doesn't enforce foreign key constraints without a pragma.
|
# SQLite doesn't enforce foreign key constraints without a pragma.
|
||||||
dialect = self.engine.url.get_dialect()
|
dialect = self.engine.url.get_dialect()
|
||||||
if dialect == sqlite.dialect:
|
if dialect == sqlite.dialect:
|
||||||
@ -9059,6 +9082,10 @@ class ArchiveTestCase(test.TestCase, ModelsObjectComparatorMixin):
|
|||||||
self.skipTest(
|
self.skipTest(
|
||||||
'sqlite version too old for reliable SQLA foreign_keys')
|
'sqlite version too old for reliable SQLA foreign_keys')
|
||||||
self.conn.execute("PRAGMA foreign_keys = ON")
|
self.conn.execute("PRAGMA foreign_keys = ON")
|
||||||
|
|
||||||
|
def test_archive_deleted_rows_fk_constraint(self):
|
||||||
|
# consoles.pool_id depends on console_pools.id
|
||||||
|
self._check_sqlite_version_less_than_3_7()
|
||||||
ins_stmt = self.console_pools.insert().values(deleted=1)
|
ins_stmt = self.console_pools.insert().values(deleted=1)
|
||||||
result = self.conn.execute(ins_stmt)
|
result = self.conn.execute(ins_stmt)
|
||||||
id1 = result.inserted_primary_key[0]
|
id1 = result.inserted_primary_key[0]
|
||||||
@ -9083,6 +9110,33 @@ class ArchiveTestCase(test.TestCase, ModelsObjectComparatorMixin):
|
|||||||
'shadow_consoles'
|
'shadow_consoles'
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def test_archive_deleted_rows_for_migrations(self):
|
||||||
|
# migrations.instance_uuid depends on instances.uuid
|
||||||
|
self._check_sqlite_version_less_than_3_7()
|
||||||
|
instance_uuid = uuidsentinel.instance
|
||||||
|
ins_stmt = self.instances.insert().values(uuid=instance_uuid,
|
||||||
|
deleted=1)
|
||||||
|
self.conn.execute(ins_stmt)
|
||||||
|
ins_stmt = self.migrations.insert().values(instance_uuid=instance_uuid,
|
||||||
|
deleted=0)
|
||||||
|
self.conn.execute(ins_stmt)
|
||||||
|
# The first try to archive instances should fail, due to FK.
|
||||||
|
num = sqlalchemy_api._archive_deleted_rows_for_table("instances",
|
||||||
|
max_rows=None)
|
||||||
|
self.assertEqual(0, num)
|
||||||
|
# Then archiving migrations should work.
|
||||||
|
num = sqlalchemy_api._archive_deleted_rows_for_table("migrations",
|
||||||
|
max_rows=None)
|
||||||
|
self.assertEqual(1, num)
|
||||||
|
# Then archiving instances should work.
|
||||||
|
num = sqlalchemy_api._archive_deleted_rows_for_table("instances",
|
||||||
|
max_rows=None)
|
||||||
|
self.assertEqual(1, num)
|
||||||
|
self._assert_shadow_tables_empty_except(
|
||||||
|
'shadow_instances',
|
||||||
|
'shadow_migrations'
|
||||||
|
)
|
||||||
|
|
||||||
def test_archive_deleted_rows_2_tables(self):
|
def test_archive_deleted_rows_2_tables(self):
|
||||||
# Add 6 rows to each table
|
# Add 6 rows to each table
|
||||||
for uuidstr in self.uuidstrs:
|
for uuidstr in self.uuidstrs:
|
||||||
|
Loading…
Reference in New Issue
Block a user