flask-socketio实现websocket通信

3,525 阅读2分钟

本文已参与[新人创作礼]活动,一起开启掘金创作之路。

flask-socketio能够实现客户端和服务器之间的低延迟双向通信。客户端可以使用js/c++/等任何官方支持的客户端库建立与服务器的长连接。

安装

pip install flask-socketio

要求

flask-socketio与python3.6+兼容,依赖的异步服务可以从以下三种选择其一。

  • eventlet 是性能最佳的选项,支持长轮询和websocket传输。
  • gevent 性能略低于eventlet,且没有原生websocket支持,需要安装gevent-websocket或使用带websocket功能的uWSGI的web服务器。
  • 基于Werkzeug的Flask开发服务器。此种仅用于简单开发测试。 [socketio官方文档](Introduction — Flask-SocketIO documentation)

初始化

from flask import Flask
from flask_socketio import SocketIO

app = Flask(__name__)
app.config['SECRET_KEY'] = 'secret!'
socketio = SocketIO(app)

if __name__ == '__main__':
    socketio.run(app, host='0.0.0.0', debug=True)

以上用socketio.run代替了app.run

注册连接/断开/消息事件

@socketio.on("connect", namespace="/transaction_cycle_life")
def connect():
    app.logger.info('connect------')


@socketio.on("disconnect", namespace="/transaction_cycle_life")
def disconnect():
    app.logger.info("disconnect -------------------")
    
    
@socketio.on("message", namespace="/transaction_cycle_life")
def message(data):
    app.logger.info(data)

以上配合客户端就可以实现双向的通信了,可以用emit发送消息到客户端

实现一个简单的循环查询并发送给客户端

from app import socketio, emit, app
from flask import request


tasks = {}


class MonitorTask(object):

    def __init__(self, username, sid):
        self._running = True
        self.username = username
        self.sid = sid

    def terminate(self):
        self._running = False

    def get_data(self):
        return {"msg": f"hahaha------{self.username}"}

    def run(self):
        while self._running:
            data = self.get_data()
            app.logger.info(data)
            # 非上下文,使用sid回传给原来的客户端
            socketio.emit("response", {"msg": data}, namespace='/transaction_cycle_life', room=self.sid)
            socketio.sleep(3)


@socketio.on("join", namespace="/transaction_cycle_life")
def join(data):
    username = data.get('username')
    # 有上下文,emit只会发给触发事件的客户端
    emit("response", {"msg": username})
    task = MonitorTask(username, request.sid)
    thread = socketio.start_background_task(task.run)
    tasks[request.sid] = {'task': task, 'thread': thread}
    app.logger.info(tasks)


@socketio.on("leave", namespace="/transaction_cycle_life")
def leave(*args):
    tasks[request.sid]['task'].terminate()
    app.logger("trigger leave ----------")
    del tasks[request.sid]
    emit("disconnect")


@socketio.on("message", namespace="/transaction_cycle_life")
def message(data):
    app.logger.info(data)


@socketio.on("connect", namespace="/transaction_cycle_life")
def connect():
    app.logger.info('connect------')


@socketio.on("disconnect", namespace="/transaction_cycle_life")
def disconnect():
    try:
        tasks[request.sid]['task'].terminate()
        del tasks[request.sid]
        app.logger.info("disconnect -------------------")
    except KeyError as ke:
        app.logger.info("key error pass")

上面的代码实现了一个客户端连接触发一次事件,并起个线程去定时持续的发送消息。