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:
huangtianhua 2015-11-03 11:59:17 +08:00
parent a089c6724a
commit 96f7c545ee
2 changed files with 55 additions and 36 deletions

View File

@ -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):

View File

@ -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',