From a484c739736c8ccea74ec684da8971b189747193 Mon Sep 17 00:00:00 2001
From: Pavlo Shchelokovskyy <pshchelokovskyy@mirantis.com>
Date: Fri, 23 Jan 2015 18:38:57 +0200
Subject: [PATCH] Remove TaskRunner from Volume resources

All the volume_tasks logic has been redistributed as methods of either
cinder or nova client plugins.

The logic of detaching was simplified and a check for out-of-band
changes is no longer executed.

The old VolumeAttachTask is left in place as it is still used in
creating the AWS::EC2::Instance resource.

Change-Id: I13f8e03ff090dd5f5739e5af231c77d066eb9eac
Partial-Bug: #1393268
---
 heat/engine/clients/os/cinder.py              |  82 +++++++
 heat/engine/clients/os/nova.py                |  50 +++++
 .../resources/openstack/cinder/volume.py      | 202 ++++++++++++------
 heat/engine/resources/volume_base.py          | 152 ++++++++-----
 heat/engine/volume_tasks.py                   | 150 -------------
 heat/tests/aws/test_volume.py                 |  34 +--
 heat/tests/openstack/test_volume.py           |   3 -
 7 files changed, 382 insertions(+), 291 deletions(-)

diff --git a/heat/engine/clients/os/cinder.py b/heat/engine/clients/os/cinder.py
index 368ca78e9b..11f9813133 100644
--- a/heat/engine/clients/os/cinder.py
+++ b/heat/engine/clients/os/cinder.py
@@ -22,6 +22,7 @@ from heat.common.i18n import _
 from heat.common.i18n import _LI
 from heat.engine.clients import client_plugin
 from heat.engine import constraints
+from heat.engine import resource
 
 
 LOG = logging.getLogger(__name__)
@@ -127,6 +128,87 @@ class CinderClientPlugin(client_plugin.ClientPlugin):
         return (isinstance(ex, exceptions.ClientException) and
                 ex.code == 409)
 
+    def check_detach_volume_complete(self, vol_id):
+        try:
+            vol = self.client().volumes.get(vol_id)
+        except Exception as ex:
+            self.ignore_not_found(ex)
+            return True
+
+        if vol.status in ('in-use', 'detaching'):
+            LOG.debug('%s - volume still in use' % vol_id)
+            return False
+
+        LOG.debug('Volume %(id)s - status: %(status)s' % {
+            'id': vol.id, 'status': vol.status})
+
+        if vol.status not in ('available', 'deleting'):
+            LOG.debug("Detachment failed - volume %(vol)s "
+                      "is in %(status)s status" % {"vol": vol.id,
+                                                   "status": vol.status})
+            raise resource.ResourceUnknownStatus(
+                resource_status=vol.status,
+                result=_('Volume detachment failed'))
+        else:
+            return True
+
+    def check_attach_volume_complete(self, vol_id):
+        vol = self.client().volumes.get(vol_id)
+        if vol.status in ('available', 'attaching'):
+            LOG.debug("Volume %(id)s is being attached - "
+                      "volume status: %(status)s" % {'id': vol_id,
+                                                     'status': vol.status})
+            return False
+
+        if vol.status != 'in-use':
+            LOG.debug("Attachment failed - volume %(vol)s is "
+                      "in %(status)s status" % {"vol": vol_id,
+                                                "status": vol.status})
+            raise resource.ResourceUnknownStatus(
+                resource_status=vol.status,
+                result=_('Volume attachment failed'))
+
+        LOG.info(_LI('Attaching volume %(id)s complete'), {'id': vol_id})
+        return True
+
+
+# NOTE(pshchelo): these Volume*Progress classes are simple key-value storages
+# meant to be passed between handle_<action> and check_<action>_complete,
+# being mutated during subsequent check_<action>_complete calls.
+class VolumeDetachProgress(object):
+    def __init__(self, srv_id, vol_id, attach_id, val=False):
+        self.called = val
+        self.cinder_complete = val
+        self.nova_complete = val
+        self.srv_id = srv_id
+        self.vol_id = vol_id
+        self.attach_id = attach_id
+
+
+class VolumeAttachProgress(object):
+    def __init__(self, srv_id, vol_id, device, val=False):
+        self.called = val
+        self.complete = val
+        self.srv_id = srv_id
+        self.vol_id = vol_id
+        self.device = device
+
+
+class VolumeDeleteProgress(object):
+    def __init__(self, val=False):
+        self.backup = {'called': val,
+                       'complete': val}
+        self.delete = {'called': val,
+                       'complete': val}
+        self.backup_id = None
+
+
+class VolumeResizeProgress(object):
+    def __init__(self, val=False, size=None):
+        self.called = val
+        self.complete = val
+        self.size = size
+
 
 class VolumeConstraint(constraints.BaseCustomConstraint):
 
