mirror of
				https://github.com/django/django.git
				synced 2025-10-25 14:46:09 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			277 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			277 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| from datetime import datetime
 | |
| 
 | |
| from django.test import SimpleTestCase, override_settings
 | |
| 
 | |
| FULL_RESPONSE = 'Test conditional get response'
 | |
| LAST_MODIFIED = datetime(2007, 10, 21, 23, 21, 47)
 | |
| LAST_MODIFIED_STR = 'Sun, 21 Oct 2007 23:21:47 GMT'
 | |
| LAST_MODIFIED_NEWER_STR = 'Mon, 18 Oct 2010 16:56:23 GMT'
 | |
| LAST_MODIFIED_INVALID_STR = 'Mon, 32 Oct 2010 16:56:23 GMT'
 | |
| EXPIRED_LAST_MODIFIED_STR = 'Sat, 20 Oct 2007 23:21:47 GMT'
 | |
| ETAG = '"b4246ffc4f62314ca13147c9d4f76974"'
 | |
| WEAK_ETAG = 'W/"b4246ffc4f62314ca13147c9d4f76974"'  # weak match to ETAG
 | |
| EXPIRED_ETAG = '"7fae4cd4b0f81e7d2914700043aa8ed6"'
 | |
| 
 | |
| 
 | |
| @override_settings(ROOT_URLCONF='conditional_processing.urls')
 | |
| class ConditionalGet(SimpleTestCase):
 | |
| 
 | |
|     def assertFullResponse(self, response, check_last_modified=True, check_etag=True):
 | |
|         self.assertEqual(response.status_code, 200)
 | |
|         self.assertEqual(response.content, FULL_RESPONSE.encode())
 | |
|         if response.request['REQUEST_METHOD'] in ('GET', 'HEAD'):
 | |
|             if check_last_modified:
 | |
|                 self.assertEqual(response.headers['Last-Modified'], LAST_MODIFIED_STR)
 | |
|             if check_etag:
 | |
|                 self.assertEqual(response.headers['ETag'], ETAG)
 | |
|         else:
 | |
|             self.assertNotIn('Last-Modified', response.headers)
 | |
|             self.assertNotIn('ETag', response.headers)
 | |
| 
 | |
|     def assertNotModified(self, response):
 | |
|         self.assertEqual(response.status_code, 304)
 | |
|         self.assertEqual(response.content, b'')
 | |
| 
 | |
|     def test_without_conditions(self):
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertFullResponse(response)
 | |
| 
 | |
|     def test_if_modified_since(self):
 | |
|         self.client.defaults['HTTP_IF_MODIFIED_SINCE'] = LAST_MODIFIED_STR
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertNotModified(response)
 | |
|         response = self.client.put('/condition/')
 | |
|         self.assertFullResponse(response)
 | |
|         self.client.defaults['HTTP_IF_MODIFIED_SINCE'] = LAST_MODIFIED_NEWER_STR
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertNotModified(response)
 | |
|         response = self.client.put('/condition/')
 | |
|         self.assertFullResponse(response)
 | |
|         self.client.defaults['HTTP_IF_MODIFIED_SINCE'] = LAST_MODIFIED_INVALID_STR
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertFullResponse(response)
 | |
|         self.client.defaults['HTTP_IF_MODIFIED_SINCE'] = EXPIRED_LAST_MODIFIED_STR
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertFullResponse(response)
 | |
| 
 | |
|     def test_if_unmodified_since(self):
 | |
|         self.client.defaults['HTTP_IF_UNMODIFIED_SINCE'] = LAST_MODIFIED_STR
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertFullResponse(response)
 | |
|         self.client.defaults['HTTP_IF_UNMODIFIED_SINCE'] = LAST_MODIFIED_NEWER_STR
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertFullResponse(response)
 | |
|         self.client.defaults['HTTP_IF_UNMODIFIED_SINCE'] = LAST_MODIFIED_INVALID_STR
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertFullResponse(response)
 | |
