{
  "Selected_candidate": {
    "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"
      }
    ]
  },
  "Justification": "Candidate A addresses a core issue related to exception handling in the requests library, which is directly related to the CURRENT bug concerning the way urllib3's exceptions are wrapped. Both reports involve understanding the flow of exceptions and their propagation through the requests API. Since Candidate A demonstrates a specific case of failure due to handling of cookies, it highlights how different types of errors can surface, thus providing insight into similar behaviors when exceptions from urllib3 are encountered. This context could be useful for improving how requests manages and reports errors, which is at the heart of the CURRENT bug.",
  "instance_id": "psf__requests-2674",
  "repo": "psf/requests",
  "created_at": "2015-07-17T08:33:52Z",
  "problem_statement": "urllib3 exceptions passing through requests API\nI don't know if it's a design goal of requests to hide urllib3's exceptions and wrap them around requests.exceptions types.\n\n(If it's not IMHO it should be, but that's another discussion)\n\nIf it is, I have at least two of them passing through that I have to catch in addition to requests' exceptions. They are requests.packages.urllib3.exceptions.DecodeError and requests.packages.urllib3.exceptions.TimeoutError (this one I get when a proxy timeouts)\n\nThanks!\n\n",
  "patch": "diff --git a/requests/adapters.py b/requests/adapters.py\n--- a/requests/adapters.py\n+++ b/requests/adapters.py\n@@ -19,6 +19,7 @@\n from .utils import (DEFAULT_CA_BUNDLE_PATH, get_encoding_from_headers,\n                     prepend_scheme_if_needed, get_auth_from_url, urldefragauth)\n from .structures import CaseInsensitiveDict\n+from .packages.urllib3.exceptions import ClosedPoolError\n from .packages.urllib3.exceptions import ConnectTimeoutError\n from .packages.urllib3.exceptions import HTTPError as _HTTPError\n from .packages.urllib3.exceptions import MaxRetryError\n@@ -421,6 +422,9 @@ def send(self, request, stream=False, timeout=None, verify=True, cert=None, prox\n \n             raise ConnectionError(e, request=request)\n \n+        except ClosedPoolError as e:\n+            raise ConnectionError(e, request=request)\n+\n         except _ProxyError as e:\n             raise ProxyError(e)\n \n"
}