mirror of
				https://github.com/django/django.git
				synced 2025-10-25 22:56:12 +00:00 
			
		
		
		
	Fixed #29687 -- Allowed the test client to serialize list/tuple as JSON.
This commit is contained in:
		
							
								
								
									
										1
									
								
								AUTHORS
									
									
									
									
									
								
							
							
						
						
									
										1
									
								
								AUTHORS
									
									
									
									
									
								
							| @@ -206,6 +206,7 @@ answer newbie questions, and generally made Django that much better: | |||||||
|     Daniel Wiesmann <daniel.wiesmann@gmail.com> |     Daniel Wiesmann <daniel.wiesmann@gmail.com> | ||||||
|     Danilo Bargen |     Danilo Bargen | ||||||
|     Dan Johnson <danj.py@gmail.com> |     Dan Johnson <danj.py@gmail.com> | ||||||
|  |     Dan Palmer <dan@danpalmer.me> | ||||||
|     Dan Poirier <poirier@pobox.com> |     Dan Poirier <poirier@pobox.com> | ||||||
|     Dan Stephenson <http://dan.io/> |     Dan Stephenson <http://dan.io/> | ||||||
|     Dan Watson <http://danwatson.net/> |     Dan Watson <http://danwatson.net/> | ||||||
|   | |||||||
| @@ -317,10 +317,10 @@ class RequestFactory: | |||||||
|  |  | ||||||
|     def _encode_json(self, data, content_type): |     def _encode_json(self, data, content_type): | ||||||
|         """ |         """ | ||||||
|         Return encoded JSON if data is a dict and content_type is |         Return encoded JSON if data is a dict, list, or tuple and content_type | ||||||
|         application/json. |         is application/json. | ||||||
|         """ |         """ | ||||||
|         should_encode = JSON_CONTENT_TYPE_RE.match(content_type) and isinstance(data, dict) |         should_encode = JSON_CONTENT_TYPE_RE.match(content_type) and isinstance(data, (dict, list, tuple)) | ||||||
|         return json.dumps(data, cls=self.json_encoder) if should_encode else data |         return json.dumps(data, cls=self.json_encoder) if should_encode else data | ||||||
|  |  | ||||||
|     def _get_path(self, parsed): |     def _get_path(self, parsed): | ||||||
|   | |||||||
| @@ -224,6 +224,10 @@ Tests | |||||||
|   URL, ignoring the ordering of the query string. |   URL, ignoring the ordering of the query string. | ||||||
|   :meth:`~.SimpleTestCase.assertRedirects` uses the new assertion. |   :meth:`~.SimpleTestCase.assertRedirects` uses the new assertion. | ||||||
|  |  | ||||||
|  | * The test :class:`~.django.test.Client` now supports automatic JSON | ||||||
|  |   serialization of list and tuple ``data`` when | ||||||
|  |   ``content_type='application/json'``. | ||||||
|  |  | ||||||
| URLs | URLs | ||||||
| ~~~~ | ~~~~ | ||||||
|  |  | ||||||
|   | |||||||
| @@ -213,10 +213,11 @@ Use the ``django.test.Client`` class to make requests. | |||||||
|  |  | ||||||
|             name=fred&passwd=secret |             name=fred&passwd=secret | ||||||
|  |  | ||||||
|         If you provide ``content_type`` as :mimetype:`application/json`, a |         If you provide ``content_type`` as :mimetype:`application/json`, the | ||||||
|         ``data`` dictionary is serialized using :func:`json.dumps` with |         ``data`` is serialized using :func:`json.dumps` if it's a dict, list, | ||||||
|         :class:`~django.core.serializers.json.DjangoJSONEncoder`. You can |         or tuple. Serialization is performed with | ||||||
|         change the encoder by providing a ``json_encoder`` argument to |         :class:`~django.core.serializers.json.DjangoJSONEncoder` by default, | ||||||
|  |         and can be overridden by providing a ``json_encoder`` argument to | ||||||
|         :class:`Client`. This serialization also happens for :meth:`put`, |         :class:`Client`. This serialization also happens for :meth:`put`, | ||||||
|         :meth:`patch`, and :meth:`delete` requests. |         :meth:`patch`, and :meth:`delete` requests. | ||||||
|  |  | ||||||
| @@ -226,6 +227,11 @@ Use the ``django.test.Client`` class to make requests. | |||||||
|             you can call :func:`json.dumps` on ``data`` before passing it to |             you can call :func:`json.dumps` on ``data`` before passing it to | ||||||
|             ``post()`` to achieve the same thing. |             ``post()`` to achieve the same thing. | ||||||
|  |  | ||||||
|  |         .. versionchanged:: 2.2 | ||||||
|  |  | ||||||
|  |             The JSON serialization was extended to support lists and tuples. In | ||||||
|  |             older versions, only dicts are serialized. | ||||||
|  |  | ||||||
|         If you provide any other ``content_type`` (e.g. :mimetype:`text/xml` |         If you provide any other ``content_type`` (e.g. :mimetype:`text/xml` | ||||||
|         for an XML payload), the contents of ``data`` are sent as-is in the |         for an XML payload), the contents of ``data`` are sent as-is in the | ||||||
|         POST request, using ``content_type`` in the HTTP ``Content-Type`` |         POST request, using ``content_type`` in the HTTP ``Content-Type`` | ||||||
|   | |||||||
| @@ -95,13 +95,20 @@ class ClientTest(TestCase): | |||||||
|     def test_json_serialization(self): |     def test_json_serialization(self): | ||||||
|         """The test client serializes JSON data.""" |         """The test client serializes JSON data.""" | ||||||
|         methods = ('post', 'put', 'patch', 'delete') |         methods = ('post', 'put', 'patch', 'delete') | ||||||
|  |         tests = ( | ||||||
|  |             ({'value': 37}, {'value': 37}), | ||||||
|  |             ([37, True], [37, True]), | ||||||
|  |             ((37, False), [37, False]), | ||||||
|  |         ) | ||||||
|         for method in methods: |         for method in methods: | ||||||
|             with self.subTest(method=method): |             with self.subTest(method=method): | ||||||
|  |                 for data, expected in tests: | ||||||
|  |                     with self.subTest(data): | ||||||
|                         client_method = getattr(self.client, method) |                         client_method = getattr(self.client, method) | ||||||
|                         method_name = method.upper() |                         method_name = method.upper() | ||||||
|                 response = client_method('/json_view/', {'value': 37}, content_type='application/json') |                         response = client_method('/json_view/', data, content_type='application/json') | ||||||
|                         self.assertEqual(response.status_code, 200) |                         self.assertEqual(response.status_code, 200) | ||||||
|                 self.assertEqual(response.context['data'], 37) |                         self.assertEqual(response.context['data'], expected) | ||||||
|                         self.assertContains(response, 'Viewing %s page.' % method_name) |                         self.assertContains(response, 'Viewing %s page.' % method_name) | ||||||
|  |  | ||||||
|     def test_json_encoder_argument(self): |     def test_json_encoder_argument(self): | ||||||
|   | |||||||
| @@ -83,14 +83,14 @@ def post_view(request): | |||||||
| def json_view(request): | def json_view(request): | ||||||
|     """ |     """ | ||||||
|     A view that expects a request with the header 'application/json' and JSON |     A view that expects a request with the header 'application/json' and JSON | ||||||
|     data with a key named 'value'. |     data, which is deserialized and included in the context. | ||||||
|     """ |     """ | ||||||
|     if request.META.get('CONTENT_TYPE') != 'application/json': |     if request.META.get('CONTENT_TYPE') != 'application/json': | ||||||
|         return HttpResponse() |         return HttpResponse() | ||||||
|  |  | ||||||
|     t = Template('Viewing {} page. With data {{ data }}.'.format(request.method)) |     t = Template('Viewing {} page. With data {{ data }}.'.format(request.method)) | ||||||
|     data = json.loads(request.body.decode('utf-8')) |     data = json.loads(request.body.decode('utf-8')) | ||||||
|     c = Context({'data': data['value']}) |     c = Context({'data': data}) | ||||||
|     return HttpResponse(t.render(c)) |     return HttpResponse(t.render(c)) | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user