ManageController.cs 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369
  1. using IES.ExamLib.Models;
  2. using IES.ExamServer.DI;
  3. using IES.ExamServer.DI.SignalRHost;
  4. using IES.ExamServer.Filters;
  5. using IES.ExamServer.Helper;
  6. using IES.ExamServer.Helpers;
  7. using IES.ExamServer.Models;
  8. using Microsoft.AspNetCore.Mvc;
  9. using Microsoft.Extensions.Caching.Memory;
  10. using Microsoft.Extensions.Configuration;
  11. using System.Linq.Expressions;
  12. using System.Net.Http;
  13. using System.Text.Json;
  14. using System.Text.Json.Nodes;
  15. namespace IES.ExamServer.Controllers
  16. {
  17. [ApiController]
  18. [Route("manage")]
  19. public class ManageController:BaseController
  20. {
  21. private readonly IConfiguration _configuration;
  22. private readonly IHttpClientFactory _httpClientFactory;
  23. private readonly IMemoryCache _memoryCache;
  24. private readonly ILogger<ManageController> _logger;
  25. private readonly LiteDBFactory _liteDBFactory;
  26. private readonly DataCenterConnectionService _connectionService;
  27. private readonly int DelayMicro = 10;//微观数据延迟
  28. private readonly int DelayMacro = 100;//宏观数据延迟
  29. private readonly SignalRExamServerHub _signalRExamServerHub;
  30. public ManageController(LiteDBFactory liteDBFactory,ILogger<ManageController> logger, IConfiguration configuration,
  31. IHttpClientFactory httpClientFactory, IMemoryCache memoryCache, DataCenterConnectionService connectionService,SignalRExamServerHub signalRExamServerHub)
  32. {
  33. _logger = logger;
  34. _configuration=configuration;
  35. _httpClientFactory=httpClientFactory;
  36. _memoryCache=memoryCache;
  37. _liteDBFactory=liteDBFactory;
  38. _connectionService=connectionService;
  39. _signalRExamServerHub=signalRExamServerHub;
  40. }
  41. [HttpPost("download-package")]
  42. [AuthToken("admin","teacher")]
  43. public async Task<IActionResult> DownloadPackage(JsonNode json)
  44. {
  45. //C#.NET 6 后端与前端流式通信
  46. //https://www.doubao.com/chat/collection/687687510791426?type=Thread
  47. //下载日志记录:1.步骤,检查,2.获取描述信息,3.分类型,4下载文件,5.前端处理,6.返回结果 , 正在下载...==> [INFO]https://www.doubao.com/chat/collection/687687510791426?type=Thread [Size=180kb] Ok...
  48. //进度条 展示下载文件总大小和已下载,末尾展示 文件总个数和已下载个数
  49. //https://cdnjs.cloudflare.com/ajax/libs/microsoft-signalr/8.0.7/signalr.min.js
  50. return Ok();
  51. }
  52. [HttpPost("check-short-code")]
  53. public async Task<IActionResult> CheckShortCode(JsonNode json)
  54. {
  55. string shortCode = $"{json["shortCode"]}";
  56. string evaluationId = $"{json["evaluationId"]}";
  57. string deviceId = $"{json["deviceId"]}";
  58. Expression<Func<EvaluationClient, bool>> predicate = x => true;
  59. if (!string.IsNullOrEmpty(shortCode))
  60. {
  61. var codePredicate = ExpressionHelper.Or<EvaluationClient>(
  62. x => !string.IsNullOrWhiteSpace(x.shortCode) && x.shortCode == shortCode,
  63. x => !string.IsNullOrWhiteSpace(x.password) && x.password == shortCode
  64. );
  65. predicate= predicate.And(codePredicate);
  66. }
  67. else {
  68. return Ok(new { code = 400,msg="必须输入开卷码" });
  69. }
  70. if (!string.IsNullOrWhiteSpace(evaluationId))
  71. {
  72. predicate= predicate.And(x => x.id!.Equals(evaluationId));
  73. }
  74. IEnumerable<EvaluationClient> evaluationClients = _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationClient>().Find(predicate);
  75. EvaluationClient? evaluationLocal = null;
  76. EvaluationClient? evaluationCloud = null;
  77. if (evaluationClients.Count()>0)
  78. {
  79. evaluationLocal= evaluationClients.First();
  80. }
  81. //如果要访问中心,则需要教师登录联网。
  82. var token = GetAuthTokenInfo();
  83. if (token.scope.Equals(ExamConstant.ScopeTeacher))
  84. {
  85. if ( _connectionService.dataCenterIsConnected)
  86. {
  87. Teacher teacher= _liteDBFactory.GetLiteDatabase().GetCollection<Teacher>().FindOne(x => x.id!.Equals(token.id));
  88. string? CenterUrl = _configuration.GetValue<string>("ExamServer:CenterUrl");
  89. var client = _httpClientFactory.CreateClient();
  90. if (client.DefaultRequestHeaders.Contains(Constant._X_Auth_AuthToken))
  91. {
  92. client.DefaultRequestHeaders.Remove(Constant._X_Auth_AuthToken);
  93. }
  94. client.DefaultRequestHeaders.Add(Constant._X_Auth_AuthToken, teacher.x_auth_token);
  95. HttpResponseMessage message = await client.PostAsJsonAsync($"{CenterUrl}/evaluation-sync/find-sync-info",new { shortCode, evaluationId });
  96. if (message.IsSuccessStatusCode)
  97. {
  98. string content = await message.Content.ReadAsStringAsync();
  99. JsonNode? jsonNode =JsonSerializer.Deserialize<JsonNode>(content);
  100. if (jsonNode!=null)
  101. {
  102. evaluationCloud= JsonSerializer.Deserialize<EvaluationClient>(jsonNode["evaluation"]);
  103. }
  104. }
  105. }
  106. }
  107. //数据,文件,页面 0 没有更新,1 有更新
  108. int data = 0,blob=0,webview=0,status=0;
  109. long dataSize = 0, blobSize=0 , webviewSize=0;
  110. if (evaluationLocal== null && evaluationCloud==null)
  111. {
  112. //线上线下没有数据
  113. status=1;
  114. }
  115. else if (evaluationLocal!=null && evaluationCloud!=null)
  116. {
  117. //线上线下有数据
  118. status = 2;
  119. if ((!string.IsNullOrWhiteSpace(evaluationLocal.blobHash) && !evaluationLocal.blobHash.Equals(evaluationCloud.blobHash))
  120. ||(evaluationLocal.blobTime<evaluationCloud.blobTime)
  121. ||(evaluationLocal.blobCount!= evaluationCloud.blobCount)
  122. ||(evaluationLocal.blobSize!= evaluationCloud.blobSize))
  123. {
  124. blob=1;
  125. blobSize=evaluationCloud.blobSize;
  126. }
  127. if ((evaluationLocal.dataTime<evaluationCloud.dataTime)
  128. ||(evaluationLocal.dataSize!=evaluationCloud.dataSize))
  129. {
  130. data=1;
  131. dataSize=evaluationCloud.dataSize;
  132. }
  133. if ((evaluationLocal.webviewCount!=evaluationCloud.webviewCount)
  134. ||(evaluationLocal.webviewSize!= evaluationCloud.webviewSize)
  135. ||(evaluationLocal.webviewTime!= evaluationCloud.webviewTime)
  136. ||(!string.IsNullOrWhiteSpace(evaluationLocal.webviewPath)&& !evaluationLocal.webviewPath.Equals(evaluationCloud.webviewPath)))
  137. {
  138. webview=1;
  139. webviewSize=evaluationCloud.webviewSize;
  140. }
  141. }
  142. else if (evaluationLocal!=null && evaluationCloud==null)
  143. {
  144. //线下有数据,线上没有数据,可能没联网。
  145. status = 3;
  146. }
  147. else if (evaluationLocal==null && evaluationCloud!=null)
  148. {
  149. //线下没有数据,线上有数据
  150. evaluationLocal= evaluationCloud;
  151. blob=1;
  152. data=1;
  153. webview=1;
  154. blobSize=evaluationCloud.blobSize;
  155. dataSize=evaluationCloud.dataSize;
  156. webviewSize=evaluationCloud.webviewSize;
  157. status = 4;
  158. _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationClient>().Insert(evaluationLocal);
  159. }
  160. List<string> file_error = new List<string>();
  161. if (evaluationLocal!=null)
  162. {
  163. await _signalRExamServerHub.SendMessage(deviceId, Constant._Message_grant_type_check_file,
  164. new MessageContent {messageType=Constant._Message_type_message, status=0, content="开始检查评测信息文件.." });
  165. //校验本地文件数据
  166. string packagePath = Path.Combine(Directory.GetCurrentDirectory(), "wwwroot", "package");
  167. if (!Directory.Exists(packagePath))
  168. Directory.CreateDirectory(packagePath);
  169. string evaluationPath = Path.Combine(packagePath, evaluationLocal.id!);
  170. // await Task.Delay(DelayMacro);
  171. int msg_status = Constant._Message_status_info;
  172. string path_evaluation = Path.Combine(evaluationPath, "evaluation.json");
  173. if (!System.IO.File.Exists(path_evaluation))
  174. {
  175. file_error.Add("evaluation");
  176. msg_status=Constant._Message_status_error;
  177. }
  178. else
  179. {
  180. msg_status=Constant._Message_status_success;
  181. }
  182. //数据格式: [消息][信息/错误/警告][15:43]=>[开始检查评测信息文件...]
  183. //数据格式: [检查][成功/失败][15:43]=>[评测数据文件:/wwwroot/package/623a9fe6-5445-0938-ff77-aeb80066ef27/evaluation.json]
  184. //数据格式: [下载][成功/失败][15:43]=>[评测数据文件:/wwwroot/package/623a9fe6-5445-0938-ff77-aeb80066ef27/evaluation.json][1024kb][15ms]
  185. await _signalRExamServerHub.SendMessage(deviceId, Constant._Message_grant_type_check_file,
  186. new MessageContent { messageType= Constant._Message_type_check, status=msg_status,content=$"评测数据文件:{path_evaluation}" });
  187. //await Task.Delay(DelayMacro);
  188. string path_groupList = Path.Combine(evaluationPath, "groupList.json");
  189. msg_status =Constant._Message_status_info;
  190. if (!System.IO.File.Exists(path_groupList))
  191. {
  192. file_error.Add("groupList");
  193. msg_status=Constant._Message_status_error;
  194. }
  195. else
  196. {
  197. msg_status=Constant._Message_status_success;
  198. }
  199. await _signalRExamServerHub.SendMessage(deviceId, Constant._Message_grant_type_check_file,
  200. new MessageContent { messageType= Constant._Message_type_check, status=msg_status, content=$"评测名单文件:{path_groupList}" });
  201. //await Task.Delay(DelayMacro);
  202. string path_source = Path.Combine(evaluationPath, "source.json");
  203. msg_status = Constant._Message_status_info;
  204. if (!System.IO.File.Exists(path_source))
  205. {
  206. file_error.Add("source");
  207. msg_status=Constant._Message_status_error;
  208. }
  209. else
  210. {
  211. msg_status=Constant._Message_status_success;
  212. }
  213. await _signalRExamServerHub.SendMessage(deviceId, Constant._Message_grant_type_check_file,
  214. new MessageContent { messageType= Constant._Message_type_check, status=msg_status, content=$"评测原始数据:{path_source}" });
  215. // await Task.Delay(DelayMacro);
  216. msg_status =Constant._Message_status_info;
  217. try {
  218. string evaluation_str = await System.IO.File.ReadAllTextAsync(path_evaluation);
  219. JsonNode? evaluation_data = JsonSerializer.Deserialize<JsonNode>(evaluation_str);
  220. if (evaluation_data!=null)
  221. {
  222. EvaluationClient? evaluationClient = JsonSerializer.Deserialize<EvaluationClient>(evaluation_data["evaluationClient"]);
  223. if (evaluationClient!=null)
  224. {
  225. if ((!string.IsNullOrWhiteSpace(evaluationLocal.blobHash) && evaluationLocal.blobHash.Equals(evaluationClient.blobHash))
  226. &&(evaluationLocal.blobTime==evaluationClient.blobTime)
  227. &&(evaluationLocal.blobCount== evaluationClient.blobCount)
  228. &&(evaluationLocal.blobSize== evaluationClient.blobSize)&& (evaluationLocal.dataTime==evaluationClient.dataTime)
  229. &&(evaluationLocal.dataSize==evaluationClient.dataSize)&&(evaluationLocal.webviewCount==evaluationClient.webviewCount)
  230. &&(evaluationLocal.webviewSize== evaluationClient.webviewSize)
  231. &&(evaluationLocal.webviewTime== evaluationClient.webviewTime)
  232. &&(!string.IsNullOrWhiteSpace(evaluationLocal.webviewPath)&& evaluationLocal.webviewPath.Equals(evaluationClient.webviewPath)))
  233. {
  234. msg_status=1;
  235. }
  236. else
  237. {
  238. msg_status=Constant._Message_status_error;
  239. }
  240. await _signalRExamServerHub.SendMessage(deviceId, Constant._Message_grant_type_check_file,
  241. new MessageContent { messageType=Constant._Message_type_message, status=msg_status, content="校验本地数据文件..." });
  242. }
  243. List<EvaluationExam>? evaluationExams = JsonSerializer.Deserialize<List<EvaluationExam>>(evaluation_data["evaluationExams"]);
  244. if (evaluationExams.IsNotEmpty())
  245. {
  246. string path_papers = Path.Combine(evaluationPath, "papers");
  247. var papers_files = FileHelper.ListAllFiles(path_papers);
  248. foreach (var evaluationExam in evaluationExams!)
  249. {
  250. int paperIndex = 0;
  251. foreach (var paper in evaluationExam.papers)
  252. {
  253. paperIndex++;
  254. List<MessageContent> contents = new List<MessageContent>();
  255. int paper_error_count = 0;
  256. foreach (var blobInfo in paper.blobs)
  257. {
  258. msg_status=Constant._Message_status_info;
  259. if (!string.IsNullOrWhiteSpace(blobInfo.path))
  260. {
  261. var file = papers_files.Find(x => x.Contains(blobInfo.path));
  262. if (file!=null)
  263. {
  264. msg_status=1;
  265. msg_status=Constant._Message_status_success;
  266. }
  267. else {
  268. msg_status=Constant._Message_status_error;
  269. paper_error_count++;
  270. }
  271. }
  272. else {
  273. msg_status=Constant._Message_status_warning; ;
  274. paper_error_count++;
  275. }
  276. contents.Add(new MessageContent { messageType=Constant._Message_type_check, status=msg_status, content=$"试卷文件信息:{paper.paperName}" });
  277. }
  278. int paper_msg_status = Constant. _Message_status_info;
  279. if (paper_error_count>0)
  280. {
  281. paper_msg_status=Constant._Message_status_error;
  282. }
  283. else {
  284. paper_msg_status=Constant._Message_status_success;
  285. }
  286. await _signalRExamServerHub.SendMessage(deviceId, Constant._Message_grant_type_check_file,
  287. new MessageContent {
  288. messageType=Constant._Message_type_message,
  289. status=paper_msg_status,
  290. content=$"试卷名称:[{paperIndex}]{evaluationExam.examName}-{evaluationExam.subjectName}-{paper.paperName}\r\n文件数量:{paper.blobs.Count()},检测成功数量:{contents.Count(x => x.status==Constant._Message_status_success)},检测异常数量{contents.Count(x => x.status==Constant._Message_status_error)}",
  291. contents=contents
  292. });
  293. }
  294. }
  295. }
  296. }
  297. }
  298. catch (Exception e) {
  299. }
  300. await _signalRExamServerHub.SendMessage(deviceId, Constant._Message_grant_type_check_file,
  301. new MessageContent { messageType=Constant._Message_type_message, status=0, content="提取评测数据文件..." });
  302. }
  303. return Ok(new {code=200, evaluation= evaluationLocal,data,blob,webview,dataSize,blobSize,webviewSize,status });
  304. }
  305. /// <summary>
  306. /// 激活考试
  307. /// </summary>
  308. /// <param name="json"></param>
  309. /// <returns></returns>
  310. [HttpPost("activate-evaluation")]
  311. public IActionResult ActivateEvaluation(JsonNode json)
  312. {
  313. string id = $"{json["id"]}";
  314. string shortCode = $"{json["shortCode"]}";
  315. if (!string.IsNullOrWhiteSpace(id) && !string.IsNullOrWhiteSpace(shortCode))
  316. {
  317. EvaluationClient evaluationClient = _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationClient>().FindOne(x => x.id!.Equals(id) && !string.IsNullOrWhiteSpace(x.shortCode) && x.shortCode.Equals(shortCode));
  318. if (evaluationClient != null)
  319. {
  320. }
  321. }
  322. return Ok();
  323. }
  324. /// <summary>
  325. /// 加载本地的活动列表
  326. /// </summary>
  327. /// <param name="json"></param>
  328. /// <returns></returns>
  329. [HttpPost("list-local-evaluation")]
  330. public IActionResult ListLocalEvaluation(JsonNode json)
  331. {
  332. IEnumerable<EvaluationClient> evaluationClients = _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationClient>().FindAll().OrderByDescending(x=>x.activate).ThenByDescending(x=>x.stime);
  333. var result = evaluationClients.Select(client =>
  334. {
  335. var properties = client.GetType().GetProperties();
  336. var anonymousObject = new Dictionary<string, object?>();
  337. foreach (var property in properties)
  338. {
  339. if (!property.Name .Equals("password") && !property.Name.Equals("shortCode"))
  340. {
  341. anonymousObject[property.Name] = property.GetValue(client);
  342. }
  343. }
  344. return anonymousObject;
  345. });
  346. return Ok(new {code=200, evaluation= result });
  347. }
  348. }
  349. }