KnowledgeController.cs 71 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533
  1. using DocumentFormat.OpenXml.Wordprocessing;
  2. using DotNetCore.CAP;
  3. using Hotline.Api.Filter;
  4. using Hotline.Application.Bulletin;
  5. using Hotline.Application.ExportExcel;
  6. using Hotline.Application.ExportWord;
  7. using Hotline.Application.FlowEngine;
  8. using Hotline.Application.Knowledge;
  9. using Hotline.Application.Systems;
  10. using Hotline.Application.Tools;
  11. using Hotline.File;
  12. using Hotline.FlowEngine.WorkflowModules;
  13. using Hotline.KnowledgeBase;
  14. using Hotline.KnowledgeBase.Notifies;
  15. using Hotline.Permissions;
  16. using Hotline.Repository.SqlSugar.Extensions;
  17. using Hotline.Repository.SqlSugar.Knowledge;
  18. using Hotline.Repository.SqlSugar.Ts;
  19. using Hotline.Settings;
  20. using Hotline.Settings.Hotspots;
  21. using Hotline.Share.Dtos;
  22. using Hotline.Share.Dtos.File;
  23. using Hotline.Share.Dtos.FlowEngine;
  24. using Hotline.Share.Dtos.Knowledge;
  25. using Hotline.Share.Dtos.Order;
  26. using Hotline.Share.Enums.Article;
  27. using Hotline.Share.Enums.KnowledgeBase;
  28. using Hotline.Share.Mq;
  29. using Hotline.Share.Tools;
  30. using Hotline.Users;
  31. using J2N.Text;
  32. using Mapster;
  33. using MapsterMapper;
  34. using MediatR;
  35. using Microsoft.AspNetCore.Mvc;
  36. using Microsoft.EntityFrameworkCore.Metadata.Internal;
  37. using Org.BouncyCastle.Utilities.IO;
  38. using SqlSugar;
  39. using System.Threading;
  40. using XF.Domain.Authentications;
  41. using XF.Domain.Exceptions;
  42. using XF.Domain.Repository;
  43. using XF.Utility.EnumExtensions;
  44. using static Microsoft.EntityFrameworkCore.DbLoggerCategory;
  45. namespace Hotline.Api.Controllers
  46. {
  47. public class KnowledgeController : BaseController
  48. {
  49. #region 注入
  50. private readonly IExportApplication _exportApplication;
  51. private readonly IRepository<KnowledgeHotWord> _knowledgeHotWordRepository;
  52. private readonly IRepository<KnowledgeWord> _knowledgeWordRepository;
  53. private readonly IWordHelperService _wordHelperService;
  54. private readonly BaseDataApplication _baseDataApplication;
  55. private readonly IKnowledgeRepository _knowledgeRepository;
  56. private readonly ISessionContext _sessionContext;
  57. private readonly IKnowledgeDomainService _knowledgeDomainService;
  58. private readonly IMapper _mapper;
  59. private readonly IKnowApplication _knowApplication;
  60. private readonly IMediator _mediator;
  61. private readonly IWorkflowApplication _workflowApplication;
  62. private readonly IKnowledgeWorkFlowRepository _knowledgeWorkFlowRepository;
  63. private readonly IRepository<User> _userRepository;
  64. private readonly IRepository<KnowledgeType> _knowledgeTypeRepository;
  65. private readonly IRepository<Hotspot> _hotspotTypeRepository;
  66. private readonly IRepositoryTextSearch<KnowledgeTs> _repositoryts;
  67. private readonly IRepository<KnowledgeWord> _knowledgeWrodRepository;
  68. private readonly IRepository<KnowledgeQuestions> _knowledgeQuestionsRepository;
  69. private readonly IRepository<KnowledgeCorrection> _knowledgeCorrectionRepository;
  70. private readonly IRepository<KnowledgeCollect> _knowledgeCollectRepository;
  71. private readonly ISystemDomainService _systemDomainService;
  72. private readonly IRepository<KnowledgeComment> _knowledgeCommentRepository;
  73. private readonly ISystemOrganizeRepository _systemOrganizeRepository;
  74. private readonly IFileRepository _fileRepository;
  75. private readonly ICapPublisher _capPublisher;
  76. private readonly IRepository<KnowledgeRelationType> _knowledgeRelationTypeRepository;
  77. private readonly IBulletinApplication _bulletinApplication;
  78. private readonly IRepository<KnowledgeCollectGroup> _knowledgeCollectGroupRepository;
  79. private readonly IRepository<KnowledgePv> _knowledgePvepository;
  80. public KnowledgeController(
  81. IKnowledgeRepository knowledgeRepository,
  82. ISessionContext sessionContext,
  83. IKnowledgeDomainService knowledgeDomainService,
  84. IMapper mapper,
  85. IKnowApplication knowApplication,
  86. IMediator mediator,
  87. IWorkflowApplication workflowApplication,
  88. IKnowledgeWorkFlowRepository knowledgeWorkFlowRepository,
  89. IRepository<User> userRepository,
  90. IRepository<KnowledgeType> knowledgeTypeRepository,
  91. IRepository<Hotspot> hotspotTypeRepository,
  92. IRepositoryTextSearch<KnowledgeTs> repositoryts,
  93. IRepository<KnowledgeWord> knowledgeWrodRepository,
  94. IRepository<KnowledgeQuestions> knowledgeQuestionsRepository,
  95. IRepository<KnowledgeCorrection> knowledgeCorrectionRepository,
  96. IRepository<KnowledgeCollect> knowledgeCollectRepository,
  97. ISystemDomainService systemDomainService,
  98. IRepository<KnowledgeComment> knowledgeCommentRepository,
  99. ISystemOrganizeRepository systemOrganizeRepository,
  100. IFileRepository fileRepository,
  101. ICapPublisher capPublisher,
  102. IRepository<KnowledgeRelationType> knowledgeRelationTypeRepository,
  103. IBulletinApplication bulletinApplication,
  104. IRepository<KnowledgeCollectGroup> knowledgeCollectGroupRepository,
  105. IExportApplication exportApplication,
  106. BaseDataApplication baseDataApplication,
  107. IWordHelperService wordHelperService,
  108. IRepository<KnowledgePv> knowledgePvepository,
  109. IRepository<KnowledgeWord> knowledgeWordRepository,
  110. IRepository<KnowledgeHotWord> knowledgeWordHotRepository,
  111. IRepository<KnowledgeHotWord> knowledgeHotWordRepository)
  112. {
  113. _knowledgeRepository = knowledgeRepository;
  114. _sessionContext = sessionContext;
  115. _knowledgeDomainService = knowledgeDomainService;
  116. _mapper = mapper;
  117. _knowApplication = knowApplication;
  118. _mediator = mediator;
  119. _workflowApplication = workflowApplication;
  120. _knowledgeWorkFlowRepository = knowledgeWorkFlowRepository;
  121. _userRepository = userRepository;
  122. _knowledgeTypeRepository = knowledgeTypeRepository;
  123. _hotspotTypeRepository = hotspotTypeRepository;
  124. _repositoryts = repositoryts;
  125. _knowledgeWrodRepository = knowledgeWrodRepository;
  126. _knowledgeQuestionsRepository = knowledgeQuestionsRepository;
  127. _knowledgeCorrectionRepository = knowledgeCorrectionRepository;
  128. _knowledgeCollectRepository = knowledgeCollectRepository;
  129. _systemDomainService = systemDomainService;
  130. _knowledgeCommentRepository = knowledgeCommentRepository;
  131. _systemOrganizeRepository = systemOrganizeRepository;
  132. _fileRepository = fileRepository;
  133. _capPublisher = capPublisher;
  134. _knowledgeRelationTypeRepository = knowledgeRelationTypeRepository;
  135. _bulletinApplication = bulletinApplication;
  136. _knowledgeCollectGroupRepository = knowledgeCollectGroupRepository;
  137. _exportApplication = exportApplication;
  138. _baseDataApplication = baseDataApplication;
  139. _wordHelperService = wordHelperService;
  140. _knowledgePvepository = knowledgePvepository;
  141. _knowledgeWordRepository = knowledgeWordRepository;
  142. _knowledgeHotWordRepository = knowledgeHotWordRepository;
  143. }
  144. #endregion
  145. #region 知识管理
  146. /// <summary>
  147. /// 知识库-新增
  148. /// </summary>
  149. /// <param name="dto"></param>
  150. /// <returns></returns>
  151. [Permission(EPermission.AddKnowledge)]
  152. [HttpPost("add")]
  153. [LogFilter("知识新增")]
  154. public async Task<string> AddKnowledge([FromBody] AddStartFlowDto dto)
  155. {
  156. //var addDto = _mapper.Map<AddKnowledgeDto>(dto.Data);
  157. var kn = _mapper.Map<Knowledge>(dto.Data);
  158. kn.SourceOrganizeId = _sessionContext.RequiredOrgId;
  159. var any = await _knowledgeRepository.Queryable().Where(x => x.Status == EKnowledgeStatus.OnShelf && x.Title == kn.Title).AnyAsync();
  160. if (any) throw UserFriendlyException.SameMessage("当前知识标题存在重复标题!");
  161. //Code为空,从新生成Code
  162. if (string.IsNullOrEmpty(kn.Code))
  163. kn.Code = Convert.ToInt64((DateTime.Now - new DateTime(1970, 1, 1, 0, 0, 0, 0)).TotalSeconds).ToString();
  164. kn.Status = EKnowledgeStatus.Drafts;
  165. kn.InitId();
  166. if (dto.Data.Files.Any()) kn.FileJson = await _fileRepository.AddFileAsync(dto.Data.Files, kn.Id, "", HttpContext.RequestAborted);
  167. await _knowledgeRepository.AddAsync(kn, HttpContext.RequestAborted);
  168. if (dto.Data.KnowledgeType.Any())
  169. {
  170. List<KnowledgeRelationType> types = _mapper.Map<List<KnowledgeRelationType>>(dto.Data.KnowledgeType);
  171. types.ForEach(x => x.KnowledgeId = kn.Id);
  172. await _knowledgeRelationTypeRepository.AddRangeAsync(types, HttpContext.RequestAborted);
  173. }
  174. if (dto.Workflow != null && !string.IsNullOrEmpty(kn.Id))
  175. {
  176. var startDto = _mapper.Map<StartWorkflowDto>(dto.Workflow);
  177. startDto.DefinitionModuleCode = WorkflowModuleConsts.KnowledgeAdd;
  178. startDto.Title = "知识库新增";
  179. //await _workflowApplication.StartWorkflowAsync(startDto, _sessionContext, kn.Id, cancellationToken: HttpContext.RequestAborted);
  180. await StartFlow(kn.Id, WorkflowModuleConsts.KnowledgeAdd, EKnowledgeApplyType.Add, startDto);
  181. //var knowledge = await _knowledgeRepository.GetAsync(kn.Id);
  182. //knowledge.Status = EKnowledgeStatus.Auditing;
  183. //await _knowledgeRepository.UpdateAsync(knowledge, HttpContext.RequestAborted);
  184. }
  185. return kn.Id;
  186. }
  187. /// <summary>
  188. /// 知识库-知识下架
  189. /// </summary>
  190. /// <param name="Id"></param>
  191. /// <returns></returns>
  192. [Permission(EPermission.KnowledgeOffShelf)]
  193. [HttpPut("offshelf")]
  194. [LogFilter("知识下架")]
  195. public async Task KnowledgeOffShelf(string Id)
  196. {
  197. var know = await _knowledgeRepository.GetAsync(Id, HttpContext.RequestAborted);
  198. if (know != null && know.Status == EKnowledgeStatus.OnShelf)
  199. {
  200. know.Status = EKnowledgeStatus.OffShelf;
  201. know.OnShelfTime = null;
  202. know.OffShelfTime = DateTime.Now;
  203. await _knowledgeRepository.UpdateAsync(know, HttpContext.RequestAborted);
  204. var pushKnowledge = _mapper.Map<KnowledgeSendDto>(know);
  205. pushKnowledge.CategoryCode = "01";
  206. pushKnowledge.CategoryName = "公共服务";
  207. //推省上
  208. await _capPublisher.PublishAsync(EventNames.HotlineKnowledgeRemove, pushKnowledge, cancellationToken: HttpContext.RequestAborted);
  209. }
  210. else
  211. throw UserFriendlyException.SameMessage("知识下架失败");
  212. }
  213. /// <summary>
  214. /// 知识库-知识上架
  215. /// </summary>
  216. /// <param name="Id"></param>
  217. /// <returns></returns>
  218. [Permission(EPermission.KnowledgeOnTheShelf)]
  219. [HttpPut("ontheshelf")]
  220. [LogFilter("知识上架")]
  221. public async Task KnowledgeOnTheShelf(string Id)
  222. {
  223. var know = await _knowledgeRepository.GetAsync(Id, HttpContext.RequestAborted);
  224. if (know != null && know.Status == EKnowledgeStatus.OffShelf)
  225. {
  226. var any = await _knowledgeRepository.Queryable().Where(x => x.Status == EKnowledgeStatus.OnShelf && x.Title == know.Title && x.Id != know.Id).AnyAsync();
  227. if (any) throw UserFriendlyException.SameMessage("当前知识标题存在重复标题!");
  228. if (know.ExpiredTime < DateTime.Now) throw UserFriendlyException.SameMessage("知识已过期不能上架!");
  229. know.Status = EKnowledgeStatus.OnShelf;
  230. know.OnShelfTime = DateTime.Now;
  231. know.OffShelfTime = null;
  232. await _knowledgeRepository.UpdateAsync(know, HttpContext.RequestAborted);
  233. }
  234. else
  235. throw UserFriendlyException.SameMessage("知识上架失败");
  236. }
  237. /// <summary>
  238. /// 知识库-标题
  239. /// </summary>
  240. /// <param name="title"></param>
  241. /// <returns></returns>
  242. [HttpGet("title")]
  243. public async Task<bool> KnowledgeTitle([FromQuery] KnowledgeTitleDto dto)
  244. {
  245. var count = await _knowledgeRepository.Queryable()
  246. .WhereIF(!string.IsNullOrEmpty(dto.Id), x => x.Id != dto.Id)
  247. .Where(x => x.Title == dto.Title && x.Status == EKnowledgeStatus.OnShelf).CountAsync();
  248. return count > 0;
  249. }
  250. /// <summary>
  251. /// 知识库-修改
  252. /// </summary>
  253. /// <param name="dto"></param>
  254. /// <returns></returns>
  255. [HttpPut("update")]
  256. [LogFilter("知识修改")]
  257. public async Task UpdateKnowledge([FromBody] UpdateStartFlowDto dto)
  258. {
  259. var update = _mapper.Map<UpdateKnowledgeDto>(dto.Data);
  260. var knowledge = await _knowledgeRepository.GetAsync(update.Id);
  261. if (knowledge == null)
  262. throw UserFriendlyException.SameMessage("知识库数据错误");
  263. if ((knowledge.Status == EKnowledgeStatus.OnShelf || knowledge.Status == EKnowledgeStatus.Auditing) && (knowledge.ExpiredTime.HasValue && knowledge.ExpiredTime.Value > DateTime.Now))
  264. throw UserFriendlyException.SameMessage("知识库数据不可修改");
  265. var any = await _knowledgeRepository.Queryable().Where(x => x.Status == EKnowledgeStatus.OnShelf && x.Title == update.Title && x.Id != update.Id).AnyAsync();
  266. if (any) throw UserFriendlyException.SameMessage("当前知识标题存在重复标题!");
  267. _mapper.Map(dto.Data, knowledge);
  268. //if (update.Tags.Any()) await _repositoryts.UpdateVectorAsync(update.Id, update.Tags, HttpContext.RequestAborted);
  269. if (dto.Data.Files.Any())
  270. knowledge.FileJson = await _fileRepository.AddFileAsync(dto.Data.Files, knowledge.Id, "", HttpContext.RequestAborted);
  271. else
  272. knowledge.FileJson = new List<Share.Dtos.File.FileJson>();
  273. if (dto.Workflow != null) knowledge.Renewaln = knowledge.Status != EKnowledgeStatus.Drafts;
  274. await _knowledgeRepository.UpdateAsync(knowledge, HttpContext.RequestAborted);
  275. if (dto.Data.KnowledgeType.Any())
  276. {
  277. var anyRelationTypes = await _knowledgeRelationTypeRepository.Queryable().Where(x => x.KnowledgeId == knowledge.Id).ToListAsync();
  278. if (anyRelationTypes.Any())
  279. await _knowledgeRelationTypeRepository.RemoveRangeAsync(anyRelationTypes);
  280. List<KnowledgeRelationType> types = _mapper.Map<List<KnowledgeRelationType>>(dto.Data.KnowledgeType);
  281. types.ForEach(x => x.KnowledgeId = update.Id);
  282. await _knowledgeRelationTypeRepository.AddRangeAsync(types, HttpContext.RequestAborted);
  283. }
  284. if (dto.Workflow != null)
  285. {
  286. if (knowledge.Status == EKnowledgeStatus.Drafts)
  287. {
  288. var startDto = _mapper.Map<StartWorkflowDto>(dto.Workflow);
  289. startDto.DefinitionModuleCode = WorkflowModuleConsts.KnowledgeAdd;
  290. startDto.Title = "知识库新增";
  291. //await _workflowApplication.StartWorkflowAsync(startDto, _sessionContext, knowledge.Id, cancellationToken: HttpContext.RequestAborted);
  292. await StartFlow(knowledge.Id, WorkflowModuleConsts.KnowledgeAdd, EKnowledgeApplyType.Add, startDto);
  293. }
  294. else
  295. {
  296. var startDto = _mapper.Map<StartWorkflowDto>(dto.Workflow);
  297. startDto.DefinitionModuleCode = WorkflowModuleConsts.KnowledgeUpdate;
  298. startDto.Title = "知识库修改";
  299. //await _workflowApplication.StartWorkflowAsync(startDto, _sessionContext, knowledge.Id, cancellationToken: HttpContext.RequestAborted);
  300. await StartFlow(knowledge.Id, WorkflowModuleConsts.KnowledgeUpdate, EKnowledgeApplyType.Update, startDto);
  301. }
  302. }
  303. }
  304. /// <summary>
  305. /// 批量审核
  306. /// </summary>
  307. /// <returns></returns>
  308. [HttpPost("batch_audit")]
  309. public async Task KnowledgeBatchAuditAsync()
  310. {
  311. // TODO: qcy 批量审核
  312. }
  313. /// <summary>
  314. /// 根据知识标题自动分词并保存到关键词表中
  315. /// </summary>
  316. /// <param name="title"></param>
  317. /// <returns></returns>
  318. [HttpGet("participle")]
  319. public async Task<IList<KnowledgeWordOutDto>> KnowledgeKeyWord([FromQuery]string title)
  320. {
  321. return await _knowApplication.TitleParticiple(title);
  322. }
  323. /// <summary>
  324. /// 删除知识
  325. /// </summary>
  326. /// <param name="id"></param>
  327. /// <returns></returns>
  328. [HttpDelete]
  329. [LogFilter("删除知识")]
  330. public async Task Remove([FromBody] DeleteStartFlowDto dto)
  331. {
  332. var delete = _mapper.Map<KnowledgeDto>(dto.Data);
  333. var knowledge = await _knowledgeRepository.GetAsync(delete.Id, HttpContext.RequestAborted);
  334. if (knowledge == null)
  335. throw UserFriendlyException.SameMessage("无效知识库数据");
  336. if (knowledge.Status == EKnowledgeStatus.OnShelf || knowledge.Status == EKnowledgeStatus.Auditing)
  337. throw UserFriendlyException.SameMessage("知识库数据不可删除");
  338. if (knowledge.Status == EKnowledgeStatus.Drafts)
  339. {
  340. await _knowledgeRepository.RemoveAsync(knowledge, false, HttpContext.RequestAborted);
  341. }
  342. else
  343. {
  344. var startDto = _mapper.Map<StartWorkflowDto>(dto.Workflow);
  345. startDto.DefinitionModuleCode = WorkflowModuleConsts.KnowledgeDelete;
  346. startDto.Title = "知识库删除";
  347. //await _workflowApplication.StartWorkflowAsync(startDto, _sessionContext, knowledge.Id, cancellationToken: HttpContext.RequestAborted);
  348. await StartFlow(delete.Id, WorkflowModuleConsts.KnowledgeDelete, EKnowledgeApplyType.Delete, startDto);
  349. }
  350. }
  351. /// <summary>
  352. /// 增加搜索量
  353. /// </summary>
  354. /// <param name="dto"></param>
  355. /// <returns></returns>
  356. [HttpPost("search_num")]
  357. [LogFilter("知识搜索")]
  358. public async Task SearchNum([FromBody] KnowledgeSearchNumDto dto)
  359. {
  360. var knowledge = await _knowledgeRepository.GetAsync(dto.Id, HttpContext.RequestAborted);
  361. if (knowledge == null)
  362. throw UserFriendlyException.SameMessage("无效知识库数据");
  363. knowledge.SearchNum++;
  364. await _knowledgeRepository.UpdateAsync(knowledge, HttpContext.RequestAborted);
  365. }
  366. /// <summary>
  367. /// 搜索量列表
  368. /// </summary>
  369. /// <param name="dto"></param>
  370. /// <returns></returns>
  371. [HttpGet("search_num/list")]
  372. public async Task<PagedDto<KnowledgeDto>> SearchNumList([FromQuery] KnowledgeCollectListDto dto)
  373. {
  374. var query = _knowledgeRepository.Queryable(false, true, false)
  375. .Where(x => x.Status == EKnowledgeStatus.OnShelf)
  376. .Where(m => m.IsDeleted == false)
  377. .Where(x => (x.ExpiredTime != null && x.ExpiredTime >= DateTime.Now) || x.ExpiredTime == null)
  378. .OrderByDescending(x => x.SearchNum);
  379. if (_sessionContext.OrgIsCenter == false)
  380. {
  381. query = query.Where(m => m.KnowledgeType.Any(a => a.IsDeleted == false && a.KnowledgeType.KnowledgeTypeOrgs
  382. .Any(k => k.IsDeleted == false && k.OrgId.StartsWith(_sessionContext.OrgId))));
  383. query = query.Where(m => m.Attribution == "部门知识库");
  384. }
  385. var (total, items) = await query.ToPagedListAsync(dto.PageIndex, dto.PageSize, HttpContext.RequestAborted);
  386. return new PagedDto<KnowledgeDto>(total, _mapper.Map<IReadOnlyList<KnowledgeDto>>(items));
  387. }
  388. /// <summary>
  389. /// 知识库-知识修改-查询详情
  390. /// </summary>
  391. /// <param name="Id"></param>
  392. /// <returns></returns>
  393. [HttpGet("updateinfo/{Id}")]
  394. public async Task<KnowledgeInfoDto> KnowledgeUpdateInfo(string Id)
  395. {
  396. //var know = await _knowledgeRepository.GetAsync(Id, HttpContext.RequestAborted);
  397. var know = await _knowledgeDomainService.KnowledgeInfo(Id, HttpContext.RequestAborted);
  398. if (know is null)
  399. throw UserFriendlyException.SameMessage("知识查询失败!");
  400. var knowledgeInfoDto = _mapper.Map<KnowledgeInfoDto>(know);
  401. if (knowledgeInfoDto != null && !string.IsNullOrEmpty(knowledgeInfoDto.Content))
  402. knowledgeInfoDto.Content = _bulletinApplication.GetSiteUrls(knowledgeInfoDto.Content);
  403. //分类
  404. //var type = await _knowledgeTypeRepository.GetAsync(know.KnowledgeTypeId, HttpContext.RequestAborted);
  405. //if (type != null)
  406. // knowledgeInfoDto.KnowledgeTypeName = type.SpliceName;
  407. //热点
  408. var hot = await _hotspotTypeRepository.GetAsync(know.HotspotId, HttpContext.RequestAborted);
  409. if (hot != null)
  410. knowledgeInfoDto.HotspotName = hot.HotSpotFullName;
  411. //收藏
  412. var collect = await _knowledgeCollectRepository.GetAsync(x => x.KnowledgeId == Id && x.CreatorId == _sessionContext.UserId);
  413. if (collect != null)
  414. knowledgeInfoDto.Collect = _mapper.Map<KnowledgeCollectDto>(collect);
  415. return knowledgeInfoDto;
  416. }
  417. /// <summary>
  418. /// 知识库-查询详情-增加浏览量
  419. /// </summary>
  420. /// <param name="Id">知识Id</param>
  421. /// <param name="IsAddPv">默认不增加,false不增加,true增加浏览量</param>
  422. /// <returns></returns>
  423. [HttpGet("info/{Id}")]
  424. public async Task<KnowledgeInfoDto> KnowledgeInfo(string Id, bool? IsAddPv)
  425. {
  426. var knowledge = await _knowledgeDomainService.KnowledgeInfo(Id, HttpContext.RequestAborted);
  427. if (knowledge is null)
  428. throw UserFriendlyException.SameMessage("知识查询失败!");
  429. if (knowledge.Workflow != null)
  430. knowledge.IsCanHandle = knowledge.Workflow.IsCanHandle(_sessionContext.RequiredUserId, _sessionContext.RequiredOrgId, _sessionContext.Roles);
  431. //转化
  432. var knowledgeShowInfoDto = _mapper.Map<KnowledgeInfoDto>(knowledge);
  433. if (knowledgeShowInfoDto != null && !string.IsNullOrEmpty(knowledgeShowInfoDto.Content))
  434. knowledgeShowInfoDto.Content = _bulletinApplication.GetSiteUrls(knowledgeShowInfoDto.Content);
  435. //var type = await _knowledgeTypeRepository.GetAsync(knowledge.KnowledgeTypeId, HttpContext.RequestAborted);
  436. //if (type != null)
  437. //{
  438. // knowledgeShowInfoDto.KnowledgeTypeName = type.SpliceName;
  439. // knowledgeShowInfoDto.KnowledgeType = _mapper.Map<KnowledgeTypeDto>(type);
  440. //}
  441. var hot = await _hotspotTypeRepository.GetAsync(knowledge.HotspotId, HttpContext.RequestAborted);
  442. if (hot != null)
  443. knowledgeShowInfoDto.HotspotName = hot.HotSpotFullName;
  444. //收藏
  445. var collect = await _knowledgeCollectRepository.GetAsync(x => x.KnowledgeId == Id && x.CreatorId == _sessionContext.UserId);
  446. if (collect != null)
  447. knowledgeShowInfoDto.Collect = _mapper.Map<KnowledgeCollectDto>(collect);
  448. //关联知识
  449. var knowledges = await _knowledgeRepository.Queryable().In(x => x.Id, knowledge.Knowledges).Where(x => x.Status == EKnowledgeStatus.OnShelf && ((x.ExpiredTime != null && x.ExpiredTime >= DateTime.Now) || x.ExpiredTime == null)).ToListAsync();
  450. if (knowledges.Any())
  451. knowledgeShowInfoDto.KnowledgeDtos = _mapper.Map<List<KnowledgeDto>>(knowledges);
  452. //关键词
  453. var knowledgeWords = await _knowledgeWrodRepository.Queryable().In(x => x.Id, knowledge.Keywords).ToListAsync();
  454. if (knowledgeWords.Any())
  455. knowledgeShowInfoDto.KeywordsDto = _mapper.Map<List<KnowledgeWordDto>>(knowledgeWords);
  456. var files = await _fileRepository.Queryable().Where(x => x.Key == knowledge.Id).ToListAsync();
  457. if (files.Any()) knowledgeShowInfoDto.Files = _mapper.Map<List<FileDto>>(files);
  458. if (IsAddPv == true)
  459. _mediator.Publish(new GetKnowledgeInfoNotify(knowledge));
  460. return knowledgeShowInfoDto;
  461. }
  462. /// <summary>
  463. /// 知识详情--导出
  464. /// </summary>
  465. /// <param name="dto"></param>
  466. /// <returns></returns>
  467. [HttpPost("info/export")]
  468. public async Task<IActionResult> KnowledgeInfoExport([FromBody] KnowledgeInfoExportInDto dto)
  469. {
  470. if (dto.Ids.Length > 1)
  471. {
  472. var streams = await _knowApplication.KnowledgeInfoListExportAsync(dto, HttpContext.RequestAborted);
  473. byte[] fileBytes = _wordHelperService.ConvertZipStream(streams);
  474. var name = DateTime.Now.ToString("yyyyMMddHHmmss");
  475. return File(fileBytes, "application/octet-stream", $"{name}.zip");
  476. }
  477. var info = await _knowledgeRepository.GetAsync(dto.Ids[0]) ?? throw UserFriendlyException.SameMessage("知识不存在");
  478. return info.Content.HtmlToStream(dto.FileType).GetFileStreamResult(dto.FileType, "知识详情");
  479. }
  480. /// <summary>
  481. /// 知识申请-关联知识-获取知识列表
  482. /// </summary>
  483. /// <returns></returns>
  484. [HttpGet("getknowledge")]
  485. public async Task<IReadOnlyList<KnowledgeCreateBMDataDto>> GetKnowledge()
  486. {
  487. var temp = await _knowledgeRepository
  488. .Queryable()
  489. .LeftJoin<SystemOrganize>((o, sys) => o.CreatorOrgId == sys.Id)
  490. //重新构建数据
  491. .Select((o, sys) => new
  492. {
  493. index = SqlFunc.RowNumber($"{o.Version} desc ", $"{o.Code}"),
  494. DepartmentId = sys.Id,
  495. Department = sys.Name,
  496. o.Id,
  497. o.Title,
  498. o.Status,
  499. o.Code,
  500. o.IsDeleted,
  501. o.ExpiredTime
  502. })
  503. //将结果合并成一个表
  504. .MergeTable()
  505. //取第一条数据
  506. .Where(x => x.IsDeleted == false)
  507. .Where(x => (x.ExpiredTime != null && x.ExpiredTime >= DateTime.Now) || x.ExpiredTime == null)
  508. .Where(d => d.index == 1 && d.Status == EKnowledgeStatus.OnShelf)
  509. .ToListAsync();
  510. //返回数据
  511. return _mapper.Map<IReadOnlyList<KnowledgeCreateBMDataDto>>(temp);
  512. }
  513. /// <summary>
  514. /// 我的草稿箱
  515. /// </summary>
  516. /// <param name="pagedDto"></param>
  517. /// <returns></returns>
  518. [HttpGet("mydraftslist")]
  519. public async Task<PagedDto<KnowledgeDataDto>> MyDraftsList([FromQuery] MyDraftsListPagedDto pagedDto)
  520. {
  521. var (total, items) = await _knowledgeRepository
  522. .Queryable()
  523. .Includes(it => it.User)
  524. .Where(p => p.CreatorId == _sessionContext.RequiredUserId && p.Status == EKnowledgeStatus.Drafts)
  525. .WhereIF(!string.IsNullOrEmpty(pagedDto.Keyword), d => d.Title.Contains(pagedDto.Keyword!))
  526. .WhereIF(pagedDto.StartTime != null, d => d.CreationTime >= pagedDto.StartTime)
  527. .WhereIF(pagedDto.EndTime != null, d => d.CreationTime <= pagedDto.EndTime)
  528. .OrderByDescending(p => p.CreationTime)
  529. .ToPagedListAsync(pagedDto.PageIndex, pagedDto.PageSize, HttpContext.RequestAborted);
  530. return new PagedDto<KnowledgeDataDto>(total, _mapper.Map<IReadOnlyList<KnowledgeDataDto>>(items));
  531. }
  532. /// <summary>
  533. /// 知识库列表页面枚举值
  534. /// </summary>
  535. /// <returns></returns>
  536. [HttpGet("knowledge-status-data")]
  537. public async Task<object> KnowledgeStatus()
  538. {
  539. var tabNames = new List<KeyValuePair<int, string>>
  540. {
  541. new KeyValuePair<int, string>(3, "已上架"),
  542. new KeyValuePair<int, string>(4, "已下架"),
  543. new KeyValuePair<int, string>(1, "审核中"),
  544. new KeyValuePair<int, string>(5, "审核不通过")
  545. };
  546. return _baseDataApplication
  547. .FileType(EFileType.excel | EFileType.pdf)
  548. .Add("tabNames", tabNames)
  549. .Build();
  550. }
  551. /// <summary>
  552. /// 知识查询
  553. /// </summary>
  554. /// <param name="pagedDto"></param>
  555. /// <returns></returns>
  556. [HttpGet]
  557. public async Task<PagedDto<KnowledgeDataDto>> GetKnowList([FromQuery] KnowPagedListDto pagedDto)
  558. {
  559. return (await _knowApplication.GetKnowList(pagedDto, HttpContext.RequestAborted))
  560. .ToPaged();
  561. }
  562. /// <summary>
  563. /// 知识查询-导出
  564. /// </summary>
  565. /// <param name="dto"></param>
  566. /// <returns></returns>
  567. [HttpPost("export")]
  568. public async Task<IActionResult> GetKnowListExportAsync([FromBody] ExportExcelDto<KnowPagedListDto> dto)
  569. {
  570. var items = (await _knowApplication.GetKnowList(dto.QueryDto, HttpContext.RequestAborted)).Item2;
  571. return _exportApplication.GetExcelFile(dto, items, "知识明细导出");
  572. }
  573. /// <summary>
  574. /// 知识检索
  575. /// </summary>
  576. /// <param name="dto"></param>
  577. /// <returns></returns>
  578. [HttpGet("knowretrieval")]
  579. public async Task<PagedDto<KnowledgeRetrievalDataDto>> KnowRetrieval([FromQuery] KnowledgeRetrievalPagedListDto dto)
  580. {
  581. var typeSpliceName = string.Empty;
  582. var hotspotHotSpotFullName = string.Empty;
  583. if (!string.IsNullOrEmpty(dto.KnowledgeTypeId))
  584. {
  585. var type = await _knowledgeTypeRepository.GetAsync(x => x.Id == dto.KnowledgeTypeId);
  586. typeSpliceName = type?.SpliceName;
  587. }
  588. if (!string.IsNullOrEmpty(dto.HotspotId))
  589. {
  590. var hotspot = await _hotspotTypeRepository.GetAsync(x => x.Id == dto.HotspotId);
  591. hotspotHotSpotFullName = hotspot?.HotSpotFullName;
  592. }
  593. var sugar = _knowledgeRepository
  594. .Queryable(false, false, false)
  595. .Includes(x => x.User)
  596. .Includes(x => x.SystemOrganize)
  597. .Includes(x => x.HotspotType)
  598. .Where(x => x.IsDeleted == false)
  599. .Where(x => x.Status == EKnowledgeStatus.OnShelf)
  600. .Where(x => x.KnowledgeType.Any(t => t.KnowledgeType.KnowledgeTypeOrgs.Any(to => to.OrgId == _sessionContext.RequiredOrgId) || t.KnowledgeType.KnowledgeTypeOrgs.Any() == false))
  601. // .WhereIF(dto.RetrievalType == EKnowledgeRetrievalType.All && !string.IsNullOrEmpty(dto.Keyword), d => d.Title.Contains(dto.Keyword!) || d.Content.Contains(dto.Keyword!))
  602. //.WhereIF(dto.RetrievalType == EKnowledgeRetrievalType.Title && !string.IsNullOrEmpty(dto.Keyword), d => d.Title.Contains(dto.Keyword!))
  603. //.WhereIF(dto.RetrievalType == EKnowledgeRetrievalType.Content && !string.IsNullOrEmpty(dto.Keyword), d => d.Content.Contains(dto.Keyword!))
  604. //.WhereIF(dto.RetrievalType == EKnowledgeRetrievalType.Summary && !string.IsNullOrEmpty(dto.Keyword), d => d.Summary != null && d.Summary.Contains(dto.Keyword!))
  605. //.WhereIF(!string.IsNullOrEmpty(typeSpliceName), x => SqlFunc.JsonLike(x.KnowledgeType, typeSpliceName))
  606. .WhereIF(!string.IsNullOrEmpty(typeSpliceName), x => x.KnowledgeType.Any(t => t.KnowledgeTypeSpliceName.EndsWith(typeSpliceName)))
  607. .WhereIF(!string.IsNullOrEmpty(hotspotHotSpotFullName), x => x.HotspotType.HotSpotFullName.EndsWith(hotspotHotSpotFullName!))
  608. .WhereIF(!string.IsNullOrEmpty(dto.HotspotName), x => x.HotspotType.HotSpotFullName.EndsWith(dto.HotspotName!))
  609. .WhereIF(!string.IsNullOrEmpty(dto.CreateOrgId), x => x.CreatorOrgId != null && x.CreatorOrgId.EndsWith(dto.CreateOrgId!))
  610. .WhereIF(!string.IsNullOrEmpty(dto.Attribution), x => x.Attribution == dto.Attribution!);
  611. if (dto.Keyword.NotNullOrEmpty())
  612. {
  613. var keywords = dto.Keyword!.SplitKeywords();
  614. var exp = Expressionable.Create<Knowledge>();
  615. foreach (var keyword in keywords)
  616. {
  617. if (dto.RetrievalType == EKnowledgeRetrievalType.All)
  618. exp.Or(m => m.Title.Contains(keyword) || m.Content.Contains(keyword));
  619. if (dto.RetrievalType == EKnowledgeRetrievalType.Title)
  620. exp.Or(m => m.Title.Contains(keyword));
  621. if (dto.RetrievalType == EKnowledgeRetrievalType.Content)
  622. exp.Or(m => m.Content.Contains(keyword));
  623. if (dto.RetrievalType == EKnowledgeRetrievalType.Summary)
  624. exp.Or(m => m.Summary != null && m.Summary.Contains(keyword));
  625. }
  626. sugar.Where(exp.ToExpression());
  627. }
  628. switch (dto.Sort)
  629. {
  630. case "2":
  631. sugar = sugar.OrderByDescending(p => p.Score);
  632. break;
  633. case "3":
  634. sugar = sugar.OrderByDescending(p => p.CreationTime);
  635. break;
  636. default:
  637. sugar = sugar.OrderByDescending(p => p.PageView);
  638. break;
  639. }
  640. var (total, temp) = await sugar.ToPagedListAsync(dto.PageIndex, dto.PageSize);
  641. return new PagedDto<KnowledgeRetrievalDataDto>(total, _mapper.Map<IReadOnlyList<KnowledgeRetrievalDataDto>>(temp));
  642. }
  643. /// <summary>
  644. /// 获取知识审批信息
  645. /// </summary>
  646. /// <param name="id"></param>
  647. /// <returns></returns>
  648. [HttpGet("audit_log")]
  649. public async Task<PagedDto<KnowledgeWorkFlowDto>> KnowRetrieval([FromQuery] AuditLogListPagedDto pagedDto)
  650. {
  651. var (total, temp) = await _knowledgeWorkFlowRepository
  652. .Queryable()
  653. .Includes(x => x.User)
  654. .Includes(x => x.SystemOrganize)
  655. .Includes(x => x.Workflow)
  656. .Where(x => x.KnowledgeId == pagedDto.id)
  657. .Where(x => x.IsDeleted == false)
  658. .OrderBy(x => x.CreationTime)
  659. .ToPagedListAsync(pagedDto.PageIndex, pagedDto.PageSize);
  660. return new PagedDto<KnowledgeWorkFlowDto>(total, _mapper.Map<IReadOnlyList<KnowledgeWorkFlowDto>>(temp));
  661. }
  662. /// <summary>
  663. /// 知识查重
  664. /// </summary>
  665. /// <param name="id"></param>
  666. /// <returns></returns>
  667. [HttpPost("finding_duplicate")]
  668. public async Task<bool> FindingDuplicate([FromBody] KnowledgeFindingDuplicateDto dto)
  669. {
  670. var any = await _knowledgeRepository.Queryable().Where(x => x.Status == EKnowledgeStatus.Auditing || x.Status >= EKnowledgeStatus.OnShelf)
  671. .WhereIF(!string.IsNullOrEmpty(dto.Title), x => x.Title.Equals(dto.Title))
  672. .WhereIF(!string.IsNullOrEmpty(dto.Summary), x => x.Summary.Equals(dto.Summary))
  673. .WhereIF(!string.IsNullOrEmpty(dto.Content), x => x.Content.Equals(dto.Content))
  674. .WhereIF(!string.IsNullOrEmpty(dto.Id), x => x.Id != dto.Id)
  675. .AnyAsync();
  676. return any;
  677. }
  678. #endregion
  679. #region 我的知识删除列表
  680. /// <summary>
  681. /// 我的知识删除列表页面枚举值
  682. /// </summary>
  683. /// <returns></returns>
  684. [HttpGet("delete-status-data")]
  685. public async Task<object> DeleteApplyStatus()
  686. {
  687. return EnumExts.GetDescriptions<EKnowledgeWorkFlowStatus>();
  688. }
  689. /// <summary>
  690. /// 我的知识删除列表
  691. /// </summary>
  692. /// <param name="pagedDto"></param>
  693. /// <returns></returns>
  694. [HttpGet("deletelist")]
  695. public async Task<PagedDto<KnowledgeDeleteApplyDataDto>> GetDeleteApplyList([FromQuery] KnowledgeDeletelPagedListDto pagedDto)
  696. {
  697. var (total, items) = await _knowledgeWorkFlowRepository
  698. .Queryable(includeDeleted: true)
  699. .Includes(it => it.Knowledge)
  700. .Includes(it => it.User)
  701. .Includes(it => it.SystemOrganize)
  702. .Includes(it => it.Knowledge, it => it.KnowledgeType)
  703. .Includes(it => it.Knowledge, it => it.HotspotType)
  704. .Includes(it => it.Workflow)
  705. .Where(d => d.CreatorId == _sessionContext.RequiredUserId && d.WorkflowModuleStatus == EKnowledgeApplyType.Delete && d.WorkflowId != null)
  706. .WhereIF(pagedDto.EKnowledgeWorkFlowStatus.HasValue, d => d.WorkFlowApplyStatus == pagedDto.EKnowledgeWorkFlowStatus)
  707. .WhereIF(pagedDto.StartApplyTime.HasValue, d => d.CreationTime >= pagedDto.StartApplyTime)
  708. .WhereIF(pagedDto.EndApplyTime.HasValue, d => d.CreationTime <= pagedDto.EndApplyTime)
  709. .WhereIF(!string.IsNullOrEmpty(pagedDto.Keyword), d => d.Knowledge.User.Name.Contains(pagedDto.Keyword!)
  710. || d.Knowledge.SystemOrganize.Name.Contains(pagedDto.Keyword!)
  711. || d.Knowledge.Title.Contains(pagedDto.Keyword!))
  712. .OrderByDescending(p => p.CreationTime)
  713. .ToPagedListAsync(pagedDto.PageIndex, pagedDto.PageSize, HttpContext.RequestAborted);
  714. return new PagedDto<KnowledgeDeleteApplyDataDto>(total, _mapper.Map<IReadOnlyList<KnowledgeDeleteApplyDataDto>>(items));
  715. }
  716. /// <summary>
  717. /// 审核管理页面枚举值
  718. /// </summary>
  719. /// <returns></returns>
  720. [HttpGet("approval-base-data")]
  721. public async Task<object> ApprovalBaseData()
  722. {
  723. return new
  724. {
  725. EKnowledgeWorkFlowStatus = EnumExts.GetDescriptions<EKnowledgeWorkFlowStatus>(),
  726. EKnowledgeApplyType = EnumExts.GetDescriptions<EKnowledgeApplyType>()
  727. };
  728. }
  729. /// <summary>
  730. /// 审核管理
  731. /// </summary>
  732. /// <param name="pagedDto"></param>
  733. /// <returns></returns>
  734. [HttpGet("approvedlist")]
  735. public async Task<PagedDto<KnowledgeApprovalDataDto>> ApprovedList([FromQuery] KnowledgeApprovalPagedListDto pagedDto)
  736. {
  737. var (total, items) = await _knowledgeWorkFlowRepository
  738. .Queryable(includeDeleted: true)
  739. .Includes(it => it.Knowledge)
  740. .Includes(it => it.User)
  741. .Includes(it => it.SystemOrganize)
  742. .Includes(it => it.Workflow, d => d.Steps)
  743. .Where(it => it.WorkflowId != null)
  744. .WhereIF(pagedDto.EKnowledgeApplyType.HasValue, d => d.WorkflowModuleStatus == pagedDto.EKnowledgeApplyType)
  745. .WhereIF(pagedDto.EKnowledgeWorkFlowStatus.HasValue, d => d.WorkFlowApplyStatus == pagedDto.EKnowledgeWorkFlowStatus)
  746. .WhereIF(!string.IsNullOrEmpty(pagedDto.Keyword), d => d.Knowledge.User.Name.Contains(pagedDto.Keyword!)
  747. || d.Knowledge.SystemOrganize.Name.Contains(pagedDto.Keyword!)
  748. || d.Knowledge.Title.Contains(pagedDto.Keyword!))
  749. .OrderByDescending(p => p.CreationTime)
  750. .ToPagedListAsync(pagedDto.PageIndex, pagedDto.PageSize, HttpContext.RequestAborted);
  751. foreach (var item in items)
  752. {
  753. if (item.Workflow != null)
  754. //item.CanHandle = item.CanHandle(_sessionContext.RequiredUserId, _sessionContext.RequiredOrgId);
  755. item.CanHandle = item.Workflow.IsCanHandle(
  756. _sessionContext.RequiredUserId, _sessionContext.RequiredOrgId, _sessionContext.Roles);
  757. }
  758. //处理是否可以办理
  759. //items.ForEach(d => d.CanHandle = d.Workflow.CanHandle(_sessionContext.RequiredUserId, _sessionContext.RequiredOrgCode));
  760. return new PagedDto<KnowledgeApprovalDataDto>(total, _mapper.Map<IReadOnlyList<KnowledgeApprovalDataDto>>(items));
  761. }
  762. /// <summary>
  763. /// 工单受理知识检索
  764. /// </summary>
  765. /// <param name="pagedDto"></param>
  766. /// <returns></returns>
  767. [HttpGet("knowpopscreen")]
  768. public async Task<PagedDto<KnowledgeRetrievalDataDto>> KnowPopScreen([FromQuery] KnowledgePopScreenPagedListDto pagedDto)
  769. {
  770. var orgid = string.Empty;
  771. if (pagedDto.RetrievalType == EKnowledgeRetrievalType.Org && !string.IsNullOrEmpty(pagedDto.Keyword))
  772. {
  773. var organize = await _systemOrganizeRepository.GetAsync(x => x.Name == pagedDto.Keyword);
  774. orgid = organize?.Id;
  775. }
  776. var (total, temp) = await _knowledgeRepository.Queryable()
  777. .Includes(x => x.SourceOrganize)
  778. .Where(d => d.Status == EKnowledgeStatus.OnShelf)
  779. .WhereIF(pagedDto.RetrievalType == EKnowledgeRetrievalType.Title && !string.IsNullOrEmpty(pagedDto.Keyword), d => d.Title.Contains(pagedDto.Keyword!))
  780. .WhereIF(pagedDto.RetrievalType == EKnowledgeRetrievalType.Content && !string.IsNullOrEmpty(pagedDto.Keyword), d => d.Content.Contains(pagedDto.Keyword!))
  781. .WhereIF(!string.IsNullOrEmpty(orgid) && pagedDto.RetrievalType == EKnowledgeRetrievalType.Org, x => x.CreatorOrgId.EndsWith(orgid!))
  782. .WhereIF(!string.IsNullOrEmpty(pagedDto.HotspotId), p => p.HotspotId == pagedDto.HotspotId)
  783. .OrderByDescending(p => p.CreationTime)
  784. .ToPagedListAsync(pagedDto.PageIndex, pagedDto.PageSize);
  785. return new PagedDto<KnowledgeRetrievalDataDto>(total, _mapper.Map<IReadOnlyList<KnowledgeRetrievalDataDto>>(temp));
  786. }
  787. ///// <summary>
  788. ///// 新增-开始流程
  789. ///// </summary>
  790. ///// <param name="id">知识id</param>
  791. ///// <param name="dto">流程开启参数</param>
  792. ///// <returns></returns>
  793. //[Permission(EPermission.AddKnowledge)]
  794. //[HttpPost("{id}/add-startflow")]
  795. //public async Task AddStartFlow(string id, [FromBody] StartWorkflowDto dto)
  796. //{
  797. // await StartFlow(id, WorkflowModuleConsts.KnowledgeAdd, EKnowledgeApplyType.Add, dto);
  798. //}
  799. ///// <summary>
  800. ///// 删除-开始流程
  801. ///// </summary>
  802. ///// <param name="id">知识id</param>
  803. ///// <param name="dto">流程开启参数</param>
  804. ///// <returns></returns>
  805. //[Permission(EPermission.KnowledgeDelete)]
  806. //[HttpPost("{id}/remove-startflow")]
  807. //public async Task RemoveStartFlow(string id, [FromBody] StartWorkflowDto dto)
  808. //{
  809. // await StartFlow(id, WorkflowModuleConsts.KnowledgeDelete, EKnowledgeApplyType.Delete, dto);
  810. //}
  811. /// <summary>
  812. /// 查询知识库办理流程开启参数-新增
  813. /// </summary>
  814. /// <returns></returns>
  815. //[Permission(EPermission.AddKnowledge)]
  816. [HttpGet("add-flow-start")]
  817. public async Task<NextStepsDto> GetAddFlowStartOptionsAsync()
  818. {
  819. return await _workflowApplication.GetStartStepsAsync(WorkflowModuleConsts.KnowledgeAdd,
  820. HttpContext.RequestAborted);
  821. }
  822. /// <summary>
  823. /// 查询知识库办理流程开启参数-新增
  824. /// </summary>
  825. /// <returns></returns>
  826. //[Permission(EPermission.AddKnowledge)]
  827. [HttpGet("update-flow-start")]
  828. public async Task<NextStepsDto> GetUpdateFlowStartOptionsAsync()
  829. {
  830. return await _workflowApplication.GetStartStepsAsync(WorkflowModuleConsts.KnowledgeUpdate,
  831. HttpContext.RequestAborted);
  832. }
  833. /// <summary>
  834. /// 查询知识库办理流程开启参数-删除
  835. /// </summary>
  836. /// <returns></returns>
  837. //[Permission(EPermission.KnowledgeDelete)]
  838. [HttpGet("remove-flow-start")]
  839. public async Task<NextStepsDto> GetRemoveFlowStartOptionsAsync()
  840. {
  841. return await _workflowApplication.GetStartStepsAsync(WorkflowModuleConsts.KnowledgeDelete,
  842. HttpContext.RequestAborted);
  843. }
  844. /// <summary>
  845. /// 开始流程
  846. /// </summary>
  847. /// <param name="id">知识ID</param>
  848. /// <param name="moduleCode">知识模板编号</param>
  849. /// <param name="eKnowledgeApplyType">申请类型</param>
  850. /// <param name="dto">流程开启参数</param>
  851. /// <returns></returns>
  852. private async Task<string> StartFlow(string id, string moduleCode, EKnowledgeApplyType eKnowledgeApplyType, StartWorkflowDto dto)
  853. {
  854. var knowledge = await _knowledgeRepository.GetAsync(id, HttpContext.RequestAborted);
  855. if (knowledge == null)
  856. throw UserFriendlyException.SameMessage("无效知识编号");
  857. if (eKnowledgeApplyType == EKnowledgeApplyType.Delete)
  858. {
  859. if (knowledge.IsDeleted == true)
  860. throw UserFriendlyException.SameMessage("知识删除失败");
  861. }
  862. //知识审批主表
  863. await _knowledgeDomainService.AddWorkFlowAsync(id, eKnowledgeApplyType, HttpContext.RequestAborted);
  864. dto.DefinitionModuleCode = moduleCode;
  865. dto.Title = knowledge.Title;
  866. return await _workflowApplication.StartWorkflowAsync(dto, _sessionContext, id, cancellationToken: HttpContext.RequestAborted);
  867. }
  868. #endregion
  869. #region 知识库词库
  870. /// <summary>
  871. /// 新增知识库词库
  872. /// </summary>
  873. /// <param name="dtos"></param>
  874. /// <returns></returns>
  875. [Permission(EPermission.AddKnowledgeWord)]
  876. [HttpPost("knowledge_word")]
  877. [LogFilter("新增知识库词库")]
  878. public async Task Add([FromBody] KnowledgeWordAddDto dto)
  879. {
  880. var word = _mapper.Map<KnowledgeWord>(dto);
  881. await _knowledgeWrodRepository.AddAsync(word, HttpContext.RequestAborted);
  882. }
  883. /// <summary>
  884. /// 删除知识库词库
  885. /// </summary>
  886. /// <param name="dto"></param>
  887. /// <returns></returns>
  888. [Permission(EPermission.DeleteKnowledgeWord)]
  889. [HttpDelete("knowledge_word")]
  890. [LogFilter("删除知识库词库")]
  891. public async Task Delete([FromBody] KnowledgeWordDeleteDto dto)
  892. {
  893. await _knowledgeRepository.RemoveKnowledgeWrodBatchAsync(dto.Ids, HttpContext.RequestAborted);
  894. }
  895. /// <summary>
  896. /// 更新知识库词库
  897. /// </summary>
  898. /// <param name="dto"></param>
  899. /// <returns></returns>
  900. [Permission(EPermission.UpdateKnowledgeWord)]
  901. [HttpPut("knowledge_word")]
  902. [LogFilter("更新知识库词库")]
  903. public async Task Update([FromBody] KnowledgeWordUpdateDto dto)
  904. {
  905. var word = await _knowledgeWrodRepository.GetAsync(dto.Id, HttpContext.RequestAborted);
  906. if (word is null)
  907. throw UserFriendlyException.SameMessage("无效知识库词库");
  908. _mapper.Map(dto, word);
  909. word.LastModificationName = _sessionContext.UserName;
  910. await _knowledgeWrodRepository.UpdateAsync(word, HttpContext.RequestAborted);
  911. }
  912. /// <summary>
  913. /// 获取知识库词库列表
  914. /// </summary>
  915. /// <param name="dto"></param>
  916. /// <returns></returns>
  917. [HttpGet("knowledge_word/list")]
  918. public async Task<PagedDto<KnowledgeWordDto>> List([FromQuery] KnowledgeWordListDto dto)
  919. {
  920. var (total, items) = await _knowledgeWrodRepository.Queryable()
  921. .WhereIF(!string.IsNullOrEmpty(dto.Tag), x => x.Tag == dto.Tag!)
  922. .WhereIF(!string.IsNullOrEmpty(dto.Classify), x => x.Classify == dto.Classify!)
  923. .WhereIF(!string.IsNullOrEmpty(dto.Synonym), x => x.Synonym != null && x.Synonym.Contains(dto.Synonym!))
  924. .OrderByDescending(x => x.CreationTime)
  925. .ToPagedListAsync(dto.PageIndex, dto.PageSize, HttpContext.RequestAborted);
  926. return new PagedDto<KnowledgeWordDto>(total, _mapper.Map<IReadOnlyList<KnowledgeWordDto>>(items));
  927. }
  928. /// <summary>
  929. /// 获取知识库词库
  930. /// </summary>
  931. /// <param name="id"></param>
  932. /// <returns></returns>
  933. [HttpGet("knowledge_word/{id}")]
  934. public async Task<KnowledgeWord> WordEntity(string id)
  935. {
  936. return await _knowledgeWrodRepository.Queryable()
  937. .FirstAsync(x => x.Id == id);
  938. }
  939. /// <summary>
  940. /// 获取知识库词库基本信息
  941. /// </summary>
  942. /// <returns></returns>
  943. [HttpGet("knowledge_word/base")]
  944. public async Task<object> Base()
  945. {
  946. var rsp = new
  947. {
  948. KnowledgeWordClassify = await _systemDomainService.GetSysDicDataByCodeAsync(SysDicTypeConsts.KnowledgeWordClassify),
  949. };
  950. return rsp;
  951. }
  952. #endregion
  953. #region 知识纠错
  954. /// <summary>
  955. /// 新增知识纠错
  956. /// </summary>
  957. /// <param name="dtos"></param>
  958. /// <returns></returns>
  959. //[Permission(EPermission.AddKnowledgeCorrection)]
  960. [HttpPost("knowledge_correction")]
  961. [LogFilter("新增知识纠错")]
  962. public async Task Add([FromBody] KnowledgeCorrectionAddDto dto)
  963. {
  964. var correction = _mapper.Map<KnowledgeCorrection>(dto);
  965. await _knowledgeCorrectionRepository.AddAsync(correction, HttpContext.RequestAborted);
  966. }
  967. /// <summary>
  968. /// 删除知识纠错
  969. /// </summary>
  970. /// <param name="dto"></param>
  971. /// <returns></returns>
  972. //[Permission(EPermission.DeleteKnowledgeCorrection)]
  973. //[HttpDelete("knowledge_correction")]
  974. //public async Task Delete([FromBody] KnowledgeCorrectionDeleteDto dto)
  975. //{
  976. // await _knowledgeCorrectionRepository.RemoveAsync(x => x.Id == dto.Id);
  977. //}
  978. /// <summary>
  979. /// 更新知识纠错
  980. /// </summary>
  981. /// <param name="dto"></param>
  982. /// <returns></returns>
  983. //[Permission(EPermission.UpdateKnowledgeCorrection)]
  984. //[HttpPut("knowledge_correction")]
  985. //public async Task Update([FromBody] KnowledgeCorrectionUpdateDto dto)
  986. //{
  987. // var correction = await _knowledgeCorrectionRepository.GetAsync(dto.Id, HttpContext.RequestAborted);
  988. // if (correction is null)
  989. // throw UserFriendlyException.SameMessage("无效知识纠错");
  990. // _mapper.Map(dto, correction);
  991. // await _knowledgeCorrectionRepository.UpdateAsync(correction, HttpContext.RequestAborted);
  992. //}
  993. /// <summary>
  994. /// 答复知识纠错
  995. /// </summary>
  996. /// <param name="dto"></param>
  997. /// <returns></returns>
  998. [Permission(EPermission.ReplyKnowledgeCorrection)]
  999. [HttpPut("knowledge_correction/Reply")]
  1000. [LogFilter("答复知识纠错")]
  1001. public async Task Reply([FromBody] KnowledgeCorrectionUpdateDto dto)
  1002. {
  1003. var correction = await _knowledgeCorrectionRepository.GetAsync(dto.Id, HttpContext.RequestAborted);
  1004. if (correction is null)
  1005. throw UserFriendlyException.SameMessage("无效知识纠错");
  1006. _mapper.Map(dto, correction);
  1007. correction.ReplyTime = DateTime.Now;
  1008. correction.ReplyUserName = _sessionContext.UserName;
  1009. correction.State = ECorrectionState.AlreadyAnswered;
  1010. await _knowledgeCorrectionRepository.UpdateAsync(correction, HttpContext.RequestAborted);
  1011. }
  1012. /// <summary>
  1013. /// 获取知识纠错列表
  1014. /// </summary>
  1015. /// <param name="dto"></param>
  1016. /// <returns></returns>
  1017. [Permission(EPermission.KnowledgeCorrectionList)]
  1018. [HttpGet("knowledge_correction/list")]
  1019. public async Task<PagedDto<KnowledgeCorrectionDto>> List([FromQuery] KnowledgeCorrectionListDto dto)
  1020. {
  1021. var typeSpliceName = string.Empty;
  1022. if (!string.IsNullOrEmpty(dto.KnowledgeTypeId))
  1023. {
  1024. var type = await _knowledgeTypeRepository.GetAsync(x => x.Id == dto.KnowledgeTypeId);
  1025. typeSpliceName = type?.SpliceName;
  1026. }
  1027. var (total, items) = await _knowledgeCorrectionRepository.Queryable()
  1028. .Includes(x => x.Knowledge)
  1029. //.WhereIF(!string.IsNullOrEmpty(dto.KnowledgeTypeId), x => x.Knowledge.KnowledgeTypeId == dto.KnowledgeTypeId!)
  1030. .WhereIF(!string.IsNullOrEmpty(dto.CreatorName), x => x.CreatorName == dto.CreatorName!)
  1031. //.WhereIF(!string.IsNullOrEmpty(typeSpliceName), x => SqlFunc.JsonLike(x.Knowledge.KnowledgeType, typeSpliceName))
  1032. .WhereIF(!string.IsNullOrEmpty(typeSpliceName), x => x.Knowledge.KnowledgeType.Any(t => t.KnowledgeTypeSpliceName.EndsWith(typeSpliceName)))
  1033. .Where(x => !string.IsNullOrEmpty(x.Knowledge.Id))
  1034. .OrderByDescending(x => x.CreationTime)
  1035. .ToPagedListAsync(dto.PageIndex, dto.PageSize, HttpContext.RequestAborted);
  1036. return new PagedDto<KnowledgeCorrectionDto>(total, _mapper.Map<IReadOnlyList<KnowledgeCorrectionDto>>(items));
  1037. }
  1038. /// <summary>
  1039. /// 获取知识库纠错
  1040. /// </summary>
  1041. /// <param name="id"></param>
  1042. /// <returns></returns>
  1043. //[Permission(EPermission.KnowledgeCorrectionEntity)]
  1044. [HttpGet("knowledge_correction/{id}")]
  1045. public async Task<KnowledgeCorrection> CorrectionEntity(string id)
  1046. {
  1047. return await _knowledgeCorrectionRepository.Queryable()
  1048. .Includes(x => x.Knowledge)
  1049. .FirstAsync(x => x.Id == id);
  1050. }
  1051. #endregion
  1052. #region 知识提问
  1053. /// <summary>
  1054. /// 新增知识提问
  1055. /// </summary>
  1056. /// <param name="dtos"></param>
  1057. /// <returns></returns>
  1058. //[Permission(EPermission.AddKnowledgeQuestions)]
  1059. [HttpPost("knowledge_questions")]
  1060. [LogFilter("新增知识提问")]
  1061. public async Task Add([FromBody] KnowledgeQuestionsAddDto dto)
  1062. {
  1063. var questions = _mapper.Map<KnowledgeQuestions>(dto);
  1064. await _knowledgeQuestionsRepository.AddAsync(questions, HttpContext.RequestAborted);
  1065. }
  1066. /// <summary>
  1067. /// 删除知识提问
  1068. /// </summary>
  1069. /// <param name="dto"></param>
  1070. /// <returns></returns>
  1071. //[Permission(EPermission.DeleteKnowledgeQuestions)]
  1072. //[HttpDelete("knowledge_questions")]
  1073. //public async Task Delete([FromBody] KnowledgeQuestionsDeleteDto dto)
  1074. //{
  1075. // await _knowledgeQuestionsRepository.RemoveAsync(x => x.Id == dto.Id);
  1076. //}
  1077. /// <summary>
  1078. /// 更新知识提问
  1079. /// </summary>
  1080. /// <param name="dto"></param>
  1081. /// <returns></returns>
  1082. //[Permission(EPermission.UpdateKnowledgeQuestions)]
  1083. //[HttpPut("knowledge_questions")]
  1084. //public async Task Update([FromBody] KnowledgeQuestionsUpdateDto dto)
  1085. //{
  1086. // var questions = await _knowledgeQuestionsRepository.GetAsync(dto.Id, HttpContext.RequestAborted);
  1087. // if (questions is null)
  1088. // throw UserFriendlyException.SameMessage("无效知识提问");
  1089. // _mapper.Map(dto, questions);
  1090. // await _knowledgeQuestionsRepository.UpdateAsync(questions, HttpContext.RequestAborted);
  1091. //}
  1092. /// <summary>
  1093. /// 答复知识提问
  1094. /// </summary>
  1095. /// <param name="dto"></param>
  1096. /// <returns></returns>
  1097. [Permission(EPermission.ReplyKnowledgeQuestions)]
  1098. [HttpPut("knowledge_questions/Reply")]
  1099. [LogFilter("答复知识提问")]
  1100. public async Task Reply([FromBody] KnowledgeQuestionsUpdateDto dto)
  1101. {
  1102. var questions = await _knowledgeQuestionsRepository.GetAsync(dto.Id, HttpContext.RequestAborted);
  1103. if (questions is null)
  1104. throw UserFriendlyException.SameMessage("无效知识提问");
  1105. _mapper.Map(dto, questions);
  1106. questions.ReplyTime = DateTime.Now;
  1107. questions.ReplyUserName = _sessionContext.UserName;
  1108. questions.State = ECorrectionState.AlreadyAnswered;
  1109. await _knowledgeQuestionsRepository.UpdateAsync(questions, HttpContext.RequestAborted);
  1110. }
  1111. /// <summary>
  1112. /// 获取知识提问列表
  1113. /// </summary>
  1114. /// <param name="dto"></param>
  1115. /// <returns></returns>
  1116. [Permission(EPermission.KnowledgeQuestionsList)]
  1117. [HttpGet("knowledge_questions/list")]
  1118. public async Task<PagedDto<KnowledgeQuestionsDto>> List([FromQuery] KnowledgeQuestionsListDto dto)
  1119. {
  1120. var typeSpliceName = string.Empty;
  1121. if (!string.IsNullOrEmpty(dto.KnowledgeTypeId))
  1122. {
  1123. var type = await _knowledgeTypeRepository.GetAsync(x => x.Id == dto.KnowledgeTypeId);
  1124. typeSpliceName = type?.SpliceName;
  1125. }
  1126. var (total, items) = await _knowledgeQuestionsRepository.Queryable()
  1127. .Includes(x => x.Knowledge)
  1128. //.WhereIF(!string.IsNullOrEmpty(dto.KnowledgeTypeId), x => x.Knowledge.KnowledgeTypeId == dto.KnowledgeTypeId!)
  1129. //.WhereIF(!string.IsNullOrEmpty(typeSpliceName), x => SqlFunc.JsonLike(x.Knowledge.KnowledgeType, typeSpliceName))
  1130. .WhereIF(!string.IsNullOrEmpty(typeSpliceName), x => x.Knowledge.KnowledgeType.Any(t => t.KnowledgeTypeSpliceName.EndsWith(typeSpliceName)))
  1131. .WhereIF(!string.IsNullOrEmpty(dto.CreatorName), x => x.CreatorName == dto.CreatorName!)
  1132. .Where(x => !string.IsNullOrEmpty(x.Knowledge.Id))
  1133. .OrderByDescending(x => x.CreationTime)
  1134. .ToPagedListAsync(dto.PageIndex, dto.PageSize, HttpContext.RequestAborted);
  1135. return new PagedDto<KnowledgeQuestionsDto>(total, _mapper.Map<IReadOnlyList<KnowledgeQuestionsDto>>(items));
  1136. }
  1137. /// <summary>
  1138. /// 获取知识提问
  1139. /// </summary>
  1140. /// <param name="id"></param>
  1141. /// <returns></returns>
  1142. //[Permission(EPermission.KnowledgeQuestionsEntity)]
  1143. [HttpGet("knowledge_questions/{id}")]
  1144. public async Task<KnowledgeQuestions> QuestionsEntity(string id)
  1145. {
  1146. return await _knowledgeQuestionsRepository.Queryable()
  1147. .Includes(x => x.Knowledge)
  1148. .FirstAsync(x => x.Id == id);
  1149. }
  1150. #endregion
  1151. #region 知识收藏
  1152. /// <summary>
  1153. /// 知识收藏列表
  1154. /// </summary>
  1155. /// <param name="dto"></param>
  1156. /// <returns></returns>
  1157. [Permission(EPermission.KnowledgeCollectList)]
  1158. [HttpGet("knowledge_collect/list")]
  1159. public async Task<PagedDto<KnowledgeCollectDto>> List([FromQuery] KnowledgeCollectListDto dto)
  1160. {
  1161. var (total, items) = await _knowledgeCollectRepository.Queryable()
  1162. .Includes(x => x.Knowledge)
  1163. .WhereIF(!string.IsNullOrEmpty(dto.Keyword), x => x.Knowledge.Title.Contains(dto.Keyword!) || (x.Knowledge.Summary != null && x.Knowledge.Summary.Contains(dto.Keyword!)))
  1164. .WhereIF(dto.KnowledgeCollectGroupId.NotNullOrEmpty(), x => x.KnowledgeCollectGroupId == dto.KnowledgeCollectGroupId)
  1165. .Where(x => x.CreatorId == _sessionContext.UserId)
  1166. .Where(x => !string.IsNullOrEmpty(x.Knowledge.Id))
  1167. .Where(x => x.Collect!.Value)
  1168. .OrderByDescending(x => x.CreationTime)
  1169. .ToPagedListAsync(dto.PageIndex, dto.PageSize, HttpContext.RequestAborted);
  1170. return new PagedDto<KnowledgeCollectDto>(total, _mapper.Map<IReadOnlyList<KnowledgeCollectDto>>(items));
  1171. }
  1172. /// <summary>
  1173. /// 新增知识收藏
  1174. /// </summary>
  1175. /// <param name="dtos"></param>
  1176. /// <returns></returns>
  1177. //[Permission(EPermission.AddKnowledgeCollect)]
  1178. [HttpPost("knowledge_collect")]
  1179. [LogFilter("知识收藏")]
  1180. public async Task Add([FromBody] KnowledgeCollectAddDto dto)
  1181. {
  1182. var collect = await _knowledgeCollectRepository.GetAsync(x => x.KnowledgeId == dto.KnowledgeId && x.CreatorId == _sessionContext.UserId);
  1183. if (collect != null)
  1184. {
  1185. collect.Collect = dto.Collect;
  1186. await _knowledgeCollectRepository.UpdateAsync(collect, HttpContext.RequestAborted);
  1187. }
  1188. else
  1189. {
  1190. var collectNew = _mapper.Map<KnowledgeCollect>(dto);
  1191. await _knowledgeCollectRepository.AddAsync(collectNew, HttpContext.RequestAborted);
  1192. }
  1193. }
  1194. /// <summary>
  1195. /// 知识评分
  1196. /// </summary>
  1197. /// <param name="dto"></param>
  1198. /// <returns></returns>
  1199. //[Permission(EPermission.AddKnowledgeScore)]
  1200. [HttpDelete("knowledge_score")]
  1201. [LogFilter("知识评分")]
  1202. public async Task Delete([FromBody] KnowledgeCollectAddDto dto)
  1203. {
  1204. var collect = await _knowledgeCollectRepository.GetAsync(x => x.KnowledgeId == dto.KnowledgeId && x.CreatorId == _sessionContext.UserId);
  1205. if (collect != null)
  1206. {
  1207. if (collect.Score > 0)
  1208. throw UserFriendlyException.SameMessage("当前知识已经评分");
  1209. collect.Score = dto.Score;
  1210. await _knowledgeCollectRepository.UpdateAsync(collect, HttpContext.RequestAborted);
  1211. }
  1212. else
  1213. {
  1214. var questions = _mapper.Map<KnowledgeCollect>(dto);
  1215. await _knowledgeCollectRepository.AddAsync(questions, HttpContext.RequestAborted);
  1216. }
  1217. //计算总分
  1218. var sugar = _knowledgeCollectRepository.Queryable().Where(x => x.KnowledgeId == dto.KnowledgeId);
  1219. var count = await sugar.CountAsync();
  1220. var collects = await sugar.SumAsync(x => x.Score);
  1221. var score = collects / count;
  1222. var knowledge = await _knowledgeRepository.GetAsync(x => x.Id == dto.KnowledgeId);
  1223. if (knowledge != null)
  1224. {
  1225. knowledge.Score = decimal.Round(score.Value, 1);
  1226. await _knowledgeRepository.UpdateAsync(knowledge, HttpContext.RequestAborted);
  1227. }
  1228. }
  1229. #endregion
  1230. #region 知识评论
  1231. /// <summary>
  1232. /// 新增知识评论
  1233. /// </summary>
  1234. /// <param name="dto"></param>
  1235. /// <returns></returns>
  1236. [HttpPost("knowledge_comment")]
  1237. public async Task Add([FromBody] KnowledgeCommentAddDto dto)
  1238. {
  1239. var model = _mapper.Map<KnowledgeComment>(dto);
  1240. await _knowledgeCommentRepository.AddAsync(model, HttpContext.RequestAborted);
  1241. if (!string.IsNullOrEmpty(dto.ReplyId))
  1242. {
  1243. var comment = await _knowledgeCommentRepository.GetAsync(dto.ReplyId);
  1244. if (comment != null)
  1245. {
  1246. comment.ReplyNum++;
  1247. await _knowledgeCommentRepository.UpdateAsync(comment, HttpContext.RequestAborted);
  1248. }
  1249. }
  1250. }
  1251. /// <summary>
  1252. /// 删除知识评论
  1253. /// </summary>
  1254. /// <param name="dto"></param>
  1255. /// <returns></returns>
  1256. [HttpDelete("knowledge_comment")]
  1257. public async Task Delete([FromBody] KnowledgeCommentDeleteDto dto)
  1258. {
  1259. var comment = await _knowledgeCommentRepository.GetAsync(dto.Id, HttpContext.RequestAborted);
  1260. if (comment is null)
  1261. throw UserFriendlyException.SameMessage("无效评论");
  1262. if (comment.CreatorId != _sessionContext.UserId)
  1263. throw UserFriendlyException.SameMessage("只有评论者可以删除当前评论");
  1264. await _knowledgeCommentRepository.RemoveAsync(x => x.Id == dto.Id);
  1265. }
  1266. /// <summary>
  1267. /// 修改知识评论
  1268. /// </summary>
  1269. /// <param name="dto"></param>
  1270. /// <returns></returns>
  1271. [HttpPut("knowledge_comment")]
  1272. public async Task Update([FromBody] KnowledgeCommentUpdateDto dto)
  1273. {
  1274. var comment = await _knowledgeCommentRepository.GetAsync(dto.Id, HttpContext.RequestAborted);
  1275. if (comment is null)
  1276. throw UserFriendlyException.SameMessage("无效评论");
  1277. _mapper.Map(dto, comment);
  1278. await _knowledgeCommentRepository.UpdateAsync(comment, HttpContext.RequestAborted);
  1279. }
  1280. /// <summary>
  1281. /// 知识评论列表
  1282. /// </summary>
  1283. /// <param name="dto"></param>
  1284. /// <returns></returns>
  1285. [HttpGet("knowledge_comment/list")]
  1286. public async Task<List<KnowledgeCommentDto>> List([FromQuery] KnowledgeCommentListDto dto)
  1287. {
  1288. var comments = await _knowledgeCommentRepository.Queryable()
  1289. .WhereIF(!string.IsNullOrEmpty(dto.KnowledgeId), x => x.KnowledgeId == dto.KnowledgeId)
  1290. .WhereIF(!string.IsNullOrEmpty(dto.ReplyId), x => x.ReplyId == dto.ReplyId)
  1291. .WhereIF(dto.All.HasValue && dto.All == false, x => x.CreatorId == _sessionContext.UserId)
  1292. .OrderByDescending(x => x.CreationTime)
  1293. .ToListAsync();
  1294. return new List<KnowledgeCommentDto>(_mapper.Map<IReadOnlyList<KnowledgeCommentDto>>(comments));
  1295. }
  1296. #endregion
  1297. #region 收藏分组
  1298. /// <summary>
  1299. /// 增加收藏分组
  1300. /// </summary>
  1301. [HttpPost("group")]
  1302. public async Task<KnowledgeCollectGroupOutDto> SaveKnowledgeCoolectGroupAsync([FromBody] KnowledgeCollectGroupInDto dto)
  1303. {
  1304. var entity = dto.Adapt<KnowledgeCollectGroup>();
  1305. if (await _knowledgeCollectGroupRepository
  1306. .Queryable()
  1307. .Where(m => m.Name == dto.Name && m.CreatorId == _sessionContext.UserId)
  1308. .AnyAsync() == true) throw UserFriendlyException.SameMessage("分组名重复, 请重新输入!");
  1309. var key = await _knowledgeCollectGroupRepository.AddAsync(entity);
  1310. if (key.IsNullOrEmpty()) throw UserFriendlyException.SameMessage("添加失败, 请重试");
  1311. var outDto = dto.Adapt<KnowledgeCollectGroupOutDto>();
  1312. outDto.Id = key;
  1313. return outDto;
  1314. }
  1315. /// <summary>
  1316. /// 删除收藏分组
  1317. /// </summary>
  1318. [HttpDelete("group")]
  1319. public async Task DeleteKnowledgeCoolectGroupAsync([FromQuery] string id)
  1320. {
  1321. var entity = await _knowledgeCollectGroupRepository.Queryable()
  1322. .Where(m => m.Id == id && m.CreatorId == _sessionContext.UserId)
  1323. .FirstAsync() ?? throw new UserFriendlyException("数据不存在");
  1324. entity.IsDeleted = true;
  1325. await _knowledgeCollectGroupRepository.UpdateAsync(entity);
  1326. await _knowledgeCollectRepository.Updateable()
  1327. .Where(m => m.KnowledgeCollectGroupId == id && m.CreatorId == _sessionContext.UserId)
  1328. .SetColumns(m => m.IsDeleted == true)
  1329. .ExecuteCommandAsync();
  1330. }
  1331. /// <summary>
  1332. /// 获取分组收藏列表
  1333. /// </summary>
  1334. /// <returns></returns>
  1335. [HttpGet("group")]
  1336. public async Task<PagedDto<KnowledgeCollectGroupOutDto>> GetKnowledgeCollectGroupListAsync([FromQuery] KnowledgeCollectGroupListInDto dto)
  1337. {
  1338. return (await _knowledgeCollectGroupRepository.Queryable()
  1339. .WhereIF(dto.Keyword.NotNullOrEmpty(), m => m.Name.Contains(dto.Keyword))
  1340. .Where(m => m.CreatorId == _sessionContext.UserId)
  1341. .Select<KnowledgeCollectGroupOutDto>()
  1342. .ToPagedListAsync(dto.PageIndex, dto.PageSize, HttpContext.RequestAborted)
  1343. ).ToPaged();
  1344. }
  1345. #endregion
  1346. #region PageView 浏览记录
  1347. /// <summary>
  1348. /// 浏览记录集合
  1349. /// </summary>
  1350. /// <param name="dto"></param>
  1351. /// <returns></returns>
  1352. [HttpGet("pageview")]
  1353. public async Task<PagedDto<PageViewOutDto>> GetPageViewListAsync([FromQuery] PageViewInDto dto)
  1354. {
  1355. return (await _knowApplication.GetPageViewListAsync(dto, HttpContext.RequestAborted))
  1356. .ToPaged();
  1357. }
  1358. /// <summary>
  1359. /// 浏览记录集合-导出
  1360. /// </summary>
  1361. /// <param name="dto"></param>
  1362. /// <returns></returns>
  1363. [HttpPost("pageview/export")]
  1364. public async Task<FileStreamResult> GetPageViewListAsync([FromBody] ExportExcelDto<PageViewInDto> dto)
  1365. {
  1366. var items = (await _knowApplication.GetPageViewListAsync(dto.QueryDto, HttpContext.RequestAborted))
  1367. .Item2;
  1368. return _exportApplication.GetExcelFile(dto, items, "浏览记录");
  1369. }
  1370. #endregion
  1371. #region 热词
  1372. /// <summary>
  1373. /// 新增热词
  1374. /// </summary>
  1375. /// <param name="dto"></param>
  1376. /// <returns></returns>
  1377. [HttpPost("hotword")]
  1378. public async Task AddKnowledgeHotWordAsync([FromBody] AddKnowledgeHotWordInDto dto)
  1379. {
  1380. await _knowApplication.AddKnowledgeHotWordAsync(dto, HttpContext.RequestAborted);
  1381. }
  1382. /// <summary>
  1383. /// 热词集合
  1384. /// </summary>
  1385. /// <param name="dto"></param>
  1386. /// <returns></returns>
  1387. [HttpGet("hotword")]
  1388. public async Task<PagedDto<KnowledgeHotWordOutDto>> GetKnowledgeHotWordListAsync([FromQuery] KnowledgeHotWordInDto dto)
  1389. {
  1390. return (await _knowApplication.GetKnowledgeHotWordListAsync(dto, HttpContext.RequestAborted))
  1391. .ToPaged();
  1392. }
  1393. /// <summary>
  1394. /// 修改热词
  1395. /// </summary>
  1396. /// <param name="dto"></param>
  1397. /// <returns></returns>
  1398. [HttpPut("hotword")]
  1399. public async Task UpdateKnowledgeHotWordAsync([FromBody] UpdateKnowledgeHotWordInDto dto)
  1400. {
  1401. await _knowApplication.UpdateKnowledgeHotWordAsync(dto, HttpContext.RequestAborted);
  1402. }
  1403. /// <summary>
  1404. /// 删除热词
  1405. /// </summary>
  1406. /// <param name="id"></param>
  1407. /// <returns></returns>
  1408. [HttpDelete("hotword")]
  1409. public async Task DeleteKnowledgeHotWordAsync([FromQuery] string id)
  1410. {
  1411. await _knowledgeHotWordRepository
  1412. .RemoveAsync(id);
  1413. }
  1414. #endregion
  1415. }
  1416. }