Catch exceptions for restore_prev_rsrc/prepare_for_replace
Catch exceptions when calling restore_prev_rsrc or prepare_for_replace, to make sure the failure will be recorded and the resource will put into FAILED state. Change-Id: If65953d6b8cbdc135d898119c0f43a133f2e1892 Closes-Bug: #1512496
This commit is contained in:
parent
a089c6724a
commit
96f7c545ee
@ -931,7 +931,6 @@ class Resource(object):
|
||||
if not self._needs_update(after, before, after_props, before_props,
|
||||
prev_resource):
|
||||
return
|
||||
|
||||
if not cfg.CONF.convergence_engine:
|
||||
if (self.action, self.status) in (
|
||||
(self.CREATE, self.IN_PROGRESS),
|
||||
@ -958,6 +957,7 @@ class Resource(object):
|
||||
self._update_stored_properties()
|
||||
except exception.UpdateReplace as ex:
|
||||
# catch all UpdateReplace expections
|
||||
try:
|
||||
if (self.stack.action == 'ROLLBACK' and
|
||||
self.stack.status == 'IN_PROGRESS' and
|
||||
not cfg.CONF.convergence_engine):
|
||||
@ -966,6 +966,12 @@ class Resource(object):
|
||||
self.restore_prev_rsrc()
|
||||
else:
|
||||
self.prepare_for_replace()
|
||||
except Exception as e:
|
||||
# if any exception happen, we should set the resource to
|
||||
# FAILED, then raise ResourceFailure
|
||||
failure = exception.ResourceFailure(e, self, action)
|
||||
self.state_set(action, self.FAILED, six.text_type(failure))
|
||||
raise failure
|
||||
raise ex
|
||||
|
||||
def prepare_for_replace(self):
|
||||
|
@ -363,63 +363,76 @@ class ResourceTest(common.HeatTestCase):
|
||||
self.assertIsNotNone(res.updated_time)
|
||||
self.assertNotEqual(res.updated_time, stored_time)
|
||||
|
||||
def test_update_replace(self):
|
||||
def _setup_resource_for_update(self, res_name):
|
||||
class TestResource(resource.Resource):
|
||||
properties_schema = {'a_string': {'Type': 'String'}}
|
||||
update_allowed_properties = ('a_string',)
|
||||
|
||||
resource._register_class('TestResource', TestResource)
|
||||
|
||||
tmpl = rsrc_defn.ResourceDefinition('test_resource',
|
||||
tmpl = rsrc_defn.ResourceDefinition(res_name,
|
||||
'TestResource')
|
||||
res = TestResource('test_resource', tmpl, self.stack)
|
||||
|
||||
utmpl = rsrc_defn.ResourceDefinition(res_name, 'TestResource',
|
||||
{'a_string': 'foo'})
|
||||
|
||||
return res, utmpl
|
||||
|
||||
def test_update_replace(self):
|
||||
res, utmpl = self._setup_resource_for_update(
|
||||
res_name='test_update_replace')
|
||||
res.prepare_for_replace = mock.Mock()
|
||||
|
||||
utmpl = rsrc_defn.ResourceDefinition('test_resource', 'TestResource',
|
||||
{'a_string': 'foo'})
|
||||
self.assertRaises(
|
||||
exception.UpdateReplace, scheduler.TaskRunner(res.update, utmpl))
|
||||
self.assertTrue(res.prepare_for_replace.called)
|
||||
|
||||
def test_update_replace_prepare_replace_error(self):
|
||||
# test if any error happened when prepare_for_replace,
|
||||
# whether the resource will go to FAILED
|
||||
res, utmpl = self._setup_resource_for_update(
|
||||
res_name='test_update_replace_prepare_replace_error')
|
||||
res.prepare_for_replace = mock.Mock(side_effect=Exception)
|
||||
|
||||
self.assertRaises(
|
||||
exception.ResourceFailure,
|
||||
scheduler.TaskRunner(res.update, utmpl))
|
||||
self.assertTrue(res.prepare_for_replace.called)
|
||||
self.assertEqual((res.UPDATE, res.FAILED), res.state)
|
||||
|
||||
def test_update_rsrc_in_progress_raises_exception(self):
|
||||
class TestResource(resource.Resource):
|
||||
properties_schema = {'a_string': {'Type': 'String'}}
|
||||
update_allowed_properties = ('a_string',)
|
||||
res, utmpl = self._setup_resource_for_update(
|
||||
res_name='test_update_rsrc_in_progress_raises_exception')
|
||||
|
||||
cfg.CONF.set_override('convergence_engine', False)
|
||||
resource._register_class('TestResource', TestResource)
|
||||
|
||||
tmpl = rsrc_defn.ResourceDefinition('test_resource',
|
||||
'TestResource')
|
||||
res = TestResource('test_resource', tmpl, self.stack)
|
||||
|
||||
utmpl = rsrc_defn.ResourceDefinition('test_resource', 'TestResource',
|
||||
{'a_string': 'foo'})
|
||||
res.action = res.UPDATE
|
||||
res.status = res.IN_PROGRESS
|
||||
self.assertRaises(
|
||||
exception.ResourceFailure, scheduler.TaskRunner(res.update, utmpl))
|
||||
|
||||
def test_update_replace_rollback(self):
|
||||
class TestResource(resource.Resource):
|
||||
properties_schema = {'a_string': {'Type': 'String'}}
|
||||
update_allowed_properties = ('a_string',)
|
||||
|
||||
resource._register_class('TestResource', TestResource)
|
||||
|
||||
tmpl = rsrc_defn.ResourceDefinition('test_resource',
|
||||
'TestResource')
|
||||
self.stack.state_set('ROLLBACK', 'IN_PROGRESS', 'Simulate rollback')
|
||||
res = TestResource('test_resource', tmpl, self.stack)
|
||||
|
||||
res, utmpl = self._setup_resource_for_update(
|
||||
res_name='test_update_replace_rollback')
|
||||
res.restore_prev_rsrc = mock.Mock()
|
||||
self.stack.state_set('ROLLBACK', 'IN_PROGRESS', 'Simulate rollback')
|
||||
|
||||
utmpl = rsrc_defn.ResourceDefinition('test_resource', 'TestResource',
|
||||
{'a_string': 'foo'})
|
||||
self.assertRaises(
|
||||
exception.UpdateReplace, scheduler.TaskRunner(res.update, utmpl))
|
||||
self.assertTrue(res.restore_prev_rsrc.called)
|
||||
|
||||
def test_update_replace_rollback_restore_prev_rsrc_error(self):
|
||||
res, utmpl = self._setup_resource_for_update(
|
||||
res_name='restore_prev_rsrc_error')
|
||||
res.restore_prev_rsrc = mock.Mock(side_effect=Exception)
|
||||
self.stack.state_set('ROLLBACK', 'IN_PROGRESS', 'Simulate rollback')
|
||||
|
||||
self.assertRaises(
|
||||
exception.ResourceFailure, scheduler.TaskRunner(res.update, utmpl))
|
||||
self.assertTrue(res.restore_prev_rsrc.called)
|
||||
self.assertEqual((res.UPDATE, res.FAILED), res.state)
|
||||
|
||||
def test_update_replace_in_failed_without_nested(self):
|
||||
tmpl = rsrc_defn.ResourceDefinition('test_resource',
|
||||
'GenericResourceType',
|
||||
|
Loading…
Reference in New Issue
Block a user