【问题标题】:Twilio Python Module Errors After Compiling编译后的 Twilio Python 模块错误
【发布时间】:2014-01-18 07:14:46
【问题描述】:

我编写了一个简单的程序,可以打开一个 csv 文件并在其中输入所有数字。我正在使用 Twilio (twilio-python) 作为服务提供商。我的代码可以作为 python 脚本正常工作。但是,当我编译脚本(使用 py2exe)时,exe 文件出错。这是我从日志文件中收到的错误......

 Traceback (most recent call last):
 File "sms.py", line 39, in <module>
 File "twilio\rest\resources\messages.pyc", line 112, in create
 File "twilio\rest\resources\base.pyc", line 352, in create_instance
 File "twilio\rest\resources\base.pyc", line 204, in request
 File "twilio\rest\resources\base.pyc", line 129, in make_twilio_request
 File "twilio\rest\resources\base.pyc", line 101, in make_request
 File "httplib2\__init__.pyc", line 1570, in request
 File "httplib2\__init__.pyc", line 1317, in _request
 File "httplib2\__init__.pyc", line 1252, in _conn_request
 File "httplib2\__init__.pyc", line 1021, in connect
 File "httplib2\__init__.pyc", line 80, in _ssl_wrap_socket
 File "ssl.pyc", line 387, in wrap_socket
 File "ssl.pyc", line 141, in __init__
 ssl.SSLError: [Errno 185090050] _ssl.c:340: error:0B084002:x509 certificate               
 routines:X509_load_cert_crl_file:system lib

当我使用未编译的代码(如下)时,我没有收到此错误

  import sys #2 params --- /path/to/contact/file    --- up to 160 char msg
  import csv
  import time

  from twilio.rest import TwilioRestClient 
  ACCOUNT_SID = "**************************" 
  AUTH_TOKEN  = "**************************" 
  client = TwilioRestClient(ACCOUNT_SID, AUTH_TOKEN) 

  sys.argv.pop(0)
  contactFile = sys.argv[0]
  sys.argv.pop(0)
  msg = (' ').join(sys.argv)

  print contactFile
  print " "
  print msg

  info = []
  with open(contactFile,'rb') as csvfile:
   reader = csv.reader(csvfile, delimiter=',', quotechar='|')
        for row in reader:
        info.append(row)

  contactCount = len(info)-1

  if contactCount > 0:
     #remove first item from list because its not a value that is needed....
     info.pop(0)

   for i in info:
       print " "
       contactName = i[0]
       phoneNumber = i[1]
       print "Texting " + contactName + "... \n"
       client.messages.create(
       to=phoneNumber, 
       from_="+14782856136",
       body=msg   
       )
       time.sleep(1.5)
    else:
     print("SMSify Error \n The contact file doesn't have any contacts in it.")

有什么想法吗??

编辑:

这是我的 setup.py 文件

