{
  "Selected_candidate": {
    "pr_number": 1888,
    "pr_title": "Fix for 301 redirect and latest PyOpenSSL.",
    "pr_body": "Fixes #1887\n",
    "issue_id": 1887,
    "issue_title": "301 redirect broken with latest pyopenssl/SNI",
    "issue_body": "With the latest pyopenssl on Windows 64bit:\n\n```\ncryptography==0.2.dev1\nndg-httpsclient==0.3.2\npyOpenSSL==0.13\npyasn1==0.1.7\n```\n\nI get an exception raised when `GET`ing a `301` response to a HTTPS request. I see that after the redirect is received the returned URL is [decoded to a Unicode string](https://github.com/kennethreitz/requests/blob/master/requests/adapters.py#L181). Then requests passes the response to `resolve_redirects` which uses the url to make a new request. This leads to a Unicode string being passed to urllib3 and eventually pyopenssl. And because in pyopenssl they now check that the data is of type bytes, an exception is thrown. \n\nI Wrote this test:\n\n```\n    def test_pyopenssl_redirect(self):\n        requests.get('https://httpbin.org/status/301')\n```\n\nand this is the result of py.test:\n\n```\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\n\nself = <OpenSSL.SSL.Connection object at 0x000000000345CC50>\nbuf = u'GET /redirect/1 HTTP/1.1\\r\\nHost: httpbin.org\\r\\nAccept-Encoding: gzip, defl...cept: */*\\r\\nUser-Agent: python-r\nequests/2.2.1 CPython/2.7.6 Windows/8\\r\\n\\r\\n'\nflags = 0\n\n    def sendall(self, buf, flags=0):\n        \"\"\"\n            Send \"all\" data on the connection. This calls send() repeatedly until\n            all data is sent. If an error occurs, it's impossible to tell how much\n            data has been sent.\n\n            :param buf: The string to send\n            :param flags: (optional) Included for compatibility with the socket\n                          API, the value is ignored\n            :return: The number of bytes written\n            \"\"\"\n        if isinstance(buf, _memoryview):\n            buf = buf.tobytes()\n        if not isinstance(buf, bytes):\n>           raise TypeError(\"buf must be a byte string\")\nE           TypeError: buf must be a byte string\n\n..\\testreq\\lib\\site-packages\\OpenSSL\\SSL.py:968: TypeError\n=================================== 117 tests deselected by '-kpyopenssl_redirect' ====================================\n====================================== 1 failed, 117 deselected in 4.47 seconds =======================================\n```\n",
    "issue_closed_at": "2014-01-28T20:14:38Z",
    "base_commit": "19756d57f73c2062240dd477dd8f8d8a7c0c512a",
    "changes": [
      {
        "file": "requests/sessions.py",
        "type": "line",
        "name": "line 17",
        "code": "    cookiejar_from_dict, extract_cookies_to_jar, RequestsCookieJar, merge_cookies)\nfrom .models import Request, PreparedRequest\nfrom .hooks import default_hooks, dispatch_hook\nfrom .utils import to_key_val_list, default_headers\nfrom .exceptions import TooManyRedirects, InvalidSchema\nfrom .structures import CaseInsensitiveDict\n"
      },
      {
        "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\n        # ((resp.status_code is codes.see_other))\n        while ('location' in resp.headers and resp.status_code in REDIRECT_STATI):\n            prepared_request = req.copy()\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 = req.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            # The scheme should be lower case...\n            parsed = urlparse(url)\n            url = parsed.geturl()\n\n            # Facilitate non-RFC2616-compliant 'location' headers\n            # (e.g. '/path/to/resource' instead of 'http://domain.tld/path/to/resource')\n            # Compliant with RFC3986, we percent encode the url.\n            if not urlparse(url).netloc:\n                url = urljoin(resp.url, requote_uri(url))\n            else:\n                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                    method != 'HEAD'):\n                method = 'GET'\n\n            # Do what the browsers do, despite standards...\n            # First, turn 302s into GETs.\n            if resp.status_code == codes.found and method != 'HEAD':\n                method = 'GET'\n\n            # Second, if a POST is responded to with a 301, turn it into a GET.\n            # This bizarre behaviour is explained in Issue 1704.\n            if resp.status_code == codes.moved and method == 'POST':\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            extract_cookies_to_jar(prepared_request._cookies,\n                                   prepared_request, resp.raw)\n            prepared_request._cookies.update(self.cookies)\n            prepared_request.prepare_cookies(prepared_request._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 A is the most helpful because it addresses issues related to redirect handling in the `requests/sessions.py` module, similar to the CURRENT bug's focus on `Session.resolve_redirects`. Both cases involve how HTTP methods are handled during redirects. The fix provided in Candidate A for a problem in 301 redirects can give insights into the logic of managing request methods and can help identify mismanagement in the CURRENT bug where POSTs are incorrectly converted to GETs. The structural similarity with error handling in redirects makes it particularly relevant for debugging the CURRENT issue.",
  "instance_id": "psf__requests-1963",
  "repo": "psf/requests",
  "created_at": "2014-03-15T17:42:11Z",
  "problem_statement": "`Session.resolve_redirects` copies the original request for all subsequent requests, can cause incorrect method selection\nConsider the following redirection chain:\n\n```\nPOST /do_something HTTP/1.1\nHost: server.example.com\n...\n\nHTTP/1.1 303 See Other\nLocation: /new_thing_1513\n\nGET /new_thing_1513\nHost: server.example.com\n...\n\nHTTP/1.1 307 Temporary Redirect\nLocation: //failover.example.com/new_thing_1513\n```\n\nThe intermediate 303 See Other has caused the POST to be converted to\na GET.  The subsequent 307 should preserve the GET.  However, because\n`Session.resolve_redirects` starts each iteration by copying the _original_\nrequest object, Requests will issue a POST!\n\n",
  "patch": "diff --git a/requests/sessions.py b/requests/sessions.py\n--- a/requests/sessions.py\n+++ b/requests/sessions.py\n@@ -168,8 +168,11 @@ def resolve_redirects(self, resp, req, stream=False, timeout=None,\n             if new_auth is not None:\n                 prepared_request.prepare_auth(new_auth)\n \n+            # Override the original request.\n+            req = prepared_request\n+\n             resp = self.send(\n-                prepared_request,\n+                req,\n                 stream=stream,\n                 timeout=timeout,\n                 verify=verify,\n"
}