diff --git a/heat/engine/clients/os/nova.py b/heat/engine/clients/os/nova.py
index ee46c43ad1..0c9a0b4111 100644
--- a/heat/engine/clients/os/nova.py
+++ b/heat/engine/clients/os/nova.py
@@ -32,6 +32,7 @@ from six.moves.urllib import parse as urlparse
 
 from heat.common import exception
 from heat.common.i18n import _
+from heat.common.i18n import _LI
 from heat.common.i18n import _LW
 from heat.engine.clients import client_plugin
 from heat.engine import constraints
@@ -495,6 +496,55 @@ echo -e '%s\tALL=(ALL)\tNOPASSWD: ALL' >> /etc/sudoers
 
         return net_id
 
+    def attach_volume(self, server_id, volume_id, device):
+        try:
+            va = self.client().volumes.create_server_volume(
+                server_id=server_id,
+                volume_id=volume_id,
+                device=device)
+        except Exception as ex:
+            if self.is_client_exception(ex):
+                raise exception.Error(_(
+                    "Failed to attach volume %(vol)s to server %(srv)s "
+                    "- %(err)s") % {'vol': volume_id,
+                                    'srv': server_id,
+                                    'err': ex})
+            else:
+                raise
+        return va.id
+
+    def detach_volume(self, server_id, attach_id):
+        # detach the volume using volume_attachment
+        try:
+            self.client().volumes.delete_server_volume(server_id, attach_id)
+        except Exception as ex:
+            if not (self.is_not_found(ex)
+                    or self.is_bad_request(ex)):
+                raise exception.Error(
+                    _("Could not detach attachment %(att)s "
+                      "from server %(srv)s.") % {'srv': server_id,
+                                                 'att': attach_id})
+
+    def check_detach_volume_complete(self, server_id, attach_id):
+        """Check that nova server lost attachment.
+
+        This check is needed for immediate reattachment when updating:
+        there might be some time between cinder marking volume as 'available'
+        and nova removing attachment from its own objects, so we
+        check that nova already knows that the volume is detached.
+        """
+        try:
+            self.client().volumes.get_server_volume(server_id, attach_id)
+        except Exception as ex:
+            self.ignore_not_found(ex)
+            LOG.info(_LI("Volume %(vol)s is detached from server %(srv)s"),
+                     {'vol': attach_id, 'srv': server_id})
+            return True
+        else:
+            LOG.debug("Server %(srv)s still has attachment %(att)s." % {
+                'att': attach_id, 'srv': server_id})
+            return False
+
 
 class ServerConstraint(constraints.BaseCustomConstraint):
 
diff --git a/heat/engine/resources/openstack/cinder/volume.py b/heat/engine/resources/openstack/cinder/volume.py
index 1520fde7f7..8c70e12ef8 100644
--- a/heat/engine/resources/openstack/cinder/volume.py
+++ b/heat/engine/resources/openstack/cinder/volume.py
@@ -19,12 +19,12 @@ from heat.common import exception
 from heat.common.i18n import _
 from heat.common.i18n import _LI
 from heat.engine import attributes
+from heat.engine.clients.os import cinder as heat_cinder
 from heat.engine import constraints
 from heat.engine import properties
+from heat.engine import resource
 from heat.engine.resources import volume_base as vb
-from heat.engine import scheduler
 from heat.engine import support
-from heat.engine import volume_tasks as vol_task
 
 LOG = logging.getLogger(__name__)
 
@@ -236,10 +236,41 @@ class CinderVolume(vb.BaseVolume):
 
         return vol_id
 
+    def _extend_volume(self, new_size):
+        try:
+            self.client().volumes.extend(self.resource_id, new_size)
+        except Exception as ex:
+            if self.client_plugin().is_client_exception(ex):
+                raise exception.Error(_(
+                    "Failed to extend volume %(vol)s - %(err)s") % {
+                        'vol': self.resource_id, 'err': str(ex)})
+            else:
+                raise
+        return True
+
+    def _check_extend_volume_complete(self):
+        vol = self.client().volumes.get(self.resource_id)
+        if vol.status == 'extending':
+            LOG.debug("Volume %s is being extended" % vol.id)
+            return False
+
+        if vol.status != 'available':
+            LOG.info(_LI("Resize failed: Volume %(vol)s "
+                         "is in %(status)s state."),
+                     {'vol': vol.id, 'status': vol.status})
+            raise resource.ResourceUnknownStatus(
+                resource_status=vol.status,
+                result=_('Volume resize failed'))
+
+        LOG.info(_LI('Volume %(id)s resize complete'), {'id': vol.id})
+        return True
+
     def handle_update(self, json_snippet, tmpl_diff, prop_diff):
         vol = None
