ManageController.cs 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403
  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 _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 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 =content.ToObject<JsonNode>();
  100. if (jsonNode!=null)
  101. {
  102. evaluationCloud=jsonNode["evaluation"]?.ToObject<EvaluationClient>();
  103. }
  104. }
  105. }
  106. }
  107. //数据,文件,页面 0 没有更新,1 有更新
  108. int data = 0,blob=0,webview=0, groupList=0, status=0;
  109. long dataSize = 0, blobSize=0 , webviewSize=0, studentCount=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. ||(evaluationLocal.paperCount!= evaluationCloud.paperCount)
  130. )
  131. {
  132. data=1;
  133. dataSize=evaluationCloud.dataSize;
  134. }
  135. if ((evaluationLocal.webviewCount!=evaluationCloud.webviewCount)
  136. ||(evaluationLocal.webviewSize!= evaluationCloud.webviewSize)
  137. ||(evaluationLocal.webviewTime!= evaluationCloud.webviewTime)
  138. ||(!string.IsNullOrWhiteSpace(evaluationLocal.webviewPath)&& !evaluationLocal.webviewPath.Equals(evaluationCloud.webviewPath)))
  139. {
  140. webview=1;
  141. webviewSize=evaluationCloud.webviewSize;
  142. }
  143. if ((evaluationLocal.studentCount!= evaluationCloud.studentCount)||(!$"{evaluationLocal.grouplistHash}".Equals(evaluationCloud.grouplistHash)))
  144. {
  145. groupList=1;
  146. studentCount=evaluationCloud.studentCount;
  147. }
  148. }
  149. else if (evaluationLocal!=null && evaluationCloud==null)
  150. {
  151. //线下有数据,线上没有数据,可能没联网。
  152. status = 3;
  153. }
  154. else if (evaluationLocal==null && evaluationCloud!=null)
  155. {
  156. //线下没有数据,线上有数据
  157. evaluationLocal= evaluationCloud;
  158. blob=1;
  159. data=1;
  160. webview=1;
  161. groupList=1;
  162. blobSize=evaluationCloud.blobSize;
  163. dataSize=evaluationCloud.dataSize;
  164. webviewSize=evaluationCloud.webviewSize;
  165. studentCount=evaluationCloud.studentCount;
  166. status = 4;
  167. _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationClient>().Insert(evaluationLocal);
  168. }
  169. List<string> file_error = new List<string>();
  170. if (evaluationLocal!=null)
  171. {
  172. await _signalRExamServerHub.SendMessage(deviceId, Constant._Message_grant_type_check_file,
  173. new MessageContent {dataId=evaluationLocal.id,dataName=evaluationLocal.name,messageType=Constant._Message_type_message, status=0, content="开始检查评测信息文件.." });
  174. //校验本地文件数据
  175. string packagePath = Path.Combine(Directory.GetCurrentDirectory(), "wwwroot", "package");
  176. if (!Directory.Exists(packagePath))
  177. Directory.CreateDirectory(packagePath);
  178. string evaluationPath = Path.Combine(packagePath, evaluationLocal.id!);
  179. // await Task.Delay(DelayMacro);
  180. int msg_status = Constant._Message_status_info;
  181. string path_evaluation = Path.Combine(evaluationPath, "evaluation.json");
  182. if (!System.IO.File.Exists(path_evaluation))
  183. {
  184. file_error.Add("evaluation");
  185. msg_status=Constant._Message_status_error;
  186. }
  187. else
  188. {
  189. msg_status=Constant._Message_status_success;
  190. //string jsonData = await System.IO.File.ReadAllTextAsync(path_evaluation);
  191. //EvaluationClient? evaluationFile =jsonData.ToObject<EvaluationClient>();
  192. //if (evaluationFile!=null)
  193. //{
  194. // if (evaluationFile.dataSize==evaluationLocal.dataSize )
  195. // {
  196. // file_error.Add("evaluation");
  197. // msg_status=Constant._Message_status_error;
  198. // }
  199. // else
  200. // {
  201. // msg_status=Constant._Message_status_success;
  202. // }
  203. //}
  204. //else
  205. //{
  206. // file_error.Add("evaluation");
  207. // msg_status=Constant._Message_status_error;
  208. //}
  209. }
  210. //数据格式: [消息][信息/错误/警告][15:43]=>[开始检查评测信息文件...]
  211. //数据格式: [检查][成功/失败][15:43]=>[评测数据文件:/wwwroot/package/623a9fe6-5445-0938-ff77-aeb80066ef27/evaluation.json]
  212. //数据格式: [下载][成功/失败][15:43]=>[评测数据文件:/wwwroot/package/623a9fe6-5445-0938-ff77-aeb80066ef27/evaluation.json][1024kb][15ms]
  213. await _signalRExamServerHub.SendMessage(deviceId, Constant._Message_grant_type_check_file,
  214. new MessageContent { dataId=evaluationLocal.id, dataName=evaluationLocal.name, messageType= Constant._Message_type_check, status=msg_status,content=$"评测数据文件:{path_evaluation}" });
  215. //await Task.Delay(DelayMacro);
  216. string path_groupList = Path.Combine(evaluationPath, "groupList.json");
  217. msg_status =Constant._Message_status_info;
  218. if (!System.IO.File.Exists(path_groupList))
  219. {
  220. file_error.Add("groupList");
  221. msg_status=Constant._Message_status_error;
  222. }
  223. else
  224. {
  225. msg_status=Constant._Message_status_success;
  226. }
  227. await _signalRExamServerHub.SendMessage(deviceId, Constant._Message_grant_type_check_file,
  228. new MessageContent { dataId=evaluationLocal.id, dataName=evaluationLocal.name, messageType= Constant._Message_type_check, status=msg_status, content=$"评测名单文件:{path_groupList}" });
  229. //await Task.Delay(DelayMacro);
  230. string path_source = Path.Combine(evaluationPath, "source.json");
  231. msg_status = Constant._Message_status_info;
  232. if (!System.IO.File.Exists(path_source))
  233. {
  234. file_error.Add("source");
  235. msg_status=Constant._Message_status_error;
  236. }
  237. else
  238. {
  239. msg_status=Constant._Message_status_success;
  240. }
  241. await _signalRExamServerHub.SendMessage(deviceId, Constant._Message_grant_type_check_file,
  242. new MessageContent { dataId=evaluationLocal.id, dataName=evaluationLocal.name, messageType= Constant._Message_type_check, status=msg_status, content=$"评测原始数据:{path_source}" });
  243. // await Task.Delay(DelayMacro);
  244. msg_status =Constant._Message_status_info;
  245. try {
  246. string evaluation_str = await System.IO.File.ReadAllTextAsync(path_evaluation);
  247. JsonNode? evaluation_data = evaluation_str.ToObject<JsonNode>();
  248. if (evaluation_data!=null)
  249. {
  250. EvaluationClient? evaluationClient = evaluation_data["evaluationClient"]?.ToObject<EvaluationClient>();
  251. if (evaluationClient!=null)
  252. {
  253. if ((!string.IsNullOrWhiteSpace(evaluationLocal.blobHash) && evaluationLocal.blobHash.Equals(evaluationClient.blobHash))
  254. &&(evaluationLocal.blobTime==evaluationClient.blobTime)
  255. &&(evaluationLocal.blobCount== evaluationClient.blobCount)
  256. &&(evaluationLocal.blobSize== evaluationClient.blobSize)&& (evaluationLocal.dataTime==evaluationClient.dataTime)
  257. &&(evaluationLocal.dataSize==evaluationClient.dataSize)&&(evaluationLocal.webviewCount==evaluationClient.webviewCount)
  258. &&(evaluationLocal.webviewSize== evaluationClient.webviewSize)
  259. &&(evaluationLocal.webviewTime== evaluationClient.webviewTime)
  260. &&(!string.IsNullOrWhiteSpace(evaluationLocal.webviewPath)&& evaluationLocal.webviewPath.Equals(evaluationClient.webviewPath)))
  261. {
  262. msg_status=1;
  263. }
  264. else
  265. {
  266. msg_status=Constant._Message_status_error;
  267. }
  268. await _signalRExamServerHub.SendMessage(deviceId, Constant._Message_grant_type_check_file,
  269. new MessageContent { dataId=evaluationLocal.id, dataName=evaluationLocal.name, messageType=Constant._Message_type_message, status=msg_status, content="校验本地数据文件..." });
  270. }
  271. List<EvaluationExam>? evaluationExams = evaluation_data["evaluationExams"]?.ToObject<List<EvaluationExam>>();
  272. if (evaluationExams.IsNotEmpty())
  273. {
  274. string path_papers = Path.Combine(evaluationPath, "papers");
  275. var papers_files = FileHelper.ListAllFiles(path_papers);
  276. foreach (var evaluationExam in evaluationExams!)
  277. {
  278. int paperIndex = 0;
  279. foreach (var paper in evaluationExam.papers)
  280. {
  281. paperIndex++;
  282. List<MessageContent> contents = new List<MessageContent>();
  283. int paper_error_count = 0;
  284. foreach (var blobInfo in paper.blobs)
  285. {
  286. msg_status=Constant._Message_status_info;
  287. if (!string.IsNullOrWhiteSpace(blobInfo.path))
  288. {
  289. var file = papers_files.Find(x => x.Contains(blobInfo.path));
  290. if (file!=null)
  291. {
  292. msg_status=1;
  293. msg_status=Constant._Message_status_success;
  294. }
  295. else {
  296. msg_status=Constant._Message_status_error;
  297. paper_error_count++;
  298. }
  299. }
  300. else {
  301. msg_status=Constant._Message_status_warning; ;
  302. paper_error_count++;
  303. }
  304. contents.Add(new MessageContent {
  305. dataId=evaluationLocal.id,
  306. dataName=evaluationLocal.name,
  307. messageType=Constant._Message_type_check, status=msg_status, content=$"试卷文件信息:{paper.paperName}" });
  308. }
  309. int paper_msg_status = Constant. _Message_status_info;
  310. if (paper_error_count>0)
  311. {
  312. paper_msg_status=Constant._Message_status_error;
  313. }
  314. else {
  315. paper_msg_status=Constant._Message_status_success;
  316. }
  317. await _signalRExamServerHub.SendMessage(deviceId, Constant._Message_grant_type_check_file,
  318. new MessageContent {
  319. dataId=evaluationLocal.id,
  320. dataName=evaluationLocal.name,
  321. messageType=Constant._Message_type_message,
  322. status=paper_msg_status,
  323. 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)}",
  324. contents=contents
  325. });
  326. }
  327. }
  328. }
  329. }
  330. }
  331. catch (Exception e) {
  332. }
  333. //检测参考名单
  334. await _signalRExamServerHub.SendMessage(deviceId, Constant._Message_grant_type_check_file,
  335. new MessageContent {
  336. dataId=evaluationLocal.id,
  337. dataName=evaluationLocal.name,
  338. messageType=Constant._Message_type_message, status=0, content="提取评测数据文件..." });
  339. }
  340. return Ok(new {code=200, evaluation= evaluationLocal,data,blob,webview,dataSize,blobSize,webviewSize,status ,groupList,studentCount});
  341. }
  342. /// <summary>
  343. /// 激活考试
  344. /// </summary>
  345. /// <param name="json"></param>
  346. /// <returns></returns>
  347. [HttpPost("activate-evaluation")]
  348. public IActionResult ActivateEvaluation(JsonNode json)
  349. {
  350. string id = $"{json["id"]}";
  351. string shortCode = $"{json["shortCode"]}";
  352. if (!string.IsNullOrWhiteSpace(id) && !string.IsNullOrWhiteSpace(shortCode))
  353. {
  354. EvaluationClient evaluationClient = _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationClient>().FindOne(x => x.id!.Equals(id) && !string.IsNullOrWhiteSpace(x.shortCode) && x.shortCode.Equals(shortCode));
  355. if (evaluationClient != null)
  356. {
  357. }
  358. }
  359. return Ok();
  360. }
  361. /// <summary>
  362. /// 加载本地的活动列表
  363. /// </summary>
  364. /// <param name="json"></param>
  365. /// <returns></returns>
  366. [HttpPost("list-local-evaluation")]
  367. public IActionResult ListLocalEvaluation(JsonNode json)
  368. {
  369. IEnumerable<EvaluationClient> evaluationClients = _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationClient>().FindAll().OrderByDescending(x=>x.activate).ThenByDescending(x=>x.stime);
  370. var result = evaluationClients.Select(client =>
  371. {
  372. var properties = client.GetType().GetProperties();
  373. var anonymousObject = new Dictionary<string, object?>();
  374. foreach (var property in properties)
  375. {
  376. if (!property.Name .Equals("password") && !property.Name.Equals("shortCode"))
  377. {
  378. anonymousObject[property.Name] = property.GetValue(client);
  379. }
  380. }
  381. return anonymousObject;
  382. });
  383. return Ok(new {code=200, evaluation= result });
  384. }
  385. }
  386. }