Skip to content

Commit 2ed581c

Browse files
authored
Merge pull request #5968 from cjerdonek/dry-up-parse-credentials
Use split_auth_from_netloc() inside MultiDomainBasicAuth
2 parents 169cd10 + 0aa301c commit 2ed581c

File tree

6 files changed

+28
-32
lines changed

6 files changed

+28
-32
lines changed

news/5968.bugfix

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
Percent-decode special characters in SVN URL credentials.

src/pip/_internal/download.py

+6-16
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,6 @@
2626
from pip._vendor.six.moves import xmlrpc_client # type: ignore
2727
from pip._vendor.six.moves.urllib import parse as urllib_parse
2828
from pip._vendor.six.moves.urllib import request as urllib_request
29-
from pip._vendor.six.moves.urllib.parse import unquote as urllib_unquote
3029
from pip._vendor.urllib3.util import IS_PYOPENSSL
3130

3231
import pip
@@ -39,8 +38,8 @@
3938
from pip._internal.utils.logging import indent_log
4039
from pip._internal.utils.misc import (
4140
ARCHIVE_EXTENSIONS, ask_path_exists, backup_dir, call_subprocess, consume,
42-
display_path, format_size, get_installed_version, rmtree, splitext,
43-
unpack_file,
41+
display_path, format_size, get_installed_version, rmtree,
42+
split_auth_from_netloc, splitext, unpack_file,
4443
)
4544
from pip._internal.utils.setuptools_build import SETUPTOOLS_SHIM
4645
from pip._internal.utils.temp_dir import TempDirectory
@@ -142,18 +141,18 @@ def __init__(self, prompting=True):
142141
def __call__(self, req):
143142
parsed = urllib_parse.urlparse(req.url)
144143

145-
# Get the netloc without any embedded credentials
146-
netloc = parsed.netloc.rsplit("@", 1)[-1]
144+
# Split the credentials from the netloc.
145+
netloc, url_user_password = split_auth_from_netloc(parsed.netloc)
147146

148147
# Set the url of the request to the url without any credentials
149148
req.url = urllib_parse.urlunparse(parsed[:1] + (netloc,) + parsed[2:])
150149

151150
# Use any stored credentials that we have for this netloc
152151
username, password = self.passwords.get(netloc, (None, None))
153152

154-
# Extract credentials embedded in the url if we have none stored
153+
# Use the credentials embedded in the url if we have none stored
155154
if username is None:
156-
username, password = self.parse_credentials(parsed.netloc)
155+
username, password = url_user_password
157156

158157
# Get creds from netrc if we still don't have them
159158
if username is None and password is None:
@@ -213,15 +212,6 @@ def warn_on_401(self, resp, **kwargs):
213212
logger.warning('401 Error, Credentials not correct for %s',
214213
resp.request.url)
215214

216-
def parse_credentials(self, netloc):
217-
if "@" in netloc:
218-
userinfo = netloc.rsplit("@", 1)[0]
219-
if ":" in userinfo:
220-
user, pwd = userinfo.split(":", 1)
221-
return (urllib_unquote(user), urllib_unquote(pwd))
222-
return urllib_unquote(userinfo), None
223-
return None, None
224-
225215

226216
class LocalFSAdapter(BaseAdapter):
227217

src/pip/_internal/utils/misc.py

+7-2
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@
2525
from pip._vendor.six import PY2
2626
from pip._vendor.six.moves import input
2727
from pip._vendor.six.moves.urllib import parse as urllib_parse
28+
from pip._vendor.six.moves.urllib.parse import unquote as urllib_unquote
2829

