61b638e722
This patch adds REST APIs for storage pools and pool flavors. And this includes AngularJS services for these REST APIs. Change-Id: Ideb3847f6192afb9c99891f8bf4d68da11988214 Partial-Implements: blueprint support-pool Partial-Implements: blueprint support-pool-flavor
317 lines
10 KiB
Python
317 lines
10 KiB
Python
# Copyright 2015 Cisco Systems.
|
|
#
|
|
# 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.
|
|
|
|
import six
|
|
import yaml
|
|
|
|
from django.views import generic
|
|
from openstack_dashboard.api.rest import urls
|
|
from openstack_dashboard.api.rest import utils as rest_utils
|
|
from zaqar_ui.api import zaqar
|
|
|
|
|
|
def _convert_to_yaml(data, default_flow_style=False):
|
|
if not data:
|
|
return ''
|
|
try:
|
|
return yaml.safe_dump(data, default_flow_style=default_flow_style)
|
|
except Exception:
|
|
return ''
|
|
|
|
|
|
def _load_yaml(data):
|
|
if not data:
|
|
loaded_data = {}
|
|
else:
|
|
try:
|
|
loaded_data = yaml.load(data)
|
|
except Exception as ex:
|
|
raise Exception(_('The specified input is not a valid '
|
|
'YAML format: %s') % six.text_type(ex))
|
|
return loaded_data
|
|
|
|
|
|
@urls.register
|
|
class Queue(generic.View):
|
|
"""API for retrieving a single queue"""
|
|
url_regex = r'zaqar/queue/(?P<queue_name>[^/]+)$'
|
|
|
|
@rest_utils.ajax()
|
|
def get(self, request, queue_name):
|
|
"""Get a specific queue"""
|
|
return zaqar.queue_get(request, queue_name).to_dict()
|
|
|
|
@rest_utils.ajax(data_required=True)
|
|
def post(self, request, queue_name):
|
|
"""Update a queue.
|
|
|
|
Returns the updated queue object on success.
|
|
"""
|
|
queue = zaqar.queue_update(request, queue_name, **request.DATA)
|
|
location = '/api/zaqar/queue/%s' % queue._name
|
|
response = {'name': queue._name,
|
|
'metadata': queue._metadata}
|
|
return rest_utils.CreatedResponse(location, response)
|
|
|
|
|
|
@urls.register
|
|
class Queues(generic.View):
|
|
"""API for queues"""
|
|
url_regex = r'zaqar/queues/$'
|
|
|
|
@rest_utils.ajax()
|
|
def get(self, request):
|
|
"""Get a list of the Queues for a project.
|
|
|
|
The returned result is an object with property 'items' and each
|
|
item under this is a queue.
|
|
"""
|
|
result = zaqar.queue_list(request)
|
|
queues = []
|
|
for q in result:
|
|
stats = q.stats['messages']
|
|
queues.append({'name': q.name,
|
|
'claimed': stats['claimed'],
|
|
'free': stats['free'],
|
|
'total': stats['total'],
|
|
'metadata': q.metadata()})
|
|
return queues
|
|
|
|
@rest_utils.ajax(data_required=True)
|
|
def delete(self, request):
|
|
"""Delete one or more queue by name.
|
|
|
|
Returns HTTP 204 (no content) on successful deletion.
|
|
"""
|
|
for queue_name in request.DATA:
|
|
zaqar.queue_delete(request, queue_name)
|
|
|
|
@rest_utils.ajax(data_required=True)
|
|
def put(self, request):
|
|
"""Create a new queue.
|
|
|
|
Returns the new queue object on success.
|
|
"""
|
|
new_queue = zaqar.queue_create(request, **request.DATA)
|
|
location = '/api/zaqar/queues/%s' % new_queue.name
|
|
response = {'name': new_queue.name,
|
|
'claimed': 0,
|
|
'free': 0,
|
|
'total': 0,
|
|
'metadata': new_queue._metadata}
|
|
return rest_utils.CreatedResponse(location, response)
|
|
|
|
|
|
@urls.register
|
|
class Subscriptions(generic.View):
|
|
"""API for queues"""
|
|
url_regex = r'zaqar/queues/(?P<queue_name>[^/]+)/subscriptions/$'
|
|
|
|
@rest_utils.ajax()
|
|
def get(self, request, queue_name):
|
|
"""Get a list of the Subscriptions for a queue."""
|
|
return zaqar.subscription_list(request, queue_name)
|
|
|
|
@rest_utils.ajax(data_required=True)
|
|
def delete(self, request, queue_name):
|
|
"""Delete one or more queue by name.
|
|
|
|
Returns HTTP 204 (no content) on successful deletion.
|
|
"""
|
|
zaqar.subscription_delete(request, queue_name, request.DATA)
|
|
|
|
@rest_utils.ajax(data_required=True)
|
|
def put(self, request, queue_name):
|
|
"""Create a new subscription.
|
|
|
|
Returns the new queue object on success.
|
|
"""
|
|
return zaqar.subscription_create(request, queue_name, request.DATA)
|
|
|
|
|
|
@urls.register
|
|
class Subscription(generic.View):
|
|
"""API for retrieving a single subscription"""
|
|
url_regex = r'zaqar/queues/(?P<queue_name>[^/]+)/' \
|
|
r'subscription/(?P<subscriber>[^/]+)/$'
|
|
|
|
@rest_utils.ajax(data_required=True)
|
|
def post(self, request, queue_name, subscriber):
|
|
zaqar.subscription_update(request, queue_name,
|
|
{'id': subscriber}, request.DATA)
|
|
|
|
|
|
@urls.register
|
|
class Pool(generic.View):
|
|
"""API for retrieving a single pool"""
|
|
url_regex = r'zaqar/pools/(?P<pool_name>[^/]+)$'
|
|
|
|
@rest_utils.ajax()
|
|
def get(self, request, pool_name):
|
|
"""Get a specific pool"""
|
|
pool = zaqar.pool_get(request, pool_name)
|
|
pool['id'] = pool.get('name')
|
|
pool['options'] = _convert_to_yaml(pool.get('options'))
|
|
return pool
|
|
|
|
@rest_utils.ajax(data_required=True)
|
|
def post(self, request, pool_name):
|
|
"""Update a pool.
|
|
|
|
Returns the updated pool object on success.
|
|
"""
|
|
request.DATA["options"] = _load_yaml(request.DATA.get("options"))
|
|
params = request.DATA
|
|
pool_name = params.pop('name')
|
|
new_pool = zaqar.pool_update(request, pool_name, params)
|
|
location = '/api/zaqar/pools/%s' % new_pool.name
|
|
response = {'name': new_pool.name,
|
|
'uri': new_pool.uri,
|
|
'weight': new_pool.weight,
|
|
'group': new_pool.group,
|
|
'options': new_pool.options}
|
|
return rest_utils.CreatedResponse(location, response)
|
|
|
|
|
|
@urls.register
|
|
class Pools(generic.View):
|
|
"""API for pools"""
|
|
url_regex = r'zaqar/pools/$'
|
|
|
|
@rest_utils.ajax()
|
|
def get(self, request):
|
|
"""Get a list of the Pools for admin.
|
|
|
|
The returned result is an object with property 'items' and each
|
|
item under this is a pool.
|
|
"""
|
|
result = zaqar.pool_list(request)
|
|
pools = []
|
|
for p in result:
|
|
options = _convert_to_yaml(p.options)
|
|
pools.append({'id': p.name,
|
|
'name': p.name,
|
|
'uri': p.uri,
|
|
'weight': p.weight,
|
|
'group': p.group,
|
|
'options': options})
|
|
return {'items': pools}
|
|
|
|
@rest_utils.ajax(data_required=True)
|
|
def delete(self, request):
|
|
"""Delete one or more pool by name.
|
|
|
|
Returns HTTP 204 (no content) on successful deletion.
|
|
"""
|
|
for pool_name in request.DATA:
|
|
zaqar.pool_delete(request, pool_name)
|
|
|
|
@rest_utils.ajax(data_required=True)
|
|
def put(self, request):
|
|
"""Create a new pool.
|
|
|
|
Returns the new pool object on success.
|
|
"""
|
|
request.DATA['options'] = _load_yaml(request.DATA.get('options'))
|
|
params = request.DATA
|
|
pool_name = params.pop('name')
|
|
new_pool = zaqar.pool_create(request, pool_name, params)
|
|
location = '/api/zaqar/pools/%s' % new_pool.name
|
|
response = {'name': new_pool.name,
|
|
'uri': new_pool.uri,
|
|
'weight': new_pool.weight,
|
|
'group': new_pool.group,
|
|
'options': new_pool.options}
|
|
return rest_utils.CreatedResponse(location, response)
|
|
|
|
|
|
@urls.register
|
|
class Flavor(generic.View):
|
|
"""API for retrieving a single flavor"""
|
|
url_regex = r'zaqar/flavors/(?P<flavor_name>[^/]+)$'
|
|
|
|
@rest_utils.ajax()
|
|
def get(self, request, flavor_name):
|
|
"""Get a specific flavor"""
|
|
flavor = zaqar.flavor_get(request, flavor_name)
|
|
flavor['id'] = flavor.get('name')
|
|
flavor['capabilities'] = _convert_to_yaml(flavor.get('capabilities'))
|
|
return flavor
|
|
|
|
@rest_utils.ajax(data_required=True)
|
|
def post(self, request, flavor_name):
|
|
"""Update a flavor.
|
|
|
|
Returns the updated flavor object on success.
|
|
"""
|
|
capabilities = request.DATA.get('capabilities')
|
|
request.DATA['capabilities'] = _load_yaml(capabilities)
|
|
params = request.DATA
|
|
flavor_name = params.pop('name')
|
|
new_flavor = zaqar.flavor_update(request, flavor_name, params)
|
|
location = '/api/zaqar/flavors/%s' % new_flavor.name
|
|
response = {'name': new_flavor.name,
|
|
'pool_group': new_flavor.pool_group,
|
|
'capabilities': new_flavor.capabilities}
|
|
return rest_utils.CreatedResponse(location, response)
|
|
|
|
|
|
@urls.register
|
|
class Flavors(generic.View):
|
|
"""API for flavors"""
|
|
url_regex = r'zaqar/flavors/$'
|
|
|
|
@rest_utils.ajax()
|
|
def get(self, request):
|
|
"""Get a list of the Flavors for admin.
|
|
|
|
The returned result is an object with property 'items' and each
|
|
item under this is a flavor.
|
|
"""
|
|
result = zaqar.flavor_list(request)
|
|
flavors = []
|
|
for f in result:
|
|
capabilities = _convert_to_yaml(f.capabilities)
|
|
flavors.append({'id': f.name,
|
|
'name': f.name,
|
|
'pool_group': f.pool_group,
|
|
'capabilities': capabilities})
|
|
return {'items': flavors}
|
|
|
|
@rest_utils.ajax(data_required=True)
|
|
def delete(self, request):
|
|
"""Delete one or more flavor by name.
|
|
|
|
Returns HTTP 204 (no content) on successful deletion.
|
|
"""
|
|
for flavor_name in request.DATA:
|
|
zaqar.flavor_delete(request, flavor_name)
|
|
|
|
@rest_utils.ajax(data_required=True)
|
|
def put(self, request):
|
|
"""Create a new flavor.
|
|
|
|
Returns the new flavor object on success.
|
|
"""
|
|
capabilities = request.DATA.get('capabilities')
|
|
request.DATA['capabilities'] = _load_yaml(capabilities)
|
|
params = request.DATA
|
|
flavor_name = params.pop('name')
|
|
new_flavor = zaqar.flavor_create(request, flavor_name, params)
|
|
location = '/api/zaqar/flavors/%s' % new_flavor.name
|
|
response = {'name': new_flavor.name,
|
|
'pool_group': new_flavor.pool_group,
|
|
'capabilities': new_flavor.capabilities}
|
|
return rest_utils.CreatedResponse(location, response)
|