web-dev-qa-db-ja.com

python(tornado)でwebsocketループの外で関数を実行する方法

WebSocketを介した公開Twitterストリームの小さな例を設定しようとしています。これは私のwebsocket.pyであり、機能しています。

私が疑問に思っているのは、クラスWSHandlerの「外部」からWebSocketと対話するにはどうすればよいですか(つまり、websocket.jsからメッセージを受信したときに応答するだけではありません)。この同じスクリプト内で、「hello!」を投稿する他の関数を実行したいとします。 5秒ごとに、クライアント側からの操作なしでWebSocket(ブラウザー)に送信します。どうすればそれができますか?

ですから、以下のようなクラスの扱い方については、基本的な初心者の質問のようなものだと思います。あらゆる方向へのポインタは大歓迎です!

import os.path
import tornado.httpserver
import tornado.websocket
import tornado.ioloop
import tornado.web

# websocket
class FaviconHandler(tornado.web.RequestHandler):
    def get(self):
        self.redirect('/static/favicon.ico')

class WebHandler(tornado.web.RequestHandler):
    def get(self):
        self.render("websockets.html")

class WSHandler(tornado.websocket.WebSocketHandler):
    def open(self):
        print 'new connection'
        self.write_message("Hi, client: connection is made ...")

    def on_message(self, message):
        print 'message received: \"%s\"' % message
        self.write_message("Echo: \"" + message + "\"")
        if (message == "green"):
            self.write_message("green!")

    def on_close(self):
        print 'connection closed'



handlers = [
    (r"/favicon.ico", FaviconHandler),
    (r'/static/(.*)', tornado.web.StaticFileHandler, {'path': 'static'}),
    (r'/', WebHandler),
    (r'/ws', WSHandler),
]

settings = dict(
    template_path=os.path.join(os.path.dirname(__file__), "static"),
)

application = tornado.web.Application(handlers, **settings)

if __name__ == "__main__":
    http_server = tornado.httpserver.HTTPServer(application)
    http_server.listen(8888)
    tornado.ioloop.IOLoop.instance().start()
25
knutole

あなたは

_IOLoop.add_timeout(deadline, callback)
_

指定された期限のタイムアウト(ワンショットですが、スケジュールを変更できます)でコールバックを呼び出すか、

_tornado.ioloop.PeriodicCallback_より定期的なタスクがある場合。

参照: http://www.tornadoweb.org/en/stable/ioloop.html#tornado.ioloop.IOLoop.add_timeout

更新:いくつかの例

_import datetime

def test():
    print "scheduled event fired"
...

if __name__ == "__main__":
    http_server = tornado.httpserver.HTTPServer(application)
    http_server.listen(8888)
    main_loop = tornado.ioloop.IOLoop.instance()
    # Schedule event (5 seconds from now)
    main_loop.add_timeout(datetime.timedelta(seconds=5), test)
    # Start main loop
    main_loop.start()
_

5秒後にtest()を呼び出します。

更新2:

_import os.path
import tornado.httpserver
import tornado.websocket
import tornado.ioloop
import tornado.web

# websocket
class FaviconHandler(tornado.web.RequestHandler):
    def get(self):
        self.redirect('/static/favicon.ico')

class WebHandler(tornado.web.RequestHandler):
    def get(self):
        self.render("websockets.html")

class WSHandler(tornado.websocket.WebSocketHandler):
    def open(self):
        print 'new connection'
        self.write_message("Hi, client: connection is made ...")
        tornado.ioloop.IOLoop.instance().add_timeout(datetime.timedelta(seconds=5), self.test)

    def on_message(self, message):
        print 'message received: \"%s\"' % message
        self.write_message("Echo: \"" + message + "\"")
        if (message == "green"):
            self.write_message("green!")

    def on_close(self):
        print 'connection closed'

    def test(self):
        self.write_message("scheduled!")

handlers = [
    (r"/favicon.ico", FaviconHandler),
    (r'/static/(.*)', tornado.web.StaticFileHandler, {'path': 'static'}),
    (r'/', WebHandler),
    (r'/ws', WSHandler),
]

settings = dict(
    template_path=os.path.join(os.path.dirname(__file__), "static"),
)

application = tornado.web.Application(handlers, **settings)

import datetime

if __name__ == "__main__":
    http_server = tornado.httpserver.HTTPServer(application)
    http_server.listen(8888)
    tornado.ioloop.IOLoop.instance().start()
_
19
pr0gg3d

私は同様の問題に遭遇しました。これが私の解決策です。これが誰かに役立つことを願っています

wss = []
class wsHandler(tornado.websocket.WebSocketHandler):
    def open(self):
        print 'Online'
        if self not in wss:
            wss.append(self)

    def on_close(self):
        print 'Offline'
        if self in wss:
            wss.remove(self)

def wsSend(message):
    for ws in wss:
        ws.write_message(message)

WebSocketにメッセージを送信するには、次を使用します。

wsSend(message)

wsSend update

たまにwsSendで例外が発生しています。これを修正するために、コードを次のように少し変更しました。

def wsSend(message):
    for ws in wss:
        if not ws.ws_connection.stream.socket:
            print "Web socket does not exist anymore!!!"
            wss.remove(ws)
        else:
            ws.write_message(message)
14
Barmaley

これを行う1つの方法は、pub-subモジュールを使用することです。

つまり、接続subscribeがあり、接続ごとにタイムアウトを設定するのではなく、1つのタイムアウトをpublishに設定するだけです。上記の期間の後。

おそらく最も実装されているものの1つはredisです。竜巻専用のモジュールもいくつかあります。たとえば、 toredis または brükva です。

もちろん、これは単純なページだけでは必要ないかもしれませんが、非常にうまくスケーリングし、一度設定した後は維持/拡張するのにも非常に便利です。

1
Levite