我遇到了一个非常奇怪的错误,它与SSL和python有关google.com(或者更常见的是我认为有多个证书链的域).每当我尝试向我发出请求时,https://*.google.com/whatever
我都会收到以下错误:
SSLError: ("bad handshake: Error([('SSL routines', 'SSL3_GET_SERVER_CERTIFICATE', 'certificate verify failed')],)",) while doing GET request to URL: https://google.com/到目前为止我做了什么
我已经经历了很多尝试让这个工作起来的箍,并且现在因为我不知道该怎么做而试图发布到Stack Overflow.这是我尝试过的:
注意到date
返回的日期比实时落后2分钟(可能使我的证书无效).假设它会验证证书,我修复了这个问题.这并没有解决问题.
发现Python 2.7.9从Python 3向后移植了一些SSL库.我从Python 2.7.6升级到2.7.9,假设更新(包括此线程中列出的修复程序:https://serverfault.com/questions/692110/错误-python2-as-a-https-client-with-an-nginx-server-and-ssl-certificate-ch)会解决它.没有运气,同样的错误.
显然设置verify=False
工作,但我们不愿意安全,我们需要开始verify=True
工作.
curl https://google.com
也按预期工作.这就是我知道它与Python有关的方式.
$ python -V Python 2.7.9 $ pip list | grep -e requests requests (2.9.1) $ uname-a # ubuntu 14.04 Linux staging.example.com 3.13.0-48-generic #80-Ubuntu SMP Thu Mar 12 11:16:15 UTC 2015 x86_64 x86_64 x86_64 GNU/Linux例
这仅适用于https上的Google域名.这是一个例子:
$ ipython Python 2.7.9 (default, Jan 6 2016, 21:37:32) Type "copyright", "credits" or "license" for more information. IPython 4.0.1 -- An enhanced Interactive Python. ? -> Introduction and overview of IPython's features. %quickref -> Quick reference. help -> Python's own help system. object? -> Details about 'object', use 'object??' for extra details. In [1]: import requests In [2]: requests.get('https://facebook.com', verify=True) Out[2]:In [3]: requests.get('https://stackoverflow.com', verify=True) Out[3]: In [4]: requests.get('https://spotify.com', verify=True) Out[4]: In [5]: requests.get('http://google.com', verify=True) # notice the http Out[5]: In [6]: requests.get('https://google.com', verify=True) --------------------------------------------------------------------------- SSLError Traceback (most recent call last) in () ----> 1 requests.get('https://google.com', verify=True) /example/.virtualenv/example/lib/python2.7/site-packages/requests/api.pyc in get(url, params, **kwargs) 65 66 kwargs.setdefault('allow_redirects', True) ---> 67 return request('get', url, params=params, **kwargs) 68 69 /example/.virtualenv/example/lib/python2.7/site-packages/requests/api.pyc in request(method, url, **kwargs) 51 # cases, and look like a memory leak in others. 52 with sessions.Session() as session: ---> 53 return session.request(method=method, url=url, **kwargs) 54 55 /example/.virtualenv/example/lib/python2.7/site-packages/requests/sessions.pyc in request(self, method, url, params, data, headers, cookies, files, auth, timeout, allow_redirects, proxies, hooks, stream, verify, cert, json) 466 } 467 send_kwargs.update(settings) --> 468 resp = self.send(prep, **send_kwargs) 469 470 return resp /example/.virtualenv/example/lib/python2.7/site-packages/requests/sessions.pyc in send(self, request, **kwargs) 574 575 # Send the request --> 576 r = adapter.send(request, **kwargs) 577 578 # Total elapsed time of the request (approximately) /example/.virtualenv/example/lib/python2.7/site-packages/requests/adapters.pyc in send(self, request, stream, timeout, verify, cert, proxies) 445 except (_SSLError, _HTTPError) as e: 446 if isinstance(e, _SSLError): --> 447 raise SSLError(e, request=request) 448 elif isinstance(e, ReadTimeoutError): 449 raise ReadTimeout(e, request=request) SSLError: ("bad handshake: Error([('SSL routines', 'SSL3_GET_SERVER_CERTIFICATE', 'certificate verify failed')],)",)
Liam Horne.. 31
我找到了解决方案.在certifi
运行的版本中似乎存在一个主要问题.我从这个(非常长的)GitHub问题中找到了这个:https://github.com/certifi/python-certifi/issues/26
TL; DR
pip uninstall -y certifi && pip install certifi==2015.04.28
我找到了解决方案.在certifi
运行的版本中似乎存在一个主要问题.我从这个(非常长的)GitHub问题中找到了这个:https://github.com/certifi/python-certifi/issues/26
TL; DR
pip uninstall -y certifi && pip install certifi==2015.04.28