|
|
@ -1,9 +1,8 @@ |
|
|
|
from flask import Flask, request, jsonify |
|
|
|
from flask_cors import CORS |
|
|
|
from pymongo import MongoClient |
|
|
|
from ultralytics import YOLO |
|
|
|
from PIL import Image |
|
|
|
import base64 |
|
|
|
from io import BytesIO |
|
|
|
from bson import ObjectId |
|
|
|
import hashlib |
|
|
|
import json |
|
|
@ -11,25 +10,30 @@ import os |
|
|
|
from flask import Flask, request |
|
|
|
from flask_socketio import SocketIO, emit, join_room, leave_room |
|
|
|
import time |
|
|
|
import numpy as np |
|
|
|
import cv2 |
|
|
|
|
|
|
|
app = Flask(__name__) |
|
|
|
|
|
|
|
client = MongoClient("mongodb://localhost:27019/") |
|
|
|
db = client["back"] |
|
|
|
usercollection = db["users"] |
|
|
|
zuobicollection = db["zuobi"] |
|
|
|
# sockitIo解决跨域问题 |
|
|
|
app.config["SECRET_KEY"] = "secret!" |
|
|
|
CORS(app, resources={r"/socket.io/*": {"origins": "*"}}) # 允许所有来源的请求 |
|
|
|
socketio = SocketIO(app, cors_allowed_origins="*") |
|
|
|
# CORS(app) |
|
|
|
# 存储连接的客户端的房间号 |
|
|
|
clients = [] |
|
|
|
adminSid = "" |
|
|
|
# 初始化考试状态 |
|
|
|
studentStatus = {} |
|
|
|
current_dir = os.path.dirname(os.path.abspath(__file__)) |
|
|
|
weights = os.path.join(current_dir, "exam.pt") |
|
|
|
print(89744, weights) |
|
|
|
model = YOLO(weights) |
|
|
|
frame_rate_divider = 300 # 设置帧率除数 |
|
|
|
det_model = YOLO(os.path.join(current_dir, "yolov8n-pose.pt")) |
|
|
|
cls_model = YOLO(os.path.join(current_dir, "yolov8n-cls.pt")) |
|
|
|
class_name = ["normal", "raise_hand", "speak", "stand", "turn_head", "use_phone"] |
|
|
|
frame_rate_divider = 90 # 设置帧率除数 |
|
|
|
frame_count = 0 # 初始化帧计数器 |
|
|
|
|
|
|
|
|
|
|
@ -43,7 +47,7 @@ def dealStudentStatus(): |
|
|
|
dealStudentStatus() |
|
|
|
|
|
|
|
|
|
|
|
@socketio.on("connect", namespace="/ws/video") |
|
|
|
@socketio.on("connect") |
|
|
|
def on_connect(): |
|
|
|
global clients |
|
|
|
global adminSid |
|
|
@ -53,11 +57,10 @@ def on_connect(): |
|
|
|
isAdmin = request.args.get("isAdmin") |
|
|
|
if isAdmin == "1": |
|
|
|
adminSid = sid |
|
|
|
print(87874, adminSid) |
|
|
|
clients.append({"id": id, "sid": sid, "isAdmin": isAdmin}) |
|
|
|
|
|
|
|
|
|
|
|
@socketio.on("disconnect", namespace="/ws/video") |
|
|
|
@socketio.on("disconnect") |
|
|
|
def on_disconnect(): |
|
|
|
global clients |
|
|
|
for item in clients: |
|
|
@ -66,19 +69,16 @@ def on_disconnect(): |
|
|
|
print(f"Client disconnected with id {item['id']}") |
|
|
|
|
|
|
|
|
|
|
|
@socketio.on("video", namespace="/ws/video") |
|
|
|
@socketio.on("video") |
|
|
|
def handle_video_frame(message): |
|
|
|
# 广播视频帧给老师 |
|
|
|
global adminSid |
|
|
|
global studentStatus |
|
|
|
global model |
|
|
|
global frame_rate_divider |
|
|
|
global frame_count |
|
|
|
|
|
|
|
# print(f"Received video frame from {message['userId']}") |
|
|
|
socketio.emit( |
|
|
|
"teacherVideo" + message["userId"], message, to=adminSid, namespace="/ws/video" |
|
|
|
) |
|
|
|
frame_count += 1 # 更新帧计数器 |
|
|
|
frame_count %= frame_rate_divider # 更新帧计数器 |
|
|
|
if frame_count % frame_rate_divider == 0: |
|
|
|
# 这里省略了实际的base64数据 |
|
|
|
base64_str = message["data"] |
|
|
@ -86,38 +86,61 @@ def handle_video_frame(message): |
|
|
|
_, img_data = base64_str.split(",") |
|
|
|
img_data = base64.b64decode(img_data) |
|
|
|
# 将字节流转换为PIL图像对象 |
|
|
|
image = Image.open(BytesIO(img_data)) |
|
|
|
results = model.predict(source=image, iou=0.5, conf=0.25) |
|
|
|
det = results[0] |
|
|
|
checkVedioResult(message, det) |
|
|
|
frame_count += 1 # 更新帧计数器 |
|
|
|
frame_count %= frame_rate_divider # 更新帧计数器 |
|
|
|
# print(99777,frame_count) |
|
|
|
# if studentStatus[message["userId"]]: |
|
|
|
# print(f"学生{message['userId']}已作弊", f"{studentStatus[message['userId']]}") |
|
|
|
# 将字节数据转换为NumPy数组 |
|
|
|
np_data = np.frombuffer(img_data, dtype=np.uint8) |
|
|
|
|
|
|
|
# 使用cv2.imdecode将数组解码为图像 |
|
|
|
image = cv2.imdecode(np_data, cv2.IMREAD_COLOR) |
|
|
|
checkVedioResult(message, image) |
|
|
|
socketio.emit( |
|
|
|
"teacherVideo" + message["userId"], message, to=adminSid |
|
|
|
) |
|
|
|
|
|
|
|
|
|
|
|
def checkVedioResult(message, det): |
|
|
|
def checkVedioResult(message, image): |
|
|
|
# 如果有有效的检测结果 |
|
|
|
# {"cheating": "疑似作弊", "good": "良好", "normal": "正常"} |
|
|
|
if det is not None and len(det): |
|
|
|
results = [] # 初始化结果列表 |
|
|
|
for res in det.boxes: |
|
|
|
for box in res: |
|
|
|
# 提前计算并转换数据类型 |
|
|
|
class_id = int(box.cls.cpu()) |
|
|
|
bbox = box.xyxy.cpu().squeeze().tolist() |
|
|
|
bbox = [int(coord) for coord in bbox] # 转换边界框坐标为整数 |
|
|
|
result = { |
|
|
|
"bbox": bbox, # 边界框 |
|
|
|
"score": box.conf.cpu().squeeze().item(), # 置信度 |
|
|
|
"class_id": class_id, # 类别ID |
|
|
|
} |
|
|
|
results.append(result) # 将结果添加到列表 |
|
|
|
global det_model |
|
|
|
global cls_model |
|
|
|
global class_name |
|
|
|
global clients |
|
|
|
global adminSid |
|
|
|
det_results = det_model(image,conf=0.5, iou=0.25) |
|
|
|
type = "" |
|
|
|
for r in det_results: |
|
|
|
if len(r) == 0: |
|
|
|
type = "leave" |
|
|
|
break |
|
|
|
# print(57841,r.boxes) |
|
|
|
box = r.boxes.xyxy |
|
|
|
if len(box) == 1: |
|
|
|
crop_image = image[ |
|
|
|
int(box[0][1]) : int(box[0][3]), int(box[0][0]) : int(box[0][2]) |
|
|
|
] |
|
|
|
# cv2.imshow('test.png', crop_image) |
|
|
|
# cv2.waitKey(0) |
|
|
|
cls_results = cls_model(crop_image) |
|
|
|
type = class_name[cls_results[0].probs.top1] |
|
|
|
else: |
|
|
|
type = "many_humans" |
|
|
|
message["type"] = type |
|
|
|
socketio.emit( |
|
|
|
"teacherTalk" + message["userId"], |
|
|
|
message, |
|
|
|
to=adminSid, |
|
|
|
) |
|
|
|
sid = "" |
|
|
|
for item in clients: |
|
|
|
if item["id"] == message["userId"]: |
|
|
|
sid = item["sid"] |
|
|
|
socketio.emit( |
|
|
|
"studentMsg" + message["userId"], message, to=sid |
|
|
|
) |
|
|
|
# print(97444, type) |
|
|
|
zuobiItem = { |
|
|
|
"userId": message["userId"], |
|
|
|
"msg": class_id, |
|
|
|
"type": class_id, |
|
|
|
"msg": message["data"], |
|
|
|
"type": type, |
|
|
|
"create_at": time.strftime( |
|
|
|
"%Y-%m-%d %H:%M:%S", time.localtime(time.time()) |
|
|
|
), |
|
|
@ -129,13 +152,14 @@ def checkVedioResult(message, det): |
|
|
|
addzuobi(zuobiItem) |
|
|
|
|
|
|
|
|
|
|
|
@socketio.on("talk", namespace="/ws/video") |
|
|
|
@socketio.on("talk") |
|
|
|
def handle_talk(message): |
|
|
|
# 说话实时传输给老师 |
|
|
|
global adminSid |
|
|
|
# print(f"Received video frame from {message['userId']}") |
|
|
|
message["type"] = "talk" |
|
|
|
socketio.emit( |
|
|
|
"teacherTalk" + message["userId"], message, to=adminSid, namespace="/ws/video" |
|
|
|
"teacherTalk" + message["userId"], message, to=adminSid |
|
|
|
) |
|
|
|
zuobiItem = { |
|
|
|
"userId": message["userId"], |
|
|
@ -148,13 +172,13 @@ def handle_talk(message): |
|
|
|
addzuobi(zuobiItem) |
|
|
|
|
|
|
|
|
|
|
|
@socketio.on("sendMsg", namespace="/ws/video") |
|
|
|
@socketio.on("sendMsg") |
|
|
|
def handle_msg(message): |
|
|
|
# 说话实时传输给老师 |
|
|
|
global adminSid |
|
|
|
# print(f"Received video frame from {message['userId']}") |
|
|
|
socketio.emit( |
|
|
|
"teacherMsg" + message["userId"], message, to=adminSid, namespace="/ws/video" |
|
|
|
"teacherMsg" + message["userId"], message, to=adminSid |
|
|
|
) |
|
|
|
zuobiItem = { |
|
|
|
"userId": message["userId"], |
|
|
@ -167,14 +191,15 @@ def handle_msg(message): |
|
|
|
addzuobi(zuobiItem) |
|
|
|
|
|
|
|
|
|
|
|
@socketio.on("answerMsg", namespace="/ws/video") |
|
|
|
@socketio.on("answerMsg") |
|
|
|
def handle_answer_msg(message): |
|
|
|
sid = "" |
|
|
|
for item in clients: |
|
|
|
if item["id"] == message["userId"]: |
|
|
|
sid = item["sid"] |
|
|
|
message["type"] = "answer" |
|
|
|
socketio.emit( |
|
|
|
"studentMsg" + message["userId"], message, to=sid, namespace="/ws/video" |
|
|
|
"studentMsg" + message["userId"], message, to=sid |
|
|
|
) |
|
|
|
zuobiItem = { |
|
|
|
"userId": message["userId"], |
|
|
@ -208,9 +233,9 @@ def initData(): |
|
|
|
"nianji": "", |
|
|
|
"banji": "", |
|
|
|
"xueshengxingming": "老师", |
|
|
|
"xuehao": "0000", |
|
|
|
"xuehao": "admin", |
|
|
|
"kaoshengtupian": "", |
|
|
|
"mima": md5_encrypt("0000"), |
|
|
|
"mima": md5_encrypt("admin"), |
|
|
|
"chengji": "", |
|
|
|
"kaoshileixing": "", |
|
|
|
"kaoshikemu": "", |
|
|
@ -476,4 +501,4 @@ def getzuobi(): |
|
|
|
if __name__ == "__main__": |
|
|
|
initData() |
|
|
|
# app.run(debug=True) |
|
|
|
socketio.run(app, debug=True) |
|
|
|
socketio.run(app, host='0.0.0.0', port=5000, debug=True) |
|
|
|