ManageController.cs 57 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962
  1. using ICSharpCode.SharpZipLib.GZip;
  2. using IES.ExamLibrary.Models;
  3. using IES.ExamServer.DI;
  4. using IES.ExamServer.DI.SignalRHost;
  5. using IES.ExamServer.Filters;
  6. using IES.ExamServer.Helper;
  7. using IES.ExamServer.Helpers;
  8. using IES.ExamServer.Models;
  9. using IES.ExamServer.Services;
  10. using Microsoft.AspNetCore.Http;
  11. using Microsoft.AspNetCore.Mvc;
  12. using Microsoft.AspNetCore.SignalR;
  13. using Microsoft.Extensions.Caching.Memory;
  14. using Microsoft.Extensions.Configuration;
  15. using System;
  16. using System.Diagnostics.Eventing.Reader;
  17. using System.Diagnostics.Metrics;
  18. using System.Linq.Expressions;
  19. using System.Net.Http;
  20. using System.Net.Http.Json;
  21. using System.Security.Cryptography.X509Certificates;
  22. using System.Text.Json;
  23. using System.Text.Json.Nodes;
  24. using System.Text.RegularExpressions;
  25. using static IES.ExamServer.Services.ManageService;
  26. using static System.Reflection.Metadata.BlobBuilder;
  27. namespace IES.ExamServer.Controllers
  28. {
  29. [ApiController]
  30. [Route("manage")]
  31. public class ManageController : BaseController
  32. {
  33. private readonly IConfiguration _configuration;
  34. private readonly IHttpClientFactory _httpClientFactory;
  35. private readonly IMemoryCache _memoryCache;
  36. private readonly ILogger<ManageController> _logger;
  37. private readonly LiteDBFactory _liteDBFactory;
  38. private readonly CenterServiceConnectionService _connectionService;
  39. private readonly int DelayMicro = 10;//微观数据延迟
  40. private readonly int DelayMacro = 100;//宏观数据延迟
  41. private readonly IHubContext<SignalRExamServerHub> _signalRExamServerHub;
  42. private readonly DataQueue _dataQueue;
  43. public ManageController(LiteDBFactory liteDBFactory, ILogger<ManageController> logger, IConfiguration configuration,
  44. IHttpClientFactory httpClientFactory, IMemoryCache memoryCache, CenterServiceConnectionService connectionService,
  45. IHubContext<SignalRExamServerHub> signalRExamServerHub, DataQueue dataQueue)
  46. {
  47. _logger = logger;
  48. _configuration=configuration;
  49. _httpClientFactory=httpClientFactory;
  50. _memoryCache=memoryCache;
  51. _liteDBFactory=liteDBFactory;
  52. _connectionService=connectionService;
  53. _signalRExamServerHub=signalRExamServerHub;
  54. _dataQueue=dataQueue;
  55. }
  56. /// <summary>
  57. /// 导出数据
  58. ///通过线上回传数据需要鉴权验证等操作。
  59. ///通过离线包回传数据需要加密操作
  60. /// </summary>
  61. /// <param name="json"></param>
  62. /// <returns></returns>
  63. [HttpPost("export-evaluation-result")]
  64. [AuthToken("admin", "teacher", "visitor")]
  65. public IActionResult ExportEvaluationResult(JsonNode json)
  66. {
  67. string evaluationId = $"{json["evaluationId"]}";
  68. string shortCode = $"{json["shortCode"]}";
  69. string openCode = $"{json["openCode"]}";
  70. string? loginToken = HttpContext.GetXAuth("AuthToken");
  71. EvaluationClient? evaluationClient = _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationClient>()
  72. .FindOne(x => x.id!.Equals(evaluationId) && x.openCode!.Equals(openCode));
  73. //已作答
  74. int answerCount = 0;
  75. //未作答
  76. int absentCount = 0;
  77. List<SubjectPushData> subjectPushDatas = new List<SubjectPushData>();
  78. List<SubjectPushData> subjectPushNew = new List<SubjectPushData>();
  79. if (evaluationClient!=null)
  80. {
  81. var datas = _liteDBFactory.GetLiteDatabase().GetCollection<SubjectPushData>()
  82. .Find(x => !string.IsNullOrWhiteSpace(x.evaluationId) && x.evaluationId!.Equals(evaluationId)).ToList();
  83. if (datas.IsNotEmpty())
  84. {
  85. subjectPushDatas.AddRange(datas);
  86. }
  87. var data = _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationStudentResult>()
  88. .Find(x => !string.IsNullOrWhiteSpace(x.evaluationId) && x.evaluationId.Equals(evaluationId));
  89. if (data.IsNotEmpty())
  90. {
  91. foreach (var item in data)
  92. {
  93. foreach (var subjectResult in item.subjectResults)
  94. {
  95. if (subjectResult.finished==1)
  96. {
  97. SubjectPushData subjectPushData = new SubjectPushData(subjectResult, item);
  98. subjectPushData.loginToken=loginToken;
  99. if (!datas.Exists(x => x.id!.Equals(subjectPushData.id)))
  100. {
  101. //数据丢失导致的推送SubjectPushData 没有被保存的问题
  102. subjectPushDatas.Add(subjectPushData);
  103. subjectPushNew.Add(subjectPushData);
  104. }
  105. }
  106. else { absentCount++; }
  107. }
  108. }
  109. }
  110. if (subjectPushNew.IsNotEmpty())
  111. {
  112. _liteDBFactory.GetLiteDatabase().GetCollection<SubjectPushData>().Upsert(subjectPushNew);
  113. }
  114. }
  115. answerCount=subjectPushDatas.Count(x => x.finished==1 && x.answers.IsNotEmpty());
  116. absentCount+=subjectPushDatas.Count()-answerCount;
  117. //TODO 以及增加 作答文件的相关数据。
  118. return Ok(new { answerCount, absentCount, subjectPushDatas });
  119. }
  120. /// <summary>
  121. /// 手动推送
  122. ///通过线上回传数据需要鉴权验证等操作。
  123. ///通过离线包回传数据需要加密操作
  124. /// </summary>
  125. /// <param name="json"></param>
  126. /// <returns></returns>
  127. [HttpPost("manual-push")]
  128. [AuthToken("admin", "teacher", "visitor")]
  129. public async Task<IActionResult> ManualPush(JsonNode json)
  130. {
  131. string evaluationId = $"{json["evaluationId"]}";
  132. //string shortCode = $"{json["shortCode"]}";
  133. string openCode = $"{json["openCode"]}";
  134. string deviceId = $"{json["deviceId"]}";
  135. EvaluationClient? evaluationClient = _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationClient>()
  136. .FindOne(x => x.id!.Equals(evaluationId) && x.openCode!.Equals(openCode));
  137. int count = 0;
  138. string? loginToken = HttpContext.GetXAuth("AuthToken");
  139. if (evaluationClient!=null)
  140. {
  141. var datas = _liteDBFactory.GetLiteDatabase().GetCollection<SubjectPushData>()
  142. .Find(x => !string.IsNullOrWhiteSpace(x.evaluationId) && x.evaluationId!.Equals(evaluationId)).ToList();
  143. if (datas.IsNotEmpty())
  144. {
  145. foreach (var item in datas)
  146. {
  147. if (item.pushed<=1 && item.finished==1)
  148. {
  149. item.order=count;
  150. item.loginToken=loginToken;
  151. count++;
  152. //_logger.LogInformation($"推送数据加入队列=>>序号:{subjectPushData.order}--学号:{item.studentId}--姓名:{item.studentName}--科目:{subjectResult.subjectName}");
  153. await _dataQueue.TryAddAsync(item);
  154. }
  155. }
  156. }
  157. var data = _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationStudentResult>()
  158. .Find(x =>!string.IsNullOrWhiteSpace(x.evaluationId) && x.evaluationId.Equals(evaluationId) );
  159. List<SubjectPushData> subjectPushDatas = new List<SubjectPushData>();
  160. if (data.IsNotEmpty())
  161. {
  162. foreach (var item in data)
  163. {
  164. foreach (var subjectResult in item.subjectResults)
  165. {
  166. if ( subjectResult.finished==1)
  167. {
  168. SubjectPushData subjectPushData= new SubjectPushData(subjectResult, item);
  169. subjectPushData.order=count;
  170. subjectPushData.loginToken=loginToken;
  171. if (!datas.Exists(x => x.id!.Equals(subjectPushData.id)))
  172. {
  173. //数据丢失导致的推送SubjectPushData 没有被保存的问题
  174. subjectPushDatas.Add(subjectPushData);
  175. //未被推送的数据
  176. if (subjectPushData.pushed<=1 && subjectPushData.finished==1)
  177. {
  178. count++;
  179. subjectPushData.pushed=1;
  180. await _dataQueue.TryAddAsync(subjectPushData);
  181. }
  182. }
  183. }
  184. }
  185. }
  186. }
  187. if (subjectPushDatas.IsNotEmpty())
  188. {
  189. _liteDBFactory.GetLiteDatabase().GetCollection<SubjectPushData>().Upsert(subjectPushDatas);
  190. }
  191. }
  192. return Ok(new { code=200, message="推送成功!", count });
  193. }
  194. /// <summary>
  195. /// 清理缓存,列出缓存占用空间,type =list列出,type=clear清理,不能清理近期及正在激活的数据,并且提示清理中暂未上传或者导出的数据。
  196. /// </summary>
  197. /// <param name="json"></param>
  198. /// <returns></returns>
  199. [HttpPost("clean-cache")]
  200. [AuthToken("admin", "teacher", "visitor")]
  201. public async Task<IActionResult> CleanCache(JsonNode json)
  202. {
  203. return Ok();
  204. }
  205. //C#.NET 6 后端与前端流式通信
  206. //https://www.doubao.com/chat/collection/687687510791426?type=Thread
  207. //下载日志记录:1.步骤,检查,2.获取描述信息,3.分类型,4下载文件,5.前端处理,6.返回结果 , 正在下载...==> [INFO]https://www.doubao.com/chat/collection/687687510791426?type=Thread [Size=180kb] Ok...
  208. //进度条 展示下载文件总大小和已下载,末尾展示 文件总个数和已下载个数
  209. //https://cdnjs.cloudflare.com/ajax/libs/microsoft-signalr/8.0.7/signalr.min.js
  210. /* int data = 0,blob=0, groupList=0
  211. {
  212. "evaluationId":"idssss",
  213. "shortCode":"1234567890",
  214. "ownerId":"hbcn/tmdid",
  215. "data":1,
  216. "blob":1,
  217. "groupList":1
  218. }
  219. */
  220. //如果要访问中心,则需要教师登录联网。
  221. [HttpPost("download-package-music")]
  222. [AuthToken("admin", "teacher", "visitor")]
  223. public async Task<IActionResult> DownloadPackageMusic(JsonNode json)
  224. {
  225. return Ok();
  226. }
  227. [HttpPost("download-package")]
  228. [AuthToken("admin", "teacher", "visitor")]
  229. public async Task<IActionResult> DownloadPackage(JsonNode json)
  230. {
  231. var token = GetAuthTokenInfo();
  232. //检查试卷文件完整性
  233. List<string> successMsgs = new List<string>();
  234. List<string> errorMsgs = new List<string>();
  235. EvaluationCheckFileResult result = new EvaluationCheckFileResult() { successMsgs=successMsgs, errorMsgs=errorMsgs };
  236. if (token.scope.Equals(ExamConstant.ScopeTeacher) || token.scope.Equals(ExamConstant.ScopeVisitor))
  237. {
  238. if (_connectionService.centerIsConnected)
  239. {
  240. Teacher? teacher = _liteDBFactory.GetLiteDatabase().GetCollection<Teacher>().FindOne(x => x.id!.Equals(token.id));
  241. string id = $"{json["evaluationId"]}";
  242. string shortCode = $"{json["shortCode"]}";
  243. string deviceId = $"{json["deviceId"]}";
  244. EvaluationClient? evaluationClient = _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationClient>().FindOne(x => x.id!.Equals(id) && x.shortCode!.Equals(shortCode));
  245. if (teacher != null && evaluationClient!= null)
  246. {
  247. int msg_status = Constant._Message_status_info;
  248. string msg_content = msg_status.Equals(Constant._Message_status_success) ? "成功" : "失败";
  249. await _signalRExamServerHub.SendMessage(_memoryCache, _logger, deviceId, Constant._Message_grant_type_download_file,
  250. new MessageContent { dataId = evaluationClient.id, dataName = evaluationClient.name, messageType = Constant._Message_type_message, status = 0, step=1, content = "检测云端数据是否匹配..." });
  251. var dataInfo= await GetEvaluationFromCenter(GetXAuthToken(), _configuration, _httpClientFactory, shortCode, evaluationClient.id!);
  252. if (dataInfo.centerCode.Equals("200")&& dataInfo.evaluationCloud!=null)
  253. {
  254. await _signalRExamServerHub.SendMessage(_memoryCache, _logger, deviceId, Constant._Message_grant_type_download_file,
  255. new MessageContent { dataId = evaluationClient.id, dataName = evaluationClient.name, messageType = Constant._Message_type_message, status = Constant._Message_status_success, step = 1, content = $"云端数据检测结果:{dataInfo.centerMsg},状态:{dataInfo.centerCode}" },true);
  256. string? CenterUrl = _configuration.GetValue<string>("ExamServer:CenterUrl");
  257. var client = _httpClientFactory.CreateClient();
  258. if (client.DefaultRequestHeaders.Contains(Constant._X_Auth_AuthToken))
  259. {
  260. client.DefaultRequestHeaders.Remove(Constant._X_Auth_AuthToken);
  261. }
  262. client.DefaultRequestHeaders.Add(Constant._X_Auth_AuthToken, teacher.x_auth_token);
  263. HttpResponseMessage message = await client.PostAsJsonAsync($"{CenterUrl}/blob/sas-read", new { containerName = $"{dataInfo.evaluationCloud.ownerId}" });
  264. string sas = string.Empty;
  265. string url = string.Empty;
  266. string cnt = string.Empty;
  267. if (message.IsSuccessStatusCode)
  268. {
  269. //url sas timeout name
  270. string content = await message.Content.ReadAsStringAsync();
  271. JsonNode? jsonNode = content.ToObject<JsonNode>();
  272. if (jsonNode != null)
  273. {
  274. sas = $"{jsonNode["sas"]}";
  275. cnt = $"{jsonNode["name"]}";
  276. url = $"{jsonNode["url"]}";
  277. msg_status = Constant._Message_status_success;
  278. }
  279. else
  280. {
  281. msg_status = Constant._Message_status_error;
  282. }
  283. }
  284. else
  285. {
  286. msg_status = Constant._Message_status_error;
  287. }
  288. msg_content = msg_status.Equals(Constant._Message_status_success) ? "成功" : "失败";
  289. await _signalRExamServerHub.SendMessage(_memoryCache, _logger, deviceId, Constant._Message_grant_type_download_file,
  290. new MessageContent { dataId = evaluationClient.id, dataName = evaluationClient.name, messageType = Constant._Message_type_message, status = msg_status, step = 1, content = $"获取云端下载授权=>{msg_content}" }, true);
  291. var httpClient = _httpClientFactory.CreateClient();
  292. string packagePath = Path.Combine(Directory.GetCurrentDirectory(), "wwwroot", "package");
  293. string evaluationPath = Path.Combine(packagePath, dataInfo.evaluationCloud.id!);
  294. //删除文件夹
  295. FileHelper.DeleteFolder(evaluationPath);
  296. string evaluationDataPath = Path.Combine(evaluationPath, "data");
  297. if (!Directory.Exists(evaluationDataPath))
  298. {
  299. Directory.CreateDirectory(evaluationDataPath);
  300. }
  301. await _signalRExamServerHub.SendMessage(_memoryCache, _logger, deviceId, Constant._Message_grant_type_download_file,
  302. new MessageContent { dataId = evaluationClient.id, dataName = evaluationClient.name, messageType = Constant._Message_type_message, step = 1, finish=1,status = Constant._Message_status_info, content = $"下载前清理资源" });
  303. string evaluationData = string.Empty;
  304. {
  305. //evaluation
  306. string evaluationUrl = $"{url}/{cnt}/package/{json["evaluationId"]}/data/evaluation.json?{sas}";
  307. HttpResponseMessage dataMessage = await httpClient.GetAsync(evaluationUrl);
  308. if (dataMessage.IsSuccessStatusCode)
  309. {
  310. var content = await dataMessage.Content.ReadAsStringAsync();
  311. evaluationData=content;
  312. string path_evaluation = Path.Combine(evaluationDataPath, "evaluation.json");
  313. await System.IO.File.WriteAllTextAsync(path_evaluation, content);
  314. successMsgs.Add("评测信息文件evaluation.json文件下载成功!");
  315. await _signalRExamServerHub.SendMessage(_memoryCache, _logger, deviceId, Constant._Message_grant_type_download_file,
  316. new MessageContent { dataId = evaluationClient.id, dataName = evaluationClient.name, messageType = Constant._Message_type_message, step = 2, finish = 1, status = Constant._Message_status_success, content = $"评测信息文件evaluation.json文件下载成功!" });
  317. }
  318. else
  319. {
  320. errorMsgs.Add("评测信息文件evaluation.json文件下载失败!");
  321. await _signalRExamServerHub.SendMessage(_memoryCache, _logger, deviceId, Constant._Message_grant_type_download_file,
  322. new MessageContent { dataId = evaluationClient.id, dataName = evaluationClient.name, messageType = Constant._Message_type_message, step = 2, finish = 1, status = Constant._Message_status_error, content = $"评测信息文件evaluation.json文件下载失败!" }, true);
  323. }
  324. }
  325. //{
  326. // //source.json
  327. // string sourceUrl = $"{url}/{cnt}/package/{json["evaluationId"]}/data/source.json?{sas}";
  328. // HttpResponseMessage dataMessage = await httpClient.GetAsync(sourceUrl);
  329. // if (dataMessage.IsSuccessStatusCode)
  330. // {
  331. // var content = await dataMessage.Content.ReadAsStringAsync();
  332. // string path_source = Path.Combine(evaluationDataPath, "source.json");
  333. // await System.IO.File.WriteAllTextAsync(path_source, content);
  334. // successMsgs.Add("评测数据原始文件source.json文件下载成功!");
  335. // await _signalRExamServerHub.SendMessage(_memoryCache, _logger, deviceId, Constant._Message_grant_type_download_file,
  336. // new MessageContent { dataId = evaluationClient.id, dataName = evaluationClient.name, messageType = Constant._Message_type_message, status = Constant._Message_status_success, content = $"评测数据原始文件source.json文件下载成功!" });
  337. // }
  338. // else
  339. // {
  340. // errorMsgs.Add("评测数据原始文件source.json文件下载失败!");
  341. // await _signalRExamServerHub.SendMessage(_memoryCache, _logger, deviceId, Constant._Message_grant_type_download_file,
  342. // new MessageContent { dataId = evaluationClient.id, dataName = evaluationClient.name, messageType = Constant._Message_type_message, status = Constant._Message_status_error, content = $"评测数据原始文件source.json文件下载失败!" }, true);
  343. // }
  344. //}
  345. {
  346. //grouplist.json
  347. string grouplistUrl = $"{url}/{cnt}/package/{json["evaluationId"]}/data/grouplist.json?{sas}";
  348. HttpResponseMessage groupListMessage = await httpClient.GetAsync(grouplistUrl);
  349. if (groupListMessage.IsSuccessStatusCode)
  350. {
  351. var content = await groupListMessage.Content.ReadAsStringAsync();
  352. string path_groupList = Path.Combine(evaluationDataPath, "grouplist.json");
  353. await System.IO.File.WriteAllTextAsync(path_groupList, content);
  354. successMsgs.Add("评测名单grouplist.json文件下载成功!");
  355. await _signalRExamServerHub.SendMessage(_memoryCache, _logger, deviceId, Constant._Message_grant_type_download_file,
  356. new MessageContent { dataId = evaluationClient.id, dataName = evaluationClient.name, messageType = Constant._Message_type_message, step = 3, finish = 1, status = Constant._Message_status_success, content = $"评测名单grouplist.json文件下载成功!" });
  357. }
  358. else
  359. {
  360. errorMsgs.Add("评测名单grouplist.json文件下载失败!");
  361. await _signalRExamServerHub.SendMessage(_memoryCache, _logger, deviceId, Constant._Message_grant_type_download_file,
  362. new MessageContent { dataId = evaluationClient.id, dataName = evaluationClient.name, messageType = Constant._Message_type_message, step = 3, finish = 1, status = Constant._Message_status_error, content = $"评测名单grouplist.json文件下载失败!" }, true);
  363. }
  364. }
  365. {
  366. //下载试卷文件
  367. List<EvaluationExam>? evaluationExams = evaluationData.ToObject<JsonNode>()?["evaluationExams"]?.ToObject<List<EvaluationExam>>();
  368. int blobCount = evaluationExams!.SelectMany(x => x.papers).SelectMany(x=>x.blobs).Count();
  369. int currCount = 0;
  370. foreach (var evaluationExam in evaluationExams!)
  371. {
  372. foreach (var evaluationPaper in evaluationExam.papers)
  373. {
  374. string path_paper = Path.Combine(evaluationPath, $"papers/{evaluationPaper.paperId}");
  375. if (!Directory.Exists(path_paper))
  376. {
  377. Directory.CreateDirectory(path_paper);
  378. }
  379. //最多开启10个线程并行下载
  380. var parallelOptions = new ParallelOptions { MaxDegreeOfParallelism = 5 };
  381. // 使用 Parallel.ForEachAsync 并行处理每个 blob
  382. await Parallel.ForEachAsync(evaluationPaper.blobs, parallelOptions, async (blob, cancellationToken) =>
  383. {
  384. currCount++;
  385. double size = Math.Round(blob.size * 1.0 / 1024 / 1024, 2);
  386. string? fileName = Path.GetFileName(blob.path);
  387. try
  388. {
  389. // 下载 Blob 文件到本地
  390. //httpClient.Timeout = TimeSpan.FromSeconds(300);
  391. HttpResponseMessage blobMessage = await httpClient.GetAsync($"{url}/{cnt}/{blob.path}?{sas}", cancellationToken);
  392. if (blobMessage.IsSuccessStatusCode)
  393. {
  394. byte[] bytes = await blobMessage.Content.ReadAsByteArrayAsync(cancellationToken);
  395. string? extension = Path.GetExtension(blob.path);
  396. if (extension!=null)
  397. {
  398. if (extension.Equals(extension.ToUpper()))
  399. {
  400. string? fileNameWithoutExtension = Path.GetFileNameWithoutExtension(blob.path);
  401. await System.IO.File.WriteAllBytesAsync(Path.Combine(path_paper, $"{fileNameWithoutExtension!}_1{extension}"), bytes, cancellationToken);
  402. }
  403. else
  404. {
  405. await System.IO.File.WriteAllBytesAsync(Path.Combine(path_paper, fileName!), bytes, cancellationToken);
  406. }
  407. }
  408. await _signalRExamServerHub.SendMessage(_memoryCache, _logger, deviceId, Constant._Message_grant_type_download_file,
  409. new MessageContent { dataId = evaluationClient.id, dataName = evaluationClient.name, messageType = Constant._Message_type_message, step = 4, status = Constant._Message_status_success, content = $"[进度:{currCount}/{blobCount}][大小:{size}kb]{evaluationExam.subjectName},{evaluationPaper.paperName},{fileName}文件下载成功。" });
  410. }
  411. else
  412. {
  413. string? error = await blobMessage.Content.ReadAsStringAsync(cancellationToken);
  414. errorMsgs.Add($"{evaluationExam.subjectName},{evaluationPaper.paperName},{blob.path}文件下载失败,{blobMessage.StatusCode},{error}");
  415. await _signalRExamServerHub.SendMessage(_memoryCache, _logger, deviceId, Constant._Message_grant_type_download_file,
  416. new MessageContent { dataId = evaluationClient.id, dataName = evaluationClient.name, messageType = Constant._Message_type_message, step = 4, status = Constant._Message_status_error, content = $"[进度:{currCount}/{blobCount}][大小:{size}kb]{evaluationExam.subjectName},{evaluationPaper.paperName},{fileName}文件下载失败,{blobMessage.StatusCode},{error}" }, true);
  417. // Console.WriteLine($"Error downloading {blob.path},{blobMessage.StatusCode},{error}");
  418. }
  419. }
  420. catch (Exception ex)
  421. {
  422. errorMsgs.Add($"{evaluationExam.subjectName},{evaluationPaper.paperName},{blob.path}文件下载错误,{ex.Message}");
  423. await _signalRExamServerHub.SendMessage(_memoryCache, _logger, deviceId, Constant._Message_grant_type_download_file,
  424. new MessageContent { dataId = evaluationClient.id, dataName = evaluationClient.name, messageType = Constant._Message_type_message, step = 4, status = Constant._Message_status_error, content = $"[进度:{currCount}/{blobCount}][大小:{size}kb]{evaluationExam.subjectName},{evaluationPaper.paperName},{fileName}文件下载错误,{ex.Message}" }, true);
  425. // 处理异常
  426. //Console.WriteLine($"Error downloading {blob.path}: {ex.Message}");
  427. }
  428. });
  429. }
  430. }
  431. }
  432. _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationClient>().Upsert(dataInfo.evaluationCloud!);
  433. await _signalRExamServerHub.SendMessage(_memoryCache, _logger, deviceId, Constant._Message_grant_type_download_file,
  434. new MessageContent { dataId = evaluationClient.id, dataName = evaluationClient.name, messageType = Constant._Message_type_message, step = 4, finish = 1, status = Constant._Message_status_info, content = $"评测试卷下载完成" });
  435. (successMsgs, errorMsgs) = await ManageService.CheckFile(dataInfo.evaluationCloud!, successMsgs, errorMsgs, _signalRExamServerHub, _memoryCache, _logger, deviceId, evaluationPath, Constant._Message_grant_type_download_file,step:5);
  436. await _signalRExamServerHub.SendMessage(_memoryCache, _logger, deviceId, Constant._Message_grant_type_download_file,
  437. new MessageContent { dataId = evaluationClient.id, dataName = evaluationClient.name, messageType = Constant._Message_type_message, step = 5, finish = 1, status = Constant._Message_status_info, content = $"评测完整性验证完成" });
  438. await _signalRExamServerHub.SendMessage(_memoryCache, _logger, deviceId, Constant._Message_grant_type_download_file,
  439. new MessageContent { dataId = evaluationClient.id, dataName = evaluationClient.name, messageType = Constant._Message_type_message, step = 6, status = Constant._Message_status_info, content = $"正在创建压缩包,请稍等..." });
  440. //下载完成后,对数据进行检查,然后在加密压缩。
  441. string zipPath = Path.Combine(Directory.GetCurrentDirectory(), "wwwroot", "zip");
  442. if (!Directory.Exists(zipPath))
  443. {
  444. Directory.CreateDirectory(zipPath);
  445. }
  446. string zipFilePath = Path.Combine(zipPath, $"{dataInfo.evaluationCloud.id}-{dataInfo.evaluationCloud.blobHash}.zip");
  447. var zipInfo = ZipHelper.CreatePasswordProtectedZip(evaluationPath, zipFilePath, dataInfo.evaluationCloud.openCode!);
  448. if (zipInfo.res)
  449. {
  450. successMsgs.Add("评测数据压缩包创建成功!");
  451. await _signalRExamServerHub.SendMessage(_memoryCache, _logger, deviceId, Constant._Message_grant_type_download_file,
  452. new MessageContent { dataId = evaluationClient.id, dataName = evaluationClient.name, messageType = Constant._Message_type_message, step = 6, finish = 1, status = Constant._Message_status_success, content = $"评测数据压缩包创建成功!" });
  453. }
  454. else
  455. {
  456. errorMsgs.Add("评测数据压缩包创建失败!");
  457. await _signalRExamServerHub.SendMessage(_memoryCache, _logger, deviceId, Constant._Message_grant_type_download_file,
  458. new MessageContent { dataId = evaluationClient.id, dataName = evaluationClient.name, messageType = Constant._Message_type_message, step = 6, finish = 1, status = Constant._Message_status_error, content = $"评测数据压缩包创建失败!" }, true);
  459. }
  460. }
  461. else {
  462. string content = $"云端数据检测结果:{dataInfo.centerMsg},状态:{dataInfo.centerCode}";
  463. errorMsgs.Add(content);
  464. await _signalRExamServerHub.SendMessage(_memoryCache, _logger, deviceId, Constant._Message_grant_type_download_file,
  465. new MessageContent { dataId = evaluationClient.id, dataName = evaluationClient.name, messageType = Constant._Message_type_message, step = 1, finish = 1, status = Constant._Message_status_error, content = content }, true);
  466. }
  467. }
  468. else
  469. {
  470. string teacherMsg = teacher==null ? "未找到教师信息" : "";
  471. string evaluationMsg = evaluationClient==null ? "未找到评测信息" : "";
  472. errorMsgs.Add($"用户信息或未找到评测信息!{teacherMsg}{evaluationMsg}");
  473. }
  474. }
  475. else {
  476. errorMsgs.Add($"云端数据中心未连接");
  477. }
  478. }
  479. else
  480. {
  481. errorMsgs.Add($"请使用教师或访客账号登录!");
  482. }
  483. result.checkError=result.errorMsgs.Count();
  484. result.checkSuccess=result.successMsgs.Count();
  485. result.checkTotal=result.checkSuccess+result.checkError;
  486. if (result.errorMsgs.Count()==0)
  487. {
  488. return Ok(new { code = 200, msg = "下载成功!", result });
  489. }
  490. else {
  491. return Ok(new { code = 1, msg = "下载失败!", result });
  492. }
  493. }
  494. /// <summary>
  495. /// 输入开卷码,查询评测信息,并返回数据。
  496. /// </summary>
  497. /// <param name="json"></param>
  498. /// <returns></returns>
  499. [HttpPost("open-evaluation")]
  500. [AuthToken("admin", "teacher", "visitor")]
  501. public async Task<IActionResult> OpenEvaluation(JsonNode json)
  502. {
  503. string deviceId = $"{json["deviceId"]}";
  504. string evaluationId = $"{json["evaluationId"]}";
  505. string openCode = $"{json["openCode"]}";
  506. string shortCode = $"{json["shortCode"]}";
  507. var token = GetAuthTokenInfo();
  508. List<string> successMsgs = new List<string>();
  509. List<string> errorMsgs = new List<string>();
  510. EvaluationCheckFileResult result = new EvaluationCheckFileResult() { successMsgs=successMsgs, errorMsgs =errorMsgs };
  511. if (!string.IsNullOrEmpty(openCode) && !string.IsNullOrEmpty(evaluationId))
  512. {
  513. EvaluationClient? evaluationLocal = _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationClient>().FindOne(x => x.id!.Equals(evaluationId) && x.openCode!.Equals(openCode) && x.shortCode!.Equals(shortCode));
  514. if (evaluationLocal!=null)
  515. {
  516. //ManageService.CheckData(evaluationLocal, null, successMsgs, errorMsgs, _liteDBFactory);
  517. await _signalRExamServerHub.SendMessage(_memoryCache, _logger, deviceId, Constant._Message_grant_type_check_file,
  518. new MessageContent { dataId=evaluationLocal.id, dataName=evaluationLocal.name, messageType=Constant._Message_type_message, status=0, content="开始检查评测信息文件.." });
  519. //判断文件包的压缩包是否存在。
  520. string zipPath = Path.Combine(Directory.GetCurrentDirectory(), "wwwroot", "zip");
  521. //校验本地文件数据
  522. string packagePath = Path.Combine(Directory.GetCurrentDirectory(), "wwwroot", "package");
  523. string evaluationPath = Path.Combine(packagePath, evaluationLocal.id!);
  524. if (System.IO.File.Exists(Path.Combine(zipPath, $"{evaluationLocal.id}-{evaluationLocal.blobHash}.zip")))
  525. {
  526. string key = $"{Constant._KeyEvaluationZipExtract}:{evaluationId}:{openCode}";
  527. successMsgs.Add("加载评测试卷文件包!");
  528. bool needExtract = false;
  529. if (Directory.Exists(evaluationPath))
  530. {
  531. (successMsgs, errorMsgs) = await ManageService.CheckFile(evaluationLocal, successMsgs, errorMsgs, _signalRExamServerHub, _memoryCache, _logger, deviceId, evaluationPath, Constant._Message_grant_type_check_file);
  532. if (errorMsgs.Count() > 0)
  533. {
  534. //删除文件夹
  535. FileHelper.DeleteFolder(evaluationPath);
  536. needExtract = true;
  537. }
  538. else
  539. {
  540. DirectoryInfo directoryInfo = new DirectoryInfo(evaluationPath);
  541. // 获取文件夹的创建时间
  542. DateTime creationTime = directoryInfo.CreationTime;
  543. // 计算时间差
  544. DateTime currentTime = DateTime.Now;
  545. TimeSpan timeDifference = currentTime - creationTime;
  546. // 判断是否超过两天
  547. if (timeDifference.TotalDays > 2)
  548. {
  549. needExtract = true;
  550. }
  551. }
  552. }
  553. else
  554. {
  555. needExtract = true;
  556. }
  557. if (needExtract)
  558. {
  559. FileHelper.DeleteFolder(evaluationPath);
  560. if (!Directory.Exists(evaluationPath))
  561. {
  562. Directory.CreateDirectory(evaluationPath);
  563. }
  564. //解压文件包
  565. var extractRes = await ZipHelper.ExtractPasswordProtectedZip(Path.Combine(zipPath, $"{evaluationLocal.id}-{evaluationLocal.blobHash}.zip"), evaluationPath, evaluationLocal.openCode!, _signalRExamServerHub, _memoryCache, _logger, deviceId, evaluationLocal);
  566. if (extractRes.res)
  567. {
  568. successMsgs.Add("评测试卷文件包解压成功!");
  569. (successMsgs, errorMsgs) = await ManageService.CheckFile(evaluationLocal, successMsgs, errorMsgs, _signalRExamServerHub, _memoryCache, _logger, deviceId, evaluationPath, Constant._Message_grant_type_check_file);
  570. }
  571. else
  572. {
  573. errorMsgs.Add("评测试卷文件包解压失败!");
  574. //return Ok(new { code = 3, msg = "评测试卷文件包解压失败!" });
  575. }
  576. }
  577. else
  578. {
  579. successMsgs.Add("评测试卷文件包检测成功!");
  580. }
  581. }
  582. else {
  583. errorMsgs.Add("评测试卷文件包不存在!");
  584. //return Ok(new { code = 3, msg = "评测试卷文件包不存在!" });
  585. }
  586. }
  587. else {
  588. errorMsgs.Add("未找到评测信息!");
  589. // return Ok(new { code = 2, msg = "未找到评测信息!" });
  590. }
  591. }
  592. else
  593. {
  594. errorMsgs.Add("评测ID或提取码均未填写!");
  595. // return Ok(new { code = 1, msg = "评测ID或提取码均未填写!" });
  596. }
  597. result.checkTotal = result.errorMsgs.Count() + result.successMsgs.Count();
  598. result.checkError= result.errorMsgs.Count();
  599. result.checkSuccess = result.successMsgs.Count();
  600. if (result.errorMsgs.Count()==0)
  601. {
  602. return Ok(new { code = 200, msg = "校验成功!", result });
  603. }
  604. else
  605. {
  606. return Ok(new { code = 1, msg = "校验失败!", result });
  607. }
  608. }
  609. /// <summary>
  610. /// 检查数据包是否有更新,zip是否存在,数据文件是否完整,名单文件是否完整。
  611. /// 如果有智音模块,则需要检查智音音乐缓存是否完整。
  612. /// </summary>
  613. /// <param name="json"></param>
  614. /// <returns></returns>
  615. [HttpPost("check-evaluation")]
  616. [AuthToken("admin", "teacher", "visitor")]
  617. public async Task<IActionResult> CheckEvaluation(JsonNode json)
  618. {
  619. string shortCode = $"{json["shortCode"]}";
  620. string evaluationId = $"{json["evaluationId"]}";
  621. string checkCenter = $"{json["checkCenter"]}";
  622. string deviceId = $"{json["deviceId"]}";
  623. string centerCode = string.Empty, centerMsg = string.Empty;
  624. Expression<Func<EvaluationClient, bool>> predicate = x => true;
  625. var token = GetAuthTokenInfo();
  626. if (!string.IsNullOrEmpty(shortCode) || !string.IsNullOrEmpty(evaluationId))
  627. {
  628. if (!string.IsNullOrEmpty(shortCode))
  629. {
  630. predicate= predicate.And(x => !string.IsNullOrWhiteSpace(x.shortCode) && x.shortCode.Equals(shortCode));
  631. }
  632. if (!string.IsNullOrWhiteSpace(evaluationId))
  633. {
  634. predicate= predicate.And(x => x.id!.Equals(evaluationId));
  635. }
  636. }
  637. else
  638. {
  639. return Ok(new { code = 400, msg = "评测ID或提取码均未填写!" });
  640. }
  641. IEnumerable<EvaluationClient>? evaluationClients = _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationClient>().Find(predicate);
  642. EvaluationClient? evaluationLocal = null;
  643. EvaluationClient? evaluationCloud = null;
  644. if (evaluationClients!=null && evaluationClients.Count()>0)
  645. {
  646. evaluationLocal= evaluationClients.First();
  647. }
  648. List<string> successMsgs = new List<string>();
  649. List<string> errorMsgs = new List<string>();
  650. //从数据中心搜索
  651. if ("1".Equals($"{checkCenter}"))
  652. {
  653. if (_connectionService.centerIsConnected)
  654. {
  655. if (token.scope.Equals(ExamConstant.ScopeTeacher) || token.scope.Equals(ExamConstant.ScopeVisitor))//由于已经绑定学校,访客教师也可以访问中心。
  656. {
  657. Teacher? teacher = _liteDBFactory.GetLiteDatabase().GetCollection<Teacher>().FindOne(x => x.id!.Equals(token.id));
  658. if (teacher != null)
  659. {
  660. (evaluationCloud, centerCode, centerMsg)= await ManageService.GetEvaluationFromCenter(GetXAuthToken(), _configuration,_httpClientFactory,shortCode,evaluationId);
  661. }
  662. else
  663. {
  664. centerCode = $"401";
  665. centerMsg = "当前登录账号未找到";
  666. }
  667. }
  668. }
  669. else
  670. {
  671. centerCode = $"404";
  672. centerMsg = "云端数据中心未连接";
  673. }
  674. if (centerCode.Equals("200"))
  675. {
  676. successMsgs.Add($"云端数据检测结果:{centerMsg},状态:{centerCode}");
  677. }
  678. else
  679. {
  680. errorMsgs.Add($"云端数据检测结果:{centerMsg},状态:{centerCode}");
  681. }
  682. }
  683. EvaluationCheckDataResult checkDataResult;
  684. (checkDataResult, evaluationLocal) = ManageService.CheckData( evaluationLocal, evaluationCloud, successMsgs, errorMsgs, _liteDBFactory);
  685. Dictionary<string, object?>? evaluation = null;
  686. if (evaluationLocal!=null)
  687. {
  688. string zipPath = Path.Combine(Directory.GetCurrentDirectory(), "wwwroot", "zip");
  689. //判断文件包的压缩包是否存在。
  690. if (!System.IO.File.Exists(Path.Combine(zipPath, $"{evaluationLocal.id}-{evaluationLocal.blobHash}.zip")))
  691. {
  692. checkDataResult.zip =1;
  693. checkDataResult.blob=1;
  694. checkDataResult.data=1;
  695. checkDataResult.groupList=1;
  696. checkDataResult.blobSize=evaluationLocal.blobSize;
  697. checkDataResult.dataSize=evaluationLocal.dataSize;
  698. checkDataResult.studentCount=evaluationLocal.studentCount;
  699. errorMsgs.Add($"评测文件包压缩包不存在,需要重新下载评测!");
  700. }
  701. var properties = evaluationLocal.GetType().GetProperties();
  702. evaluation = new Dictionary<string, object?>();
  703. foreach (var property in properties)
  704. {
  705. if (!property.Name.Equals("openCode"))
  706. {
  707. evaluation[property.Name] = property.GetValue(evaluationLocal);
  708. }
  709. }
  710. }
  711. return Ok(new
  712. {
  713. code = 200,
  714. evaluation = evaluation,
  715. result = checkDataResult
  716. });
  717. }
  718. /// <summary>
  719. /// 获取当前评测的开考设置信息
  720. /// </summary>
  721. /// <param name="json"></param>
  722. /// <returns></returns>
  723. [HttpPost("list-evaluation-round")]
  724. [AuthToken("admin", "teacher", "visitor")]
  725. public IActionResult ListEvaluationRound(JsonNode json)
  726. {
  727. string evaluationId = $"{json["evaluationId"]}";
  728. string openCode = $"{json["openCode"]}";
  729. string shortCode = $"{json["shortCode"]}";
  730. EvaluationClient? evaluationClient = _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationClient>()
  731. .FindOne(x => x.id!.Equals(evaluationId) && x.shortCode!.Equals(shortCode) && x.openCode!.Equals(openCode));
  732. if (evaluationClient!=null)
  733. {
  734. IEnumerable<EvaluationRoundSetting>? settings = _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationRoundSetting>().Find(x => x.evaluationId!.Equals(evaluationClient.id)).OrderByDescending(x => x.activate).ThenByDescending(x => x.startline).ThenByDescending(x => x.createTime);
  735. if (settings.IsNotEmpty())
  736. {
  737. return Ok(new { code = 200, msg = "OK", settings = settings });
  738. }
  739. else
  740. {
  741. return Ok(new { code = 2, msg = "未设置开考信息!" });
  742. }
  743. }
  744. else
  745. {
  746. return Ok(new { code = 1, msg = "未找到评测信息!" });
  747. }
  748. }
  749. /// <summary>
  750. /// 获取当前评测的开考设置信息
  751. /// </summary>
  752. /// <param name="json"></param>
  753. /// <returns></returns>
  754. [HttpPost("load-evaluation-round")]
  755. [AuthToken("admin", "teacher", "visitor")]
  756. public IActionResult LoadEvaluationRound(JsonNode json)
  757. {
  758. string evaluationId = $"{json["evaluationId"]}";
  759. string openCode = $"{json["openCode"]}";
  760. string shortCode = $"{json["shortCode"]}";
  761. string settingId = $"{json["settingId"]}";
  762. EvaluationClient? evaluationClient = _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationClient>()
  763. .FindOne(x => x.id!.Equals(evaluationId) && x.shortCode!.Equals(shortCode) && x.openCode!.Equals(openCode));
  764. EvaluationRoundSetting? setting = null;
  765. if (evaluationClient!=null)
  766. {
  767. setting = _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationRoundSetting>().FindOne(x =>x.id!.Equals(settingId) && x.evaluationId!.Equals(evaluationClient.id));
  768. if (setting!=null)
  769. {
  770. IEnumerable<EvaluationStudentResult>? results = null;
  771. var members = _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationMember>().Find(x => x.evaluationId!.Equals(evaluationClient.id) && x.roundId!.Equals(setting.id));
  772. //并获取学生的作答信息
  773. //顺便返回本轮的学生名单
  774. if (members!=null && members.Count()>0)
  775. {
  776. results = _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationStudentResult>()
  777. .Find(x => members.Select(x => x.id).Contains(x.studentId)&&!string.IsNullOrWhiteSpace(x.evaluationId) && x.evaluationId.Equals(evaluationClient.id));
  778. if (results.Count()==members.Count())
  779. {
  780. return Ok(new { code = 200, setting, results });
  781. }
  782. else {
  783. return Ok(new { code = 200,msg="学生作答信息数量不匹配", setting, results });
  784. }
  785. }
  786. else
  787. {
  788. return Ok(new { code = 200,msg="未分配学生名单,或名单没有学生!", setting , results });
  789. }
  790. }
  791. else
  792. {
  793. return Ok(new { code = 2, msg = "未设置开考信息!" });
  794. }
  795. }
  796. else {
  797. return Ok(new { code = 1, msg = "未找到评测信息!" });
  798. }
  799. }
  800. /// <summary>
  801. /// 设置评测开考信息(本轮名单,计时规则,分配试卷等)
  802. /// </summary>
  803. /// <param name="json"></param>
  804. /// <returns></returns>
  805. [HttpPost("setting-evaluation-round")]
  806. [AuthToken("admin", "teacher", "visitor")]
  807. public IActionResult SettingEvaluationRound(JsonNode json)
  808. {
  809. EvaluationRoundSetting? setting = json.ToObject<EvaluationRoundSetting>();
  810. if (setting!=null)
  811. {
  812. var db = _liteDBFactory.GetLiteDatabase();
  813. var collection = db.GetCollection<EvaluationClient>() ;
  814. //&& x.openCode!.Equals($"{json["openCode"]}")&& x.shortCode!.Equals($"{json["shortCode"]}")
  815. string shortCode = $"{json["shortCode"]}";
  816. string openCode = $"{json["openCode"]}";
  817. EvaluationClient ? evaluationClient = collection.FindOne(x => x.id!.Equals(setting.evaluationId)
  818. && !string.IsNullOrWhiteSpace(x.shortCode) && x.shortCode.Equals(x.shortCode)
  819. && !string.IsNullOrWhiteSpace(x.openCode) && x.openCode.Equals(openCode) );
  820. if (evaluationClient!=null)
  821. {
  822. IEnumerable<EvaluationRoundSetting> settings = _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationRoundSetting>().Find(x => x.evaluationId!.Equals(evaluationClient.id));
  823. if (settings != null && settings.Count() > 0)
  824. {
  825. var datas = settings.ToList();
  826. foreach (EvaluationRoundSetting item in datas)
  827. {
  828. item.activate = 0;
  829. }
  830. _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationRoundSetting>().Upsert(datas);
  831. }
  832. /// 判断是否包含所有分组
  833. bool isAllContained = setting.groupList.All(x => evaluationClient.grouplist.Any(y => y.id == x.id));
  834. if (isAllContained && evaluationClient.grouplist.IsNotEmpty())
  835. {
  836. var ids= setting.groupList.Select(x => x.id).OrderBy(x=>x);
  837. //增加排序,保证id的唯一性
  838. setting.id=ShaHashHelper.GetSHA1($"{evaluationClient.id}_{string.Join("", ids) }");
  839. setting.createTime= DateTimeOffset.Now.ToUnixTimeMilliseconds();
  840. _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationRoundSetting>().Upsert(setting);
  841. _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationClient>().Upsert(evaluationClient);
  842. /// 分配试卷
  843. var(roundStudentPapers, members, results, code, msg)= ManageService.AssignStudentPaper(evaluationClient, setting,_connectionService, _liteDBFactory, _logger);
  844. //var (roundStudentPapers, members, results,code,msg) = AssignStudentPaper(evaluationClient, setting);
  845. return Ok(new { code = code, msg =msg , setting, results, roundStudentPapers });
  846. }
  847. else
  848. {
  849. return Ok(new { code = 3, msg = "开考名单不在当前评测中!" });
  850. }
  851. }
  852. else { return Ok(new { code = 2, msg = "未找到评测,请确认评测ID、提取码、开卷码是否正确!" }); }
  853. }
  854. else
  855. {
  856. return Ok(new { code = 1, msg = "未完善设置信息!" });
  857. }
  858. }
  859. /// <summary>
  860. /// 加载本地的活动列表
  861. /// </summary>
  862. /// <param name="json"></param>
  863. /// <returns></returns>
  864. [HttpPost("list-local-evaluation")]
  865. [AuthToken("admin", "teacher", "visitor")]
  866. public IActionResult ListLocalEvaluation(JsonNode json)
  867. {
  868. if (!string.IsNullOrWhiteSpace($"{_connectionService.serverDevice?.school?.id}"))
  869. {
  870. string code = $"{_connectionService!.serverDevice!.school!.id}";
  871. IEnumerable<EvaluationClient>? evaluationClients = _liteDBFactory.GetLiteDatabase().GetCollection<EvaluationClient>().Find(x => x.ownerId!.Equals(code)).OrderByDescending(x => x.stime);
  872. if (evaluationClients != null)
  873. {
  874. var result = evaluationClients.Select(client =>
  875. {
  876. var properties = client.GetType().GetProperties();
  877. var anonymousObject = new Dictionary<string, object?>();
  878. foreach (var property in properties)
  879. {
  880. if (!property.Name.Equals("openCode"))
  881. {
  882. anonymousObject[property.Name] = property.GetValue(client);
  883. }
  884. }
  885. return anonymousObject;
  886. });
  887. return Ok(new { code = 200, evaluation = result });
  888. }
  889. else
  890. {
  891. return Ok(new { code = 200, evaluation = new List<EvaluationClient>() });
  892. }
  893. }
  894. else
  895. {
  896. return Ok(new { code = 200, evaluation = new List<EvaluationClient>() });
  897. }
  898. }
  899. }
  900. }