¡@

Home 

OpenStack Study: backups.py

OpenStack Index

**** CubicPower OpenStack Study ****

# Copyright (C) 2012 Hewlett-Packard Development Company, L.P.

# All Rights Reserved.

#

# 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.

"""The backups api."""

import webob

from webob import exc

from cinder.api import common

from cinder.api import extensions

from cinder.api.openstack import wsgi

from cinder.api.views import backups as backup_views

from cinder.api import xmlutil

from cinder import backup as backupAPI

from cinder import exception

from cinder.openstack.common import log as logging

from cinder import utils

LOG = logging.getLogger(__name__)

**** CubicPower OpenStack Study ****

def make_backup(elem):

    elem.set('id')

    elem.set('status')

    elem.set('size')

    elem.set('container')

    elem.set('volume_id')

    elem.set('object_count')

    elem.set('availability_zone')

    elem.set('created_at')

    elem.set('name')

    elem.set('description')

    elem.set('fail_reason')

**** CubicPower OpenStack Study ****

def make_backup_restore(elem):

    elem.set('backup_id')

    elem.set('volume_id')

**** CubicPower OpenStack Study ****

def make_backup_export_import_record(elem):

    elem.set('backup_service')

    elem.set('backup_url')

**** CubicPower OpenStack Study ****

class BackupTemplate(xmlutil.TemplateBuilder):

**** CubicPower OpenStack Study ****

    def construct(self):

        root = xmlutil.TemplateElement('backup', selector='backup')

        make_backup(root)

        alias = Backups.alias

        namespace = Backups.namespace

        return xmlutil.MasterTemplate(root, 1, nsmap={alias: namespace})

**** CubicPower OpenStack Study ****

class BackupsTemplate(xmlutil.TemplateBuilder):

**** CubicPower OpenStack Study ****

    def construct(self):

        root = xmlutil.TemplateElement('backups')

        elem = xmlutil.SubTemplateElement(root, 'backup', selector='backups')

        make_backup(elem)

        alias = Backups.alias

        namespace = Backups.namespace

        return xmlutil.MasterTemplate(root, 1, nsmap={alias: namespace})

**** CubicPower OpenStack Study ****

class BackupRestoreTemplate(xmlutil.TemplateBuilder):

**** CubicPower OpenStack Study ****

    def construct(self):

        root = xmlutil.TemplateElement('restore', selector='restore')

        make_backup_restore(root)

        alias = Backups.alias

        namespace = Backups.namespace

        return xmlutil.MasterTemplate(root, 1, nsmap={alias: namespace})

**** CubicPower OpenStack Study ****

class BackupExportImportTemplate(xmlutil.TemplateBuilder):

**** CubicPower OpenStack Study ****

    def construct(self):

        root = xmlutil.TemplateElement('backup-record',

                                       selector='backup-record')

        make_backup_export_import_record(root)

        alias = Backups.alias

        namespace = Backups.namespace

        return xmlutil.MasterTemplate(root, 1, nsmap={alias: namespace})

**** CubicPower OpenStack Study ****

class CreateDeserializer(wsgi.MetadataXMLDeserializer):

**** CubicPower OpenStack Study ****

    def default(self, string):

        dom = utils.safe_minidom_parse_string(string)

        backup = self._extract_backup(dom)

        return {'body': {'backup': backup}}

**** CubicPower OpenStack Study ****

    def _extract_backup(self, node):

        backup = {}

        backup_node = self.find_first_child_named(node, 'backup')

        attributes = ['container', 'display_name',

                      'display_description', 'volume_id']

        for attr in attributes:

            if backup_node.getAttribute(attr):

                backup[attr] = backup_node.getAttribute(attr)

        return backup

**** CubicPower OpenStack Study ****

class RestoreDeserializer(wsgi.MetadataXMLDeserializer):

**** CubicPower OpenStack Study ****

    def default(self, string):

        dom = utils.safe_minidom_parse_string(string)

        restore = self._extract_restore(dom)

        return {'body': {'restore': restore}}

**** CubicPower OpenStack Study ****

    def _extract_restore(self, node):

        restore = {}

        restore_node = self.find_first_child_named(node, 'restore')

        if restore_node.getAttribute('volume_id'):

            restore['volume_id'] = restore_node.getAttribute('volume_id')

        return restore

**** CubicPower OpenStack Study ****

class BackupImportDeserializer(wsgi.MetadataXMLDeserializer):

**** CubicPower OpenStack Study ****

    def default(self, string):

        dom = utils.safe_minidom_parse_string(string)

        backup = self._extract_backup(dom)

        retval = {'body': {'backup-record': backup}}

        return retval