-        checkers = []
         cinder = self.client()
+        prg_resize = None
+        prg_attach = None
+        prg_detach = None
         # update the name and description for cinder volume
         if self.NAME in prop_diff or self.DESCRIPTION in prop_diff:
             vol = cinder.volumes.get(self.resource_id)
@@ -268,6 +299,10 @@ class CinderVolume(vb.BaseVolume):
                     vol = cinder.volumes.get(self.resource_id)
                 new_vol_type = prop_diff.get(self.VOLUME_TYPE)
                 cinder.volumes.retype(vol, new_vol_type, 'never')
+        # update read_only access mode
+        if self.READ_ONLY in prop_diff:
+            flag = prop_diff.get(self.READ_ONLY)
+            cinder.volumes.update_readonly_flag(self.resource_id, flag)
         # extend volume size
         if self.SIZE in prop_diff:
             if not vol:
@@ -278,6 +313,7 @@ class CinderVolume(vb.BaseVolume):
                 raise exception.NotSupported(feature=_("Shrinking volume"))
 
             elif new_size > vol.size:
+                prg_resize = heat_cinder.VolumeResizeProgress(size=new_size)
                 if vol.attachments:
                     # NOTE(pshchelo):
                     # this relies on current behavior of cinder attachments,
@@ -289,35 +325,59 @@ class CinderVolume(vb.BaseVolume):
                     server_id = vol.attachments[0]['server_id']
                     device = vol.attachments[0]['device']
                     attachment_id = vol.attachments[0]['id']
-                    detach_task = vol_task.VolumeDetachTask(
-                        self.stack, server_id, attachment_id)
-                    checkers.append(scheduler.TaskRunner(detach_task))
-                    extend_task = vol_task.VolumeExtendTask(
-                        self.stack, vol.id, new_size)
-                    checkers.append(scheduler.TaskRunner(extend_task))
-                    attach_task = vol_task.VolumeAttachTask(
-                        self.stack, server_id, vol.id, device)
-                    checkers.append(scheduler.TaskRunner(attach_task))
+                    prg_detach = heat_cinder.VolumeDetachProgress(
+                        server_id, vol.id, attachment_id)
+                    prg_attach = heat_cinder.VolumeAttachProgress(
+                        server_id, vol.id, device)
 
-                else:
-                    extend_task = vol_task.VolumeExtendTask(
-                        self.stack, vol.id, new_size)
-                    checkers.append(scheduler.TaskRunner(extend_task))
-        # update read_only access mode
-        if self.READ_ONLY in prop_diff:
-            flag = prop_diff.get(self.READ_ONLY)
-            cinder.volumes.update_readonly_flag(self.resource_id, flag)
+        return prg_detach, prg_resize, prg_attach
 
-        if checkers:
-            checkers[0].start()
-        return checkers
+    def _detach_volume_to_complete(self, prg_detach):
+        if not prg_detach.called:
+            self.client_plugin('nova').detach_volume(prg_detach.srv_id,
+                                                     prg_detach.attach_id)
+            prg_detach.called = True
+            return False
+        if not prg_detach.cinder_complete:
+            cinder_complete_res = self.client_plugin(
+            ).check_detach_volume_complete(prg_detach.vol_id)
+            prg_detach.cinder_complete = cinder_complete_res
+            return False
+        if not prg_detach.nova_complete:
+            prg_detach.nova_complete = self.client_plugin(
+                'nova').check_detach_volume_complete(prg_detach.srv_id,
+                                                     prg_detach.attach_id)
+            return False
+
+    def _attach_volume_to_complete(self, prg_attach):
+        if not prg_attach.called:
+            prg_attach.called = self.client_plugin('nova').attach_volume(
+                prg_attach.srv_id, prg_attach.vol_id, prg_attach.device)
+            return False
+        if not prg_attach.complete:
+            prg_attach.complete = self.client_plugin(
+            ).check_attach_volume_complete(prg_attach.vol_id)
+            return prg_attach.complete
 
     def check_update_complete(self, checkers):
-        for checker in checkers:
-            if not checker.started():
-                checker.start()
-            if not checker.step():
+        prg_detach, prg_resize, prg_attach = checkers
+        if not prg_resize:
+            return True
+        # detach volume
+        if prg_detach:
+            if not prg_detach.nova_complete:
+                self._detach_volume_to_complete(prg_detach)
                 return False
+        # resize volume
+        if not prg_resize.called:
+            prg_resize.called = self._extend_volume(prg_resize.size)
+            return False
+        if not prg_resize.complete:
+            prg_resize.complete = self._check_extend_volume_complete()
+            return prg_resize.complete and not prg_attach
+        # reattach volume back
+        if prg_attach:
+            return self._attach_volume_to_complete(prg_attach)
         return True
 
     def handle_snapshot(self):
