26dca9ff02
Fix bug 1160644 Fix bug 1119790 When try to serialize an exception object as json, it failed with 'ValueError: Circular reference detected' that is same error as bug 119790. So this patch fixes bug 119790 too. Change-Id: I0da2616f4ec59da31be054b8be2f7a140f59d63a
118 lines
4.4 KiB
Python
118 lines
4.4 KiB
Python
# Copyright 2012 OpenStack Foundation.
|
|
# 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.
|
|
|
|
"""
|
|
Utility methods for working with WSGI servers redux
|
|
"""
|
|
|
|
import netaddr
|
|
import webob.dec
|
|
import webob.exc
|
|
|
|
from quantum.api.v2 import attributes
|
|
from quantum.common import exceptions
|
|
from quantum.openstack.common import log as logging
|
|
from quantum import wsgi
|
|
|
|
|
|
LOG = logging.getLogger(__name__)
|
|
|
|
|
|
class Request(wsgi.Request):
|
|
pass
|
|
|
|
|
|
def Resource(controller, faults=None, deserializers=None, serializers=None):
|
|
"""Represents an API entity resource and the associated serialization and
|
|
deserialization logic
|
|
"""
|
|
xml_deserializer = wsgi.XMLDeserializer(attributes.get_attr_metadata())
|
|
default_deserializers = {'application/xml': xml_deserializer,
|
|
'application/json': wsgi.JSONDeserializer()}
|
|
xml_serializer = wsgi.XMLDictSerializer(attributes.get_attr_metadata())
|
|
default_serializers = {'application/xml': xml_serializer,
|
|
'application/json': wsgi.JSONDictSerializer()}
|
|
format_types = {'xml': 'application/xml',
|
|
'json': 'application/json'}
|
|
action_status = dict(create=201, delete=204)
|
|
|
|
default_deserializers.update(deserializers or {})
|
|
default_serializers.update(serializers or {})
|
|
|
|
deserializers = default_deserializers
|
|
serializers = default_serializers
|
|
faults = faults or {}
|
|
|
|
@webob.dec.wsgify(RequestClass=Request)
|
|
def resource(request):
|
|
route_args = request.environ.get('wsgiorg.routing_args')
|
|
if route_args:
|
|
args = route_args[1].copy()
|
|
else:
|
|
args = {}
|
|
|
|
# NOTE(jkoelker) by now the controller is already found, remove
|
|
# it from the args if it is in the matchdict
|
|
args.pop('controller', None)
|
|
fmt = args.pop('format', None)
|
|
action = args.pop('action', None)
|
|
content_type = format_types.get(fmt,
|
|
request.best_match_content_type())
|
|
deserializer = deserializers.get(content_type)
|
|
serializer = serializers.get(content_type)
|
|
|
|
try:
|
|
if request.body:
|
|
args['body'] = deserializer.deserialize(request.body)['body']
|
|
|
|
method = getattr(controller, action)
|
|
|
|
result = method(request=request, **args)
|
|
except (exceptions.QuantumException,
|
|
netaddr.AddrFormatError) as e:
|
|
LOG.exception(_('%s failed'), action)
|
|
body = serializer.serialize({'QuantumError': e})
|
|
kwargs = {'body': body, 'content_type': content_type}
|
|
for fault in faults:
|
|
if isinstance(e, fault):
|
|
raise faults[fault](**kwargs)
|
|
raise webob.exc.HTTPInternalServerError(**kwargs)
|
|
except webob.exc.HTTPException as e:
|
|
LOG.exception(_('%s failed'), action)
|
|
e.body = serializer.serialize({'QuantumError': e})
|
|
e.content_type = content_type
|
|
raise
|
|
except Exception as e:
|
|
# NOTE(jkoelker) Everyting else is 500
|
|
LOG.exception(_('%s failed'), action)
|
|
# Do not expose details of 500 error to clients.
|
|
msg = _('Request Failed: internal server error while '
|
|
'processing your request.')
|
|
body = serializer.serialize({'QuantumError': msg})
|
|
kwargs = {'body': body, 'content_type': content_type}
|
|
raise webob.exc.HTTPInternalServerError(**kwargs)
|
|
|
|
status = action_status.get(action, 200)
|
|
body = serializer.serialize(result)
|
|
# NOTE(jkoelker) Comply with RFC2616 section 9.7
|
|
if status == 204:
|
|
content_type = ''
|
|
body = None
|
|
|
|
return webob.Response(request=request, status=status,
|
|
content_type=content_type,
|
|
body=body)
|
|
return resource
|