from distutils.core import setup
import py2exe, sys, os
sys.argv.append('py2exe')
Mydata_files = [('cacert.pem', ['C:\\Python27\\Lib\\site-      
packages\\twilio\\conf\\cacert.pem'])]

setup(
   console=['sms.py'],
   data_files = Mydata_files,
   options={
              "py2exe":{
                    "bundle_files": 1,
                    "compressed": True
                }
         }
 )

【问题讨论】:

  • 一个简单的解决方案是 - 您可以将证书物理地放置在您的可分发文件中,并在代码中明确传递证书位置。见stackoverflow.com/a/34227183/318700

标签: python sms twilio


【解决方案1】:

这是因为 self-signed certificate 文件在包中丢失。

requestshttplib2 模块同样存在此问题。

例如,如果您有一个名为 req_example.py 的文件使用 request 模块:

import requests
url = 'https://google.com/'
requests.get(url)

当你以python req_example.py 运行它时它可以工作,但是当它捆绑它时,它就不起作用了。

或者,如果您有一个名为 http2_example.py 的文件使用 http2 模块:

import httplib2
url = 'https://google.com/'
http = httplib2.Http()
http.request(url)

当你以 python http2_example.py 运行它时它可以工作,但是当它捆绑它时,它就不起作用了。

要解决这个问题,您有两种选择,一种是bad,另一种是good

  1. 禁用验证 SSL 证书:

    requests 模块执行此操作:

    import requests
    url = 'https://google.com/'
    requests.get(url, verify=False)
    

    对于httplib2 模块:

    import httplib2
    http = httplib2.Http(disable_ssl_certificate_validation=True)
    http.request(url)
    
  2. self-signed certificate 文件添加到捆绑包中:

    对于requests 模块,文件cacert.pem 位于:

    .../PythonXX/lib/site-packages/requests/cacert.pem

    对于httplib2,模块位于:

    .../PythonXX/lib/site-packages/httplib2/cacerts.txt

    对于它们中的每一个,您都可以将其复制到您的项目内部(或只是地址),

    并配置 setup.py 以包含它:

    setup(console=['temp.py'],
        # for `requests` module
        data_files=['cacert.pem'] ) # or data_files=['cacerts.txt'] ) for `httplib2`
    

    并将您的代码更改为使用该代码,用于request 模块:

    import os 
    import requests 
    url = 'https://google.com/' 
    cert ='cacert.pem'
    # or os.environ['REQUESTS_CA_BUNDLE'] = cert 
    os.environ['REQUESTS_CA_BUNDLE'] = os.path.join(os.getcwd(), cert)
    requests.get(url)
    

    对于httplib2 模块:

    import httplib2 
    cert = 'cacerts.txt' 
    http = httplib2.Http(ca_certs=cert) 
    http.request(url)
    

    或者如果你的httplib2 版本是0.8,你可以创建一个文件 should 命名为ca_certs_locater.py,并定义一个get 函数, ca_certs 文件的返回路径。

    def get():
        return 'cacerts.txt'
    

好的,现在对于您的错误,对于 twilio 模块,它是 use httplib2cacert.pem 它在:

.../twilio/conf/cacert.pem

所以你需要把这个文件添加到setup.py,如上所述。

但是twilio 本身有一个名为get_cert_file 的函数,它将ca_cert 文件传递​​给httplib2

我认为,如果您使用上述ca_certs_locater.py,它也适用, 但如果没有,你还有一个ugly 选项,所以你可以对twilioget_cert_file 函数进行猴子补丁:

from twilio.rest.resources.base import get_cert_file
get_cert_file = lambda: 'cacert.pem'

请注意,这可能是twilio 甚至py2exePyInstaller 的问题。

【讨论】:

  • 好的。所以我在 setup.py 文件中添加了 twilio cacert.pem 文件。我的 setup.py 文件发布在上面。我仍然遇到同样的错误。
  • 这是行不通的,因为twilio想从自己的路径加载它,即.../twilio/conf/cacert.pem,但捆绑包中不存在(在library.zip文件中),试试ca_certs_locater.py方式,如果不行,试试patch方式。
  • 好的。所以我创建了文件 ca_certs_locater.py 并添加了 get 函数。现在,我应该将它捆绑在 setup.py 中吗?答案有点模糊。感谢您的帮助!
  • 无需将ca_certs_locater.py文件添加到setup.py,只需将此文件添加到项目的根目录内,然后此文件会自动添加到捆绑包中。如果这种方法不起作用,请尝试第二种方法:只需将上述补丁的代码(上面的两行)添加到项目的主文件中。另请注意,对于这两种方式,您应该将cacert.pem 文件包含到setup.py
【解决方案2】:

我在使用 twilio 和 pyinstaller 时遇到了同样的问题,并且能够通过修改 twilio\rest\resources 中的 base.py 模块来修复它:

   def get_cert_file():
 """ Get the cert file location or bail """
# XXX - this currently fails test coverage because we don't actually go
# over the network anywhere. Might be good to have a test that stands up a
# local server and authenticates against it.
  try:
    # Apparently __file__ is not available in all places so wrapping this
    # in a try/catch
    current_path = os.path.realpath(__file__)
    #ca_cert_path = os.path.join(current_path, "..", "..", "..",    (old path)
     #                           "conf", "cacert.pem")
    ca_cert_path = os.getcwd() + '\Config\cacert.pem' (my new path)

    return os.path.abspath(ca_cert_path)

(我将我的 cacert.pem 文件存储在我的主脚本目录之外的 Config 文件夹中)

【讨论】:

    【解决方案3】:

    py2exe 应该有一种方法可以捆绑非 Python 文件,例如模板,或者在本例中是存储在 cacert.pem 中的 SSL 证书。通常这是使用 MANIFEST.in 自动完成的,但我不确定该项目如何处理它。查看那里的文档以获取更多信息。

    【讨论】:

      猜你喜欢
      • 2016-05-29
      • 2012-09-17
      • 2012-11-19
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2021-08-29
      • 2012-11-21
      相关资源
      最近更新 更多