@@ -334,24 +394,25 @@ class CinderVolume(vb.BaseVolume):
         raise exception.Error(backup.fail_reason)
 
     def handle_delete_snapshot(self, snapshot):
-        backup_id = snapshot['resource_data'].get('backup_id')
+        backup_id = snapshot['resource_data']['backup_id']
+        try:
+            self.client().backups.delete(backup_id)
+        except Exception as ex:
+            self.client_plugin().ignore_not_found(ex)
+            return
+        else:
+            return backup_id
 
-        def delete():
-            cinder = self.client()
-            try:
-                cinder.backups.delete(backup_id)
-                while True:
-                    yield
-                    cinder.backups.get(backup_id)
-            except Exception as ex:
-                self.client_plugin().ignore_not_found(ex)
-
-        delete_task = scheduler.TaskRunner(delete)
-        delete_task.start()
-        return delete_task
-
-    def check_delete_snapshot_complete(self, delete_task):
-        return delete_task.step()
+    def check_delete_snapshot_complete(self, backup_id):
+        if not backup_id:
+            return True
+        try:
+            self.client().backups.get(backup_id)
+        except Exception as ex:
+            self.client_plugin().ignore_not_found(ex)
+            return True
+        else:
+            return False
 
     def _build_exclusive_options(self):
         exclusive_options = []
@@ -463,13 +524,21 @@ class CinderVolumeAttachment(vb.BaseVolumeAttachment):
     }
 
     def handle_update(self, json_snippet, tmpl_diff, prop_diff):
-        checkers = []
+        prg_attach = None
+        prg_detach = None
         if prop_diff:
             # Even though some combinations of changed properties
             # could be updated in UpdateReplace manner,
             # we still first detach the old resource so that
             # self.resource_id is not replaced prematurely
             volume_id = self.properties[self.VOLUME_ID]
+            server_id = self._stored_properties_data.get(self.INSTANCE_ID)
+            self.client_plugin('nova').detach_volume(server_id,
+                                                     self.resource_id)
+            prg_detach = heat_cinder.VolumeDetachProgress(
+                server_id, volume_id, self.resource_id)
+            prg_detach.called = True
+
             if self.VOLUME_ID in prop_diff:
                 volume_id = prop_diff.get(self.VOLUME_ID)
 
@@ -477,29 +546,36 @@ class CinderVolumeAttachment(vb.BaseVolumeAttachment):
             if self.DEVICE in prop_diff:
                 device = prop_diff.get(self.DEVICE)
 
-            server_id = self._stored_properties_data.get(self.INSTANCE_ID)
-            detach_task = vol_task.VolumeDetachTask(
-                self.stack, server_id, self.resource_id)
-            checkers.append(scheduler.TaskRunner(detach_task))
-
             if self.INSTANCE_ID in prop_diff:
                 server_id = prop_diff.get(self.INSTANCE_ID)
-            attach_task = vol_task.VolumeAttachTask(
-                self.stack, server_id, volume_id, device)
+            prg_attach = heat_cinder.VolumeAttachProgress(
+                server_id, volume_id, device)
 
-            checkers.append(scheduler.TaskRunner(attach_task))
-
-        if checkers:
-            checkers[0].start()
-        return checkers
+        return prg_detach, prg_attach
 
     def check_update_complete(self, checkers):
-        for checker in checkers:
-            if not checker.started():
-                checker.start()
-            if not checker.step():
-                return False
-        self.resource_id_set(checkers[-1]._task.attachment_id)
+        prg_detach, prg_attach = checkers
+        if not (prg_detach and prg_attach):
+            return True
+        if not prg_detach.cinder_complete:
+            prg_detach.cinder_complete = self.client_plugin(
+            ).check_detach_volume_complete(prg_detach.vol_id)
+            return False
+        if not prg_detach.nova_complete:
+            prg_detach.nova_complete = self.client_plugin(
+                'nova').check_detach_volume_complete(prg_detach.srv_id,
+                                                     self.resource_id)
+            return False
+        if not prg_attach.called:
+            prg_attach.called = self.client_plugin('nova').attach_volume(
+                prg_attach.srv_id, prg_attach.vol_id, prg_attach.device)
+            return False
+        if not prg_attach.complete:
+            prg_attach.complete = self.client_plugin(
+            ).check_attach_volume_complete(prg_attach.vol_id)
+            if prg_attach.complete:
+                self.resource_id_set(prg_attach.called)
+            return prg_attach.complete
         return True
 
 
diff --git a/heat/engine/resources/volume_base.py b/heat/engine/resources/volume_base.py
index ed4aeaa7e9..7172735010 100644
--- a/heat/engine/resources/volume_base.py
+++ b/heat/engine/resources/volume_base.py
@@ -13,9 +13,8 @@
 
 from heat.common import exception
 from heat.common.i18n import _
