{
  "Selected_candidate": {
    "pr_number": 1361,
    "pr_title": "Always percent-encode location headers.",
    "pr_body": "If we don't, stupid servers that don't do it themselves can hurt us badly.\n\nResolves #1360.\n",
    "issue_id": 1360,
    "issue_title": "Problem with 301-redirect",
    "issue_body": "I have a problem with downloading a file, when the server returns the 301 code:\n\n``` python\nr = requests.get('http://servag.rupsy.ru/track.php?url=2013-04-11/Haatzumaniak_-_Dark_Sadhu_[2013]/03.%20Haatzumaniak_-_777_Dark%20Sadhu%202013.mp3&arj=Haatzumaniak_-_Dark_Sadhu_[2013].ZIP')\nprint(r)\n```\n\nIf you do redirect manually - it works fine:\n\n``` python\nr = requests.get('http://servag.rupsy.ru/track.php?url=2013-04-11/Haatzumaniak_-_Dark_Sadhu_[2013]/03.%20Haatzumaniak_-_777_Dark%20Sadhu%202013.mp3&arj=Haatzumaniak_-_Dark_Sadhu_[2013].ZIP', allow_redirects=False)\nr = requests.get(r.headers['Location'])\nprint(r)\n```\n\nThe problem appears to Python 3.3 and python-requests-1.2.0/python-requests-dev\n\n```\nTraceback (most recent call last):\n  File \"C:\\Dropbox\\projects\\Helpers\\ph\\.envwin32\\lib\\site-packages\\requests\\packages\\urllib3\\connectionpool.py\", line 428, in urlopen\n    body=body, headers=headers)\n  File \"C:\\Dropbox\\projects\\Helpers\\ph\\.envwin32\\lib\\site-packages\\requests\\packages\\urllib3\\connectionpool.py\", line 290, in _make_request\n    httplib_response = conn.getresponse()\n  File \"c:\\Python33\\lib\\http\\client.py\", line 1143, in getresponse\n    response.begin()\n  File \"c:\\Python33\\lib\\http\\client.py\", line 354, in begin\n    version, status, reason = self._read_status()\n  File \"c:\\Python33\\lib\\http\\client.py\", line 324, in _read_status\n    raise BadStatusLine(line)\nhttp.client.BadStatusLine: ''\n\nDuring handling of the above exception, another exception occurred:\n\nTraceback (most recent call last):\n  File \"C:\\Dropbox\\projects\\Helpers\\ph\\.envwin32\\lib\\site-packages\\requests\\adapters.py\", line 283, in send\n    timeout=timeout\n  File \"C:\\Dropbox\\projects\\Helpers\\ph\\.envwin32\\lib\\site-packages\\requests\\packages\\urllib3\\connectionpool.py\", line 474, in urlopen\n    raise MaxRetryError(self, url, e)\nrequests.packages.urllib3.exceptions.MaxRetryError: HTTPConnectionPool(host='servag.rupsy.ru', port=80): Max retries exceeded with url: /media/2/Haatzumaniak_-_Dark_Sadhu_[2013]/03. Haatzumaniak_-_777_Dark Sadhu 2013.mp3 (Caused by <class 'http.client.BadStatusLine'>: '')\n\nDuring handling of the above exception, another exception occurred:\n\nTraceback (most recent call last):\n  File \"C:/Dropbox/projects/InfoStreams/test.py\", line 25, in <module>\n    allow_redirects=True)\n  File \"C:\\Dropbox\\projects\\Helpers\\ph\\.envwin32\\lib\\site-packages\\requests\\api.py\", line 55, in get\n    return request('get', url, **kwargs)\n  File \"C:\\Dropbox\\projects\\Helpers\\ph\\.envwin32\\lib\\site-packages\\requests\\api.py\", line 44, in request\n    return session.request(method=method, url=url, **kwargs)\n  File \"C:\\Dropbox\\projects\\Helpers\\ph\\.envwin32\\lib\\site-packages\\requests\\sessions.py\", line 345, in request\n    resp = self.send(prep, **send_kwargs)\n  File \"C:\\Dropbox\\projects\\Helpers\\ph\\.envwin32\\lib\\site-packages\\requests\\sessions.py\", line 464, in send\n    history = [resp for resp in gen] if allow_redirects else []\n  File \"C:\\Dropbox\\projects\\Helpers\\ph\\.envwin32\\lib\\site-packages\\requests\\sessions.py\", line 464, in <listcomp>\n    history = [resp for resp in gen] if allow_redirects else []\n  File \"C:\\Dropbox\\projects\\Helpers\\ph\\.envwin32\\lib\\site-packages\\requests\\sessions.py\", line 154, in resolve_redirects\n    allow_redirects=False,\n  File \"C:\\Dropbox\\projects\\Helpers\\ph\\.envwin32\\lib\\site-packages\\requests\\sessions.py\", line 448, in send\n    r = adapter.send(request, **kwargs)\n  File \"C:\\Dropbox\\projects\\Helpers\\ph\\.envwin32\\lib\\site-packages\\requests\\adapters.py\", line 318, in send\n    raise ConnectionError(e)\nrequests.exceptions.ConnectionError: HTTPConnectionPool(host='servag.rupsy.ru', port=80): Max retries exceeded with url: /media/2/Haatzumaniak_-_Dark_Sadhu_[2013]/03. Haatzumaniak_-_777_Dark Sadhu 2013.mp3 (Caused by <class 'http.client.BadStatusLine'>: '')\n```\n",
    "issue_closed_at": "2013-05-20T20:04:08Z",
    "base_commit": "eacb91afb1c5dee8a6fcdadc9473463008e791c6",
    "changes": [
      {
        "file": "requests/sessions.py",
        "type": "function",
        "name": "resolve_redirects",
        "class_name": "SessionRedirectMixin",
        "code": "def resolve_redirects(self, resp, req, stream=False, timeout=None,\n                          verify=True, cert=None, proxies=None):\n        \"\"\"Receives a Response. Returns a generator of Responses.\"\"\"\n\n        i = 0\n        prepared_request = PreparedRequest()\n        prepared_request.body = req.body\n        prepared_request.headers = req.headers.copy()\n        prepared_request.hooks = req.hooks\n        prepared_request.method = req.method\n        prepared_request.url = req.url\n\n        # ((resp.status_code is codes.see_other))\n        while (('location' in resp.headers and resp.status_code in REDIRECT_STATI)):\n\n            resp.content  # Consume socket so it can be released\n\n            if i >= self.max_redirects:\n                raise TooManyRedirects('Exceeded %s redirects.' % self.max_redirects)\n\n            # Release the connection back into the pool.\n            resp.close()\n\n            url = resp.headers['location']\n            method = prepared_request.method\n\n            # Handle redirection without scheme (see: RFC 1808 Section 4)\n            if url.startswith('//'):\n                parsed_rurl = urlparse(resp.url)\n                url = '%s:%s' % (parsed_rurl.scheme, url)\n\n            # Facilitate non-RFC2616-compliant 'location' headers\n            # (e.g. '/path/to/resource' instead of 'http://domain.tld/path/to/resource')\n            if not urlparse(url).netloc:\n                # Compliant with RFC3986, we percent encode the url.\n                url = urljoin(resp.url, requote_uri(url))\n\n            prepared_request.url = url\n\n            # http://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html#sec10.3.4\n            if (resp.status_code == codes.see_other and\n                    prepared_request.method != 'HEAD'):\n                method = 'GET'\n\n            # Do what the browsers do, despite standards...\n            if (resp.status_code in (codes.moved, codes.found) and\n                    prepared_request.method not in ('GET', 'HEAD')):\n                method = 'GET'\n\n            prepared_request.method = method\n\n            # https://github.com/kennethreitz/requests/issues/1084\n            if resp.status_code not in (codes.temporary, codes.resume):\n                if 'Content-Length' in prepared_request.headers:\n                    del prepared_request.headers['Content-Length']\n\n                prepared_request.body = None\n\n            headers = prepared_request.headers\n            try:\n                del headers['Cookie']\n            except KeyError:\n                pass\n\n            prepared_request.prepare_cookies(self.cookies)\n\n            resp = self.send(\n                prepared_request,\n                stream=stream,\n                timeout=timeout,\n                verify=verify,\n                cert=cert,\n                proxies=proxies,\n                allow_redirects=False,\n            )\n\n            extract_cookies_to_jar(self.cookies, prepared_request, resp.raw)\n\n            i += 1\n            yield resp"
      }
    ]
  },
  "Justification": "Candidate E is most relevant because it directly addresses a problem related to handling HTTP 301 redirects in the requests library, which is closely related to the CURRENT bug that involves handling the method as a string and potentially returning unexpected response codes. Candidate E’s bug involves similar stack traces and call flows, as both involve the request process and error handling in the requests library. This report provides insights into handling redirects, which could be beneficial for resolving the CURRENT bug where a mismanaged method value leads to a 404 response instead of the expected behavior. The structural, symptom, and impact similarities make this candidate the most helpful for debugging the CURRENT issue.",
  "instance_id": "psf__requests-2317",
  "repo": "psf/requests",
  "created_at": "2014-11-01T02:20:16Z",
  "problem_statement": "method = builtin_str(method) problem\nIn requests/sessions.py is a command:\n\nmethod = builtin_str(method)\nConverts method from\nb’GET’\nto\n\"b'GET’\"\n\nWhich is the literal string, no longer a binary string.  When requests tries to use the method \"b'GET’”, it gets a 404 Not Found response.\n\nI am using python3.4 and python-neutronclient (2.3.9) with requests (2.4.3).  neutronclient is broken because it uses this \"args = utils.safe_encode_list(args)\" command which converts all the values to binary string, including method.\n\nI'm not sure if this is a bug with neutronclient or a bug with requests, but I'm starting here.  Seems if requests handled the method value being a binary string, we wouldn't have any problem.\n\nAlso, I tried in python2.6 and this bug doesn't exist there. Some difference between 2.6 and 3.4 makes this not work right.\n\n",
  "patch": "diff --git a/requests/sessions.py b/requests/sessions.py\n--- a/requests/sessions.py\n+++ b/requests/sessions.py\n@@ -13,7 +13,7 @@\n from datetime import datetime\n \n from .auth import _basic_auth_str\n-from .compat import cookielib, OrderedDict, urljoin, urlparse, builtin_str\n+from .compat import cookielib, OrderedDict, urljoin, urlparse\n from .cookies import (\n     cookiejar_from_dict, extract_cookies_to_jar, RequestsCookieJar, merge_cookies)\n from .models import Request, PreparedRequest, DEFAULT_REDIRECT_LIMIT\n@@ -425,7 +425,7 @@ def request(self, method, url,\n             If Tuple, ('cert', 'key') pair.\n         \"\"\"\n \n-        method = builtin_str(method)\n+        method = to_native_string(method)\n \n         # Create the Request.\n         req = Request(\n"
}