Python Gevent SocketIO内存泄漏断开未触发

时间:2012-09-13 16:57:27

标签: python socket.io gevent

我几个小时以来一直在努力。这是一个非常简单的应用程序,类似于基本聊天,但服务器仅向客户端发送消息。我的问题的基础是内存泄漏。我尝试将应用程序重新分解为更无状态。我跟踪this guide来跟踪内存泄漏。我无法理解图表。我可能会在以后更深入。首先,我注意到断开连接没有触发,它确实做了一些清理工作,例如杀死greenlets。 为什么断开连接没有解雇?如果有人能给我一些见解,我们非常感激。

https://gist.github.com/a53d1acc83b06b16d263

from gevent import monkey; monkey.patch_all()
import gevent
import redis
import json
import phpserialize

from socketio import socketio_manage
from socketio.server import SocketIOServer
from socketio.namespace import BaseNamespace
from socketio.mixins import BroadcastMixin, RoomsMixin

r = redis.StrictRedis(host='localhost', port=6379)

class ShoutsNamespace(BaseNamespace):

    def listener(self, room):
        r = redis.StrictRedis()
        r = r.pubsub()

        r.subscribe(room)

        for m in r.listen():
            if m['type'] == 'message':
                data = json.loads(m['data'])
                self.emit("shouts", data)

    def initialize(self):
        print 'Connected'

    def on_join(self, room):
        self.emit("shouts", json.loads(r.hget('ffxi.shouts', room)))
        self.spawn(self.listener, 'ffxi.shouts:'+room)
        return True

    def recv_disconnect(self):
        print 'Disconnected'
        self.disconnect(silent=True)

class PingNamespace(BaseNamespace):
    def on_ping(self, message):
        print message

class Application(object):
    def __init__(self):
        self.buffer = []

    def __call__(self, environ, start_response):
        path = environ['PATH_INFO'].strip('/')
        if path.startswith('static/'):
            try:
                data = open(path).read()
            except Exception:
                return not_found(start_response)

            if path.endswith(".js"):
                content_type = "text/javascript"
            elif path.endswith(".css"):
                content_type = "text/css"
            elif path.endswith(".swf"):
                content_type = "application/x-shockwave-flash"
            else:
                content_type = "text/html"

            start_response('200 OK', [('Content-Type', content_type)])
            return [data]

        if path.startswith("socket.io"):
            socketio_manage(environ, {'/ping': PingNamespace, '/shouts': ShoutsNamespace })
        else:
            return not_found(start_response)

def not_found(start_response):
    start_response('404 Not Found', [])
    return ['<h1>Not Found</h1>']


if __name__ == '__main__':
    SocketIOServer(('216.144.246.171', 443), Application(), resource="socket.io", policy_server=False, transports=['xhr-polling']).serve_forever()

1 个答案:

答案 0 :(得分:3)

recv_disconnect由用户而不是服务器触发。您是否必须断开用户连接(添加链接或按钮?)或使用&#34; beforeunload&#34;之类的事件。或者&#34;卸载&#34;打电话给它。在这两种情况下,您都需要调用socket.disconnect()。

示例:

$(window).on('beforeunload',function(){socket.disconnect();});

希望这对你有所帮助。

致以最诚挚的问候,

安德烈亚斯