如何在Tornado中随意发送websocket消息?

时间:2015-09-07 20:33:43

标签: python websocket tornado

我对Tornado很新,想知道是否可以从我的Python程序中随意向所有客户端发送消息(write_message)?例如,假设我的程序正在监视目录以查看文件是否存在。当它出现时,我想向浏览器客户端发送该文件存在的Web套接字消息。我似乎无法理解如何在没有收到websocket消息(on_message handler)的情况下调用“write_message”方法。

即使我使用“PeriodicCallback”方法,我仍然不清楚我是如何实际调用“write_message”方法的。有没有关于如何在“on_message”处理程序中调用“write_message”的示例?

1 个答案:

答案 0 :(得分:6)

您需要保留一组打开的websockets并随意遍历该集合以发送消息。

例如,我会在客户端连接到your.domain.example/test/的任何时候发送消息,但只要您想发送内容,这个想法就是相同的:

import os.path
import logging

from tornado import ioloop, web, websocket


SERVER_FOLDER = os.path.abspath(os.path.dirname(__file__))
LOGGER = logging.getLogger('tornado.application')


class TestHandler(web.RequestHandler):
    def get(self):
        server = ioloop.IOLoop.current()
        data = "whatever"
        server.add_callback(DefaultWebSocket.send_message, data)
        self.set_status(200)
        self.finish()


class DefaultWebSocket(websocket.WebSocketHandler):
    live_web_sockets = set()

    def open(self):
        LOGGER.debug("WebSocket opened")
        self.set_nodelay(True)
        self.live_web_sockets.add(self)
        self.write_message("you've been connected. Congratz.")

    def on_message(self, message):
        LOGGER.debug('Message incomming: %s', message)

    def on_close(self):
        LOGGER.debug("WebSocket closed")

    @classmethod
    def send_message(cls, message):
        removable = set()
        for ws in cls.live_web_sockets:
            if not ws.ws_connection or not ws.ws_connection.stream.socket:
                removable.add(ws)
            else:
                ws.write_message(message)
        for ws in removable:
            cls.live_web_sockets.remove(ws)


def serve_forever(port=80, address=''):
    application = web.Application([
            (r"/test/", TestHandler),
            (r"/websocket/", DefaultWebSocket),
            ...
        ],
        static_path=os.path.join(SERVER_FOLDER, ...),
        debug=True,
    )
    application.listen(port, address)
    LOGGER.debug(
            'Server listening at http://%s:%d/',
            address or 'localhost', port)
    ioloop.IOLoop.current().start()


if __name__ == "__main__":
    serve_forever()

您显然需要使用以下JavaScript在浏览器中创建一个websocket:

socket = new WebSocket('ws://your.domain.example:80/websocket/');

并相应地管理它。