**** CubicPower OpenStack Study ****

    def _extract_backup(self, node):

        backup = {}

        backup_node = self.find_first_child_named(node, 'backup-record')

        attributes = ['backup_service', 'backup_url']

        for attr in attributes:

            if backup_node.getAttribute(attr):

                backup[attr] = backup_node.getAttribute(attr)

        return backup

**** CubicPower OpenStack Study ****

class BackupsController(wsgi.Controller):

"""The Backups API controller for the OpenStack API."""

_view_builder_class = backup_views.ViewBuilder

**** CubicPower OpenStack Study ****

    def __init__(self):

        self.backup_api = backupAPI.API()

        super(BackupsController, self).__init__()

    @wsgi.serializers(xml=BackupTemplate)

**** CubicPower OpenStack Study ****

    def show(self, req, id):

        """Return data about the given backup."""

        LOG.debug(_('show called for member %s'), id)

        context = req.environ['cinder.context']

        try:

            backup = self.backup_api.get(context, backup_id=id)

        except exception.BackupNotFound as error:

            raise exc.HTTPNotFound(explanation=error.msg)

        return self._view_builder.detail(req, backup)

**** CubicPower OpenStack Study ****

    def delete(self, req, id):

        """Delete a backup."""

        LOG.debug(_('delete called for member %s'), id)

        context = req.environ['cinder.context']

        LOG.audit(_('Delete backup with id: %s'), id, context=context)

        try:

            self.backup_api.delete(context, id)

        except exception.BackupNotFound as error:

            raise exc.HTTPNotFound(explanation=error.msg)

        except exception.InvalidBackup as error:

            raise exc.HTTPBadRequest(explanation=error.msg)

        return webob.Response(status_int=202)

    @wsgi.serializers(xml=BackupsTemplate)

**** CubicPower OpenStack Study ****

    def index(self, req):

        """Returns a summary list of backups."""

        return self._get_backups(req, is_detail=False)

    @wsgi.serializers(xml=BackupsTemplate)

**** CubicPower OpenStack Study ****

    def detail(self, req):

        """Returns a detailed list of backups."""

        return self._get_backups(req, is_detail=True)

**** CubicPower OpenStack Study ****

    def _get_backups(self, req, is_detail):

        """Returns a list of backups, transformed through view builder."""

        context = req.environ['cinder.context']

        backups = self.backup_api.get_all(context)

        limited_list = common.limited(backups, req)

        if is_detail:

            backups = self._view_builder.detail_list(req, limited_list)

        else:

            backups = self._view_builder.summary_list(req, limited_list)

        return backups

    # TODO(frankm): Add some checks here including

    # - whether requested volume_id exists so we can return some errors

    #   immediately

    # - maybe also do validation of swift container name

    @wsgi.response(202)

    @wsgi.serializers(xml=BackupTemplate)

    @wsgi.deserializers(xml=CreateDeserializer)

**** CubicPower OpenStack Study ****

    def create(self, req, body):

        """Create a new backup."""

        LOG.debug(_('Creating new backup %s'), body)

        if not self.is_valid_body(body, 'backup'):

            raise exc.HTTPBadRequest()

        context = req.environ['cinder.context']

        try:

            backup = body['backup']

            volume_id = backup['volume_id']

        except KeyError:

            msg = _("Incorrect request body format")

            raise exc.HTTPBadRequest(explanation=msg)

        container = backup.get('container', None)

        name = backup.get('name', None)

        description = backup.get('description', None)

        LOG.audit(_("Creating backup of volume %(volume_id)s in container"

                    " %(container)s"),

                  {'volume_id': volume_id, 'container': container},

                  context=context)

        try:

            new_backup = self.backup_api.create(context, name, description,

                                                volume_id, container)

        except exception.InvalidVolume as error:

            raise exc.HTTPBadRequest(explanation=error.msg)

        except exception.VolumeNotFound as error:

            raise exc.HTTPNotFound(explanation=error.msg)

        except exception.ServiceNotFound as error:

            raise exc.HTTPInternalServerError(explanation=error.msg)

        retval = self._view_builder.summary(req, dict(new_backup.iteritems()))

        return retval

    @wsgi.response(202)

    @wsgi.serializers(xml=BackupRestoreTemplate)

    @wsgi.deserializers(xml=RestoreDeserializer)

