lot_manager/video_task.py

184 lines
6.5 KiB
Python
Raw Normal View History

2024-01-03 18:28:27 +08:00
import websockets
import asyncio
import json
import paho.mqtt.client as mqtt
from pydantic import BaseModel
broker = 'mqtt.lihaink.cn'
port = 1883
2024-01-06 15:15:52 +08:00
APP = "app"
SCREEN = "screen"
WEB = "web"
SCENE_NAME = "scene"
user_sched = {}
user_msg = {}
2024-01-06 16:29:19 +08:00
user_scene = []
2024-01-03 18:28:27 +08:00
class BaseResponse(BaseModel):
code: int = 200
msg: str = "success"
class MQTTClient:
def __init__(self, broker, port, topic, username, password):
self.broker = broker
self.port = port
self.topic = topic
self.username = username
self.password = password
# 千万不要指定client_id 不然死翘翘
self.client = mqtt.Client()
self.client.username_pw_set(self.username, self.password)
def push(self):
self.client.publish(self.topic, payload=json.dumps({"msg": "open_led"}, ensure_ascii=False),
qos=0)
self.client.publish(self.topic, payload=json.dumps({"msg": "push_stream"}, ensure_ascii=False),
qos=0)
def close(self):
self.client.publish(self.topic, payload=json.dumps({"msg": "close_stream"}, ensure_ascii=False),
qos=0)
self.client.publish(self.topic, payload=json.dumps({"msg": "close_led"}, ensure_ascii=False),
qos=0)
def start(self):
self.client.connect(self.broker, self.port)
async def close(username, device):
# 倒计时600秒
2024-01-06 09:22:40 +08:00
await asyncio.sleep(30)
2024-01-03 18:28:27 +08:00
print(username + "结束推流")
user_sched.pop(username)
close_stream(username, device)
async def task_start(username, device):
print(username + "创建结束推流定时任务")
2024-01-05 10:01:04 +08:00
task = asyncio.create_task(close(username, device))
2024-01-03 18:28:27 +08:00
user_sched[username] = task
async def start(username, device):
try:
# 如果定时任务不存在,那么直接进行推流。当用户退出界面时,创建任务
if username not in user_sched:
print(username + "开始推流")
push_stream(username, device)
# 如果定时任务存在,那么取消上次的任务。当用户退出界面时,创建任务
else:
t = user_sched[username]
# 取消任务
if t is not None:
print(username + "取消定时任务")
t.cancel()
2024-01-04 18:32:36 +08:00
return BaseResponse(code=200, msg="success").json()
2024-01-03 18:28:27 +08:00
except Exception as e:
2024-01-04 18:32:36 +08:00
return BaseResponse(code=500, msg=str(e)).json()
2024-01-03 18:28:27 +08:00
async def stop(username, device):
try:
await task_start(username, device)
2024-01-04 18:32:36 +08:00
return BaseResponse(code=200, msg="success").json()
2024-01-03 18:28:27 +08:00
except Exception as e:
2024-01-04 18:32:36 +08:00
return BaseResponse(code=500, msg=str(e)).json()
2024-01-03 18:28:27 +08:00
def push_stream(username, device):
MQTT = MQTTClient(broker, port, device, username, username)
MQTT.start()
MQTT.push()
def close_stream(username, device):
MQTT = MQTTClient(broker, port, device, username, username)
MQTT.start()
MQTT.close()
# 定义函数A
async def function_A(client, data):
2024-01-04 18:32:36 +08:00
data = json.loads(data)
2024-01-03 18:28:27 +08:00
username = data['username']
device = data['device']
2024-01-05 10:01:04 +08:00
# print("A", username, device)
2024-01-04 18:32:36 +08:00
return await start(username, device)
2024-01-03 18:28:27 +08:00
2024-01-03 19:10:56 +08:00
2024-01-03 18:28:27 +08:00
# 定义函数B
async def function_B(client, data):
2024-01-04 18:32:36 +08:00
data = json.loads(data)
2024-01-03 18:28:27 +08:00
username = data['username']
device = data['device']
2024-01-05 10:01:04 +08:00
# print("B", username, device)
2024-01-03 18:28:27 +08:00
print(f"{client}的连接已断开")
2024-01-04 18:32:36 +08:00
await stop(username, device)
2024-01-03 18:28:27 +08:00
# 创建WebSocket连接的处理函数
async def handler(websocket, path):
2024-01-06 16:29:19 +08:00
the_user_scene = None
username = None
2024-01-03 18:28:27 +08:00
try:
# 在循环中等待客户端的消息
async for message in websocket:
2024-01-04 18:32:36 +08:00
user_msg[websocket] = message
2024-01-06 15:15:52 +08:00
data = json.loads(message)
2024-01-06 16:29:19 +08:00
if ('username' not in data
or 'device' not in data
or 'scene' not in data):
await websocket.send(json.dumps({"code": 200, "msg": "请正确传参"}, ensure_ascii=False))
2024-01-06 15:15:52 +08:00
continue
2024-01-06 16:29:19 +08:00
if (data['username'] is None
or data['device'] is None
or data['scene'] is None):
await websocket.send(json.dumps({"code": 200, "msg": "请传参数,不要为空"}, ensure_ascii=False))
2024-01-06 15:15:52 +08:00
continue
2024-01-06 16:29:19 +08:00
username = data['username']
the_user_scene = username + data[SCENE_NAME]
# 如果该用户的场景已经开启,那么不在开启了
if user_scene[the_user_scene] is True:
await websocket.send(
json.dumps({"code": 200, "msg": the_user_scene + "该场景已经开启了,不要再请求我了!"},
ensure_ascii=False))
continue
if user_scene[username + APP] is True or user_scene[username + WEB] is True or user_scene[
username + SCREEN] is True:
user_scene[the_user_scene] = True
await websocket.send(
json.dumps({"code": 200, "msg": "已经有其他场景开启了,不需要再推流了!"}, ensure_ascii=False))
continue
# 如果该用户的场景没有开启,那么进行开启
user_scene[the_user_scene] = True
2024-01-03 18:28:27 +08:00
# 当接收到消息时调用函数A
response = await function_A(websocket, message)
await websocket.send(response)
2024-01-06 16:29:19 +08:00
# 没有消息了,则首先关闭该用户的场景
if the_user_scene is not None:
user_scene[the_user_scene] = False
# 查看该用户的所有场景是否在线,只有都不在线就关闭推流
if user_scene[username + APP] is False and user_scene[username + WEB] is False and user_scene[
username + SCREEN] is False:
await websocket.send(json.dumps({"code": 200, "msg": "开始关闭场景" + the_user_scene}, ensure_ascii=False))
2024-01-06 15:15:52 +08:00
await function_B(websocket, user_msg[websocket])
2024-01-03 18:28:27 +08:00
except Exception as e:
2024-01-06 16:29:19 +08:00
if username is not None:
if the_user_scene is not None:
user_scene[the_user_scene] = False
# 查看所有场景是否在线,只有都不在线就关闭推流
if user_scene[username + APP] is False and user_scene[username + WEB] is False and user_scene[
username + SCREEN] is False:
await function_B(websocket, user_msg[websocket])
2024-01-03 18:28:27 +08:00
2024-01-06 15:15:52 +08:00
if __name__ == '__main__':
# 启动WebSocket服务器
start_server = websockets.serve(handler, "0.0.0.0", 8765)
2024-01-03 18:28:27 +08:00
2024-01-06 15:15:52 +08:00
asyncio.get_event_loop().run_until_complete(start_server)
asyncio.get_event_loop().run_forever()