From 26dca9ff02dc9eb9b431ea66ec60fb828b4ae0a5 Mon Sep 17 00:00:00 2001 From: He Jie Xu Date: Wed, 27 Mar 2013 07:42:20 +0800 Subject: [PATCH] Return unicode for object in json and xml serializer 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 --- quantum/api/v2/resource.py | 4 +- quantum/tests/unit/test_api_v2_resource.py | 94 +++++++++++++++++++--- quantum/tests/unit/test_wsgi.py | 68 ++++++++++++++++ quantum/wsgi.py | 9 ++- 4 files changed, 162 insertions(+), 13 deletions(-) diff --git a/quantum/api/v2/resource.py b/quantum/api/v2/resource.py index be71096302..97fc8a8ada 100644 --- a/quantum/api/v2/resource.py +++ b/quantum/api/v2/resource.py @@ -83,7 +83,7 @@ def Resource(controller, faults=None, deserializers=None, serializers=None): except (exceptions.QuantumException, netaddr.AddrFormatError) as e: LOG.exception(_('%s failed'), action) - body = serializer.serialize({'QuantumError': str(e)}) + body = serializer.serialize({'QuantumError': e}) kwargs = {'body': body, 'content_type': content_type} for fault in faults: if isinstance(e, fault): @@ -91,7 +91,7 @@ def Resource(controller, faults=None, deserializers=None, serializers=None): raise webob.exc.HTTPInternalServerError(**kwargs) except webob.exc.HTTPException as e: LOG.exception(_('%s failed'), action) - e.body = serializer.serialize({'QuantumError': str(e)}) + e.body = serializer.serialize({'QuantumError': e}) e.content_type = content_type raise except Exception as e: diff --git a/quantum/tests/unit/test_api_v2_resource.py b/quantum/tests/unit/test_api_v2_resource.py index 7328c1c6d0..f4580de4d0 100644 --- a/quantum/tests/unit/test_api_v2_resource.py +++ b/quantum/tests/unit/test_api_v2_resource.py @@ -100,27 +100,81 @@ class RequestTestCase(base.BaseTestCase): class ResourceTestCase(base.BaseTestCase): - def test_unmapped_quantum_error(self): + def test_unmapped_quantum_error_with_json(self): + msg = u'\u7f51\u7edc' + + class TestException(q_exc.QuantumException): + message = msg + expected_res = {'body': {'QuantumError': msg}} controller = mock.MagicMock() - controller.test.side_effect = q_exc.QuantumException() + controller.test.side_effect = TestException() resource = webtest.TestApp(wsgi_resource.Resource(controller)) - environ = {'wsgiorg.routing_args': (None, {'action': 'test'})} + environ = {'wsgiorg.routing_args': (None, {'action': 'test', + 'format': 'json'})} res = resource.get('', extra_environ=environ, expect_errors=True) self.assertEqual(res.status_int, exc.HTTPInternalServerError.code) + self.assertEqual(wsgi.JSONDeserializer().deserialize(res.body), + expected_res) - def test_mapped_quantum_error(self): + def test_unmapped_quantum_error_with_xml(self): + msg = u'\u7f51\u7edc' + + class TestException(q_exc.QuantumException): + message = msg + expected_res = {'body': {'QuantumError': msg}} controller = mock.MagicMock() - controller.test.side_effect = q_exc.QuantumException() + controller.test.side_effect = TestException() - faults = {q_exc.QuantumException: exc.HTTPGatewayTimeout} + resource = webtest.TestApp(wsgi_resource.Resource(controller)) + + environ = {'wsgiorg.routing_args': (None, {'action': 'test', + 'format': 'xml'})} + res = resource.get('', extra_environ=environ, expect_errors=True) + self.assertEqual(res.status_int, exc.HTTPInternalServerError.code) + self.assertEqual(wsgi.XMLDeserializer().deserialize(res.body), + expected_res) + + def test_mapped_quantum_error_with_json(self): + msg = u'\u7f51\u7edc' + + class TestException(q_exc.QuantumException): + message = msg + expected_res = {'body': {'QuantumError': msg}} + controller = mock.MagicMock() + controller.test.side_effect = TestException() + + faults = {TestException: exc.HTTPGatewayTimeout} resource = webtest.TestApp(wsgi_resource.Resource(controller, faults=faults)) - environ = {'wsgiorg.routing_args': (None, {'action': 'test'})} + environ = {'wsgiorg.routing_args': (None, {'action': 'test', + 'format': 'json'})} res = resource.get('', extra_environ=environ, expect_errors=True) self.assertEqual(res.status_int, exc.HTTPGatewayTimeout.code) + self.assertEqual(wsgi.JSONDeserializer().deserialize(res.body), + expected_res) + + def test_mapped_quantum_error_with_xml(self): + msg = u'\u7f51\u7edc' + + class TestException(q_exc.QuantumException): + message = msg + expected_res = {'body': {'QuantumError': msg}} + controller = mock.MagicMock() + controller.test.side_effect = TestException() + + faults = {TestException: exc.HTTPGatewayTimeout} + resource = webtest.TestApp(wsgi_resource.Resource(controller, + faults=faults)) + + environ = {'wsgiorg.routing_args': (None, {'action': 'test', + 'format': 'xml'})} + res = resource.get('', extra_environ=environ, expect_errors=True) + self.assertEqual(res.status_int, exc.HTTPGatewayTimeout.code) + self.assertEqual(wsgi.XMLDeserializer().deserialize(res.body), + expected_res) def test_http_error(self): controller = mock.MagicMock() @@ -132,15 +186,37 @@ class ResourceTestCase(base.BaseTestCase): res = resource.get('', extra_environ=environ, expect_errors=True) self.assertEqual(res.status_int, exc.HTTPGatewayTimeout.code) - def test_unhandled_error(self): + def test_unhandled_error_with_json(self): + expected_res = {'body': {'QuantumError': + _('Request Failed: internal server error ' + 'while processing your request.')}} controller = mock.MagicMock() controller.test.side_effect = Exception() resource = webtest.TestApp(wsgi_resource.Resource(controller)) - environ = {'wsgiorg.routing_args': (None, {'action': 'test'})} + environ = {'wsgiorg.routing_args': (None, {'action': 'test', + 'format': 'json'})} res = resource.get('', extra_environ=environ, expect_errors=True) self.assertEqual(res.status_int, exc.HTTPInternalServerError.code) + self.assertEqual(wsgi.JSONDeserializer().deserialize(res.body), + expected_res) + + def test_unhandled_error_with_xml(self): + expected_res = {'body': {'QuantumError': + _('Request Failed: internal server error ' + 'while processing your request.')}} + controller = mock.MagicMock() + controller.test.side_effect = Exception() + + resource = webtest.TestApp(wsgi_resource.Resource(controller)) + + environ = {'wsgiorg.routing_args': (None, {'action': 'test', + 'format': 'xml'})} + res = resource.get('', extra_environ=environ, expect_errors=True) + self.assertEqual(res.status_int, exc.HTTPInternalServerError.code) + self.assertEqual(wsgi.XMLDeserializer().deserialize(res.body), + expected_res) def test_status_200(self): controller = mock.MagicMock() diff --git a/quantum/tests/unit/test_wsgi.py b/quantum/tests/unit/test_wsgi.py index c3d7561023..b775a83eca 100644 --- a/quantum/tests/unit/test_wsgi.py +++ b/quantum/tests/unit/test_wsgi.py @@ -614,6 +614,24 @@ class JSONDictSerializerTest(base.BaseTestCase): self.assertEqual(result, expected_json) + def test_json_with_utf8(self): + input_dict = dict(servers=dict(a=(2, '\xe7\xbd\x91\xe7\xbb\x9c'))) + expected_json = '{"servers":{"a":[2,"\\u7f51\\u7edc"]}}' + serializer = wsgi.JSONDictSerializer() + result = serializer.serialize(input_dict) + result = result.replace('\n', '').replace(' ', '') + + self.assertEqual(result, expected_json) + + def test_json_with_unicode(self): + input_dict = dict(servers=dict(a=(2, u'\u7f51\u7edc'))) + expected_json = '{"servers":{"a":[2,"\\u7f51\\u7edc"]}}' + serializer = wsgi.JSONDictSerializer() + result = serializer.serialize(input_dict) + result = result.replace('\n', '').replace(' ', '') + + self.assertEqual(result, expected_json) + class TextDeserializerTest(base.BaseTestCase): @@ -654,6 +672,20 @@ class JSONDeserializerTest(base.BaseTestCase): self.assertRaises( exception.MalformedRequestBody, deserializer.default, data_string) + def test_json_with_utf8(self): + data = '{"a": "\xe7\xbd\x91\xe7\xbb\x9c"}' + as_dict = {'body': {'a': u'\u7f51\u7edc'}} + deserializer = wsgi.JSONDeserializer() + self.assertEqual( + deserializer.deserialize(data), as_dict) + + def test_json_with_unicode(self): + data = '{"a": "\u7f51\u7edc"}' + as_dict = {'body': {'a': u'\u7f51\u7edc'}} + deserializer = wsgi.JSONDeserializer() + self.assertEqual( + deserializer.deserialize(data), as_dict) + class XMLDeserializerTest(base.BaseTestCase): def test_xml_empty(self): @@ -681,6 +713,14 @@ class XMLDeserializerTest(base.BaseTestCase): self.assertRaises( exception.MalformedRequestBody, deserializer.default, data_string) + def test_xml_with_utf8(self): + xml = '\xe7\xbd\x91\xe7\xbb\x9c' + as_dict = {'body': {'a': u'\u7f51\u7edc'}} + deserializer = wsgi.XMLDeserializer() + + self.assertEqual( + deserializer.deserialize(xml), as_dict) + class RequestHeadersDeserializerTest(base.BaseTestCase): @@ -1044,6 +1084,34 @@ class XMLDictSerializerTest(base.BaseTestCase): result = result.replace('\n', '').replace(' ', '') self.assertEqual(expected, result) + def test_xml_with_utf8(self): + data = {'servers': '\xe7\xbd\x91\xe7\xbb\x9c'} + serializer = wsgi.XMLDictSerializer() + expected = ( + '' + '' + '\xe7\xbd\x91\xe7\xbb\x9c' + ) + result = serializer(data) + result = result.replace('\n', '').replace(' ', '') + self.assertEqual(expected, result) + + def test_xml_with_unicode(self): + data = {'servers': u'\u7f51\u7edc'} + serializer = wsgi.XMLDictSerializer() + expected = ( + '' + '' + '\xe7\xbd\x91\xe7\xbb\x9c' + ) + result = serializer(data) + result = result.replace('\n', '').replace(' ', '') + self.assertEqual(expected, result) + class TestWSGIServerWithSSL(base.BaseTestCase): """WSGI server tests.""" diff --git a/quantum/wsgi.py b/quantum/wsgi.py index 5daec8b584..8558729f85 100644 --- a/quantum/wsgi.py +++ b/quantum/wsgi.py @@ -334,7 +334,9 @@ class JSONDictSerializer(DictSerializer): """Default JSON request body serialization""" def default(self, data): - return jsonutils.dumps(data) + def sanitizer(obj): + return unicode(obj) + return jsonutils.dumps(data, default=sanitizer) class XMLDictSerializer(DictSerializer): @@ -467,7 +469,10 @@ class XMLDictSerializer(DictSerializer): LOG.debug(_("Data %(data)s type is %(type)s"), {'data': data, 'type': type(data)}) - result.text = str(data) + if isinstance(data, str): + result.text = unicode(data, 'utf-8') + else: + result.text = unicode(data) return result def _create_link_nodes(self, xml_doc, links):