summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorDenis Cornehl <syphar@fastmail.fm>2016-01-05 08:09:10 +0100
committerTim Graham <timograham@gmail.com>2016-01-05 09:41:13 -0500
commitee2835e69c47d02e8bce5a496bf8bd9f93b04fb4 (patch)
treedccba382988875d2e87f531f3042dba24df7e643
parent775291d4c767cd17dc5bbd2e8a605fd4fdfd8aed (diff)
downloaddjango-ee2835e69c47d02e8bce5a496bf8bd9f93b04fb4.tar.gz
[1.9.x] Fixed #26024 -- Fixed regression in ConditionalGetMiddleware ETag support.
Backport of 186b6c61bfe85afa4d6bf213d04a28dd2853fed2 from master
-rw-r--r--django/middleware/common.py5
-rw-r--r--django/middleware/http.py4
-rw-r--r--django/utils/http.py7
-rw-r--r--docs/releases/1.9.2.txt3
-rw-r--r--tests/middleware/tests.py5
-rw-r--r--tests/utils_tests/test_http.py4
6 files changed, 21 insertions, 7 deletions
diff --git a/django/middleware/common.py b/django/middleware/common.py
index 3db925c894..a8958a80e3 100644
--- a/django/middleware/common.py
+++ b/django/middleware/common.py
@@ -8,6 +8,7 @@ from django.core.exceptions import PermissionDenied
from django.core.mail import mail_managers
from django.utils.cache import get_conditional_response, set_response_etag
from django.utils.encoding import force_text
+from django.utils.http import unquote_etag
from django.utils.six.moves.urllib.parse import urlparse
logger = logging.getLogger('django.request')
@@ -120,9 +121,7 @@ class CommonMiddleware(object):
if response.has_header('ETag'):
return get_conditional_response(
request,
- # get_conditional_response() requires an unquoted version
- # of the response's ETag.
- etag=response['ETag'].strip('"'),
+ etag=unquote_etag(response['ETag']),
response=response,
)
diff --git a/django/middleware/http.py b/django/middleware/http.py
index 7a6f237c96..7912200634 100644
--- a/django/middleware/http.py
+++ b/django/middleware/http.py
@@ -1,5 +1,5 @@
from django.utils.cache import get_conditional_response
-from django.utils.http import http_date, parse_http_date_safe
+from django.utils.http import http_date, parse_http_date_safe, unquote_etag
class ConditionalGetMiddleware(object):
@@ -23,7 +23,7 @@ class ConditionalGetMiddleware(object):
if etag or last_modified:
return get_conditional_response(
request,
- etag=etag,
+ etag=unquote_etag(etag),
last_modified=last_modified,
response=response,
)
diff --git a/django/utils/http.py b/django/utils/http.py
index 8bbafaedec..70bcbd90ac 100644
--- a/django/utils/http.py
+++ b/django/utils/http.py
@@ -253,6 +253,13 @@ def quote_etag(etag):
return '"%s"' % etag.replace('\\', '\\\\').replace('"', '\\"')
+def unquote_etag(etag):
+ """
+ Unquote an ETag string; i.e. revert quote_etag().
+ """
+ return etag.strip('"').replace('\\"', '"').replace('\\\\', '\\') if etag else etag
+
+
def is_same_domain(host, pattern):
"""
Return ``True`` if the host is either an exact match or a match
diff --git a/docs/releases/1.9.2.txt b/docs/releases/1.9.2.txt
index 484fb6e4c2..8e04de6e29 100644
--- a/docs/releases/1.9.2.txt
+++ b/docs/releases/1.9.2.txt
@@ -9,4 +9,5 @@ Django 1.9.2 fixes several bugs in 1.9.1.
Bugfixes
========
-* ...
+* Fixed a regression in ``ConditionalGetMiddleware`` causing ``If-None-Match`` checks
+ to always return HTTP 200 (:ticket:`26024`).
diff --git a/tests/middleware/tests.py b/tests/middleware/tests.py
index 2804e743f7..bf96f05048 100644
--- a/tests/middleware/tests.py
+++ b/tests/middleware/tests.py
@@ -470,6 +470,11 @@ class ConditionalGetMiddlewareTest(SimpleTestCase):
self.resp = ConditionalGetMiddleware().process_response(self.req, self.resp)
self.assertEqual(self.resp.status_code, 304)
+ def test_if_none_match_and_same_etag_with_quotes(self):
+ self.req.META['HTTP_IF_NONE_MATCH'] = self.resp['ETag'] = '"spam"'
+ self.resp = ConditionalGetMiddleware().process_response(self.req, self.resp)
+ self.assertEqual(self.resp.status_code, 304)
+
def test_if_none_match_and_different_etag(self):
self.req.META['HTTP_IF_NONE_MATCH'] = 'spam'
self.resp['ETag'] = 'eggs'
diff --git a/tests/utils_tests/test_http.py b/tests/utils_tests/test_http.py
index f90486003a..6051818958 100644
--- a/tests/utils_tests/test_http.py
+++ b/tests/utils_tests/test_http.py
@@ -158,8 +158,10 @@ class ETagProcessingTests(unittest.TestCase):
self.assertEqual(etags, ['', 'etag', 'e"t"ag', r'e\tag', 'weak'])
def test_quoting(self):
- quoted_etag = http.quote_etag(r'e\t"ag')
+ original_etag = r'e\t"ag'
+ quoted_etag = http.quote_etag(original_etag)
self.assertEqual(quoted_etag, r'"e\\t\"ag"')
+ self.assertEqual(http.unquote_etag(quoted_etag), original_etag)
class HttpDateProcessingTests(unittest.TestCase):