3

我想使用 rethinkdb .changes() 功能向用户推送一些消息。消息应该在没有来自用户的任何请求的情况下发送。

我正在将 rethinkdb 与 aiohttp 和 websockets 一起使用。这个怎么运作:

  1. 用户发送消息
  2. 服务器放入rethinkdb
  3. 我需要什么:一个额外的循环使用 rethinkdb.changes函数向连接的用户发送更新

这就是我启动应用程序的方式:

@asyncio.coroutine
def init(loop):
    app = Application(loop=loop)
    app['sockets'] = []
    app['susers'] = []
    app.router.add_route('GET', '/', wshandler)
    handler = app.make_handler()
    srv = yield from loop.create_server(handler, '127.0.0.1', 9080)
    print("Server started at http://127.0.0.1:9080")
    return app, srv, handler

wshandler我有一个循环,它处理传入的消息:

@asyncio.coroutine
def wshandler(request):
    resp = WebSocketResponse()
    if not resp.can_prepare(request):
        return Response(
            body=bytes(json.dumps({"error_code": 401}), 'utf-8'),
            content_type='application/json'
        )
    yield from resp.prepare(request)
    request.app['sockets'].append(resp)
    print('Someone connected')
    while True:
        msg = yield from resp.receive()
        if msg.tp == MsgType.text:
            runCommand(msg, resp, request)
        else:
            break
    request.app['sockets'].remove(resp)
    print('Someone disconnected.')
    return resp

如何创建第二个循环将消息发送到同一个打开的连接池?如何使其线程安全?

4

1 回答 1

3

一般来说,在运行事件循环时,您应该尽量避免使用线程。

不幸的rethinkdb是不支持asyncio开箱即用,但它确实支持Tornado 和 Twisted框架。因此,您可以桥接Tornado &asyncio并使其在不使用线程的情况下工作。

编辑

正如安德鲁指出的rethinkdb 那样支持asyncio。在2.1.0之后,您大概可以这样做:

rethinkdb.set_loop_type("asyncio")

然后在您的网络处理程序中:

res = await rethinkdb.table(tbl).changes().run(connection)
while await res.fetch_next():
   ...
于 2015-11-23T20:54:59.320 回答