Fix database purge query
Fix raw_template purge query on MySQL, and handle stack tags before removing stacks. This also removes a bunch of race conditions where we deleted incorrect data. Change-Id: I7b7a1d94acefbaeeed86f1833c979819361c8988 Closes-Bug: #1524387
This commit is contained in:
parent
f84d2ea795
commit
6cece5bb7e
@ -1064,6 +1064,7 @@ def purge_deleted(age, granularity='days'):
|
|||||||
|
|
||||||
stack = sqlalchemy.Table('stack', meta, autoload=True)
|
stack = sqlalchemy.Table('stack', meta, autoload=True)
|
||||||
stack_lock = sqlalchemy.Table('stack_lock', meta, autoload=True)
|
stack_lock = sqlalchemy.Table('stack_lock', meta, autoload=True)
|
||||||
|
stack_tag = sqlalchemy.Table('stack_tag', meta, autoload=True)
|
||||||
resource = sqlalchemy.Table('resource', meta, autoload=True)
|
resource = sqlalchemy.Table('resource', meta, autoload=True)
|
||||||
resource_data = sqlalchemy.Table('resource_data', meta, autoload=True)
|
resource_data = sqlalchemy.Table('resource_data', meta, autoload=True)
|
||||||
event = sqlalchemy.Table('event', meta, autoload=True)
|
event = sqlalchemy.Table('event', meta, autoload=True)
|
||||||
@ -1073,46 +1074,67 @@ def purge_deleted(age, granularity='days'):
|
|||||||
syncpoint = sqlalchemy.Table('sync_point', meta, autoload=True)
|
syncpoint = sqlalchemy.Table('sync_point', meta, autoload=True)
|
||||||
|
|
||||||
# find the soft-deleted stacks that are past their expiry
|
# find the soft-deleted stacks that are past their expiry
|
||||||
stack_where = sqlalchemy.select([stack.c.id]).where(
|
stack_where = sqlalchemy.select([stack.c.id, stack.c.raw_template_id,
|
||||||
|
stack.c.prev_raw_template_id,
|
||||||
|
stack.c.user_creds_id]).where(
|
||||||
stack.c.deleted_at < time_line)
|
stack.c.deleted_at < time_line)
|
||||||
|
stacks = list(engine.execute(stack_where))
|
||||||
|
if stacks:
|
||||||
|
stack_ids = [i[0] for i in stacks]
|
||||||
# delete stack locks (just in case some got stuck)
|
# delete stack locks (just in case some got stuck)
|
||||||
stack_lock_del = stack_lock.delete().where(
|
stack_lock_del = stack_lock.delete().where(
|
||||||
stack_lock.c.stack_id.in_(stack_where))
|
stack_lock.c.stack_id.in_(stack_ids))
|
||||||
engine.execute(stack_lock_del)
|
engine.execute(stack_lock_del)
|
||||||
|
# delete stack tags
|
||||||
|
stack_tag_del = stack_tag.delete().where(
|
||||||
|
stack_tag.c.stack_id.in_(stack_ids))
|
||||||
|
engine.execute(stack_tag_del)
|
||||||
# delete resource_data
|
# delete resource_data
|
||||||
res_where = sqlalchemy.select([resource.c.id]).where(
|
res_where = sqlalchemy.select([resource.c.id]).where(
|
||||||
resource.c.stack_id.in_(stack_where))
|
resource.c.stack_id.in_(stack_ids))
|
||||||
res_data_del = resource_data.delete().where(
|
res_data_del = resource_data.delete().where(
|
||||||
resource_data.c.resource_id.in_(res_where))
|
resource_data.c.resource_id.in_(res_where))
|
||||||
engine.execute(res_data_del)
|
engine.execute(res_data_del)
|
||||||
# delete resources
|
# delete resources
|
||||||
res_del = resource.delete().where(resource.c.stack_id.in_(stack_where))
|
res_del = resource.delete().where(resource.c.stack_id.in_(stack_ids))
|
||||||
engine.execute(res_del)
|
engine.execute(res_del)
|
||||||
# delete events
|
# delete events
|
||||||
event_del = event.delete().where(event.c.stack_id.in_(stack_where))
|
event_del = event.delete().where(event.c.stack_id.in_(stack_ids))
|
||||||
engine.execute(event_del)
|
engine.execute(event_del)
|
||||||
# clean up any sync_points that may have lingered
|
# clean up any sync_points that may have lingered
|
||||||
sync_del = syncpoint.delete().where(syncpoint.c.stack_id.in_(stack_where))
|
sync_del = syncpoint.delete().where(
|
||||||
|
syncpoint.c.stack_id.in_(stack_ids))
|
||||||
engine.execute(sync_del)
|
engine.execute(sync_del)
|
||||||
# delete the stacks
|
# delete the stacks
|
||||||
stack_del = stack.delete().where(stack.c.deleted_at < time_line)
|
stack_del = stack.delete().where(stack.c.id.in_(stack_ids))
|
||||||
engine.execute(stack_del)
|
engine.execute(stack_del)
|
||||||
# delete orphaned raw templates
|
# delete orphaned raw templates
|
||||||
raw_templ_sel = raw_template.c.id.in_(
|
raw_template_ids = [i[1] for i in stacks if i[1] is not None]
|
||||||
sqlalchemy.select([raw_template.c.id]).select_from(
|
raw_template_ids.extend(i[2] for i in stacks if i[2] is not None)
|
||||||
sqlalchemy.join(
|
if raw_template_ids:
|
||||||
raw_template,
|
# keep those still referenced
|
||||||
stack,
|
raw_tmpl_sel = sqlalchemy.select([stack.c.raw_template_id]).where(
|
||||||
sqlalchemy.or_(
|
stack.c.raw_template_id.in_(raw_template_ids))
|
||||||
stack.c.prev_raw_template_id == raw_template.c.id,
|
raw_tmpl = [i[0] for i in engine.execute(raw_tmpl_sel)]
|
||||||
stack.c.raw_template_id == raw_template.c.id),
|
raw_template_ids = set(raw_template_ids) - set(raw_tmpl)
|
||||||
isouter=True)).where(stack.c.id == None)) # noqa
|
raw_tmpl_sel = sqlalchemy.select(
|
||||||
raw_templ_del = raw_template.delete().where(raw_templ_sel)
|
[stack.c.prev_raw_template_id]).where(
|
||||||
|
stack.c.prev_raw_template_id.in_(raw_template_ids))
|
||||||
|
raw_tmpl = [i[0] for i in engine.execute(raw_tmpl_sel)]
|
||||||
|
raw_template_ids = raw_template_ids - set(raw_tmpl)
|
||||||
|
raw_templ_del = raw_template.delete().where(
|
||||||
|
raw_template.c.id.in_(raw_template_ids))
|
||||||
engine.execute(raw_templ_del)
|
engine.execute(raw_templ_del)
|
||||||
# purge any user creds that are no longer referenced
|
# purge any user creds that are no longer referenced
|
||||||
stack_creds_sel = sqlalchemy.select([stack.c.user_creds_id])
|
user_creds_ids = [i[3] for i in stacks if i[3] is not None]
|
||||||
user_creds_sel = sqlalchemy.not_(user_creds.c.id.in_(stack_creds_sel))
|
if user_creds_ids:
|
||||||
usr_creds_del = user_creds.delete().where(user_creds_sel)
|
# keep those still referenced
|
||||||
|
user_sel = sqlalchemy.select([stack.c.user_creds_id]).where(
|
||||||
|
stack.c.user_creds_id.in_(user_creds_ids))
|
||||||
|
users = [i[0] for i in engine.execute(user_sel)]
|
||||||
|
user_creds_ids = set(user_creds_ids) - set(users)
|
||||||
|
usr_creds_del = user_creds.delete().where(
|
||||||
|
user_creds.c.id.in_(user_creds_ids))
|
||||||
engine.execute(usr_creds_del)
|
engine.execute(usr_creds_del)
|
||||||
# Purge deleted services
|
# Purge deleted services
|
||||||
srvc_del = service.delete().where(service.c.deleted_at < time_line)
|
srvc_del = service.delete().where(service.c.deleted_at < time_line)
|
||||||
|
@ -189,6 +189,9 @@ class Stack(
|
|||||||
def refresh(self):
|
def refresh(self):
|
||||||
db_stack = db_api.stack_get(
|
db_stack = db_api.stack_get(
|
||||||
self._context, self.id, show_deleted=True)
|
self._context, self.id, show_deleted=True)
|
||||||
|
if db_stack is None:
|
||||||
|
message = _('No stack exists with id "%s"') % str(self.id)
|
||||||
|
raise exception.NotFound(message)
|
||||||
db_stack.refresh()
|
db_stack.refresh()
|
||||||
return self.__class__._from_db_object(
|
return self.__class__._from_db_object(
|
||||||
self._context,
|
self._context,
|
||||||
|
47
heat_integrationtests/functional/test_purge.py
Normal file
47
heat_integrationtests/functional/test_purge.py
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
# not use this file except in compliance with the License. You may obtain
|
||||||
|
# a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
# License for the specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
|
||||||
|
from oslo_concurrency import processutils
|
||||||
|
|
||||||
|
from heat_integrationtests.functional import functional_base
|
||||||
|
|
||||||
|
|
||||||
|
class PurgeTest(functional_base.FunctionalTestsBase):
|
||||||
|
template = '''
|
||||||
|
heat_template_version: 2014-10-16
|
||||||
|
parameters:
|
||||||
|
resources:
|
||||||
|
test_resource:
|
||||||
|
type: OS::Heat::TestResource
|
||||||
|
'''
|
||||||
|
|
||||||
|
def test_purge(self):
|
||||||
|
stack_identifier = self.stack_create(template=self.template)
|
||||||
|
self._stack_delete(stack_identifier)
|
||||||
|
stacks = dict((stack.id, stack) for stack in
|
||||||
|
self.client.stacks.list(show_deleted=True))
|
||||||
|
self.assertIn(stack_identifier.split('/')[1], stacks)
|
||||||
|
cmd = "heat-manage purge_deleted 0"
|
||||||
|
processutils.execute(cmd, shell=True)
|
||||||
|
stacks = dict((stack.id, stack) for stack in
|
||||||
|
self.client.stacks.list(show_deleted=True))
|
||||||
|
self.assertNotIn(stack_identifier.split('/')[1], stacks)
|
||||||
|
|
||||||
|
# Test with tags
|
||||||
|
stack_identifier = self.stack_create(template=self.template,
|
||||||
|
tags="foo,bar")
|
||||||
|
self._stack_delete(stack_identifier)
|
||||||
|
cmd = "heat-manage purge_deleted 0"
|
||||||
|
processutils.execute(cmd, shell=True)
|
||||||
|
stacks = dict((stack.id, stack) for stack in
|
||||||
|
self.client.stacks.list(show_deleted=True))
|
||||||
|
self.assertNotIn(stack_identifier.split('/')[1], stacks)
|
Loading…
Reference in New Issue
Block a user