Я пытаюсь реализовать веб-сокеты на сервере openshift.com (который должен их поддерживать).
openshift.com предоставляет мне WSGI, поэтому я встраиваю в него свой cherrypy
, чтобы мой wsgi.py
script определял объект application
. Кроме того, cherrypy
имеет инструмент веб-сокета, как определено ws4py
.
Это минимальное приложение cherrypy, которое работает под WSGI в OpenShift и которое должно также использовать веб-сокеты!
import cherrypy
from ws4py.server.cherrypyserver import WebSocketPlugin, WebSocketTool
from ws4py.websocket import EchoWebSocket
import atexit
import logging
# see http://tools.cherrypy.org/wiki/ModWSGI
cherrypy.config.update({'environment': 'embedded'})
if cherrypy.__version__.startswith('3.0') and cherrypy.engine.state == 0:
cherrypy.engine.start(blocking=False)
atexit.register(cherrypy.engine.stop)
class Root(object):
def index(self): return 'I work!'
def ws(self): print('THIS IS NEVER PRINTED :(')
index.exposed=True
ws.exposed=True
# registering the websocket
conf={'/ws':{'tools.websocket.on': True,'tools.websocket.handler_cls': EchoWebSocket}}
WebSocketPlugin(cherrypy.engine).subscribe()
cherrypy.tools.websocket = WebSocketTool()
#show stacktraces in console (for some reason this is not default in cherrypy+WSGI)
logger = logging.getLogger()
logger.setLevel(logging.INFO)
stream = logging.StreamHandler()
stream.setLevel(logging.INFO)
logger.addHandler(stream)
application = cherrypy.Application(Root(), script_name='', config=conf)
Все работает прекрасно, за исключением случаев, когда я создаю веб-сокет (подключаясь к ws://myserver:8000/ws
), я получаю трассировку стека:
cherrypy/_cplogging.py, 214, HTTP Traceback (most recent call last):
File "cherrypy/_cprequest.py", line 661, in respond
self.hooks.run('before_request_body')
File "cherrypy/_cprequest.py", line 114, in run
raise exc
File "cherrypy/_cprequest.py", line 104, in run
hook()
File "cherrypy/_cprequest.py", line 63, in __call__
return self.callback(**self.kwargs)
File "ws4py/server/cherrypyserver.py", line 200, in upgrade
ws_conn = get_connection(request.rfile.rfile)
AttributeError: 'mod_wsgi.Input' object has no attribute 'rfile'
(Я вручную удалил абсолютный путь из имен файлов) PS: Я использую python3.3
, cherrypy==3.5.0
, ws4py==0.3.4
.
Мне непонятно:
- если это отсутствие совместимости между cherrypy и ws4py в среде WSGI.
- если это проблема ws4py в среде WSGI
- если это потому, что веб-сокеты Openshift имеют другой порт, чем http
PPS: это полноценный проект OpenShift, который вы можете запустить и попробовать сами: https://github.com/spocchio/wsgi-cherrypy-ws4py