|         self.client.defaults['HTTP_IF_UNMODIFIED_SINCE'] = EXPIRED_LAST_MODIFIED_STR
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertEqual(response.status_code, 412)
 | |
| 
 | |
|     def test_if_none_match(self):
 | |
|         self.client.defaults['HTTP_IF_NONE_MATCH'] = ETAG
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertNotModified(response)
 | |
|         response = self.client.put('/condition/')
 | |
|         self.assertEqual(response.status_code, 412)
 | |
|         self.client.defaults['HTTP_IF_NONE_MATCH'] = EXPIRED_ETAG
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertFullResponse(response)
 | |
| 
 | |
|         # Several etags in If-None-Match is a bit exotic but why not?
 | |
|         self.client.defaults['HTTP_IF_NONE_MATCH'] = '%s, %s' % (ETAG, EXPIRED_ETAG)
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertNotModified(response)
 | |
| 
 | |
|     def test_weak_if_none_match(self):
 | |
|         """
 | |
|         If-None-Match comparisons use weak matching, so weak and strong ETags
 | |
|         with the same value result in a 304 response.
 | |
|         """
 | |
|         self.client.defaults['HTTP_IF_NONE_MATCH'] = ETAG
 | |
|         response = self.client.get('/condition/weak_etag/')
 | |
|         self.assertNotModified(response)
 | |
|         response = self.client.put('/condition/weak_etag/')
 | |
|         self.assertEqual(response.status_code, 412)
 | |
| 
 | |
|         self.client.defaults['HTTP_IF_NONE_MATCH'] = WEAK_ETAG
 | |
|         response = self.client.get('/condition/weak_etag/')
 | |
|         self.assertNotModified(response)
 | |
|         response = self.client.put('/condition/weak_etag/')
 | |
|         self.assertEqual(response.status_code, 412)
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertNotModified(response)
 | |
|         response = self.client.put('/condition/')
 | |
|         self.assertEqual(response.status_code, 412)
 | |
| 
 | |
|     def test_all_if_none_match(self):
 | |
|         self.client.defaults['HTTP_IF_NONE_MATCH'] = '*'
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertNotModified(response)
 | |
|         response = self.client.put('/condition/')
 | |
|         self.assertEqual(response.status_code, 412)
 | |
|         response = self.client.get('/condition/no_etag/')
 | |
|         self.assertFullResponse(response, check_last_modified=False, check_etag=False)
 | |
| 
 | |
|     def test_if_match(self):
 | |
|         self.client.defaults['HTTP_IF_MATCH'] = ETAG
 | |
|         response = self.client.put('/condition/')
 | |
|         self.assertFullResponse(response)
 | |
|         self.client.defaults['HTTP_IF_MATCH'] = EXPIRED_ETAG
 | |
|         response = self.client.put('/condition/')
 | |
|         self.assertEqual(response.status_code, 412)
 | |
| 
 | |
|     def test_weak_if_match(self):
 | |
|         """
 | |
|         If-Match comparisons use strong matching, so any comparison involving
 | |
|         a weak ETag return a 412 response.
 | |
|         """
 | |
|         self.client.defaults['HTTP_IF_MATCH'] = ETAG
 | |
|         response = self.client.get('/condition/weak_etag/')
 | |
|         self.assertEqual(response.status_code, 412)
 | |
| 
 | |
|         self.client.defaults['HTTP_IF_MATCH'] = WEAK_ETAG
 | |
|         response = self.client.get('/condition/weak_etag/')
 | |
|         self.assertEqual(response.status_code, 412)
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertEqual(response.status_code, 412)
 | |
| 
 | |
|     def test_all_if_match(self):
 | |
|         self.client.defaults['HTTP_IF_MATCH'] = '*'
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertFullResponse(response)
 | |
|         response = self.client.get('/condition/no_etag/')
 | |
|         self.assertEqual(response.status_code, 412)
 | |
| 
 | |
|     def test_both_headers(self):
 | |