**** CubicPower OpenStack Study ****

    def restore(self, req, id, body):

        """Restore an existing backup to a volume."""

        LOG.debug(_('Restoring backup %(backup_id)s (%(body)s)'),

                  {'backup_id': id, 'body': body})

        if not self.is_valid_body(body, 'restore'):

            msg = _("Incorrect request body format")

            raise exc.HTTPBadRequest(explanation=msg)

        context = req.environ['cinder.context']

        restore = body['restore']

        volume_id = restore.get('volume_id', None)

        LOG.audit(_("Restoring backup %(backup_id)s to volume %(volume_id)s"),

                  {'backup_id': id, 'volume_id': volume_id},

                  context=context)

        try:

            new_restore = self.backup_api.restore(context,

                                                  backup_id=id,

                                                  volume_id=volume_id)

        except exception.InvalidInput as error:

            raise exc.HTTPBadRequest(explanation=error.msg)

        except exception.InvalidVolume as error:

            raise exc.HTTPBadRequest(explanation=error.msg)

        except exception.InvalidBackup as error:

            raise exc.HTTPBadRequest(explanation=error.msg)

        except exception.BackupNotFound as error:

            raise exc.HTTPNotFound(explanation=error.msg)

        except exception.VolumeNotFound as error:

            raise exc.HTTPNotFound(explanation=error.msg)

        except exception.VolumeSizeExceedsAvailableQuota as error:

            raise exc.HTTPRequestEntityTooLarge(

                explanation=error.msg, headers={'Retry-After': 0})

        except exception.VolumeLimitExceeded as error:

            raise exc.HTTPRequestEntityTooLarge(

                explanation=error.msg, headers={'Retry-After': 0})

        retval = self._view_builder.restore_summary(

            req, dict(new_restore.iteritems()))

        return retval

    @wsgi.response(200)

    @wsgi.serializers(xml=BackupExportImportTemplate)

**** CubicPower OpenStack Study ****

    def export_record(self, req, id):

        """Export a backup."""

        LOG.debug(_('export record called for member %s.'), id)

        context = req.environ['cinder.context']

        try:

            backup_info = self.backup_api.export_record(context, id)

        except exception.BackupNotFound as error:

            raise exc.HTTPNotFound(explanation=error.msg)

        except exception.InvalidBackup as error:

            raise exc.HTTPBadRequest(explanation=error.msg)

        retval = self._view_builder.export_summary(

            req, dict(backup_info.iteritems()))

        LOG.debug(_('export record output: %s.'), retval)

        return retval

    @wsgi.response(201)

    @wsgi.serializers(xml=BackupTemplate)

    @wsgi.deserializers(xml=BackupImportDeserializer)

**** CubicPower OpenStack Study ****

    def import_record(self, req, body):

        """Import a backup."""

        LOG.debug(_('Importing record from %s.'), body)

        if not self.is_valid_body(body, 'backup-record'):

            msg = _("Incorrect request body format.")

            raise exc.HTTPBadRequest(explanation=msg)

        context = req.environ['cinder.context']

        import_data = body['backup-record']

        #Verify that body elements are provided

        try:

            backup_service = import_data['backup_service']

            backup_url = import_data['backup_url']

        except KeyError:

            msg = _("Incorrect request body format.")

            raise exc.HTTPBadRequest(explanation=msg)

        LOG.debug(_('Importing backup using %(service)s and url %(url)s.'),

                  {'service': backup_service, 'url': backup_url})

        try:

            new_backup = self.backup_api.import_record(context,

                                                       backup_service,

                                                       backup_url)

        except exception.BackupNotFound as error:

            raise exc.HTTPNotFound(explanation=error.msg)

        except exception.InvalidBackup as error:

            raise exc.HTTPBadRequest(explanation=error.msg)

        except exception.ServiceNotFound as error:

            raise exc.HTTPInternalServerError(explanation=error.msg)

        retval = self._view_builder.summary(req, dict(new_backup.iteritems()))

        LOG.debug(_('import record output: %s.'), retval)

        return retval

**** CubicPower OpenStack Study ****

class Backups(extensions.ExtensionDescriptor):

"""Backups support."""

name = 'Backups'

alias = 'backups'

namespace = 'http://docs.openstack.org/volume/ext/backups/api/v1'

updated = '2012-12-12T00:00:00+00:00'

**** CubicPower OpenStack Study ****

    def get_resources(self):

        resources = []

        res = extensions.ResourceExtension(

            Backups.alias, BackupsController(),

            collection_actions={'detail': 'GET', 'import_record': 'POST'},

            member_actions={'restore': 'POST', 'export_record': 'GET'})

        resources.append(res)

        return resources