Reb*_*ner 1 python sockets google-app-engine
我正在使用PyAPNS从Python连接到APN.当作为独立脚本运行时,一切正常并且推送消息得以传递.在Google App Engine开发环境中运行时,出现以下错误:
Traceback (most recent call last):
File "/usr/local/google_appengine/lib/webapp2-2.5.2/webapp2.py", line 1535, in __call__
rv = self.handle_exception(request, response, e)
File "/usr/local/google_appengine/lib/webapp2-2.5.2/webapp2.py", line 1529, in __call__
rv = self.router.dispatch(request, response)
File "/usr/local/google_appengine/lib/webapp2-2.5.2/webapp2.py", line 1278, in default_dispatcher
return route.handler_adapter(request, response)
File "/usr/local/google_appengine/lib/webapp2-2.5.2/webapp2.py", line 1102, in __call__
return handler.dispatch()
File "/usr/local/google_appengine/lib/webapp2-2.5.2/webapp2.py", line 572, in dispatch
return self.handle_exception(e, self.app.debug)
File "/usr/local/google_appengine/lib/webapp2-2.5.2/webapp2.py", line 570, in dispatch
return method(*args, **kwargs)
File "/mnt/dev/OmegaHandler.py", line 173, in apns
apns.gateway_server.send_notification("70f23022d76aae0176844087c97f0ff068dd3e3686dbac086b8f82a630d0196a", payload)
File "/mnt/dev/apns.py", line 544, in send_notification
self.write(self._get_notification(token_hex, payload))
File "/mnt/dev/apns.py", line 273, in write
return self._connection().write(string)
File "/mnt/dev/apns.py", line 254, in _connection
self._connect()
File "/mnt/dev/apns.py", line 230, in _connect
self._ssl = wrap_socket(self._socket, self.key_file, self.cert_file)
File "/usr/local/lib/python2.7/ssl.py", line 911, in wrap_socket
ciphers=ciphers)
File "/usr/local/lib/python2.7/ssl.py", line 535, in __init__
if sock.getsockopt(SOL_SOCKET, SO_TYPE) != SOCK_STREAM:
File "/mnt/dev/rsocket.py", line 225, in meth
return getattr(self._sock,name)(*args)
error: [Errno 22] Invalid argument
Run Code Online (Sandbox Code Playgroud)
我使用此答案来帮助开发环境中的"权限被拒绝"错误.同样,当独立运行时,使用完全相同的库和代码都可以正常运行.错误似乎来自于getsockopt
使用选项调用(1, 3)
.我再也不知道了.
为什么会在开发环境中发生这种情况,使用库存Python socket.py修补猴子并且不会出现在独立脚本中.请不要回答我需要启用计费才能使用Socket API,这只发生在开发环境中.谢谢.
编辑:这也发生在不同的计算机上和不同的服务器上(常规SSL套接字连接到Google IP地址范围之外的某些服务器).
小智 7
不确定这是否仍然有用或者您是否在其他地方找到了答案.
我有一个非常类似的问题,试图requests
在相当标准的应用程序引擎代码中使用该库.
tl; dr - App Engine补丁/提供沙盒socket
.该ssl
模块socket
通过自己的导入从命名空间中快照对象.如果socket
(和_socket
)在ssl
导入时不是正确的版本,ssl
将最终使用基于GAE-patch/sandbox的值SOL_SOCKET
和SO_TYPE
(可能不正确),而不是应该来自基本系统安装的值.如果你确保你的socket
(和_socket
)副本是正确的,那就行了ssl
.
我提交了支持票与谷歌,得到了回推来实现这个.
对我来说实际上交换物理socket.py
实现并不是那么微不足道,所以我选择了这个解决方案的路线.
这让我大部分都在那里,但由于SSL问题仍然出现问题.
在App Engine开发沙箱(即dev_appserver.py
)中发出HTTP请求时,我已经测试了与以下代码几乎完全相同的内容,以使SSL使用原始套接字(而不是URLFetch ):
import os
def _patch_ssl_support():
# Only patch SSL support if it's local dev.
if not os.environ.get('SERVER_SOFTWARE', '').startswith('Development'):
return
import imp
import inspect
from google.appengine.tools.devappserver2.python import sandbox
# Allow the sandbox to read _ssl and _socket.
sandbox._WHITE_LIST_C_MODULES += ['_socket', '_ssl']
# Use the system socket.
# I used inspect here, but many core modules should work.
# It ultimately depends on your python installation.
runtime_path = os.path.realpath(inspect.getsourcefile(inspect))
runtime_dir = os.path.dirname(runtime_path)
# Patch and reload the socket module implementation.
system_socket = os.path.join(runtime_dir, 'socket.py')
imp.load_source('socket', system_socket)
# Patch and reload the ssl module implementation.
system_ssl = os.path.join(runtime_dir, 'ssl.py')
imp.load_source('ssl', system_ssl)
# Patch and/or reload any other libraries you suspect may have copied values
# from the socket or ssl namespaces.
# Patch SSL support before you do anything else.
_patch_ssl_support()
import webapp2
# Setup app engine application, or something that runs in an app engine runtime:
app = webapp2.WSGIApplication(routes)
Run Code Online (Sandbox Code Playgroud)
归档时间: |
|
查看次数: |
612 次 |
最近记录: |