|         # see https://tools.ietf.org/html/rfc7232#section-6
 | |
|         self.client.defaults['HTTP_IF_MODIFIED_SINCE'] = LAST_MODIFIED_STR
 | |
|         self.client.defaults['HTTP_IF_NONE_MATCH'] = ETAG
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertNotModified(response)
 | |
| 
 | |
|         self.client.defaults['HTTP_IF_MODIFIED_SINCE'] = EXPIRED_LAST_MODIFIED_STR
 | |
|         self.client.defaults['HTTP_IF_NONE_MATCH'] = ETAG
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertNotModified(response)
 | |
| 
 | |
|         self.client.defaults['HTTP_IF_MODIFIED_SINCE'] = LAST_MODIFIED_STR
 | |
|         self.client.defaults['HTTP_IF_NONE_MATCH'] = EXPIRED_ETAG
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertFullResponse(response)
 | |
| 
 | |
|         self.client.defaults['HTTP_IF_MODIFIED_SINCE'] = EXPIRED_LAST_MODIFIED_STR
 | |
|         self.client.defaults['HTTP_IF_NONE_MATCH'] = EXPIRED_ETAG
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertFullResponse(response)
 | |
| 
 | |
|     def test_both_headers_2(self):
 | |
|         self.client.defaults['HTTP_IF_UNMODIFIED_SINCE'] = LAST_MODIFIED_STR
 | |
|         self.client.defaults['HTTP_IF_MATCH'] = ETAG
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertFullResponse(response)
 | |
| 
 | |
|         self.client.defaults['HTTP_IF_UNMODIFIED_SINCE'] = EXPIRED_LAST_MODIFIED_STR
 | |
|         self.client.defaults['HTTP_IF_MATCH'] = ETAG
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertFullResponse(response)
 | |
| 
 | |
|         self.client.defaults['HTTP_IF_UNMODIFIED_SINCE'] = EXPIRED_LAST_MODIFIED_STR
 | |
|         self.client.defaults['HTTP_IF_MATCH'] = EXPIRED_ETAG
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertEqual(response.status_code, 412)
 | |
| 
 | |
|         self.client.defaults['HTTP_IF_UNMODIFIED_SINCE'] = LAST_MODIFIED_STR
 | |
|         self.client.defaults['HTTP_IF_MATCH'] = EXPIRED_ETAG
 | |
|         response = self.client.get('/condition/')
 | |
|         self.assertEqual(response.status_code, 412)
 | |
| 
 | |
|     def test_single_condition_1(self):
 | |
|         self.client.defaults['HTTP_IF_MODIFIED_SINCE'] = LAST_MODIFIED_STR
 | |
|         response = self.client.get('/condition/last_modified/')
 | |
|         self.assertNotModified(response)
 | |
|         response = self.client.get('/condition/etag/')
 | |
|         self.assertFullResponse(response, check_last_modified=False)
 | |
| 
 | |
|     def test_single_condition_2(self):
 | |
|         self.client.defaults['HTTP_IF_NONE_MATCH'] = ETAG
 | |
|         response = self.client.get('/condition/etag/')
 | |
|         self.assertNotModified(response)
 | |
|         response = self.client.get('/condition/last_modified/')
 | |
|         self.assertFullResponse(response, check_etag=False)
 | |
| 
 | |
|     def test_single_condition_3(self):
 | |
|         self.client.defaults['HTTP_IF_MODIFIED_SINCE'] = EXPIRED_LAST_MODIFIED_STR
 | |
|         response = self.client.get('/condition/last_modified/')
 | |
|         self.assertFullResponse(response, check_etag=False)
 | |
| 
 | |
|     def test_single_condition_4(self):
 | |
|         self.client.defaults['HTTP_IF_NONE_MATCH'] = EXPIRED_ETAG
 | |
|         response = self.client.get('/condition/etag/')
 | |
|         self.assertFullResponse(response, check_last_modified=False)
 | |
| 
 | |
|     def test_single_condition_5(self):
 | |