+from heat.engine.clients.os import cinder as heat_cinder
 from heat.engine import resource
-from heat.engine import scheduler
-from heat.engine import volume_tasks as vol_task
 
 
 class BaseVolume(resource.Resource):
@@ -82,55 +81,80 @@ class BaseVolume(resource.Resource):
         ]
         self._verify_check_conditions(checks)
 
-    def _backup(self):
-        cinder = self.client()
-        backup = cinder.backups.create(self.resource_id)
-        while backup.status == 'creating':
-            yield
-            backup = cinder.backups.get(backup.id)
-        if backup.status != 'available':
+    def handle_snapshot_delete(self, state):
+        backup = state not in ((self.CREATE, self.FAILED),
+                               (self.UPDATE, self.FAILED))
+        progress = heat_cinder.VolumeDeleteProgress()
+        progress.backup['called'] = not backup
+        progress.backup['complete'] = not backup
+        return progress
+
+    def handle_delete(self):
+        if self.resource_id is None:
+            return heat_cinder.VolumeDeleteProgress(True)
+        progress = heat_cinder.VolumeDeleteProgress()
+        progress.backup['called'] = True
+        progress.backup['complete'] = True
+        return progress
+
+    def _create_backup(self):
+        backup = self.client().backups.create(self.resource_id)
+        return backup.id
+
+    def _check_create_backup_complete(self, prg):
+        backup = self.client().backups.get(prg.backup_id)
+        if backup.status == 'creating':
+            return False
+        if backup.status == 'available':
+            return True
+        else:
             raise resource.ResourceUnknownStatus(
                 resource_status=backup.status,
                 result=_('Volume backup failed'))
 
-    @scheduler.wrappertask
-    def _delete(self, backup=False):
-        if self.resource_id is not None:
+    def _delete_volume(self):
+        try:
             cinder = self.client()
+            vol = cinder.volumes.get(self.resource_id)
+            if vol.status == 'in-use':
+                raise exception.Error(_('Volume in use'))
+            # if the volume is already in deleting status,
+            # just wait for the deletion to complete
+            if vol.status != 'deleting':
+                cinder.volumes.delete(self.resource_id)
+            else:
+                return True
+        except Exception as ex:
+            self.client_plugin().ignore_not_found(ex)
+            return True
+        else:
+            return False
+
+    def check_delete_complete(self, prg):
+        if not prg.backup['called']:
+            prg.backup_id = self._create_backup()
+            prg.backup['called'] = True
+            return False
+
+        if not prg.backup['complete']:
+            prg.backup['complete'] = self._check_create_backup_complete(prg)
+            return False
+
+        if not prg.delete['called']:
+            prg.delete['complete'] = self._delete_volume()
+            prg.delete['called'] = True
+            return False
+
+        if not prg.delete['complete']:
             try:
-                vol = cinder.volumes.get(self.resource_id)
-
-                if backup:
-                    yield self._backup()
-                    vol = cinder.volumes.get(self.resource_id)
-
-                if vol.status == 'in-use':
-                    raise exception.Error(_('Volume in use'))
-                # if the volume is already in deleting status,
-                # just wait for the deletion to complete
-                if vol.status != 'deleting':
-                    cinder.volumes.delete(self.resource_id)
-                while True:
-                    yield
-                    vol = cinder.volumes.get(self.resource_id)
+                self.client().volumes.get(self.resource_id)
             except Exception as ex:
                 self.client_plugin().ignore_not_found(ex)
-
-    def handle_snapshot_delete(self, state):
-        backup = state not in ((self.CREATE, self.FAILED),
-                               (self.UPDATE, self.FAILED))
-
-        delete_task = scheduler.TaskRunner(self._delete, backup=backup)
-        delete_task.start()
-        return delete_task
-
-    def handle_delete(self):
-        delete_task = scheduler.TaskRunner(self._delete)
-        delete_task.start()
-        return delete_task
-
-    def check_delete_complete(self, delete_task):
-        return delete_task.step()
+                prg.delete['complete'] = True
+                return True
+            else:
+                return False
+        return True
 
 
 class BaseVolumeAttachment(resource.Resource):
