diff options
author | Коренберг Марк <socketpair@gmail.com> | 2017-12-21 12:16:17 (GMT) |
---|---|---|
committer | Andrew Svetlov <andrew.svetlov@gmail.com> | 2017-12-21 12:16:17 (GMT) |
commit | fbd605151fcf2899b14575f4ddb9ce3c55e684ab (patch) | |
tree | 33c8d17b6cc9fd44c5471860c16b7fdc02e67eaa | |
parent | a8d25a16452f7ee8dfc350cd028b3ae172d28ada (diff) | |
download | cpython-fbd605151fcf2899b14575f4ddb9ce3c55e684ab.zip cpython-fbd605151fcf2899b14575f4ddb9ce3c55e684ab.tar.gz cpython-fbd605151fcf2899b14575f4ddb9ce3c55e684ab.tar.bz2 |
bpo-32323: urllib.parse.urlsplit() must not lowercase() IPv6 scope value (#4867)
-rw-r--r-- | Lib/test/test_urlparse.py | 9 | ||||
-rw-r--r-- | Lib/urllib/parse.py | 10 | ||||
-rw-r--r-- | Misc/NEWS.d/next/Library/2017-12-14-10-10-10.bpo-32323.ideco.rst | 2 |
3 files changed, 17 insertions, 4 deletions
diff --git a/Lib/test/test_urlparse.py b/Lib/test/test_urlparse.py index e5f6130..ddee1c3 100644 --- a/Lib/test/test_urlparse.py +++ b/Lib/test/test_urlparse.py @@ -520,6 +520,15 @@ class UrlParseTestCase(unittest.TestCase): self.assertEqual(result.url, defrag) self.assertEqual(result.fragment, frag) + def test_urlsplit_scoped_IPv6(self): + p = urllib.parse.urlsplit('http://[FE80::822a:a8ff:fe49:470c%tESt]:1234') + self.assertEqual(p.hostname, "fe80::822a:a8ff:fe49:470c%tESt") + self.assertEqual(p.netloc, '[FE80::822a:a8ff:fe49:470c%tESt]:1234') + + p = urllib.parse.urlsplit(b'http://[FE80::822a:a8ff:fe49:470c%tESt]:1234') + self.assertEqual(p.hostname, b"fe80::822a:a8ff:fe49:470c%tESt") + self.assertEqual(p.netloc, b'[FE80::822a:a8ff:fe49:470c%tESt]:1234') + def test_urlsplit_attributes(self): url = "HTTP://WWW.PYTHON.ORG/doc/#frag" p = urllib.parse.urlsplit(url) diff --git a/Lib/urllib/parse.py b/Lib/urllib/parse.py index 76086e5..58460f9 100644 --- a/Lib/urllib/parse.py +++ b/Lib/urllib/parse.py @@ -155,10 +155,12 @@ class _NetlocResultMixinBase(object): def hostname(self): hostname = self._hostinfo[0] if not hostname: - hostname = None - elif hostname is not None: - hostname = hostname.lower() - return hostname + return None + # Scoped IPv6 address may have zone info, which must not be lowercased + # like http://[fe80::822a:a8ff:fe49:470c%tESt]:1234/keys + separator = '%' if isinstance(hostname, str) else b'%' + hostname, percent, zone = hostname.partition(separator) + return hostname.lower() + percent + zone @property def port(self): diff --git a/Misc/NEWS.d/next/Library/2017-12-14-10-10-10.bpo-32323.ideco.rst b/Misc/NEWS.d/next/Library/2017-12-14-10-10-10.bpo-32323.ideco.rst new file mode 100644 index 0000000..3077d7c --- /dev/null +++ b/Misc/NEWS.d/next/Library/2017-12-14-10-10-10.bpo-32323.ideco.rst @@ -0,0 +1,2 @@ +:func:`urllib.parse.urlsplit()` does not convert zone-id (scope) to lower case +for scoped IPv6 addresses in hostnames now. |