From e083751d45f3f0de5387ee50b70885a310240300 Mon Sep 17 00:00:00 2001 From: lichong <18518571399@163.com> Date: Mon, 15 Jul 2024 17:38:11 +0800 Subject: [PATCH] 12454 --- api/main.py | 212 +++++++++++++++++++++++++++++------- front/package.json | 3 +- front/src/api/student.js | 31 +++++- front/src/api/teacher.js | 12 +- front/src/main.js | 2 +- front/src/views/login.vue | 16 ++- front/src/views/student.vue | 184 +++++++++++++++++++++++++++++-- front/src/views/teacher.vue | 89 +++++++++++---- 8 files changed, 470 insertions(+), 79 deletions(-) diff --git a/api/main.py b/api/main.py index 9801119..0dfb193 100644 --- a/api/main.py +++ b/api/main.py @@ -10,13 +10,26 @@ import time app = Flask(__name__) client = MongoClient("mongodb://localhost:27019/") db = client["back"] -collection = db["users"] +usercollection = db["users"] +zuobicollection = db["zuobi"] # sockitIo解决跨域问题 app.config["SECRET_KEY"] = "secret!" socketio = SocketIO(app, cors_allowed_origins="*") # 存储连接的客户端的房间号 clients = [] adminSid = "" +# 初始化考试状态 +studentStatus = {} + + +def dealStudentStatus(): + global studentStatus + userObj = usercollection.find({"isExit": 1, "isAdmin": 0}) + for studentItem in userObj: + studentStatus[str(studentItem["_id"])] = False + + +dealStudentStatus() @socketio.on("connect", namespace="/ws/video") @@ -46,9 +59,71 @@ def on_disconnect(): def handle_video_frame(message): # 广播视频帧给老师 global adminSid + global studentStatus # print(f"Received video frame from {message['userId']}") - socketio.emit("teacherVideo", message, to=adminSid, namespace="/ws/video") - # + socketio.emit( + "teacherVideo" + message["userId"], message, to=adminSid, namespace="/ws/video" + ) + if studentStatus[message["userId"]]: + print(f"学生{message['userId']}已作弊", f"{studentStatus[message['userId']]}") + + +@socketio.on("talk", namespace="/ws/video") +def handle_talk(message): + # 说话实时传输给老师 + global adminSid + # print(f"Received video frame from {message['userId']}") + socketio.emit( + "teacherTalk" + message["userId"], message, to=adminSid, namespace="/ws/video" + ) + zuobiItem = { + "userId": message["userId"], + "msg": message["data"], + "type": "talk", + "create_at": time.strftime("%Y-%m-%d %H:%M:%S", time.localtime(time.time())), + "update_at": time.strftime("%Y-%m-%d %H:%M:%S", time.localtime(time.time())), + "isExit": 1, + } + addzuobi(zuobiItem) + + +@socketio.on("sendMsg", namespace="/ws/video") +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" + ) + zuobiItem = { + "userId": message["userId"], + "msg": message["data"], + "type": "qustion", + "create_at": time.strftime("%Y-%m-%d %H:%M:%S", time.localtime(time.time())), + "update_at": time.strftime("%Y-%m-%d %H:%M:%S", time.localtime(time.time())), + "isExit": 1, + } + addzuobi(zuobiItem) + + +@socketio.on("answerMsg", namespace="/ws/video") +def handle_answer_msg(message): + sid = "" + for item in clients: + if item["id"] == message["userId"]: + sid = item["sid"] + socketio.emit( + "studentMsg" + message["userId"], message, to=sid, namespace="/ws/video" + ) + zuobiItem = { + "userId": message["userId"], + "msg": message["data"], + "type": "answer", + "create_at": time.strftime("%Y-%m-%d %H:%M:%S", time.localtime(time.time())), + "update_at": time.strftime("%Y-%m-%d %H:%M:%S", time.localtime(time.time())), + "isExit": 1, + } + addzuobi(zuobiItem) # md5加密 @@ -60,10 +135,10 @@ def md5_encrypt(data): # 初始化数据 def initData(): - admin = collection.find_one({"isAdmin": 1, "isExit": 1}) + admin = usercollection.find_one({"isAdmin": 1, "isExit": 1}) if admin is None: - # 学校名称、学校代号、专业名称、专业代号、年级、班级、学生姓名、学号、密码、成绩、作弊情况(作弊类型、作弊时间、作弊图片)、考试类型、考试科目、考试时间段、考试链接、是否是管理员、是否删除、创建时间、更新时间 - # xuexiaomingcheng、xuexiaodaihao、zhuanyemingcheng、zhuanyedaihao、nianji、banji、xueshengxingming、xuehao、mima、chengji、zuobiqingkuang(zuobileixing、zuobishijian、zuobitupian)、kaoshileixing、kaoshikemu、kaoshishijianduan、kaoshilianjie、isAdmin、isExit、chuangjianshijian、gengxinshijian + # 学校名称、学校代号、专业名称、专业代号、年级、班级、学生姓名、学号、考生图片、密码、成绩、考试类型、考试科目、考试时间段、考试链接、是否是管理员、是否删除、创建时间、更新时间 + # xuexiaomingcheng、xuexiaodaihao、zhuanyemingcheng、zhuanyedaihao、nianji、banji、xueshengxingming、xuehao、kaoshengtupian、mima、chengji、zuobiqingkuang(zuobileixing、zuobishijian、zuobitupian)、kaoshileixing、kaoshikemu、kaoshishijianduan、kaoshilianjie、isAdmin、isExit、chuangjianshijian、gengxinshijian user = { "xuexiaomingcheng": "", "xuexiaodaihao": "", @@ -73,19 +148,23 @@ def initData(): "banji": "", "xueshengxingming": "老师", "xuehao": "0000", + "kaoshengtupian": "", "mima": md5_encrypt("0000"), "chengji": "", - "zuobiqingkuang": [], "kaoshileixing": "", "kaoshikemu": "", "kaoshishijianduan": "", "kaoshilianjie": "", "isAdmin": 1, "isExit": 1, - "chuangjianshijian": "", - "gengxinshijian": "", + "chuangjianshijian": time.strftime( + "%Y-%m-%d %H:%M:%S", time.localtime(time.time()) + ), + "gengxinshijian": time.strftime( + "%Y-%m-%d %H:%M:%S", time.localtime(time.time()) + ), } - collection.insert_one(user) + usercollection.insert_one(user) print("管理账号'老师'已被初始化新建") else: print(f"管理账号'{admin['xueshengxingming']}'已存在") @@ -94,7 +173,7 @@ def initData(): # 判断是否是管理员 def isAdmin(id): mongoId = ObjectId(id) - userObj = collection.find_one({"_id": mongoId, "isExit": 1}) + userObj = usercollection.find_one({"_id": mongoId, "isExit": 1}) # 检查 userObj 是否为 None if userObj is None: return False @@ -103,6 +182,28 @@ def isAdmin(id): return bool(adminBool) +# 开始检测 +@app.route("/startCheck", methods=["post"]) +def startCheck(): + global studentStatus + resData = request.data + userJson = json.loads(resData) + print(f"学生{userJson['_id']}开始检测", studentStatus[userJson["_id"]]) + studentStatus[userJson["_id"]] = True + print(f"学生{userJson['_id']}开始检测", studentStatus[userJson["_id"]]) + return {"code": 200, "msg": "开始检测", "list": [], "hasError": False} + + +# 停止检测 +@app.route("/stopCheck", methods=["post"]) +def stopCheck(): + global studentStatus + resData = request.data + userJson = json.loads(resData) + studentStatus[userJson["_id"]] = False + return {"code": 200, "msg": "停止检测", "list": [], "hasError": False} + + # 用户登录 @app.route("/userLogin", methods=["post"]) def login(): @@ -112,14 +213,11 @@ def login(): return {"code": 200, "msg": "请填写信息后登录", "list": [], "hasError": True} # 获取到POST过来的数据,转为json形式 userJson = json.loads(resData) - res = collection.find_one({"xuehao": userJson["xuehao"], "mima": userJson["mima"]}) + res = usercollection.find_one( + {"xuehao": userJson["xuehao"], "mima": userJson["mima"]} + ) if res is None: return {"code": 200, "msg": "暂无此用户", "list": [], "hasError": True} - # serialized_data = [] - # for item in data: - # item["_id"] = str(item["_id"]) # 将ObjectId转换为字符串 - # serialized_data.append(item) - # 将ObjectId转换为字符串 res["_id"] = str(res["_id"]) return {"code": 200, "msg": "登陆成功", "list": [res], "hasError": False} @@ -128,9 +226,6 @@ def login(): @app.route("/addUser", methods=["post"]) def insert_data(): resData = request.data - # 检测是否有数据 - if not resData: - return {"code": 200, "msg": "无数据", "list": [], "hasError": True} # 获取到POST过来的数据,转为json形式 userJson = json.loads(resData) admin = isAdmin(userJson["id"]) @@ -162,7 +257,8 @@ def insert_data(): "%Y-%m-%d %H:%M:%S", time.localtime(time.time()) ), } - collection.insert_one(userItem) + usercollection.insert_one(userItem) + dealStudentStatus() return "新增成功" @@ -170,14 +266,11 @@ def insert_data(): @app.route("/getUser", methods=["post"]) def query_data(): resData = request.data - # 检测是否有数据 - if not resData: - return {"code": 200, "msg": "请填写信息后登录", "list": [], "hasError": True} # 获取到POST过来的数据,转为json形式 userJson = json.loads(resData) admin = isAdmin(userJson["id"]) if admin: - users = collection.find() + users = usercollection.find() serialized_data = [] for item in users: item["_id"] = str(item["_id"]) # 将ObjectId转换为字符串 @@ -190,7 +283,7 @@ def query_data(): } else: mongoId = ObjectId(userJson["id"]) - userObj = collection.find_one({"_id": mongoId, "isExit": 1}) + userObj = usercollection.find_one({"_id": mongoId, "isExit": 1}) userObj["_id"] = str(userObj["_id"]) # 将ObjectId转换为字符串 return { "code": 200, @@ -204,9 +297,6 @@ def query_data(): @app.route("/updateUser", methods=["post"]) def update_data(): resData = request.data - # 检测是否有数据 - if not resData: - return {"code": 200, "msg": "无数据", "list": [], "hasError": True} # 获取到POST过来的数据,转为json形式 userJson = json.loads(resData) admin = isAdmin(userJson["id"]) @@ -229,7 +319,7 @@ def update_data(): "%Y-%m-%d %H:%M:%S", time.localtime(time.time()) ), } - collection.update_one( + usercollection.update_one( {"_id": ObjectId(userJson["userid"])}, {"$set": userItem}, ) @@ -240,9 +330,6 @@ def update_data(): @app.route("/delUser", methods=["post"]) def delete_data(): resData = request.data - # 检测是否有数据 - if not resData: - return {"code": 200, "msg": "无数据", "list": [], "hasError": True} # 获取到POST过来的数据,转为json形式 userJson = json.loads(resData) admin = isAdmin(userJson["id"]) @@ -250,7 +337,7 @@ def delete_data(): serialized_data = [] for item in userJson["ids"]: serialized_data.append(ObjectId(item)) # 将ObjectId转换为字符串 - collection.delete_many({"_id": {"$in": serialized_data}}) + usercollection.delete_many({"_id": {"$in": serialized_data}}) return "删除成功" @@ -258,14 +345,11 @@ def delete_data(): @app.route("/updatePassword", methods=["post"]) def updatePassword(): resData = request.data - # 检测是否有数据 - if not resData: - return {"code": 200, "msg": "无数据", "list": [], "hasError": True} # 获取到POST过来的数据,转为json形式 userJson = json.loads(resData) admin = isAdmin(userJson["id"]) if admin: - collection.update_one( + usercollection.update_one( {"_id": ObjectId(userJson["userid"])}, { "$set": {"mima": md5_encrypt(userJson["mima"])}, @@ -274,6 +358,58 @@ def updatePassword(): return {"code": 200, "msg": "修改成功", "list": [], "hasError": False} +def addzuobi(zuobiItem): + zuobicollection.insert_one(zuobiItem) + + +# 删除作弊 +@app.route("/delzuobi", methods=["post"]) +def delzuobi(): + resData = request.data + # 获取到POST过来的数据,转为json形式 + userJson = json.loads(resData) + admin = isAdmin(userJson["id"]) + if admin: + serialized_data = [] + for item in userJson["ids"]: + serialized_data.append(ObjectId(item)) # 将ObjectId转换为字符串 + zuobicollection.delete_many({"_id": {"$in": serialized_data}}) + return "删除成功" + + +# 查询作弊 +@app.route("/getzuobi", methods=["post"]) +def getzuobi(): + resData = request.data + # 获取到POST过来的数据,转为json形式 + userJson = json.loads(resData) + admin = isAdmin(userJson["id"]) + if admin: + users = zuobicollection.find() + serialized_data = [] + for item in users: + item["_id"] = str(item["_id"]) # 将ObjectId转换为字符串 + serialized_data.append(item) + return { + "code": 200, + "msg": "查询成功", + "list": serialized_data, + "hasError": False, + } + else: + users = zuobicollection.find({"userId": userJson["id"], "isExit": 1}) + serialized_data = [] + for item in users: + item["_id"] = str(item["_id"]) # 将ObjectId转换为字符串 + serialized_data.append(item) + return { + "code": 200, + "msg": "查询成功", + "list": serialized_data, + "hasError": False, + } + + if __name__ == "__main__": initData() # app.run(debug=True) diff --git a/front/package.json b/front/package.json index 92ec342..114cf54 100644 --- a/front/package.json +++ b/front/package.json @@ -11,6 +11,7 @@ "@element-plus/icons-vue": "^2.3.1", "axios": "^1.6.6", "dayjs": "^1.11.10", + "echarts": "^5.5.1", "element-plus": "^2.5.3", "lodash": "^4.17.21", "md5": "^2.3.0", @@ -28,4 +29,4 @@ "@vitejs/plugin-vue": "^4.6.2", "vite": "^5.0.8" } -} \ No newline at end of file +} diff --git a/front/src/api/student.js b/front/src/api/student.js index a93007a..ee2faf5 100644 --- a/front/src/api/student.js +++ b/front/src/api/student.js @@ -1,7 +1,7 @@ import _axios from "@/plugins/axios"; //修改用户 -export function updateUser(data) { +export function updateUser(data = {}) { return _axios({ url: `/v1/updateUser`, method: "POST", @@ -9,7 +9,7 @@ export function updateUser(data) { }); } //查找用户 -export function getUser(data) { +export function getUser(data = {}) { return _axios({ url: `/v1/getUser`, method: "POST", @@ -17,10 +17,35 @@ export function getUser(data) { }); } //修改密码 -export function updatePassword(data) { +export function updatePassword(data = {}) { return _axios({ url: `/v1/updatePassword`, method: "POST", data: { id: localStorage.getItem("userId"), ...data }, }); } +//开始检测 +export function startCheck() { + return _axios({ + url: `/v1/startCheck`, + method: "POST", + data: { _id: localStorage.getItem("userId") }, + }); +} +//停止检测 +export function stopCheck() { + return _axios({ + url: `/v1/stopCheck`, + method: "POST", + data: { _id: localStorage.getItem("userId") }, + }); +} + +//查找作弊信息 +export function getzuobi(data = {}) { + return _axios({ + url: `/v1/getzuobi`, + method: "POST", + data: { id: localStorage.getItem("userId"), ...data }, + }); +} \ No newline at end of file diff --git a/front/src/api/teacher.js b/front/src/api/teacher.js index 6480ad0..71f7812 100644 --- a/front/src/api/teacher.js +++ b/front/src/api/teacher.js @@ -1,7 +1,7 @@ import _axios from "@/plugins/axios"; //登录、注册 -export function loginFun(data) { +export function loginFun(data = {}) { return _axios({ url: `/v1/userLogin`, method: "POST", @@ -11,7 +11,7 @@ export function loginFun(data) { //增加用户 -export function addUser(data) { +export function addUser(data = {}) { return _axios({ url: `/v1/addUser`, method: "POST", @@ -20,7 +20,7 @@ export function addUser(data) { } //删除用户 -export function delUser(data) { +export function delUser(data = {}) { return _axios({ url: `/v1/delUser`, method: "POST", @@ -28,7 +28,7 @@ export function delUser(data) { }); } //修改用户 -export function updateUser(data) { +export function updateUser(data = {}) { return _axios({ url: `/v1/updateUser`, method: "POST", @@ -36,7 +36,7 @@ export function updateUser(data) { }); } //查找用户 -export function getUser(data) { +export function getUser(data = {}) { return _axios({ url: `/v1/getUser`, method: "POST", @@ -44,7 +44,7 @@ export function getUser(data) { }); } //修改密码 -export function updatePassword(data) { +export function updatePassword(data = {}) { return _axios({ url: `/v1/updatePassword`, method: "POST", diff --git a/front/src/main.js b/front/src/main.js index 5c591f7..340e030 100644 --- a/front/src/main.js +++ b/front/src/main.js @@ -10,7 +10,7 @@ import dayjs from "dayjs"; import "@/assets/css/base.css"; //全局引入element-plus所有图标 import * as ElementPlusIconsVue from '@element-plus/icons-vue' - +import 'element-plus/dist/index.css' const app = createApp(App); for (const [key, component] of Object.entries(ElementPlusIconsVue)) { app.component(key, component) diff --git a/front/src/views/login.vue b/front/src/views/login.vue index eafc112..fa4e89a 100644 --- a/front/src/views/login.vue +++ b/front/src/views/login.vue @@ -59,15 +59,25 @@ export default { let formData = this.dealForm() let res = await loginFun(formData) if (!res.hasError) { - ElMessage.success(res.msg); localStorage.setItem("userId", this.$_.get(res, ["list", 0, "_id"], "")) localStorage.setItem("userName", this.$_.get(res, ["list", 0, "xueshengxingming"], "")) localStorage.setItem("isAdmin", this.$_.get(res, ["list", 0, "isAdmin"], 0)) if (this.$_.get(res, ["list", 0, "isAdmin"], 0)) { this.$router.push("/teacher") } else { - this.$router.push(localStorage.getItem("routerFullPath")) - localStorage.removeItem("routerFullPath") + console.log(78744, localStorage.getItem("routerFullPath")) + let idStr = localStorage.getItem("routerFullPath") || "" + let id = this.$_.split(idStr, "/")[2] + if (this.$_.get(res, ["list", 0, "_id"], "") === id) { + ElMessage.success(res.msg); + this.$router.push(localStorage.getItem("routerFullPath")) + localStorage.removeItem("routerFullPath") + } else { + ElMessage.error("请登录自己的考试信息"); + localStorage.removeItem("userId") + localStorage.removeItem("userName") + localStorage.removeItem("isAdmin") + } } } else { ElMessage.error(res.msg); diff --git a/front/src/views/student.vue b/front/src/views/student.vue index 2ea55a6..deb226d 100644 --- a/front/src/views/student.vue +++ b/front/src/views/student.vue @@ -23,7 +23,9 @@
+
+
+ {{ zuobiItem.label }}
+ :
- 实时作弊信息
+ {{ (zuobiObj[zuobiItem.prop] || []).length }}
-