Websocket(Python 3.5中的ws4py)无法在Docker容器中运行

时间:2015-10-23 10:20:46

标签: python-3.x websocket docker cherrypy ws4py

我有一个简单的Web应用程序,它使用CherryPy作为RESTful后端,使用ws4py作为websockets。

class WebSocketHandler(WebSocket):
def __init__(self, *args, **kw):
    WebSocket.__init__(self, *args, **kw)
    print('Connected to websocket server!')
    SUBSCRIBERS.add(self)

def received_message(self, message):
    print('Websocket identified itself?')
    SUBSCRIBERS_WITH_ID[message.data] = self

Python websocket代码:

cherrypy.config.update({'server.socket_host': 0.0.0.0,
                        'server.socket_port': 8080})

cherrypy.tools.CORS = cherrypy.Tool('before_handler', CORS)

WebSocketPlugin(cherrypy.engine).subscribe()
cherrypy.tools.websocket = WebSocketTool()

cherrypy.tree.mount(
    DummyAPI(), '/api/dummy', config={
        '/': {
            'request.dispatch': cherrypy.dispatch.MethodDispatcher(),
            'tools.CORS.on': True
        }
    }
)

cherrypy.tree.mount(
    WebSocketAPI(), '/ws', config={
        '/': {
            'tools.websocket.on': True,
            'tools.websocket.handler_cls': WebSocketHandler
        }
    }
)

cherrypy.engine.start()
cherrypy.engine.block()

Python CherryPy代码:

var req = new XMLHttpRequest();
req.onload = function (e) {
    console.log('Response code: ' + req.responseText);
    result_id = req.responseText;
    websocketConnect();
};
req.open('POST', api_url);
req.send();

Javascript代码:

POST到API:

function websocketConnect() {
websocket = new WebSocket(websocket_url);
websocket.onopen = function (evt) { 
      console.log("Connected to WebSocket server.");
      websocket.send(result_id)};

Websocket部分:

FROM python
MAINTAINER xxx

RUN pip3.5 install cherrypy ws4py
RUN mkdir /var/www

COPY python /var/www/python

EXPOSE 8080

CMD ["/usr/local/bin/python3.5", "/var/www/python/simplews.py"]

这在我的机器上正常工作并部署在Amazon EC2上。但是,当我尝试将其部署在Docker容器中时,websockets无法正常工作。

这是我的Dockerfile:

sudo docker run  -v /tmp/dockertmp/:/tmp/ -p 8080:8080 -i 563a51d13f59
[23/Oct/2015:10:09:00] ENGINE Bus STARTING
[23/Oct/2015:10:09:00] ENGINE Starting WebSocket processing
[23/Oct/2015:10:09:00] ENGINE Started monitor thread '_TimeoutMonitor'.
[23/Oct/2015:10:09:00] ENGINE Started monitor thread 'Autoreloader'.
[23/Oct/2015:10:09:01] ENGINE Serving on http://0.0.0.0:8080
[23/Oct/2015:10:09:01] ENGINE Bus STARTED
172.17.42.1 - - [23/Oct/2015:10:09:08] "POST ..."
Connected to websocket server!

从Docker运行时的输出:

enter code here
sudo docker run  -v /tmp/dockertmp/:/tmp/ -p 8080:8080 -i 563a51d13f59
[23/Oct/2015:10:15:30] ENGINE Bus STARTING
[23/Oct/2015:10:15:30] ENGINE Starting WebSocket processing
[23/Oct/2015:10:15:30] ENGINE Started monitor thread '_TimeoutMonitor'.
[23/Oct/2015:10:15:30] ENGINE Started monitor thread 'Autoreloader'.
[23/Oct/2015:10:15:30] ENGINE Serving on http://0.0.0.0:8080
[23/Oct/2015:10:15:30] ENGINE Bus STARTED
172.17.42.1 - - [23/Oct/2015:10:15:33] "POST ..."
Connected to websocket server!
^C[23/Oct/2015:10:15:35] ENGINE Keyboard Interrupt: shutting down bus
[23/Oct/2015:10:15:35] ENGINE Bus STOPPING
[23/Oct/2015:10:15:40] ENGINE HTTP Server     cherrypy._cpwsgi_server.CPWSGIServer(('0.0.0.0', 8080)) shut down
[23/Oct/2015:10:15:40] ENGINE Stopped thread '_TimeoutMonitor'.
[23/Oct/2015:10:15:40] ENGINE Terminating WebSocket processing
[23/Oct/2015:10:15:40] ENGINE Stopped thread 'Autoreloader'.
[23/Oct/2015:10:15:40] ENGINE Bus STOPPED
[23/Oct/2015:10:15:40] ENGINE Bus EXITING
[23/Oct/2015:10:15:40] ENGINE Bus EXITED
[23/Oct/2015:10:15:40] ENGINE Waiting for child threads to terminate...
Websocket identified itself?

它永远不会到达Javascript文件能够通过websocket发送数据的阶段。 Chrome调试将WS状态显示为"待定"。

然而,当我退出Docker中的CherryPy应用程序时,websocket消息会通过:

python3.5 python/simplews.py 
[23/Oct/2015:06:09:45] ENGINE Bus STARTING
[23/Oct/2015:06:09:45] ENGINE Starting WebSocket processing
[23/Oct/2015:06:09:45] ENGINE Started monitor thread '_TimeoutMonitor'.
[23/Oct/2015:06:09:45] ENGINE Started monitor thread 'Autoreloader'.
[23/Oct/2015:06:09:45] ENGINE Serving on http://0.0.0.0:8080
[23/Oct/2015:06:09:45] ENGINE Bus STARTED
127.0.0.1 - - [23/Oct/2015:06:09:48] "POST ..."
Connected to websocket server!
Websocket identified itself?

在没有Docker的情况下运行时的输出:

[23/Oct/2015:10:15:40] ENGINE Waiting for child threads to terminate...
Websocket identified itself?

在非Docker案例中,websocket立即识别自己。我无法解决这个问题。 websocket接收器线程是否以某种方式卡在Docker中?

Docker日志中的可疑部分是当应用程序退出时,websocket收到的消息最终到达:

import static org.hamcrest.collection.IsArrayContainingInOrder.arrayContaining;

...

@Test
public void inOrder1() {
    assertThat(new String[]{"foo", "bar"}, arrayContaining(equalTo("foo"), equalTo("bar")));
}

@Test(expected =  AssertionError.class)
public void inOrder2() {
    assertThat(new String[]{"bar", "foo"}, arrayContaining(equalTo("foo"), equalTo("bar")));
    // Expected: ["foo", "bar"]
    //      but: item 0: was "bar"
}

@Test(expected =  AssertionError.class)
public void inOrder3() {
    assertThat(new String[]{"foo", "bar", "lux"}, arrayContaining(equalTo("foo"), equalTo("bar")));
    // Expected: ["foo", "bar"] in any order
    //      but: Not matched: "lux"
}

@Test(expected =  AssertionError.class)
public void inOrder4() {
    assertThat(new String[]{"foo", "bar"}, arrayContaining(equalTo("foo"), equalTo("bar"), equalTo("lux")));
    // Expected: ["foo", "bar", "lux"] in any order
    //      but: No item matched: "lux" in ["foo", "bar"]
}

1 个答案:

答案 0 :(得分:0)

用threading.Thread替换Python多处理模块修复了问题。我在POST方法处理程序中使用了多处理。

这是对实际项目的修复提交:

https://github.com/sevagh/youtube-audio-dl/commit/9e23ae174f8791206b934608a1d4d702b411d18e