티스토리 수익 글 보기

티스토리 수익 글 보기

[4.2.x] Fixed CVE-2023-41164 — Fixed potential DoS in django.utils.e… · django/django@9c51b4d · GitHub
Skip to content

Commit 9c51b4d

Browse files
felixxmnessita
andcommitted
[4.2.x] Fixed CVE-2023-41164 — Fixed potential DoS in django.utils.encoding.uri_to_iri().
Thanks MProgrammer (https://hackerone.com/mprogrammer) for the report. Co-authored-by: nessita <124304+nessita@users.noreply.github.com>
1 parent acfb427 commit 9c51b4d

File tree

5 files changed

+43
5
lines changed

5 files changed

+43
5
lines changed

django/utils/encoding.py

Lines changed: 4 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -219,16 +219,18 @@ def repercent_broken_unicode(path):
219219
repercent-encode any octet produced that is not part of a strictly legal
220220
UTF-8 octet sequence.
221221
"""
222+
changed_parts = []
222223
while True:
223224
try:
224225
path.decode()
225226
except UnicodeDecodeError as e:
226227
# CVE-2019-14235: A recursion shouldn't be used since the exception
227228
# handling uses massive amounts of memory
228229
repercent = quote(path[e.start : e.end], safe=b"/#%[]=:;$&()+,!?*@'~")
229-
path = path[: e.start] + repercent.encode() + path[e.end :]
230+
changed_parts.append(path[: e.start] + repercent.encode())
231+
path = path[e.end :]
230232
else:
231-
return path
233+
return b"".join(changed_parts) + path
232234

233235

234236
def filepath_to_uri(path):

docs/releases/3.2.21.txt

Lines changed: 6 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -6,4 +6,9 @@ Django 3.2.21 release notes
66

77
Django 3.2.21 fixes a security issue with severity "moderate" in 3.2.20.
88

9-
...
9+
CVE-2023-41164: Potential denial of service vulnerability in ``django.utils.encoding.uri_to_iri()``
10+
===================================================================================================
11+
12+
``django.utils.encoding.uri_to_iri()`` was subject to potential denial of
13+
service attack via certain inputs with a very large number of Unicode
14+
characters.

docs/releases/4.1.11.txt

Lines changed: 6 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -6,4 +6,9 @@ Django 4.1.11 release notes
66

77
Django 4.1.11 fixes a security issue with severity "moderate" in 4.1.10.
88

9-
...
9+
CVE-2023-41164: Potential denial of service vulnerability in ``django.utils.encoding.uri_to_iri()``
10+
===================================================================================================
11+
12+
``django.utils.encoding.uri_to_iri()`` was subject to potential denial of
13+
service attack via certain inputs with a very large number of Unicode
14+
characters.

docs/releases/4.2.5.txt

Lines changed: 7 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,13 @@ Django 4.2.5 release notes
77
Django 4.2.5 fixes a security issue with severity "moderate" and several bugs
88
in 4.2.4.
99

10+
CVE-2023-41164: Potential denial of service vulnerability in ``django.utils.encoding.uri_to_iri()``
11+
===================================================================================================
12+
13+
``django.utils.encoding.uri_to_iri()`` was subject to potential denial of
14+
service attack via certain inputs with a very large number of Unicode
15+
characters.
16+
1017
Bugfixes
1118
========
1219

tests/utils_tests/test_encoding.py

Lines changed: 20 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,10 @@
11
import datetime
2+
import inspect
23
import sys
34
import unittest
45
from pathlib import Path
56
from unittest import mock
6-
from urllib.parse import quote_plus
7+
from urllib.parse import quote, quote_plus
78

89
from django.test import SimpleTestCase
910
from django.utils.encoding import (
@@ -120,6 +121,24 @@ def test_repercent_broken_unicode_recursion_error(self):
120121
except RecursionError:
121122
self.fail("Unexpected RecursionError raised.")
122123

124+
def test_repercent_broken_unicode_small_fragments(self):
125+
data = b"test\xfctest\xfctest\xfc"
126+
decoded_paths = []
127+
128+
def mock_quote(*args, **kwargs):
129+
# The second frame is the call to repercent_broken_unicode().
130+
decoded_paths.append(inspect.currentframe().f_back.f_locals["path"])
131+
return quote(*args, **kwargs)
132+
133+
with mock.patch("django.utils.encoding.quote", mock_quote):
134+
self.assertEqual(repercent_broken_unicode(data), b"test%FCtest%FCtest%FC")
135+
136+
# decode() is called on smaller fragment of the path each time.
137+
self.assertEqual(
138+
decoded_paths,
139+
[b"test\xfctest\xfctest\xfc", b"test\xfctest\xfc", b"test\xfc"],
140+
)
141+
123142

124143
class TestRFC3987IEncodingUtils(unittest.TestCase):
125144
def test_filepath_to_uri(self):

0 commit comments

Comments
 (0)