flask-socketio能够实现客户端和服务器之间的低延迟双向通信。客户端可以使用js/c++/等任何官方支持的客户端库建立与服务器的长连接。
pip install flask-socketio
flask-socketio与python3.6+兼容,依赖的异步服务可以从以下三种选择其一。
[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 = Trueself.username = usernameself.sid = siddef terminate(self):self._running = Falsedef 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")
上面的代码实现了一个客户端连接触发一次事件,并起个线程去定时持续的发送消息。