ActivityService.cs 57 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116
  1. using Azure.Cosmos;
  2. using Azure.Storage.Blobs.Models;
  3. using Azure.Storage.Sas;
  4. using DocumentFormat.OpenXml.Bibliography;
  5. using DocumentFormat.OpenXml.Drawing;
  6. using DocumentFormat.OpenXml.Drawing.Charts;
  7. using DocumentFormat.OpenXml.Math;
  8. using DocumentFormat.OpenXml.Spreadsheet;
  9. using DocumentFormat.OpenXml.Wordprocessing;
  10. using HTEXLib.COMM.Helpers;
  11. using OpenXmlPowerTools;
  12. using System;
  13. using System.Collections.Generic;
  14. using System.Diagnostics;
  15. using System.Linq;
  16. using System.Net.NetworkInformation;
  17. using System.Text;
  18. using System.Text.Json;
  19. using System.Threading.Tasks;
  20. using TEAMModelOS.SDK.DI;
  21. using TEAMModelOS.SDK.Extension;
  22. using TEAMModelOS.SDK.Models;
  23. using TEAMModelOS.Services;
  24. using static TEAMModelOS.SDK.Models.Cosmos.Student.StudentAnalysis;
  25. using Activity = TEAMModelOS.SDK.Models.Activity;
  26. namespace TEAMModelOS.SDK
  27. {
  28. public static class ActivityService
  29. {
  30. public static async Task<List<ActivityScoreLevel>> ActivityScores (AzureCosmosFactory _azureCosmos,JsonElement _activityId,string teacherId= null) {
  31. List<ActivityScoreLevel> scoreLevels = new List<ActivityScoreLevel>();
  32. Azure.Response responseContest = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Common).ReadItemStreamAsync(_activityId.GetString(), new PartitionKey("Contest"));
  33. if (responseContest.Status == 200)
  34. {
  35. long now = DateTimeOffset.Now.ToUnixTimeMilliseconds();
  36. Contest contest = JsonDocument.Parse(responseContest.Content).RootElement.ToObject<Contest>();
  37. string sql = $"select value c from c ";
  38. if (!string.IsNullOrWhiteSpace(teacherId)) {
  39. sql=$"{sql} where c.id ='{teacherId}'";
  40. }
  41. var result = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Teacher).GetList<ActivityEnroll>(sql, $"ActivityEnroll-{_activityId}");
  42. //List<ActivityTeacherScoreLevel> teacherScoreLevels = new List<ActivityTeacherScoreLevel>();
  43. List<ActivityTeacherScore> teacherScores = new List<ActivityTeacherScore>();
  44. if (result.list.IsNotEmpty())
  45. {
  46. if (contest.sign.type==1)
  47. {
  48. var teams = result.list.Where(x => !string.IsNullOrEmpty(x.contest?.cipher)).GroupBy(z => z.contest.cipher);
  49. foreach (var team in teams)
  50. {
  51. var members = team.ToList();
  52. var leader = members.FindAll(z => z.contest.leader==1)?.FirstOrDefault();
  53. if (leader==null)
  54. {
  55. leader=members.First();
  56. }
  57. double score = leader.upload?.score??-1;
  58. if (leader.upload?.showScore==1)
  59. {
  60. score= leader.upload.maskScore;
  61. }
  62. teacherScores.Add(new ActivityTeacherScore
  63. {
  64. uploadId=leader.upload?.uploadId,
  65. uploadName=!string.IsNullOrWhiteSpace(leader.upload?.name) ? leader.upload?.name : $"{leader.contest.teamName}",
  66. score=score,
  67. tmdid=leader.id,
  68. nickname=leader.tmdName,
  69. name=leader.contest.enrollInfos.Find(z => z.code.Equals("name"))?.val,
  70. picture=leader.tmdPicture,
  71. school=leader.schoolId,
  72. schoolName=leader.schoolName,
  73. schoolPicture=leader.schoolPicture,
  74. cipher=leader.contest.cipher,
  75. teamName=leader.contest.teamName,
  76. type=leader.contest.type,
  77. members=members.Select(z => new IdNameCode { id=z.id, name=z.contest.enrollInfos.Find(x => x.code.Equals("name"))?.val, picture=z.tmdPicture, nickname=z.tmdName }).ToList(),
  78. });
  79. }
  80. }
  81. else
  82. {
  83. ///个人组
  84. foreach (var item in result.list)
  85. {
  86. double score = item.upload?.score??-1;
  87. if (item.upload?.showScore==1)
  88. {
  89. score= item.upload.maskScore;
  90. }
  91. teacherScores.Add(new ActivityTeacherScore
  92. {
  93. uploadId=item.upload?.uploadId,
  94. uploadName=!string.IsNullOrWhiteSpace(item.upload?.name) ? item.upload?.name : $"{item.contest.enrollInfos.Find(z => z.code.Equals("name"))?.val}",
  95. score=score,
  96. tmdid=item.id,
  97. nickname=item.tmdName,
  98. name=item.contest.enrollInfos.Find(z => z.code.Equals("name"))?.val,
  99. picture=item.tmdPicture,
  100. school=item.schoolId,
  101. schoolName=item.schoolName,
  102. schoolPicture=item.schoolPicture,
  103. type=item.contest.type,
  104. });
  105. }
  106. }
  107. if(contest.modules.Contains("score") && contest.score!= null)
  108. {
  109. if (contest.score.showType==1)
  110. {
  111. teacherScores= teacherScores.Where(x => x.score>=0).OrderByDescending(x => x.score).ToList();
  112. if (contest.score.levelType==1)
  113. {
  114. var levels = contest.score.scoreLevels.OrderBy(x => x.order).ToList();
  115. foreach (var level in levels)
  116. {
  117. var datas = teacherScores.Take(level.top);
  118. var scores = datas.ToJsonString().ToObject<List<ActivityTeacherScoreLevel>>();
  119. scores.ForEach(z => { z.scoreLevel=level.lable; });
  120. scoreLevels.Add(new ActivityScoreLevel { lable=level.lable, scores=scores });
  121. teacherScores.RemoveAll(x => scores.Select(v => v.uploadId).Contains(x.uploadId));
  122. }
  123. }
  124. else
  125. {
  126. var levels = contest.score.scoreLevels.OrderByDescending(x => x.max);
  127. foreach (var level in levels)
  128. {
  129. scoreLevels.Add(new ActivityScoreLevel { lable=level.lable });
  130. foreach (var item in teacherScores)
  131. {
  132. string label = string.Empty;
  133. if (level.max==100)
  134. {
  135. if (item.score>level.min && item.score<=level.max)
  136. {
  137. label=level.lable;
  138. }
  139. }
  140. else
  141. {
  142. if (item.score>level.min && item.score<level.max)
  143. {
  144. label=level.lable;
  145. }
  146. }
  147. if (!string.IsNullOrWhiteSpace(label))
  148. {
  149. var score = item.ToJsonString().ToObject<ActivityTeacherScoreLevel>();
  150. score.scoreLevel=label;
  151. score.score=item.score;
  152. var data = scoreLevels.Find(x => x.lable.Equals(label));
  153. if (data!=null)
  154. {
  155. data.scores.Add(score);
  156. }
  157. else
  158. {
  159. scoreLevels.Add(new ActivityScoreLevel { lable=label, scores= new List<ActivityTeacherScoreLevel>() { score } });
  160. }
  161. }
  162. }
  163. }
  164. }
  165. }
  166. else
  167. {
  168. if (contest.score.top>-1)
  169. {
  170. teacherScores= teacherScores.Where(x => x.score>=0).OrderByDescending(x => x.score).Take(contest.score.top).ToList();
  171. }
  172. else
  173. {
  174. teacherScores= teacherScores.Where(x => x.score>=0).ToList();
  175. }
  176. var scores = new List<ActivityTeacherScoreLevel>();
  177. foreach (var item in teacherScores)
  178. {
  179. double score = item.score;
  180. var teacher = item.ToJsonString().ToObject<ActivityTeacherScoreLevel>();
  181. teacher.scoreLevel=$"{score}";
  182. teacher.score=score;
  183. scores.Add(teacher);
  184. }
  185. scoreLevels.Add(new ActivityScoreLevel { lable="", scores=scores });
  186. }
  187. }
  188. }
  189. }
  190. return scoreLevels;
  191. }
  192. public static async Task<(List<ExpertContestTaskDto> expertContestTasks, List<ExpertContestTaskDto> activityEnrollsInvalid, List<ExpertContestTaskDto> expertContestTasksDB)>
  193. AllocationTask(AzureCosmosFactory _azureCosmos, IEnumerable<ExpertPeriodSubjectDto> experts, Contest contest, HashSet<string> periodSubjectKeys, string distribute, int taskCount)
  194. {
  195. //先获取已经分配完成的
  196. List<ExpertContestTaskDto> worksDB = new List<ExpertContestTaskDto>();
  197. IEnumerable<ExpertContestTaskDto> complete = new List<ExpertContestTaskDto>();
  198. string taskSQL = $"select value c from c where c.pk='ActivityExpertTask'";
  199. var result = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Teacher).GetList<ActivityExpertTask>(taskSQL, $"ActivityExpertTask-{contest.id}");
  200. if (result.list.IsNotEmpty())
  201. {
  202. foreach (var item in result.list)
  203. {
  204. worksDB.AddRange(item.contestTasks.Select(z =>
  205. new ExpertContestTaskDto
  206. {
  207. expertName=item.name ,
  208. expertPicture=item.picture,
  209. expertTmdname=item.tmdname,
  210. expertId = item.id,
  211. available = 1,
  212. uploadId = z.uploadId,
  213. name = z.name,
  214. uploadTypes = z.uploadTypes,
  215. count = z.count,
  216. cipher=z.cipher,
  217. type=z.type,
  218. leader=z.leader,
  219. members=z.members,
  220. tmdid=z.tmdid,
  221. score=z.score,
  222. status=z.status,
  223. detailScore=z.detailScore,
  224. activityId=contest.id
  225. }));
  226. }
  227. //处理已经分配完成(taskCount)的作品
  228. complete = worksDB.GroupBy(task => task.uploadId).Where(group => group.Count() >= taskCount).SelectMany(group => group);
  229. }
  230. string enrollSQL = "select value c from c where c.pk='ActivityEnroll'";
  231. //已经满足分配的无需再去处理检查。
  232. if (complete!=null && complete.Count()>0)
  233. {
  234. var uploadIds = complete.Select(z => z.uploadId).ToHashSet();
  235. enrollSQL =$"{enrollSQL} where c.upload.uploadId not in ({string.Join(",", uploadIds.Select(z => $"'{z}'"))})";
  236. }
  237. //检查报名,学段和科目是否匹配
  238. var resultActivityEnroll = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Teacher).GetList<ActivityEnroll>(enrollSQL, $"ActivityEnroll-{contest.id}");
  239. List<ExpertContestTaskDto> expertContestTasks = new List<ExpertContestTaskDto>();
  240. foreach (ActivityEnroll enroll in resultActivityEnroll.list)
  241. {
  242. //如果是队长统一上传
  243. if (contest.upload.captainUpload == 1 && !string.IsNullOrWhiteSpace(enroll.contest?.cipher))
  244. {
  245. var team = resultActivityEnroll.list.FindAll(z => z.contest != null && z.contest.type == 1 && enroll.contest?.type == 1 && z.contest.cipher.Equals(enroll.contest.cipher));
  246. //获取队长信息
  247. var leaders = team.FindAll(z => z.contest.leader == 1);
  248. if (leaders.IsNotEmpty())
  249. {
  250. leaders = new List<ActivityEnroll>() { enroll };
  251. }
  252. List<IdNameCode> members = team.Select(z => new IdNameCode { id = z.id, code = z.schoolId, picture = z.tmdPicture, nickname = z.tmdName, name = z.contest?.enrollInfos?.Find(e => e.code.Equals("name")).val }).ToList();
  253. //队长的报名信息
  254. var leader = leaders?.First();
  255. string name = leader?.contest?.teamName;
  256. var period = leader?.contest?.enrollInfos?.Find(z => z.code.Equals("period"));
  257. var subject = leader?.contest?.enrollInfos?.Find(z => z.code.Equals("subject"));
  258. string uploadId = string.Empty;
  259. if (string.IsNullOrEmpty(name))
  260. {
  261. name = $"{leader?.contest?.enrollInfos?.Find(z => z.code.Equals("name"))?.val}({team.Count})";
  262. }
  263. int count = 0;
  264. int available = -1;
  265. var checkResult = ActivityService.CheckPeriodSubject(distribute, period, subject, periodSubjectKeys);
  266. if (checkResult.available == 0)
  267. {
  268. if (leader?.upload != null && leader.upload.sokrates.IsNotEmpty())
  269. {
  270. count += leader.upload.sokrates.Count;
  271. }
  272. if (leader?.upload != null && leader.upload.files.IsNotEmpty())
  273. {
  274. count += leader.upload.files.Count;
  275. }
  276. if (leader?.upload != null && leader.upload.lessons.IsNotEmpty())
  277. {
  278. count += leader.upload.lessons.Count;
  279. }
  280. if (leader?.upload!=null && leader.upload.complexes.IsNotEmpty())
  281. {
  282. count+=leader.upload.complexes.Count();
  283. }
  284. if (count <= 0)
  285. {
  286. uploadId = enroll.contest.cipher;
  287. available = 5;
  288. }
  289. else
  290. {
  291. uploadId = leader?.upload?.uploadId;
  292. available = checkResult.available;
  293. }
  294. }
  295. else
  296. {
  297. uploadId = enroll.contest.cipher;
  298. available = checkResult.available;
  299. }
  300. ExpertContestTaskDto expertContestTask = expertContestTasks.Find(z => !string.IsNullOrWhiteSpace(z.uploadId) && z.uploadId.Equals(leader.upload));
  301. if (expertContestTask == null)
  302. {
  303. expertContestTasks.Add(new ExpertContestTaskDto
  304. {
  305. uploadId = leader.upload?.uploadId,
  306. name = $"{leader.schoolName}-{name}",
  307. // uploadTypes = new List<string> { leader?.upload.type },
  308. uploadTypes =leader.upload.uploadType,
  309. count = count,
  310. cipher = leader.contest?.cipher,
  311. type = 1,
  312. leader = 1,
  313. tmdid = leader.id,
  314. status = -1,
  315. score = -1,
  316. //detailScore=new List<RuleConfig>()
  317. members = members,
  318. periodSubjectKey = checkResult.periodSubjectKey,
  319. period = period?.val,
  320. subject = subject?.val,
  321. available = available,
  322. activityId=contest.id,
  323. });
  324. }
  325. else
  326. {
  327. expertContestTask.available = available;
  328. expertContestTask.name = name;
  329. expertContestTask.count = count;
  330. expertContestTask.members = members;
  331. expertContestTask.periodSubjectKey = checkResult.periodSubjectKey;
  332. expertContestTask.activityId=contest?.id;
  333. }
  334. }
  335. else
  336. {
  337. if (enroll.upload!= null)
  338. {
  339. //如果不是队长统一上传
  340. //学段
  341. var period = enroll.contest?.enrollInfos?.Find(z => z.code.Equals("period"));
  342. //科目
  343. var subject = enroll.contest?.enrollInfos?.Find(z => z.code.Equals("subject"));
  344. string periodSubjectKey = string.Empty;
  345. string name = enroll.contest?.enrollInfos?.Find(z => z.code.Equals("name"))?.val;
  346. string uploadId = string.Empty;
  347. if (!string.IsNullOrEmpty(name))
  348. {
  349. name = $"({name})";
  350. }
  351. else
  352. {
  353. name = string.Empty;
  354. }
  355. int count = 0;
  356. int available = -1;
  357. var checkResult = ActivityService.CheckPeriodSubject(distribute, period, subject, periodSubjectKeys);
  358. if (checkResult.available == 0)
  359. {
  360. if (enroll.upload != null && enroll.upload.sokrates.IsNotEmpty())
  361. {
  362. count += enroll.upload.sokrates.Count;
  363. }
  364. if (enroll.upload != null && enroll.upload.files.IsNotEmpty())
  365. {
  366. count += enroll.upload.files.Count;
  367. }
  368. if (enroll.upload != null && enroll.upload.lessons.IsNotEmpty())
  369. {
  370. count += enroll.upload.lessons.Count;
  371. }
  372. if (enroll.upload != null && enroll.upload.complexes.IsNotEmpty())
  373. {
  374. count+=enroll.upload.complexes.Count();
  375. }
  376. if (count <= 0)
  377. {
  378. available = 5;
  379. uploadId = enroll.id;
  380. }
  381. else
  382. {
  383. available = checkResult.available;
  384. uploadId = enroll.upload.uploadId;
  385. }
  386. }
  387. else
  388. {
  389. uploadId = enroll.id;
  390. available = checkResult.available;
  391. }
  392. ExpertContestTaskDto expertContestTask = expertContestTasks.Find(z => !string.IsNullOrWhiteSpace(z.uploadId) && z.uploadId.Equals(enroll.upload));
  393. if (expertContestTask == null)
  394. {
  395. expertContestTasks.Add(new ExpertContestTaskDto
  396. {
  397. uploadId = uploadId,
  398. name = $"{enroll.schoolName}-{name}",
  399. //uploadTypes = new List<string> { enroll.upload?.type },
  400. uploadTypes =enroll.upload.uploadType,
  401. count = count,
  402. cipher = enroll.contest?.cipher,
  403. type = 1,
  404. leader = 1,
  405. tmdid = enroll.id,
  406. status = -1,
  407. score = -1,
  408. periodSubjectKey = checkResult.periodSubjectKey,
  409. period = period?.val,
  410. subject = subject?.val,
  411. available = available,
  412. activityId = contest?.id,
  413. });
  414. }
  415. else
  416. {
  417. expertContestTask.available = available;
  418. expertContestTask.name = name;
  419. expertContestTask.count = count;
  420. expertContestTask.periodSubjectKey = checkResult.periodSubjectKey;
  421. expertContestTask.activityId=contest?.id;
  422. }
  423. }
  424. }
  425. }
  426. List<ExpertContestTaskDto> invalids = expertContestTasks.FindAll(z => z.available!=0);
  427. List<ExpertContestTaskDto> works = expertContestTasks.FindAll(z => z.available==0);
  428. var data = AssignWorksToExperts(works, worksDB, experts, taskCount);
  429. if (invalids.IsNotEmpty()) {
  430. invalids.AddRange(data.assignmentsInvalid);
  431. }
  432. List<ExpertContestTaskDto> okTask = new List<ExpertContestTaskDto>();
  433. okTask.AddRange(data.assignmentsAdd);
  434. okTask.AddRange(worksDB);
  435. var groups= okTask.GroupBy(z => z.uploadId).Select(x => new { key = x.Key, list = x.ToList() });
  436. foreach (var item in groups) {
  437. if (item.list.Count()<taskCount) {
  438. ExpertContestTaskDto taskDto= item.list.First().ToJsonString().ToObject<ExpertContestTaskDto>();
  439. taskDto.expertName=null;
  440. taskDto.expertId=null;
  441. taskDto.expertPicture=null;
  442. taskDto.expertTmdname=null;
  443. //,8分配次数不足
  444. taskDto.available=8;
  445. invalids.Add(taskDto);
  446. }
  447. }
  448. return (data.assignmentsAdd, invalids, worksDB);
  449. }
  450. public static (int available, string periodSubjectKey) CheckPeriodSubject(string distribute , EnrollInfo period, EnrollInfo subject, HashSet<string> periodSubjectKeys) {
  451. int available = -1;
  452. string periodSubjectKey= string.Empty;
  453. if (distribute.Equals("period"))
  454. {
  455. if (period != null)
  456. {
  457. periodSubjectKey = $"{period.val}-";
  458. }
  459. else
  460. {
  461. available = 2;
  462. }
  463. }
  464. else if (distribute.Equals("subject"))
  465. {
  466. if (subject != null)
  467. {
  468. periodSubjectKey = $"-{subject.val}";
  469. }
  470. else
  471. {
  472. available = 3;
  473. }
  474. }
  475. else if (distribute.Equals("periodAndSubject"))
  476. {
  477. if (period != null && subject != null)
  478. {
  479. periodSubjectKey = $"{period.val}-{subject.val}";
  480. }
  481. else
  482. {
  483. available = 4;
  484. }
  485. }
  486. else
  487. {
  488. periodSubjectKey = "-";
  489. }
  490. if (!string.IsNullOrWhiteSpace(periodSubjectKey)) {
  491. if (periodSubjectKeys.Contains(periodSubjectKey))
  492. {
  493. available = 0;
  494. }
  495. else
  496. {
  497. //学段学科不匹配
  498. available = 6;
  499. }
  500. }
  501. return (available, periodSubjectKey);
  502. }
  503. /// <summary>
  504. /// 将作品按分配次数,分配给不同的专家
  505. /// </summary>
  506. /// <param name="works"></param>
  507. /// <param name="experts"></param>
  508. /// <param name="N"></param>
  509. /// <returns></returns>
  510. static (List<ExpertContestTaskDto> assignmentsAdd, List<ExpertContestTaskDto> assignmentsInvalid) AssignWorksToExperts(List<ExpertContestTaskDto> works,List<ExpertContestTaskDto> worksDB, IEnumerable<ExpertPeriodSubjectDto> experts, int N)
  511. {
  512. //增加的
  513. var assignmentsAdd = new List<ExpertContestTaskDto>();
  514. //未分配到的
  515. var assignmentsInvalid = new List<ExpertContestTaskDto>();
  516. var random = new Random();
  517. //用于处理数据库和新分配的结果中检查已经分配过的作品和专家
  518. var assignmentsAll = worksDB.ToJsonString().ToObject<List<ExpertContestTaskDto>>();
  519. foreach (var work in works)
  520. {
  521. for (int i = 0; i < N; i++)
  522. {
  523. //分配次数已经Ok,分配结果完善的,不需要再去处理
  524. if (assignmentsAll.Count(z => z.uploadId.Equals(work.uploadId))>=N) {
  525. continue;
  526. }
  527. var availableExperts = experts
  528. .OrderBy(expert => assignmentsAll.Count(a => a.expertId == expert.expertId))
  529. .Where(expert => expert.periodSubjects.Contains(work.periodSubjectKey) && !assignmentsAll.Any(a => a.uploadId == work.uploadId && a.expertId == expert.expertId))
  530. .ToList();
  531. if (availableExperts.Count > 0)
  532. {
  533. var selectedExpert = availableExperts[random.Next(availableExperts.Count)];
  534. var newWork= work.DeepCopy<ExpertContestTaskDto>();
  535. newWork.expertId = selectedExpert.expertId;
  536. newWork.expertName = selectedExpert.expertName;
  537. newWork.expertPicture = selectedExpert.expertPicture;
  538. newWork.expertTmdname = selectedExpert.expertTmdname;
  539. newWork.available = 1;
  540. newWork.turn = i + 1;
  541. assignmentsAdd.Add(newWork);
  542. assignmentsAll.Add(newWork);
  543. }
  544. else
  545. {
  546. var newWork = work.DeepCopy<ExpertContestTaskDto>();
  547. newWork.expertId =null;
  548. newWork.available = 7;
  549. newWork.turn = i + 1;
  550. //作品在第几轮未匹配到
  551. assignmentsInvalid.Add(newWork);
  552. //Console.WriteLine($"No available expert for WorkId {work.uploadId} and subject {work.periodSubjectKey} in attempt {i + 1}.");
  553. break;
  554. }
  555. }
  556. }
  557. return (assignmentsAdd, assignmentsInvalid);
  558. }
  559. /// <summary>
  560. /// 删除活动关联的数据
  561. /// </summary>
  562. /// <param name="_azureCosmos"></param>
  563. /// <param name="activity"></param>
  564. /// <returns></returns>
  565. public static async Task DeleteActivityRelated(AzureCosmosFactory _azureCosmos,Activity activity)
  566. {
  567. //删除模块,Contest, Training ,Research, ReviewRule规则(ReviewRule-disposable"; 存为活动)
  568. await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Common).DeleteItemStreamAsync(activity.id, new PartitionKey("Contest"));
  569. await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Common).DeleteItemStreamAsync(activity.id, new PartitionKey("Training"));
  570. await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Common).DeleteItemStreamAsync(activity.id, new PartitionKey("Research"));
  571. await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Normal).DeleteItemStreamAsync(activity.id, new PartitionKey("ReviewRule-disposable"));
  572. //删除邀请教师 ActivityTeacher,
  573. var resultTeacher = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Teacher)
  574. .GetList<IdCode>("select c.id ,c.code from c where c.pk='ActivityEnroll' and c.", $"ActivityTeacher-{activity.id}");
  575. if (resultTeacher.list.IsNotEmpty())
  576. {
  577. await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Teacher).DeleteItemsStreamAsync(resultTeacher.list.Select(z => z.id).ToList(), $"ActivityTeacher-{activity.id}");
  578. }
  579. //删除报名数据 ActivityEnroll
  580. var resultEnroll = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Teacher)
  581. .GetList<IdCode>("select c.id ,c.code from c where c.pk='ActivityEnroll' and c.", $"ActivityEnroll-{activity.id}");
  582. if (resultEnroll.list.IsNotEmpty())
  583. {
  584. await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Teacher).DeleteItemsStreamAsync(resultEnroll.list.Select(z => z.id).ToList(), $"ActivityEnroll-{activity.id}");
  585. }
  586. //删除专家数据 ActivityExpert
  587. await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Teacher).DeleteItemStreamAsync(activity.id, new PartitionKey("ActivityExpert"));
  588. }
  589. /// <summary>
  590. /// 生成组队口令
  591. /// </summary>
  592. /// <param name="client"></param>
  593. /// <param name="_dingDing"></param>
  594. /// <param name="_option"></param>
  595. /// <param name="_activityId"></param>
  596. /// <returns></returns>
  597. /// <exception cref="Exception"></exception>
  598. public static async Task<string> GenCipher(CosmosClient client, DingDing _dingDing, TEAMModelOS.Models.Option _option, string _activityId)
  599. {
  600. string _num09 = "123456789";
  601. string no = $"{Utils.CreatSaltString(7, _num09)}";
  602. for (int i = 0; i < 10; i++)
  603. {
  604. List<SheetConfig> sheets = new List<SheetConfig>();
  605. bool hasCurrFrom = false;
  606. string cipherSQL = $"select value c.id from c where c.contest!=null and c.activityId='{_activityId}' and c.contest.type=1 and c.contest.cipher='{no}' ";
  607. var cipherResult = await client.GetContainer(Constant.TEAMModelOS, Constant.Teacher).GetList<string>(cipherSQL, $"ActivityEnroll-{_activityId}");
  608. if (cipherResult.list.IsNotEmpty() && cipherResult.list.Count>0) {
  609. hasCurrFrom = true;
  610. }
  611. if (hasCurrFrom)
  612. {
  613. if (i == 9)
  614. {
  615. string msg = $"{_option.Location},ActivityService/GenCipher\n 组队口令生成异常,重复生成次数超过10次";
  616. await _dingDing.SendBotMsg($"OS,{_option.Location},{msg}", GroupNames.醍摩豆服務運維群組);
  617. throw new Exception(msg);
  618. }
  619. else
  620. {
  621. no = $"{Utils.CreatSaltString(7, _num09)}";
  622. }
  623. }
  624. else { break; }
  625. }
  626. return no;
  627. }
  628. public static async Task<IEnumerable<ActivityDto>> AreaActivityList(AzureCosmosFactory _azureCosmos, AzureStorageFactory _azureStorage, JsonElement request, string _areaId, int isCount = 0)
  629. {
  630. string yearSql = string.Empty;
  631. if (isCount==0) {
  632. yearSql = $" and c.year={DateTimeOffset.Now.Year}";
  633. if (request.TryGetProperty("year", out JsonElement _year))
  634. {
  635. yearSql = $" and c.year={_year}";
  636. }
  637. }
  638. string sql = $"select value c from c where c.owner='{_areaId}' {yearSql} ";
  639. var result = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Common).GetList<ActivityDto>(sql, "Activity");
  640. result.list.ForEach(z => {
  641. var (blob_uri, blob_sas) = _azureStorage.GetBlobContainerSAS(z.owner, BlobContainerSasPermissions.Read | BlobContainerSasPermissions.List);
  642. z.sas=blob_sas;
  643. z.url=blob_uri;
  644. });
  645. return result.list.OrderByDescending(z => z.stime);
  646. }
  647. public static async Task<List<ActivityDto>> SchoolActivityList(AzureCosmosFactory _azureCosmos, AzureStorageFactory _azureStorage, JsonElement request, string school, int isCount = 0) {
  648. School schoolbase = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.School).ReadItemAsync<School>(school, new PartitionKey("Base"));
  649. List<ActivityDto> activities = new List<ActivityDto>();
  650. string yearSql = string.Empty;
  651. if (isCount==0)
  652. {
  653. yearSql = $" and c.year={DateTimeOffset.Now.Year}";
  654. if (request.TryGetProperty("year", out JsonElement _year))
  655. {
  656. yearSql = $" and c.year={_year}";
  657. }
  658. }
  659. //获取开放的
  660. {
  661. //完全开放 所有的学校
  662. string sqlOpen = $"select value c from c where c.scope='public' {yearSql} and (c.publish=1 or c.publish=2 ) and( ARRAY_LENGTH(c.invitedSchools)=0 or IS_DEFINED(c.invitedSchools) = false ) ";
  663. var resultOpen = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Common).GetList<ActivityDto>(sqlOpen, "Activity");
  664. activities.AddRange(resultOpen.list);
  665. //部分学校
  666. string sqlSchool = $"select value c from c join s in c.invitedSchools where c.scope='public' {yearSql} and (c.publish=1 or c.publish=2 ) and s.id='{school}' ";
  667. var resultSchool = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Common).GetList<ActivityDto>(sqlSchool, "Activity");
  668. resultSchool.list.ForEach(z =>
  669. {
  670. var confirmedSchool = z.confirmedSchools.Find(z => z.id.Equals(school));
  671. if (confirmedSchool==null)
  672. {
  673. z.confirmedSchools.Add(new ActivityConfirmedSchool
  674. {
  675. id=schoolbase.id,
  676. name=schoolbase.name,
  677. picture=schoolbase.picture,
  678. status=0
  679. });
  680. }
  681. });
  682. activities.AddRange(resultSchool.list);
  683. }
  684. //获取区级下放的
  685. {
  686. if (!string.IsNullOrWhiteSpace(schoolbase.areaId))
  687. {
  688. //区级所有学校
  689. string sqlOpen = $"select value c from c where c.scope='area'{yearSql} and (c.publish=1 or c.publish=2 ) and c.owner='{schoolbase.areaId}' and( ARRAY_LENGTH(c.invitedSchools)=0 or IS_DEFINED(c.invitedSchools) = false) ";
  690. var resultOpen = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Common).GetList<ActivityDto>(sqlOpen, "Activity");
  691. resultOpen.list.ForEach(z =>
  692. {
  693. var confirmedSchool = z.confirmedSchools.Find(z => z.id.Equals(school));
  694. if (confirmedSchool==null)
  695. {
  696. z.confirmedSchools.Add(new ActivityConfirmedSchool
  697. {
  698. id=schoolbase.id,
  699. name=schoolbase.name,
  700. picture=schoolbase.picture,
  701. status=0
  702. });
  703. }
  704. });
  705. activities.AddRange(resultOpen.list);
  706. //区级部分学校
  707. string sqlSchool = $"select value c from c join s in c.invitedSchools where c.scope='area'{yearSql} and (c.publish=1 or c.publish=2 ) and c.owner='{schoolbase.areaId}' and s.id='{school}' ";
  708. var resultSchool = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Common).GetList<ActivityDto>(sqlSchool, "Activity");
  709. resultSchool.list.ForEach(z =>
  710. {
  711. var confirmedSchool = z.confirmedSchools.Find(z => z.id.Equals(school));
  712. if (confirmedSchool==null)
  713. {
  714. z.confirmedSchools.Add(new ActivityConfirmedSchool
  715. {
  716. id=schoolbase.id,
  717. name=schoolbase.name,
  718. picture=schoolbase.picture,
  719. status=0
  720. });
  721. }
  722. });
  723. activities.AddRange(resultSchool.list);
  724. }
  725. }
  726. //获取学校自己的
  727. {
  728. string sqlSchool = $"select value c from c where c.scope='school'{yearSql} and c.owner='{school}' ";
  729. var resultSchool = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Common).GetList<ActivityDto>(sqlSchool, "Activity");
  730. activities.AddRange(resultSchool.list);
  731. }
  732. long now = DateTimeOffset.Now.ToUnixTimeMilliseconds();
  733. activities.ForEach(z =>
  734. {
  735. var (blob_uri, blob_sas) = _azureStorage.GetBlobContainerSAS(z.owner, BlobContainerSasPermissions.Read | BlobContainerSasPermissions.List);
  736. z.sas=blob_sas;
  737. z.url = blob_uri;
  738. if (z.etime<now)
  739. {
  740. z.publish=2;
  741. }
  742. });
  743. return activities;
  744. }
  745. public static async Task<List<TeacherActivityDto>> TeacherActivityList(AzureCosmosFactory _azureCosmos,AzureStorageFactory _azureStorage, JsonElement request, string tmdid, int isCount = 0)
  746. {
  747. List<TeacherActivityDto> activities = new List<TeacherActivityDto>();
  748. HashSet<string> inviteActivityIds = new HashSet<string>();
  749. string yearSql = string.Empty;
  750. if (isCount==0)
  751. {
  752. yearSql = $" and c.year={DateTimeOffset.Now.Year}";
  753. if (request.TryGetProperty("year", out JsonElement _year) && int.TryParse($"{_year}", out int year))
  754. {
  755. yearSql = $" and c.year={year}";
  756. }
  757. }
  758. //先获取邀请制的
  759. string sqlInvite = $"select value c from c join t in c.inviteTeachers where t.id='{tmdid}' and c.pk='ActivityTeacher'";
  760. var result = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Teacher).GetList<ActivityTeacher>(sqlInvite);
  761. inviteActivityIds= result.list.Where(x=>!string.IsNullOrWhiteSpace(x.activityId)).Select(z => z.activityId).ToHashSet();
  762. if (inviteActivityIds.Count>0)
  763. {
  764. string sqlActivity = $"select value c from c where c.id in ({string.Join(",", inviteActivityIds.Select(z => $"'{z}'"))}) {yearSql} ";
  765. var resultActivity = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Common).GetList<TeacherActivityDto>(sqlActivity, "Activity");
  766. if (resultActivity.list.IsNotEmpty())
  767. {
  768. activities.AddRange(resultActivity.list);
  769. }
  770. }
  771. Teacher teacher = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Teacher).ReadItemAsync<Teacher>(tmdid, new PartitionKey("Base"));
  772. string schoolOwnerIn = string.Empty;
  773. string schoolIdIn = string.Empty;
  774. if (teacher.schools.IsNotEmpty())
  775. {
  776. schoolIdIn = $"and i.id in ({string.Join(",", teacher.schools.Select(z => $"'{z.schoolId}'"))})";
  777. schoolOwnerIn= $"and c.owner in ({string.Join(",", teacher.schools.Select(z => $"'{z.schoolId}'"))})";
  778. }
  779. //获取开放的
  780. {
  781. //完全开放 所有的学校
  782. string sqlOpen = $"select value c from c where c.scope='public' and (c.publish=1 or c.publish=2 ) and( ARRAY_LENGTH(c.invitedSchools)=0 or IS_DEFINED(c.invitedSchools) = false ) {yearSql} ";
  783. var resultOpen = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Common).GetList<TeacherActivityDto>(sqlOpen, "Activity");
  784. activities.AddRange(resultOpen.list);
  785. if (!string.IsNullOrWhiteSpace(schoolIdIn))
  786. { //部分学校
  787. string sqlSchool = $"select value c from c join i in c.confirmedSchools where c.scope='public' and c.joinMode='enroll' and (c.publish=1 or c.publish=2 ) and i.status=1 {yearSql} {schoolIdIn} ";
  788. var resultSchool = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Common).GetList<TeacherActivityDto>(sqlSchool, "Activity");
  789. activities.AddRange(resultSchool.list);
  790. }
  791. }
  792. string areaOwnerIn = string.Empty;
  793. var hasAreaSchools = teacher.schools.FindAll(z => !string.IsNullOrWhiteSpace(z.areaId));
  794. if (hasAreaSchools.IsNotEmpty())
  795. {
  796. areaOwnerIn = $"and c.owner in ({string.Join(",", hasAreaSchools.Select(z => $"'{z.areaId}'"))})";
  797. schoolIdIn = $"and i.id in ({string.Join(",", hasAreaSchools.Select(z => $"'{z.schoolId}'"))})";
  798. }
  799. //获取所有区级的
  800. if (!string.IsNullOrWhiteSpace(areaOwnerIn) && !string.IsNullOrEmpty(schoolIdIn))
  801. {
  802. string sqlOpen = $"select value c from c join i in c.confirmedSchools where c.scope='area' and c.joinMode='enroll' and (c.publish=1 or c.publish=2 ) and i.status=1 {yearSql} {areaOwnerIn} {schoolIdIn} ";
  803. var resultOpen = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Common).GetList<TeacherActivityDto>(sqlOpen, "Activity");
  804. activities.AddRange(resultOpen.list);
  805. }
  806. //获取所有学校的
  807. if (!string.IsNullOrWhiteSpace(schoolOwnerIn))
  808. {
  809. string sqlSchool = $"select value c from c where c.scope='school' and c.joinMode='enroll' and (c.publish=1 or c.publish=2 ) {yearSql} {schoolOwnerIn} ";
  810. var resultSchool = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Common).GetList<TeacherActivityDto>(sqlSchool, "Activity");
  811. activities.AddRange(resultSchool.list);
  812. }
  813. long now=DateTimeOffset.Now.ToUnixTimeMilliseconds();
  814. activities.ForEach(z =>
  815. {
  816. var (blob_uri, blob_sas) = _azureStorage.GetBlobContainerSAS(z.owner, BlobContainerSasPermissions.Read);
  817. z.sas=blob_sas;
  818. z.url=blob_uri;
  819. if (z.etime<now)
  820. {
  821. z.publish=2;
  822. }
  823. });
  824. if (activities.IsNotEmpty())
  825. {
  826. string sql = $"select value c from c where c.pk='ActivityEnroll' and contains(c.code,'ActivityEnroll-') and c.id='{tmdid}' and c.activityId in ({string.Join(",", activities.Select(z => $"'{z.id}'"))})";
  827. var resultEnroll = await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Teacher).GetList<ActivityEnroll>(sql);
  828. foreach (var enroll in resultEnroll.list)
  829. {
  830. var activityDto = activities.Find(z => z.id.Equals(enroll.activityId));
  831. if (activityDto!=null)
  832. {
  833. if (enroll.contest!=null && enroll.contest.enrollTime>0)
  834. {
  835. activityDto.contestSign=1;
  836. activityDto.signTime=enroll.contest.enrollTime;
  837. activityDto.contestType=enroll.contest.type;
  838. }
  839. //if (enroll.upload!=null && (enroll.upload.files.IsNotEmpty() || enroll.upload.sokrates.IsNotEmpty() || enroll.upload.lessons.IsNotEmpty()) )
  840. //{
  841. // activityDto.contestUpload=1;
  842. // activityDto.uploadTime=enroll.upload.uploadTime;
  843. // activityDto.uploadType=enroll.upload.type;
  844. //}
  845. if (enroll.upload!=null &&enroll.upload.complexes.IsNotEmpty())
  846. {
  847. activityDto.contestUpload=1;
  848. activityDto.uploadTime=enroll.upload.uploadTime;
  849. activityDto.uploadTypes=enroll.upload.uploadType;
  850. }
  851. }
  852. }
  853. }
  854. return activities;
  855. }
  856. public static async Task<(ReviewRule reviewRule, int invalidCode, string msg)> UpsertReviewRule(ReviewRuleTree reviewRuleTree,Activity activity, Contest contest, AzureCosmosFactory _azureCosmos)
  857. {
  858. var nodes = new List<RuleConfig>();
  859. nodes= TreeToList(reviewRuleTree.trees, nodes);
  860. ReviewRule reviewRule = new ReviewRule() {
  861. id= activity.id,
  862. code="ReviewRule-disposable",
  863. pk="ReviewRule",
  864. name=reviewRuleTree.name,
  865. desc=reviewRuleTree.desc,
  866. owner=activity.owner,
  867. type="disposable",
  868. configs=nodes,
  869. sourceName=activity.name,
  870. taskCount=reviewRuleTree.taskCount,
  871. scoreRule=reviewRuleTree.scoreRule,
  872. distribute=reviewRuleTree.distribute,
  873. scoreDetail=reviewRuleTree.scoreDetail,
  874. };
  875. //代码顺序不能动
  876. var checkReult = CheckReviewRule(reviewRule, contest);
  877. if (checkReult.invalidCode==200) {
  878. await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Normal).UpsertItemAsync(reviewRule, new Azure.Cosmos.PartitionKey(reviewRule.code));
  879. if (reviewRuleTree.upsertAsTemplate==1)
  880. {
  881. reviewRule.code="ReviewRule-template";
  882. reviewRule.type="template";
  883. reviewRule.taskCount = reviewRuleTree.taskCount;
  884. reviewRule.scoreRule = reviewRuleTree.scoreRule;
  885. reviewRule.distribute = reviewRuleTree.distribute;
  886. reviewRule.scoreDetail = reviewRuleTree.scoreDetail;
  887. await _azureCosmos.GetCosmosClient().GetContainer(Constant.TEAMModelOS, Constant.Normal).UpsertItemAsync(reviewRule, new Azure.Cosmos.PartitionKey(reviewRule.code));
  888. }
  889. }
  890. return (reviewRule,checkReult.invalidCode,checkReult.msg);
  891. }
  892. public static (int invalidCode, string msg) CheckReviewRule(ReviewRule reviewRule,Contest contest)
  893. {
  894. int invalidCode = -1;
  895. string msg = "";
  896. if (reviewRule.taskCount<=0) {
  897. invalidCode=28;
  898. msg="作品分配次数至少一次!";
  899. }
  900. if (reviewRule.taskCount==1 )
  901. {
  902. if (reviewRule.scoreRule.Equals("only"))
  903. {
  904. invalidCode=200;
  905. }
  906. else
  907. {
  908. invalidCode=21;//分配次数一次,必须匹配only
  909. msg="分配次数1次,必须匹配【默认统分】规则";
  910. }
  911. }
  912. if (reviewRule.taskCount>=2 )
  913. {
  914. if ((reviewRule.scoreRule.Equals("avg")||reviewRule.scoreRule.Equals("top"))) {
  915. invalidCode=200;
  916. }
  917. else
  918. {
  919. invalidCode=22;//分配次数2次,必须匹配avg,top
  920. msg="分配次数2次,必须匹配【按平均分】,【按最高分】";
  921. }
  922. }
  923. if (reviewRule.taskCount>=3 )
  924. {if ((reviewRule.scoreRule.Equals("avg")||reviewRule.scoreRule.Equals("top")||reviewRule.scoreRule.Equals("rmLowAvg")||reviewRule.scoreRule.Equals("rmTopAvg")))
  925. {
  926. invalidCode=200;
  927. }
  928. else
  929. {
  930. invalidCode=23;//分配次数2次,必须匹配avg,top,rmLowAvg,rmTopAvg
  931. msg="分配次数3次,必须匹配必须匹配【按平均分】,【按最高分】,【去掉最低分的平均分】,【去掉最高分的平均分】";
  932. }
  933. }
  934. if (reviewRule.taskCount>=4 )
  935. {
  936. if ((reviewRule.scoreRule.Equals("avg")||reviewRule.scoreRule.Equals("top")||reviewRule.scoreRule.Equals("rmLowAvg")||reviewRule.scoreRule.Equals("rmTopAvg")||reviewRule.scoreRule.Equals("rmLowTopAvg")))
  937. {
  938. invalidCode=200;
  939. }
  940. else
  941. {
  942. invalidCode=24;//分配次数2次,必须匹配avg,top,rmLowAvg,rmTopAvg,rmLowTopAvg
  943. msg="分配次数4次,必须匹配必须匹配【按平均分】,【按最高分】,【去掉最低分的平均分】,【去掉最高分的平均分】,【去掉最高分和最低分的平均分】";
  944. }
  945. }
  946. if (invalidCode==200) {
  947. if (!string.IsNullOrWhiteSpace(reviewRule.distribute))
  948. {
  949. if (reviewRule.distribute.Equals("period"))
  950. {
  951. var period = contest.sign?.fields?.Find(z => z.field.Equals("period"));
  952. if (period!= null)
  953. {
  954. invalidCode = 200;
  955. }
  956. else
  957. {
  958. invalidCode=25;
  959. msg="作品分配匹配规则为【学段】,但报名填写表单未配置。";
  960. }
  961. }
  962. else if (reviewRule.distribute.Equals("subject"))
  963. {
  964. var subject = contest.sign?.fields?.Find(z => z.field.Equals("subject"));
  965. if (subject!= null)
  966. {
  967. invalidCode = 200;
  968. }
  969. else
  970. {
  971. invalidCode=26;
  972. msg="作品分配匹配规则为【学科】,但报名填写表单未配置。";
  973. }
  974. }
  975. else if (reviewRule.distribute.Equals("periodAndSubject"))
  976. {
  977. var period = contest.sign?.fields?.Find(z => z.field.Equals("period"));
  978. var subject = contest.sign?.fields?.Find(z => z.field.Equals("subject"));
  979. if (subject!= null && period!= null)
  980. {
  981. invalidCode = 200;
  982. }
  983. else
  984. {
  985. invalidCode=27;
  986. msg="作品分配匹配规则为【学段,学科】,但报名填写表单未配置。";
  987. }
  988. }
  989. else if (reviewRule.distribute.Equals("none"))
  990. {
  991. invalidCode=200;
  992. }
  993. else
  994. {
  995. invalidCode=30;
  996. msg=$"作品分配匹配规则未识别【{reviewRule.distribute}】。";
  997. }
  998. }
  999. else {
  1000. invalidCode=29;
  1001. msg="作品分配匹配规则不能为空。";
  1002. }
  1003. }
  1004. return (invalidCode,msg);
  1005. }
  1006. public static List<RuleConfig> TreeToList(List<RuleConfigTree> trees, List<RuleConfig> nodes) {
  1007. trees = trees.OrderBy(x => x.order).ToList();
  1008. List<RuleConfig> list = new List<RuleConfig>();
  1009. trees.ForEach(x => {
  1010. List<string> cids = new List<string>();
  1011. if (x.children.IsNotEmpty())
  1012. {
  1013. x.children.ForEach(y => cids.Add(y.id));
  1014. }
  1015. var node = new RuleConfig
  1016. {
  1017. id = x.id,
  1018. pid = x.pid,
  1019. cids= cids,
  1020. label = x.label,
  1021. desc = x.desc,
  1022. score = x.score,
  1023. order = x.order,
  1024. };
  1025. list.Add(node);
  1026. });
  1027. nodes.AddRange(list);
  1028. foreach (RuleConfigTree tree in trees)
  1029. {
  1030. if (null != tree.children && tree.children.Count > 0)
  1031. {
  1032. TreeToList(tree.children, nodes);
  1033. }
  1034. }
  1035. return nodes;
  1036. }
  1037. public static List<RuleConfigTree> ListToTree(List<RuleConfig> noes)
  1038. {
  1039. List<RuleConfigTree> list = noes.ToJsonString().ToObject<List<RuleConfigTree>>();
  1040. var res = from r in list group r by r.id into g select g;
  1041. Dictionary<string, RuleConfigTree> blockDict = new Dictionary<string, RuleConfigTree>();
  1042. foreach (var s in res)
  1043. {
  1044. blockDict.TryAdd(s.First().id, s.First());
  1045. }
  1046. return GetChild(list, blockDict);
  1047. }
  1048. private static List<RuleConfigTree> GetChild(List<RuleConfigTree> list, Dictionary<string, RuleConfigTree> dict)
  1049. {
  1050. List<RuleConfigTree> trees = new List<RuleConfigTree>();
  1051. trees = trees.OrderBy(x => x.order).ToList();
  1052. foreach (RuleConfigTree node in list)
  1053. {
  1054. bool flag = dict.TryGetValue(node.pid, out RuleConfigTree syllabus);
  1055. if (flag && syllabus != null)
  1056. {
  1057. syllabus.children.Add(node);
  1058. }
  1059. else
  1060. {
  1061. trees.Add(node);
  1062. }
  1063. }
  1064. return trees;
  1065. }
  1066. }
  1067. }