|         self.client.defaults['HTTP_IF_MODIFIED_SINCE'] = LAST_MODIFIED_STR
 | |
|         response = self.client.get('/condition/last_modified2/')
 | |
|         self.assertNotModified(response)
 | |
|         response = self.client.get('/condition/etag2/')
 | |
|         self.assertFullResponse(response, check_last_modified=False)
 | |
| 
 | |
|     def test_single_condition_6(self):
 | |
|         self.client.defaults['HTTP_IF_NONE_MATCH'] = ETAG
 | |
|         response = self.client.get('/condition/etag2/')
 | |
|         self.assertNotModified(response)
 | |
|         response = self.client.get('/condition/last_modified2/')
 | |
|         self.assertFullResponse(response, check_etag=False)
 | |
| 
 | |
|     def test_single_condition_7(self):
 | |
|         self.client.defaults['HTTP_IF_UNMODIFIED_SINCE'] = EXPIRED_LAST_MODIFIED_STR
 | |
|         response = self.client.get('/condition/last_modified/')
 | |
|         self.assertEqual(response.status_code, 412)
 | |
|         response = self.client.get('/condition/etag/')
 | |
|         self.assertEqual(response.status_code, 412)
 | |
| 
 | |
|     def test_single_condition_8(self):
 | |
|         self.client.defaults['HTTP_IF_UNMODIFIED_SINCE'] = LAST_MODIFIED_STR
 | |
|         response = self.client.get('/condition/last_modified/')
 | |
|         self.assertFullResponse(response, check_etag=False)
 | |
| 
 | |
|     def test_single_condition_9(self):
 | |
|         self.client.defaults['HTTP_IF_UNMODIFIED_SINCE'] = EXPIRED_LAST_MODIFIED_STR
 | |
|         response = self.client.get('/condition/last_modified2/')
 | |
|         self.assertEqual(response.status_code, 412)
 | |
|         response = self.client.get('/condition/etag2/')
 | |
|         self.assertEqual(response.status_code, 412)
 | |
| 
 | |
|     def test_single_condition_head(self):
 | |
|         self.client.defaults['HTTP_IF_MODIFIED_SINCE'] = LAST_MODIFIED_STR
 | |
|         response = self.client.head('/condition/')
 | |
|         self.assertNotModified(response)
 | |
| 
 | |
|     def test_unquoted(self):
 | |
|         """
 | |
|         The same quoted ETag should be set on the header regardless of whether
 | |
|         etag_func() in condition() returns a quoted or an unquoted ETag.
 | |
|         """
 | |
|         response_quoted = self.client.get('/condition/etag/')
 | |
|         response_unquoted = self.client.get('/condition/unquoted_etag/')
 | |
|         self.assertEqual(response_quoted['ETag'], response_unquoted['ETag'])
 | |
| 
 | |
|     # It's possible that the matching algorithm could use the wrong value even
 | |
|     # if the ETag header is set correctly correctly (as tested by
 | |
|     # test_unquoted()), so check that the unquoted value is matched.
 | |
|     def test_unquoted_if_none_match(self):
 | |
|         self.client.defaults['HTTP_IF_NONE_MATCH'] = ETAG
 | |
|         response = self.client.get('/condition/unquoted_etag/')
 | |
|         self.assertNotModified(response)
 | |
|         response = self.client.put('/condition/unquoted_etag/')
 | |
|         self.assertEqual(response.status_code, 412)
 | |
|         self.client.defaults['HTTP_IF_NONE_MATCH'] = EXPIRED_ETAG
 | |
|         response = self.client.get('/condition/unquoted_etag/')
 | |
|         self.assertFullResponse(response, check_last_modified=False)
 | |
| 
 | |
|     def test_invalid_etag(self):
 | |
|         self.client.defaults['HTTP_IF_NONE_MATCH'] = '"""'
 | |
|         response = self.client.get('/condition/etag/')
 | |
|         self.assertFullResponse(response, check_last_modified=False)
 |