@@ -138,31 +162,41 @@ class BaseVolumeAttachment(resource.Resource):
     Base Volume Attachment Manager.
     '''
 
+    default_client_name = 'cinder'
+
     def handle_create(self):
         server_id = self.properties[self.INSTANCE_ID]
         volume_id = self.properties[self.VOLUME_ID]
         dev = self.properties[self.DEVICE]
 
-        attach_task = vol_task.VolumeAttachTask(
-            self.stack, server_id, volume_id, dev)
-        attach_runner = scheduler.TaskRunner(attach_task)
+        attach_id = self.client_plugin('nova').attach_volume(
+            server_id, volume_id, dev)
 
-        attach_runner.start()
+        self.resource_id_set(attach_id)
 
-        self.resource_id_set(attach_task.attachment_id)
+        return volume_id
 
-        return attach_runner
-
-    def check_create_complete(self, attach_runner):
-        return attach_runner.step()
+    def check_create_complete(self, volume_id):
+        return self.client_plugin().check_attach_volume_complete(volume_id)
 
     def handle_delete(self):
         server_id = self.properties[self.INSTANCE_ID]
-        detach_task = vol_task.VolumeDetachTask(
-            self.stack, server_id, self.resource_id)
-        detach_runner = scheduler.TaskRunner(detach_task)
-        detach_runner.start()
-        return detach_runner
+        vol_id = self.properties[self.VOLUME_ID]
+        self.client_plugin('nova').detach_volume(server_id,
+                                                 self.resource_id)
+        prg = heat_cinder.VolumeDetachProgress(
+            server_id, vol_id, self.resource_id)
+        prg.called = True
+        return prg
 
-    def check_delete_complete(self, detach_runner):
-        return detach_runner.step()
+    def check_delete_complete(self, prg):
+        if not prg.cinder_complete:
+            prg.cinder_complete = self.client_plugin(
+            ).check_detach_volume_complete(prg.vol_id)
+            return False
+        if not prg.nova_complete:
+            prg.nova_complete = self.client_plugin(
+                'nova').check_detach_volume_complete(prg.srv_id,
+                                                     prg.attach_id)
+            return prg.nova_complete
+        return True
diff --git a/heat/engine/volume_tasks.py b/heat/engine/volume_tasks.py
index 0a0c4f4947..afe709ce80 100644
--- a/heat/engine/volume_tasks.py
+++ b/heat/engine/volume_tasks.py
@@ -13,7 +13,6 @@
 
 from oslo_log import log as logging
 
-from heat.common import exception
 from heat.common.i18n import _
 from heat.common.i18n import _LI
 from heat.engine import resource
@@ -21,56 +20,6 @@ from heat.engine import resource
 LOG = logging.getLogger(__name__)
 
 
-class VolumeExtendTask(object):
-    """A task to resize volume using Cinder API."""
-
-    def __init__(self, stack, volume_id, size):
-        self.clients = stack.clients
-        self.volume_id = volume_id
-        self.size = size
-
-    def __str__(self):
-        return _("Resizing volume %(vol)s to size %(size)i") % {
-            'vol': self.volume_id, 'size': self.size}
-
-    def __repr__(self):
-        return "%s(%s +-> %i)" % (type(self).__name__, self.volume_id,
-                                  self.size)
-
-    def __call__(self):
-        LOG.debug(str(self))
-
-        cinder = self.clients.client('cinder').volumes
-        vol = cinder.get(self.volume_id)
-
-        try:
-            cinder.extend(self.volume_id, self.size)
-        except Exception as ex:
-            if self.clients.client_plugin('cinder').is_client_exception(ex):
-                raise exception.Error(_(
-                    "Failed to extend volume %(vol)s - %(err)s") % {
-                        'vol': vol.id, 'err': ex})
-            else:
-                raise
-
-        yield
-
-        vol = cinder.get(self.volume_id)
-        while vol.status == 'extending':
-            LOG.debug("Volume %s is being extended" % self.volume_id)
-            yield
-            vol = cinder.get(self.volume_id)
-
-        if vol.status != 'available':
-            LOG.info(_LI("Resize failed: Volume %(vol)s is in %(status)s "
-                         "state."), {'vol': vol.id, 'status': vol.status})
-            raise resource.ResourceUnknownStatus(
-                resource_status=vol.status,
-                result=_('Volume resize failed'))
-
-        LOG.info(_LI('%s - complete'), str(self))
-
-
 class VolumeAttachTask(object):
     """A task for attaching a volume to a Nova server."""
 
@@ -128,102 +77,3 @@ class VolumeAttachTask(object):
                 result=_('Volume attachment failed'))
 
         LOG.info(_LI('%s - complete'), str(self))
-
-
-class VolumeDetachTask(object):
-    """A task for detaching a volume from a Nova server."""
-
-    def __init__(self, stack, server_id, attachment_id):
-        """
-        Initialise with the stack (for obtaining the clients), and the IDs of
-        the server and volume.
-        """
-        self.clients = stack.clients
-        self.server_id = server_id
-        self.attachment_id = attachment_id
-
-    def __str__(self):
-        """Return a human-readable string description of the task."""
-        return _('Removing attachment %(att)s from Instance %(srv)s') % {
-            'att': self.attachment_id, 'srv': self.server_id}
-
-    def __repr__(self):
-        """Return a brief string description of the task."""
-        return '%s(%s -/> %s)' % (type(self).__name__,
-                                  self.attachment_id,
-                                  self.server_id)
-
-    def __call__(self):
-        """Return a co-routine which runs the task."""
-        LOG.debug(str(self))
-
-        nova_plugin = self.clients.client_plugin('nova')
-        cinder_plugin = self.clients.client_plugin('cinder')
-        server_api = self.clients.client('nova').volumes
-        cinder = self.clients.client('cinder')
-        # get reference to the volume while it is attached
-        try:
-            nova_vol = server_api.get_server_volume(self.server_id,
-                                                    self.attachment_id)
-            vol = cinder.volumes.get(nova_vol.id)
-        except Exception as ex:
-            if (cinder_plugin.is_not_found(ex) or
-                    nova_plugin.is_not_found(ex) or
-                    nova_plugin.is_bad_request(ex)):
-                return
-            else:
-                raise
-
-        if vol.status == 'deleting':
-            return
-
-        # detach the volume using volume_attachment
-        try:
-            server_api.delete_server_volume(self.server_id, self.attachment_id)
-        except Exception as ex:
-            if nova_plugin.is_not_found(ex) or nova_plugin.is_bad_request(ex):
-                pass
-            else:
-                raise
-
-        yield
-
-        try:
-            while vol.status in ('in-use', 'detaching'):
-                LOG.debug('%s - volume still in use' % str(self))
-                yield
-                vol = cinder.volumes.get(nova_vol.id)
-
-            LOG.info(_LI('%(name)s - status: %(status)s'),
-                     {'name': str(self), 'status': vol.status})
-            if vol.status not in ['available', 'deleting']:
-                LOG.info(_LI("Detachment failed - volume %(vol)s "
-                             "is in %(status)s status"),
-                         {"vol": vol.id,
-                          "status": vol.status})
-                raise resource.ResourceUnknownStatus(
-                    resource_status=vol.status,
-                    result=_('Volume detachment failed'))
-
-        except Exception as ex:
-            cinder_plugin.ignore_not_found(ex)
-
-        # The next check is needed for immediate reattachment when updating:
-        # there might be some time between cinder marking volume as 'available'
-        # and nova removing attachment from its own objects, so we
-        # check that nova already knows that the volume is detached
-        def server_has_attachment(server_id, attachment_id):
-            try:
-                server_api.get_server_volume(server_id, attachment_id)
-            except Exception as ex:
-                nova_plugin.ignore_not_found(ex)
-                return False
-            return True
-
-        while server_has_attachment(self.server_id, self.attachment_id):
-            LOG.info(_LI("Server %(srv)s still has attachment %(att)s."),
-                     {'att': self.attachment_id, 'srv': self.server_id})
-            yield
-
-        LOG.info(_LI("Volume %(vol)s is detached from server %(srv)s"),
-                 {'vol': vol.id, 'srv': self.server_id})
diff --git a/heat/tests/aws/test_volume.py b/heat/tests/aws/test_volume.py
index 892c7f3f04..acd43afcad 100644
--- a/heat/tests/aws/test_volume.py
+++ b/heat/tests/aws/test_volume.py
@@ -308,10 +308,13 @@ class VolumeTest(vt_base.BaseVolumeTest):
         self._mock_create_server_volume_script(fva)
         self.stub_VolumeConstraint_validate()
         # delete script
-        self.fc.volumes.get_server_volume(u'WikiDatabase',
-                                          'vol-123').AndReturn(fva)
+        self.fc.volumes.delete_server_volume(u'WikiDatabase',
+                                             'vol-123').AndReturn(None)
         self.cinder_fc.volumes.get(fva.id).AndRaise(
             cinder_exp.NotFound('Not found'))
+        self.fc.volumes.get_server_volume(u'WikiDatabase', 'vol-123'
+                                          ).AndRaise(
+                                              fakes_nova.fake_exception())
 
         self.m.ReplayAll()
 
@@ -333,9 +336,12 @@ class VolumeTest(vt_base.BaseVolumeTest):
         self._mock_create_server_volume_script(fva)
         self.stub_VolumeConstraint_validate()
         # delete script
-        self.fc.volumes.get_server_volume(u'WikiDatabase',
-                                          'vol-123').AndReturn(fva)
+        self.fc.volumes.delete_server_volume(u'WikiDatabase',
+                                             'vol-123').AndReturn(None)
         self.cinder_fc.volumes.get(fva.id).AndReturn(fva)
+        self.fc.volumes.get_server_volume(u'WikiDatabase', 'vol-123'
+                                          ).AndRaise(
+                                              fakes_nova.fake_exception())
 
         self.m.ReplayAll()
 
@@ -395,11 +401,8 @@ class VolumeTest(vt_base.BaseVolumeTest):
         self.stub_VolumeConstraint_validate()
         # delete script
         fva = vt_base.FakeVolume('in-use')
-        self.fc.volumes.get_server_volume(u'WikiDatabase',
-                                          'vol-123').AndReturn(fva)
-        self.cinder_fc.volumes.get(fva.id).AndReturn(fva)
         self.fc.volumes.delete_server_volume(
-            'WikiDatabase', 'vol-123').MultipleTimes().AndReturn(None)
+            'WikiDatabase', 'vol-123').AndReturn(None)
         self.cinder_fc.volumes.get(fva.id).AndReturn(
             vt_base.FakeVolume('error', id=fva.id))
         self.m.ReplayAll()
@@ -444,13 +447,9 @@ class VolumeTest(vt_base.BaseVolumeTest):
         fv = self._mock_create_volume(vt_base.FakeVolume('creating'),
                                       stack_name)
 
-        self.cinder_fc.volumes.get(fv.id).AndReturn(
-            vt_base.FakeVolume('available'))
-        self.cinder_fc.volumes.delete(fv.id).AndReturn(True)
         self.cinder_fc.volumes.get(fv.id).AndReturn(
             vt_base.FakeVolume('deleting'))
-        self.cinder_fc.volumes.get(fv.id).AndRaise(
-            cinder_exp.NotFound('Not found'))
+
         self.m.ReplayAll()
 
         stack = utils.parse_stack(self.t, stack_name=stack_name)
@@ -533,8 +532,10 @@ class VolumeTest(vt_base.BaseVolumeTest):
 
         # snapshot script
         self.m.StubOutWithMock(self.cinder_fc.backups, 'create')
-        self.cinder_fc.backups.create(fv.id).AndReturn(
-            vt_base.FakeBackup('available'))
+        self.m.StubOutWithMock(self.cinder_fc.backups, 'get')
+        fb = vt_base.FakeBackup('available')
+        self.cinder_fc.backups.create(fv.id).AndReturn(fb)
+        self.cinder_fc.backups.get(fb.id).AndReturn(fb)
         self.cinder_fc.volumes.get(fv.id).AndReturn(fv)
         self._mock_delete_volume(fv)
 
@@ -555,10 +556,11 @@ class VolumeTest(vt_base.BaseVolumeTest):
                                       stack_name)
 
         # snapshot script
-        self.cinder_fc.volumes.get(fv.id).AndReturn(fv)
         self.m.StubOutWithMock(self.cinder_fc.backups, 'create')
+        self.m.StubOutWithMock(self.cinder_fc.backups, 'get')
         fb = vt_base.FakeBackup('error')
         self.cinder_fc.backups.create(fv.id).AndReturn(fb)
+        self.cinder_fc.backups.get(fb.id).AndReturn(fb)
         self.m.ReplayAll()
 
         self.t['Resources']['DataVolume']['DeletionPolicy'] = 'Snapshot'
diff --git a/heat/tests/openstack/test_volume.py b/heat/tests/openstack/test_volume.py
index b905faf7c0..0bd1e87470 100644
--- a/heat/tests/openstack/test_volume.py
+++ b/heat/tests/openstack/test_volume.py
@@ -337,7 +337,6 @@ class CinderVolumeTest(vt_base.BaseVolumeTest):
         fv = vt_base.FakeVolume('available',
                                 size=1, attachments=[])
         self.cinder_fc.volumes.get(fv.id).AndReturn(fv)
-        self.cinder_fc.volumes.get(fv.id).AndReturn(fv)
         self.cinder_fc.volumes.extend(fv.id, 2)
         self.cinder_fc.volumes.get(fv.id).AndReturn(
             vt_base.FakeVolume('extending'))
@@ -371,7 +370,6 @@ class CinderVolumeTest(vt_base.BaseVolumeTest):
         fv = vt_base.FakeVolume('available',
                                 size=1, attachments=[])
         self.cinder_fc.volumes.get(fv.id).AndReturn(fv)
-        self.cinder_fc.volumes.get(fv.id).AndReturn(fv)
         self.cinder_fc.volumes.extend(fv.id, 2).AndRaise(
             cinder_exp.OverLimit(413))
         self.m.ReplayAll()
@@ -400,7 +398,6 @@ class CinderVolumeTest(vt_base.BaseVolumeTest):
         fv = vt_base.FakeVolume('available',
                                 size=1, attachments=[])
         self.cinder_fc.volumes.get(fv.id).AndReturn(fv)
-        self.cinder_fc.volumes.get(fv.id).AndReturn(fv)
         self.cinder_fc.volumes.extend(fv.id, 2)
         self.cinder_fc.volumes.get(fv.id).AndReturn(
             vt_base.FakeVolume('extending'))