{
  "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\u2019GET\u2019\nto\n\"b'GET\u2019\"\n\nWhich is the literal string, no longer a binary string.  When requests tries to use the method \"b'GET\u2019\u201d, 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",
  "similar_bug_items": [
    {
      "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"
        }
      ]
    },
    {
      "pr_number": 1537,
      "pr_title": "Allow non-string objects to be posted as data alongside files.",
      "pr_body": "This should resolve #1462.\n",
      "issue_id": 1462,
      "issue_title": "multipart/form-data and datetime data",
      "issue_body": "I raise an bug that you already fix in the past on this issue : https://github.com/kennethreitz/requests/issues/661 or https://github.com/kennethreitz/requests/issues/737\n\nI tried the same methodology with that code :\n\n```\nimport requets\n\nrequests.post(\"http://httpbin.org/post\", data={'a': 0})\nrequests.post(\"http://httpbin.org/post\", data={'a': 0.0})\nrequests.post(\"http://httpbin.org/post\", data={'a': 0}, files={'b': 'foo'})\nrequests.post(\"http://httpbin.org/post\", data={'a': 0.0}, files={'b': 'foo'})\n```\n\nWith the 1.2.0 version, no error is raised.\n\nWith 1.2.3 version, I have that traceback :\n\n```\nTraceback (most recent call last):\n  File \"test.py\", line 8, in <module>\n    requests.post(\"http://httpbin.org/post\", data={'a': 0.0}, files={'b': 'foo'})\n  File \".../dev/lib/python2.7/site-packages/requests/api.py\", line 88, in post\n    return request('post', url, data=data, **kwargs)\n  File \".../dev/lib/python2.7/site-packages/requests/api.py\", line 44, in request\n    return session.request(method=method, url=url, **kwargs)\n  File \".../dev/lib/python2.7/site-packages/requests/sessions.py\", line 324, in request\n    prep = req.prepare()\n  File \".../dev/lib/python2.7/site-packages/requests/models.py\", line 225, in prepare\n    p.prepare_body(self.data, self.files)\n  File \".../dev/lib/python2.7/site-packages/requests/models.py\", line 385, in prepare_body\n    (body, content_type) = self._encode_files(files, data)\n  File \".../dev/lib/python2.7/site-packages/requests/models.py\", line 133, in _encode_files\n    body, content_type = encode_multipart_formdata(new_fields)\n  File \".../dev/lib/python2.7/site-packages/requests/packages/urllib3/filepost.py\", line 90, in encode_multipart_formdata\n    body.write(data)\nTypeError: 'float' does not have the buffer interface\n```\n\nMy original problem was with a python datetime in the data dict\nThanks,\n",
      "issue_closed_at": "2013-08-27T07:59:06Z",
      "base_commit": "d8268fb7b44da7b8aa225eb1ca6fbdb4f9dc2457",
      "changes": [
        {
          "file": "requests/models.py",
          "type": "function",
          "name": "_encode_files",
          "class_name": "RequestEncodingMixin",
          "code": "def _encode_files(files, data):\n        \"\"\"Build the body for a multipart/form-data request.\n\n        Will successfully encode files when passed as a dict or a list of\n        2-tuples. Order is retained if data is a list of 2-tuples but abritrary\n        if parameters are supplied as a dict.\n\n        \"\"\"\n        if (not files) or isinstance(data, str):\n            return None\n\n        new_fields = []\n        fields = to_key_val_list(data or {})\n        files = to_key_val_list(files or {})\n\n        for field, val in fields:\n            if isinstance(val, basestring) or not hasattr(val, '__iter__'):\n                val = [val]\n            for v in val:\n                if v is not None:\n                    new_fields.append(\n                        (field.decode('utf-8') if isinstance(field, bytes) else field,\n                         v.encode('utf-8') if isinstance(v, str) else v))\n\n        for (k, v) in files:\n            # support for explicit filename\n            ft = None\n            if isinstance(v, (tuple, list)):\n                if len(v) == 2:\n                    fn, fp = v\n                else:\n                    fn, fp, ft = v\n            else:\n                fn = guess_filename(v) or k\n                fp = v\n            if isinstance(fp, str):\n                fp = StringIO(fp)\n            if isinstance(fp, bytes):\n                fp = BytesIO(fp)\n\n            if ft:\n                new_v = (fn, fp.read(), ft)\n            else:\n                new_v = (fn, fp.read())\n            new_fields.append((k, new_v))\n\n        body, content_type = encode_multipart_formdata(new_fields)\n\n        return body, content_type"
        }
      ]
    },
    {
      "pr_number": 2148,
      "pr_title": "Handle socket errors in iter_content",
      "pr_body": "My first PR for Requests, so I probably did something terribly wrong.\n\nCloses #2144.\n",
      "issue_id": 2144,
      "issue_title": "socket.error exception not caught/wrapped in a requests exception (ConnectionError perhaps?)",
      "issue_body": "I just noticed a case where I had a socket reset on me, and was raised to me as a raw socket error as opposed to something like a requests.exceptions.ConnectionError:\n\n```\n  File \"/home/rtdean/***/***/***/***/***/***.py\", line 67, in dir_parse\n    root = ElementTree.fromstring(response.text)\n  File \"/home/rtdean/.pyenv/versions/2.7.6/lib/python2.7/site-packages/requests-2.3.0-py2.7.egg/requests/models.py\", line 721, in text\n    if not self.content:\n  File \"/home/rtdean/.pyenv/versions/2.7.6/lib/python2.7/site-packages/requests-2.3.0-py2.7.egg/requests/models.py\", line 694, in content\n    self._content = bytes().join(self.iter_content(CONTENT_CHUNK_SIZE)) or bytes()\n  File \"/home/rtdean/.pyenv/versions/2.7.6/lib/python2.7/site-packages/requests-2.3.0-py2.7.egg/requests/models.py\", line 627, in generate\n    for chunk in self.raw.stream(chunk_size, decode_content=True):\n  File \"/home/rtdean/.pyenv/versions/2.7.6/lib/python2.7/site-packages/requests-2.3.0-py2.7.egg/requests/packages/urllib3/response.py\", line 240, in stream\n    data = self.read(amt=amt, decode_content=decode_content)\n  File \"/home/rtdean/.pyenv/versions/2.7.6/lib/python2.7/site-packages/requests-2.3.0-py2.7.egg/requests/packages/urllib3/response.py\", line 187, in read\n    data = self._fp.read(amt)\n  File \"/home/rtdean/.pyenv/versions/2.7.6/lib/python2.7/httplib.py\", line 543, in read\n    return self._read_chunked(amt)\n  File \"/home/rtdean/.pyenv/versions/2.7.6/lib/python2.7/httplib.py\", line 612, in _read_chunked\n    value.append(self._safe_read(chunk_left))\n  File \"/home/rtdean/.pyenv/versions/2.7.6/lib/python2.7/httplib.py\", line 658, in _safe_read\n    chunk = self.fp.read(min(amt, MAXAMOUNT))\n  File \"/home/rtdean/.pyenv/versions/2.7.6/lib/python2.7/socket.py\", line 380, in read\n    data = self._sock.recv(left)\n  File \"/home/rtdean/.pyenv/versions/2.7.6/lib/python2.7/site-packages/gevent-1.0.1-py2.7-linux-x86_64.egg/gevent/socket.py\", line 385, in recv\n    return sock.recv(*args)\nsocket.error: [Errno 104] Connection reset by peer\n```\n\nNot sure if this is by accident or design... in general, I guess I'd expect a requests exception when using requests, but I can start looking for socket errors and the like as well.\n",
      "issue_closed_at": "2014-07-27T13:59:08Z",
      "base_commit": "fe693c492242ae532211e0c173324f09ca8cf227",
      "changes": [
        {
          "file": "requests/models.py",
          "type": "line",
          "name": "line 9",
          "code": "\nimport collections\nimport datetime\n\nfrom io import BytesIO, UnsupportedOperation\nfrom .hooks import default_hooks"
        },
        {
          "file": "requests/models.py",
          "type": "line",
          "name": "line 22",
          "code": "from .packages.urllib3.exceptions import DecodeError\nfrom .exceptions import (\n    HTTPError, RequestException, MissingSchema, InvalidURL,\n    ChunkedEncodingError, ContentDecodingError)\nfrom .utils import (\n    guess_filename, get_auth_from_url, requote_uri,\n    stream_decode_response_unicode, to_key_val_list, parse_header_links,"
        },
        {
          "file": "requests/models.py",
          "type": "function",
          "name": "generate",
          "class_name": "Response",
          "code": "def generate():\n            try:\n                # Special case for urllib3.\n                try:\n                    for chunk in self.raw.stream(chunk_size, decode_content=True):\n                        yield chunk\n                except IncompleteRead as e:\n                    raise ChunkedEncodingError(e)\n                except DecodeError as e:\n                    raise ContentDecodingError(e)\n            except AttributeError:\n                # Standard file-like object.\n                while True:\n                    chunk = self.raw.read(chunk_size)\n                    if not chunk:\n                        break\n                    yield chunk\n\n            self._content_consumed = True"
        }
      ]
    },
    {
      "pr_number": 1713,
      "pr_title": "Fixed #1711",
      "pr_body": "I think i missed the point that cookies can be Dict or CookieJar.\nThis fix will work for MozillaCookieJar.\n",
      "issue_id": 1711,
      "issue_title": "Regression 2.0.1: Using MozillaCookieJar does not work",
      "issue_body": "Could not find an issue raised for this, not sure if this was an expected change either. This is reproducible on master.\n\nExisting code fails on update to `requests-2.0.1`. The cause seems to be triggered by the change at https://github.com/kennethreitz/requests/commit/012f0334ce43fe23044fc58e4246a804db88650d#diff-28e67177469c0d36b068d68d9f6043bfR326\n\nThe parameter `cookies` expects either `Dict` or `CookieJar`. Treating `MozillaCookieJar` as a dict triggers the error in this instance.\n\nThe following code highlights the issue:\n\n``` py\nimport sys\nimport requests\nfrom os.path import expanduser\n\nif sys.version_info.major >= 3:\n    from http.cookiejar import MozillaCookieJar\nelse:\n    from cookielib import MozillaCookieJar\n\nURL = 'https://bugzilla.redhat.com'\nCOOKIE_FILE = expanduser('~/.bugzillacookies')\n\ncookiejar = MozillaCookieJar(COOKIE_FILE)\ncookiejar.load()\n\nrequests.get(URL, cookies=cookiejar)\n```\n\nThe following `AttributeError` is thrown:\n\n```\nTraceback (most recent call last):\n  File \"rtest.py\", line 16, in <module>\n    requests.get(URL, cookies=cookiejar)\n  File \"/tmp/rtestenv/lib/python2.7/site-packages/requests/api.py\", line 55, in get\n    return request('get', url, **kwargs)\n  File \"/tmp/rtestenv/lib/python2.7/site-packages/requests/api.py\", line 44, in request\n    return session.request(method=method, url=url, **kwargs)\n  File \"/tmp/rtestenv/lib/python2.7/site-packages/requests/sessions.py\", line 327, in request\n    self.cookies = cookiejar_from_dict(cookies, cookiejar=self.cookies, overwrite=False)\n  File \"/tmp/rtestenv/lib/python2.7/site-packages/requests/cookies.py\", line 410, in cookiejar_from_dict\n    cookiejar.set_cookie(create_cookie(name, cookie_dict[name]))\nAttributeError: MozillaCookieJar instance has no attribute '__getitem__'\n```\n",
      "issue_closed_at": "2013-12-04T01:33:33Z",
      "base_commit": "340b2459031feb421d678c3c75865c3b11c07938",
      "changes": [
        {
          "file": "requests/cookies.py",
          "type": "function",
          "name": "cookiejar_from_dict",
          "class_name": null,
          "code": "def cookiejar_from_dict(cookie_dict, cookiejar=None, overwrite=True):\n    \"\"\"Returns a CookieJar from a key/value dictionary.\n\n    :param cookie_dict: Dict of key/values to insert into CookieJar.\n    :param cookiejar: (optional) A cookiejar to add the cookies to.\n    :param overwrite: (optional) If False, will not replace cookies\n        already in the jar with new ones.\n    \"\"\"\n    if cookiejar is None:\n        cookiejar = RequestsCookieJar()\n\n    if cookie_dict is not None:\n        names_from_jar = [cookie.name for cookie in cookiejar]\n        for name in cookie_dict:\n            if overwrite or (name not in names_from_jar):\n                cookiejar.set_cookie(create_cookie(name, cookie_dict[name]))\n\n    return cookiejar"
        },
        {
          "file": "requests/sessions.py",
          "type": "line",
          "name": "line 13",
          "code": "from datetime import datetime\n\nfrom .compat import cookielib, OrderedDict, urljoin, urlparse, builtin_str\nfrom .cookies import cookiejar_from_dict, extract_cookies_to_jar, RequestsCookieJar\nfrom .models import Request, PreparedRequest\nfrom .hooks import default_hooks, dispatch_hook\nfrom .utils import to_key_val_list, default_headers"
        },
        {
          "file": "requests/sessions.py",
          "type": "function",
          "name": "prepare_request",
          "class_name": "Session",
          "code": "def prepare_request(self, request):\n        \"\"\"Constructs a :class:`PreparedRequest <PreparedRequest>` for\n        transmission and returns it. The :class:`PreparedRequest` has settings\n        merged from the :class:`Request <Request>` instance and those of the\n        :class:`Session`.\n\n        :param request: :class:`Request` instance to prepare with this\n        session's settings.\n        \"\"\"\n        cookies = request.cookies or {}\n\n        # Bootstrap CookieJar.\n        if not isinstance(cookies, cookielib.CookieJar):\n            cookies = cookiejar_from_dict(cookies)\n\n        # Merge with session cookies\n        merged_cookies = RequestsCookieJar()\n        merged_cookies.update(self.cookies)\n        merged_cookies.update(cookies)\n\n\n        # Set environment's basic authentication if not explicitly set.\n        auth = request.auth\n        if self.trust_env and not auth and not self.auth:\n            auth = get_netrc_auth(request.url)\n\n        p = PreparedRequest()\n        p.prepare(\n            method=request.method.upper(),\n            url=request.url,\n            files=request.files,\n            data=request.data,\n            headers=merge_setting(request.headers, self.headers, dict_class=CaseInsensitiveDict),\n            params=merge_setting(request.params, self.params),\n            auth=merge_setting(auth, self.auth),\n            cookies=merged_cookies,\n            hooks=merge_setting(request.hooks, self.hooks),\n        )\n        return p"
        },
        {
          "file": "requests/sessions.py",
          "type": "function",
          "name": "request",
          "class_name": "Session",
          "code": "def request(self, method, url,\n        params=None,\n        data=None,\n        headers=None,\n        cookies=None,\n        files=None,\n        auth=None,\n        timeout=None,\n        allow_redirects=True,\n        proxies=None,\n        hooks=None,\n        stream=None,\n        verify=None,\n        cert=None):\n        \"\"\"Constructs a :class:`Request <Request>`, prepares it and sends it.\n        Returns :class:`Response <Response>` object.\n\n        :param method: method for the new :class:`Request` object.\n        :param url: URL for the new :class:`Request` object.\n        :param params: (optional) Dictionary or bytes to be sent in the query\n            string for the :class:`Request`.\n        :param data: (optional) Dictionary or bytes to send in the body of the\n            :class:`Request`.\n        :param headers: (optional) Dictionary of HTTP Headers to send with the\n            :class:`Request`.\n        :param cookies: (optional) Dict or CookieJar object to send with the\n            :class:`Request`.\n        :param files: (optional) Dictionary of 'filename': file-like-objects\n            for multipart encoding upload.\n        :param auth: (optional) Auth tuple or callable to enable\n            Basic/Digest/Custom HTTP Auth.\n        :param timeout: (optional) Float describing the timeout of the\n            request.\n        :param allow_redirects: (optional) Boolean. Set to True by default.\n        :param proxies: (optional) Dictionary mapping protocol to the URL of\n            the proxy.\n        :param stream: (optional) whether to immediately download the response\n            content. Defaults to ``False``.\n        :param verify: (optional) if ``True``, the SSL cert will be verified.\n            A CA_BUNDLE path can also be provided.\n        :param cert: (optional) if String, path to ssl client cert file (.pem).\n            If Tuple, ('cert', 'key') pair.\n        \"\"\"\n\n        method = builtin_str(method)\n\n        # Create the Request.\n        req = Request(\n            method = method.upper(),\n            url = url,\n            headers = headers,\n            files = files,\n            data = data or {},\n            params = params or {},\n            auth = auth,\n            cookies = cookies,\n            hooks = hooks,\n        )\n        prep = self.prepare_request(req)\n\n        # Add param cookies to session cookies\n        self.cookies = cookiejar_from_dict(cookies, cookiejar=self.cookies, overwrite=False)\n\n        proxies = proxies or {}\n\n        # Gather clues from the surrounding environment.\n        if self.trust_env:\n            # Set environment's proxies.\n            env_proxies = get_environ_proxies(url) or {}\n            for (k, v) in env_proxies.items():\n                proxies.setdefault(k, v)\n\n            # Look for configuration.\n            if not verify and verify is not False:\n                verify = os.environ.get('REQUESTS_CA_BUNDLE')\n\n            # Curl compatibility.\n            if not verify and verify is not False:\n                verify = os.environ.get('CURL_CA_BUNDLE')\n\n        # Merge all the kwargs.\n        proxies = merge_setting(proxies, self.proxies)\n        stream = merge_setting(stream, self.stream)\n        verify = merge_setting(verify, self.verify)\n        cert = merge_setting(cert, self.cert)\n\n        # Send the request.\n        send_kwargs = {\n            'stream': stream,\n            'timeout': timeout,\n            'verify': verify,\n            'cert': cert,\n            'proxies': proxies,\n            'allow_redirects': allow_redirects,\n        }\n        resp = self.send(prep, **send_kwargs)\n\n        return resp"
        }
      ]
    },
    {
      "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"
        }
      ]
    }
  ]
}