2930
from pip._internal.exceptions import CommandError, InstallationError
3031
from pip._internal.locations import (
@@ -880,10 +881,14 @@ def split_auth_from_netloc(netloc):
880881
# Split from the left because that's how urllib.parse.urlsplit()
881882
# behaves if more than one : is present (which again can be checked
882883
# using the password attribute of the return value)
883-
user_pass = tuple(auth.split(':', 1))
884+
user_pass = auth.split(':', 1)
884885
else:
885886
user_pass = auth, None
886887

888+
user_pass = tuple(
889+
None if x is None else urllib_unquote(x) for x in user_pass
890+
)
891+
887892
return netloc, user_pass
888893

889894

@@ -897,7 +902,7 @@ def redact_netloc(netloc):
897902
if user is None:
898903
return netloc
899904
password = '' if password is None else ':****'
900-
return '{user}{password}@{netloc}'.format(user=user,
905+
return '{user}{password}@{netloc}'.format(user=urllib_parse.quote(user),
901906
password=password,
902907
netloc=netloc)
903908

tests/unit/test_download.py

+2-13
Original file line numberDiff line numberDiff line change
@@ -10,8 +10,8 @@
1010

1111
import pip
1212
from pip._internal.download import (
13-
MultiDomainBasicAuth, PipSession, SafeFileCache, path_to_url,
14-
unpack_file_url, unpack_http_url, url_to_path,
13+
PipSession, SafeFileCache, path_to_url, unpack_file_url, unpack_http_url,
14+
url_to_path,
1515
)
1616
from pip._internal.exceptions import HashMismatch
1717
from pip._internal.models.link import Link
@@ -328,14 +328,3 @@ def test_insecure_host_cache_is_not_enabled(self, tmpdir):
328328
)
329329

330330
assert not hasattr(session.adapters["https://example.com/"], "cache")
331-
332-
333-
def test_parse_credentials():
334-
auth = MultiDomainBasicAuth()
335-
assert auth.parse_credentials("foo:bar@example.com") == ('foo', 'bar')
336-
assert auth.parse_credentials("foo@example.com") == ('foo', None)
337-
assert auth.parse_credentials("example.com") == (None, None)
338-
339-
# URL-encoded reserved characters:
340-
assert auth.parse_credentials("user%3Aname:%23%40%5E@example.com") \
341-
== ("user:name", "#@^")

tests/unit/test_utils.py

+9-1
Original file line numberDiff line numberDiff line change
@@ -665,6 +665,9 @@ def test_make_vcs_requirement_url(args, expected):
665665
('user:pass@word@example.com', ('example.com', ('user', 'pass@word'))),
666666
# Test the password containing a : symbol.
667667
('user:pass:word@example.com', ('example.com', ('user', 'pass:word'))),
668+
# Test URL-encoded reserved characters.
669+
('user%3Aname:%23%40%5E@example.com',
670+
('example.com', ('user:name', '#@^'))),
668671
])
669672
def test_split_auth_from_netloc(netloc, expected):
670673
actual = split_auth_from_netloc(netloc)
@@ -684,6 +687,8 @@ def test_split_auth_from_netloc(netloc, expected):
684687
('user:pass@word@example.com', 'user:****@example.com'),
685688
# Test the password containing a : symbol.
686689
('user:pass:word@example.com', 'user:****@example.com'),
690+
# Test URL-encoded reserved characters.
691+
('user%3Aname:%23%40%5E@example.com', 'user%3Aname:****@example.com'),
687692
])
688693
def test_redact_netloc(netloc, expected):
689694
actual = redact_netloc(netloc)
@@ -715,7 +720,10 @@ def test_remove_auth_from_url(auth_url, expected_url):
715720
('https://user@example.com/abc', 'https://user@example.com/abc'),
716721
('https://user:password@example.com', 'https://user:****@example.com'),
717722
('https://user:@example.com', 'https://user:****@example.com'),
718-
('https://example.com', 'https://example.com')
723+
('https://example.com', 'https://example.com'),
724+
# Test URL-encoded reserved characters.
725+
('https://user%3Aname:%23%40%5E@example.com',
726+
'https://user%3Aname:****@example.com'),
719727
])
720728
def test_redact_password_from_url(auth_url, expected_url):
721729
url = redact_password_from_url(auth_url)

tests/unit/test_vcs.py

+3
Original file line numberDiff line numberDiff line change
@@ -192,6 +192,9 @@ def test_git__get_netloc_and_auth(args, expected):
192192
(('user@example.com', 'https'), ('example.com', ('user', None))),
193193
# Test https with username and password.
194194
(('user:pass@example.com', 'https'), ('example.com', ('user', 'pass'))),
195+
# Test https with URL-encoded reserved characters.
196+
(('user%3Aname:%23%40%5E@example.com', 'https'),
197+
('example.com', ('user:name', '#@^'))),
195198
# Test ssh with username and password.
196199
(('user:pass@example.com', 'ssh'),
197200
('user:pass@example.com', (None, None))),

0 commit comments

Comments
 (0)