0

我正在尝试使用 python zmq 发布/订阅模式。我在客户端面临一个奇怪的问题。在客户端,我使用 pyzmq、gevent-websocket 和 bottle 作为 wsgi 服务器。虽然它对一个客户端非常有效,但其他客户端正在等待第一个客户端断开连接。在使用一个客户端时,如果我断开连接并重新连接,每次重新连接都会收到两次或更多消息。

#!/usr/bin/python
from gevent import monkey; monkey.patch_all()
import zmq
import gevent
from bottle import route, run, request, abort, Bottle ,static_file
from gevent import sleep
from gevent.pywsgi import WSGIServer
import geventwebsocket
from geventwebsocket import WebSocketHandler, WebSocketError

host = "127.0.0.1"
port = 8000
mqport = "8082"

context = zmq.Context()
socket = context.socket(zmq.SUB)

app = Bottle()

@app.route('/v1/streams/device/<id>')
def handle_websocket(id):
    socket.setsockopt(zmq.UNSUBSCRIBE, '')
    socket.connect ("tcp://localhost:%s" % mqport)
    socket.setsockopt(zmq.SUBSCRIBE, id)
    wsock = request.environ.get('wsgi.websocket')
    if wsock is None:
        logger.info("Error creating websocket")
    try :
        while True:
            string = socket.recv()
            logger.info("%s" % string)
            id, data = string.split("  ")
            wsock.send("%s" % data)
            sleep(0.1)
    except geventwebsocket.WebSocketError, ex:
        wsock.close()
        sock.close()
server = WSGIServer((host, port), app,
                handler_class=WebSocketHandler)
server.serve_forever()

我见过的所有示例都使用 while 循环来接收消息。我对这个while循环和寻找像socket.on_message这样的回调函数感到不舒服。出于实验目的,我编写了一个 node.js 版本,没有像这样的 while 循环,但在我的项目中节点被排除在外。节点版本在这里:

var WebSocketServer = require('websocket').server;

var http = require('http');

global.client = 0;

var server = http.createServer(function(request, response) {

});

server.listen(8000, function() { });

wsServer = new WebSocketServer({

    httpServer: server

});



// WebSocket server

wsServer.on('request', function(request) {

    console.log("connected client :", global.client++)

    var connection = request.accept(null, request.origin); 

    var url = request.resourceURL.href

    var device_id = url.substr(url.lastIndexOf('/') + 1)

    var zeromq = require('zmq');

    var sock = zeromq.socket('sub');

    sock.connect('tcp://127.0.0.1:8082');

    sock.subscribe(device_id);

    sock.on('message', function(data) {
        console.log(data.toString());

        var msg = data.toString();

        var rjson = msg.split("  ") 

        connection.send(rjson.pop());

    });

    connection.on('close', function(connection) {
    });
});

这里我不使用while循环。

我在 python 代码上做错了什么?

4

1 回答 1

0

经过几天的碰壁后,我们发现了一个名为 gevent-zeromq 的非阻塞 zeromq 客户端。

教训是任何使用 gevent-websocket 和 zmq 客户端的人都不要使用 pyzmq,而是使用 gevent-zeromq。

现在 gevent-zeromq 与 pyzmq 合并。import zmq而不是做做from zmq.green import zmq

于 2013-11-16T14:14:07.677 回答