ActivityService.cs 51 KB

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