KnowledgeController.cs 72 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572
  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<Dictionary<string, dynamic>> 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. if (dto.RetrievalType == EKnowledgeRetrievalType.KeyWord)
  626. {
  627. var keywordEntity = await _knowledgeWordRepository.GetAsync(m => m.Tag == keyword && m.IsEnable == 0);
  628. if (keywordEntity is null) continue;
  629. exp.Or(m => SqlFunc.JsonArrayAny(m.Keywords, keywordEntity.Id));
  630. }
  631. }
  632. sugar.Where(exp.ToExpression());
  633. }
  634. switch (dto.Sort)
  635. {
  636. case "2":
  637. sugar = sugar.OrderByDescending(p => p.CollectCount);
  638. break;
  639. case "3":
  640. sugar = sugar.OrderByDescending(p => p.CreationTime);
  641. break;
  642. default:
  643. sugar = sugar.OrderByDescending(p => p.PageView);
  644. break;
  645. }
  646. var (total, temp) = await sugar.ToPagedListAsync(dto.PageIndex, dto.PageSize);
  647. return new PagedDto<KnowledgeRetrievalDataDto>(total, _mapper.Map<IReadOnlyList<KnowledgeRetrievalDataDto>>(temp));
  648. }
  649. /// <summary>
  650. /// 知识检索页面基础数据
  651. /// </summary>
  652. /// <returns></returns>
  653. [HttpGet("knowretrieval/base_data")]
  654. public async Task<Dictionary<string, dynamic>> GetKnowretrievalBaseData()
  655. {
  656. return _baseDataApplication
  657. .KnowledgeRetrievalType(new[] { 3, 4})
  658. .Build();
  659. }
  660. /// <summary>
  661. /// 获取知识审批信息
  662. /// </summary>
  663. /// <param name="id"></param>
  664. /// <returns></returns>
  665. [HttpGet("audit_log")]
  666. public async Task<PagedDto<KnowledgeWorkFlowDto>> KnowRetrieval([FromQuery] AuditLogListPagedDto pagedDto)
  667. {
  668. var (total, temp) = await _knowledgeWorkFlowRepository
  669. .Queryable()
  670. .Includes(x => x.User)
  671. .Includes(x => x.SystemOrganize)
  672. .Includes(x => x.Workflow)
  673. .Where(x => x.KnowledgeId == pagedDto.id)
  674. .Where(x => x.IsDeleted == false)
  675. .OrderBy(x => x.CreationTime)
  676. .ToPagedListAsync(pagedDto.PageIndex, pagedDto.PageSize);
  677. return new PagedDto<KnowledgeWorkFlowDto>(total, _mapper.Map<IReadOnlyList<KnowledgeWorkFlowDto>>(temp));
  678. }
  679. /// <summary>
  680. /// 知识查重
  681. /// </summary>
  682. /// <param name="id"></param>
  683. /// <returns></returns>
  684. [HttpPost("finding_duplicate")]
  685. public async Task<bool> FindingDuplicate([FromBody] KnowledgeFindingDuplicateDto dto)
  686. {
  687. var any = await _knowledgeRepository.Queryable().Where(x => x.Status == EKnowledgeStatus.Auditing || x.Status >= EKnowledgeStatus.OnShelf)
  688. .WhereIF(!string.IsNullOrEmpty(dto.Title), x => x.Title.Equals(dto.Title))
  689. .WhereIF(!string.IsNullOrEmpty(dto.Summary), x => x.Summary.Equals(dto.Summary))
  690. .WhereIF(!string.IsNullOrEmpty(dto.Content), x => x.Content.Equals(dto.Content))
  691. .WhereIF(!string.IsNullOrEmpty(dto.Id), x => x.Id != dto.Id)
  692. .AnyAsync();
  693. return any;
  694. }
  695. #endregion
  696. #region 我的知识删除列表
  697. /// <summary>
  698. /// 我的知识删除列表页面枚举值
  699. /// </summary>
  700. /// <returns></returns>
  701. [HttpGet("delete-status-data")]
  702. public async Task<object> DeleteApplyStatus()
  703. {
  704. return EnumExts.GetDescriptions<EKnowledgeWorkFlowStatus>();
  705. }
  706. /// <summary>
  707. /// 我的知识删除列表
  708. /// </summary>
  709. /// <param name="pagedDto"></param>
  710. /// <returns></returns>
  711. [HttpGet("deletelist")]
  712. public async Task<PagedDto<KnowledgeDeleteApplyDataDto>> GetDeleteApplyList([FromQuery] KnowledgeDeletelPagedListDto pagedDto)
  713. {
  714. var (total, items) = await _knowledgeWorkFlowRepository
  715. .Queryable(includeDeleted: true)
  716. .Includes(it => it.Knowledge)
  717. .Includes(it => it.User)
  718. .Includes(it => it.SystemOrganize)
  719. .Includes(it => it.Knowledge, it => it.KnowledgeType)
  720. .Includes(it => it.Knowledge, it => it.HotspotType)
  721. .Includes(it => it.Workflow)
  722. .Where(d => d.CreatorId == _sessionContext.RequiredUserId && d.WorkflowModuleStatus == EKnowledgeApplyType.Delete && d.WorkflowId != null)
  723. .WhereIF(pagedDto.EKnowledgeWorkFlowStatus.HasValue, d => d.WorkFlowApplyStatus == pagedDto.EKnowledgeWorkFlowStatus)
  724. .WhereIF(pagedDto.StartApplyTime.HasValue, d => d.CreationTime >= pagedDto.StartApplyTime)
  725. .WhereIF(pagedDto.EndApplyTime.HasValue, d => d.CreationTime <= pagedDto.EndApplyTime)
  726. .WhereIF(!string.IsNullOrEmpty(pagedDto.Keyword), d => d.Knowledge.User.Name.Contains(pagedDto.Keyword!)
  727. || d.Knowledge.SystemOrganize.Name.Contains(pagedDto.Keyword!)
  728. || d.Knowledge.Title.Contains(pagedDto.Keyword!))
  729. .OrderByDescending(p => p.CreationTime)
  730. .ToPagedListAsync(pagedDto.PageIndex, pagedDto.PageSize, HttpContext.RequestAborted);
  731. return new PagedDto<KnowledgeDeleteApplyDataDto>(total, _mapper.Map<IReadOnlyList<KnowledgeDeleteApplyDataDto>>(items));
  732. }
  733. /// <summary>
  734. /// 审核管理页面枚举值
  735. /// </summary>
  736. /// <returns></returns>
  737. [HttpGet("approval-base-data")]
  738. public async Task<object> ApprovalBaseData()
  739. {
  740. return new
  741. {
  742. EKnowledgeWorkFlowStatus = EnumExts.GetDescriptions<EKnowledgeWorkFlowStatus>(),
  743. EKnowledgeApplyType = EnumExts.GetDescriptions<EKnowledgeApplyType>()
  744. };
  745. }
  746. /// <summary>
  747. /// 审核管理
  748. /// </summary>
  749. /// <param name="pagedDto"></param>
  750. /// <returns></returns>
  751. [HttpGet("approvedlist")]
  752. public async Task<PagedDto<KnowledgeApprovalDataDto>> ApprovedList([FromQuery] KnowledgeApprovalPagedListDto pagedDto)
  753. {
  754. var (total, items) = await _knowledgeWorkFlowRepository
  755. .Queryable(includeDeleted: true)
  756. .Includes(it => it.Knowledge)
  757. .Includes(it => it.User)
  758. .Includes(it => it.SystemOrganize)
  759. .Includes(it => it.Workflow, d => d.Steps)
  760. .Where(it => it.WorkflowId != null)
  761. .WhereIF(pagedDto.EKnowledgeApplyType.HasValue, d => d.WorkflowModuleStatus == pagedDto.EKnowledgeApplyType)
  762. .WhereIF(pagedDto.EKnowledgeWorkFlowStatus.HasValue, d => d.WorkFlowApplyStatus == pagedDto.EKnowledgeWorkFlowStatus)
  763. .WhereIF(!string.IsNullOrEmpty(pagedDto.Keyword), d => d.Knowledge.User.Name.Contains(pagedDto.Keyword!)
  764. || d.Knowledge.SystemOrganize.Name.Contains(pagedDto.Keyword!)
  765. || d.Knowledge.Title.Contains(pagedDto.Keyword!))
  766. .OrderByDescending(p => p.CreationTime)
  767. .ToPagedListAsync(pagedDto.PageIndex, pagedDto.PageSize, HttpContext.RequestAborted);
  768. foreach (var item in items)
  769. {
  770. if (item.Workflow != null)
  771. //item.CanHandle = item.CanHandle(_sessionContext.RequiredUserId, _sessionContext.RequiredOrgId);
  772. item.CanHandle = item.Workflow.IsCanHandle(
  773. _sessionContext.RequiredUserId, _sessionContext.RequiredOrgId, _sessionContext.Roles);
  774. }
  775. //处理是否可以办理
  776. //items.ForEach(d => d.CanHandle = d.Workflow.CanHandle(_sessionContext.RequiredUserId, _sessionContext.RequiredOrgCode));
  777. return new PagedDto<KnowledgeApprovalDataDto>(total, _mapper.Map<IReadOnlyList<KnowledgeApprovalDataDto>>(items));
  778. }
  779. /// <summary>
  780. /// 工单受理知识检索
  781. /// </summary>
  782. /// <param name="pagedDto"></param>
  783. /// <returns></returns>
  784. [HttpGet("knowpopscreen")]
  785. public async Task<PagedDto<KnowledgeRetrievalDataDto>> KnowPopScreen([FromQuery] KnowledgePopScreenPagedListDto pagedDto)
  786. {
  787. var orgid = string.Empty;
  788. if (pagedDto.RetrievalType == EKnowledgeRetrievalType.Org && !string.IsNullOrEmpty(pagedDto.Keyword))
  789. {
  790. var organize = await _systemOrganizeRepository.GetAsync(x => x.Name == pagedDto.Keyword);
  791. orgid = organize?.Id;
  792. }
  793. var (total, temp) = await _knowledgeRepository.Queryable()
  794. .Includes(x => x.SourceOrganize)
  795. .Where(d => d.Status == EKnowledgeStatus.OnShelf)
  796. .WhereIF(pagedDto.RetrievalType == EKnowledgeRetrievalType.Title && !string.IsNullOrEmpty(pagedDto.Keyword), d => d.Title.Contains(pagedDto.Keyword!))
  797. .WhereIF(pagedDto.RetrievalType == EKnowledgeRetrievalType.Content && !string.IsNullOrEmpty(pagedDto.Keyword), d => d.Content.Contains(pagedDto.Keyword!))
  798. .WhereIF(!string.IsNullOrEmpty(orgid) && pagedDto.RetrievalType == EKnowledgeRetrievalType.Org, x => x.CreatorOrgId.EndsWith(orgid!))
  799. .WhereIF(!string.IsNullOrEmpty(pagedDto.HotspotId), p => p.HotspotId == pagedDto.HotspotId)
  800. .OrderByDescending(p => p.CreationTime)
  801. .ToPagedListAsync(pagedDto.PageIndex, pagedDto.PageSize);
  802. return new PagedDto<KnowledgeRetrievalDataDto>(total, _mapper.Map<IReadOnlyList<KnowledgeRetrievalDataDto>>(temp));
  803. }
  804. ///// <summary>
  805. ///// 新增-开始流程
  806. ///// </summary>
  807. ///// <param name="id">知识id</param>
  808. ///// <param name="dto">流程开启参数</param>
  809. ///// <returns></returns>
  810. //[Permission(EPermission.AddKnowledge)]
  811. //[HttpPost("{id}/add-startflow")]
  812. //public async Task AddStartFlow(string id, [FromBody] StartWorkflowDto dto)
  813. //{
  814. // await StartFlow(id, WorkflowModuleConsts.KnowledgeAdd, EKnowledgeApplyType.Add, dto);
  815. //}
  816. ///// <summary>
  817. ///// 删除-开始流程
  818. ///// </summary>
  819. ///// <param name="id">知识id</param>
  820. ///// <param name="dto">流程开启参数</param>
  821. ///// <returns></returns>
  822. //[Permission(EPermission.KnowledgeDelete)]
  823. //[HttpPost("{id}/remove-startflow")]
  824. //public async Task RemoveStartFlow(string id, [FromBody] StartWorkflowDto dto)
  825. //{
  826. // await StartFlow(id, WorkflowModuleConsts.KnowledgeDelete, EKnowledgeApplyType.Delete, dto);
  827. //}
  828. /// <summary>
  829. /// 查询知识库办理流程开启参数-新增
  830. /// </summary>
  831. /// <returns></returns>
  832. //[Permission(EPermission.AddKnowledge)]
  833. [HttpGet("add-flow-start")]
  834. public async Task<NextStepsDto> GetAddFlowStartOptionsAsync()
  835. {
  836. return await _workflowApplication.GetStartStepsAsync(WorkflowModuleConsts.KnowledgeAdd,
  837. HttpContext.RequestAborted);
  838. }
  839. /// <summary>
  840. /// 查询知识库办理流程开启参数-新增
  841. /// </summary>
  842. /// <returns></returns>
  843. //[Permission(EPermission.AddKnowledge)]
  844. [HttpGet("update-flow-start")]
  845. public async Task<NextStepsDto> GetUpdateFlowStartOptionsAsync()
  846. {
  847. return await _workflowApplication.GetStartStepsAsync(WorkflowModuleConsts.KnowledgeUpdate,
  848. HttpContext.RequestAborted);
  849. }
  850. /// <summary>
  851. /// 查询知识库办理流程开启参数-删除
  852. /// </summary>
  853. /// <returns></returns>
  854. //[Permission(EPermission.KnowledgeDelete)]
  855. [HttpGet("remove-flow-start")]
  856. public async Task<NextStepsDto> GetRemoveFlowStartOptionsAsync()
  857. {
  858. return await _workflowApplication.GetStartStepsAsync(WorkflowModuleConsts.KnowledgeDelete,
  859. HttpContext.RequestAborted);
  860. }
  861. /// <summary>
  862. /// 开始流程
  863. /// </summary>
  864. /// <param name="id">知识ID</param>
  865. /// <param name="moduleCode">知识模板编号</param>
  866. /// <param name="eKnowledgeApplyType">申请类型</param>
  867. /// <param name="dto">流程开启参数</param>
  868. /// <returns></returns>
  869. private async Task<string> StartFlow(string id, string moduleCode, EKnowledgeApplyType eKnowledgeApplyType, StartWorkflowDto dto)
  870. {
  871. var knowledge = await _knowledgeRepository.GetAsync(id, HttpContext.RequestAborted);
  872. if (knowledge == null)
  873. throw UserFriendlyException.SameMessage("无效知识编号");
  874. if (eKnowledgeApplyType == EKnowledgeApplyType.Delete)
  875. {
  876. if (knowledge.IsDeleted == true)
  877. throw UserFriendlyException.SameMessage("知识删除失败");
  878. }
  879. //知识审批主表
  880. await _knowledgeDomainService.AddWorkFlowAsync(id, eKnowledgeApplyType, HttpContext.RequestAborted);
  881. dto.DefinitionModuleCode = moduleCode;
  882. dto.Title = knowledge.Title;
  883. return await _workflowApplication.StartWorkflowAsync(dto, id, cancellationToken: HttpContext.RequestAborted);
  884. }
  885. #endregion
  886. #region 知识库词库
  887. /// <summary>
  888. /// 新增知识库词库
  889. /// </summary>
  890. /// <param name="dtos"></param>
  891. /// <returns></returns>
  892. [Permission(EPermission.AddKnowledgeWord)]
  893. [HttpPost("knowledge_word")]
  894. [LogFilter("新增知识库词库")]
  895. public async Task Add([FromBody] KnowledgeWordAddDto dto)
  896. {
  897. var word = _mapper.Map<KnowledgeWord>(dto);
  898. await _knowledgeWrodRepository.AddAsync(word, HttpContext.RequestAborted);
  899. }
  900. /// <summary>
  901. /// 删除知识库词库
  902. /// </summary>
  903. /// <param name="dto"></param>
  904. /// <returns></returns>
  905. [Permission(EPermission.DeleteKnowledgeWord)]
  906. [HttpDelete("knowledge_word")]
  907. [LogFilter("删除知识库词库")]
  908. public async Task Delete([FromBody] KnowledgeWordDeleteDto dto)
  909. {
  910. await _knowledgeRepository.RemoveKnowledgeWrodBatchAsync(dto.Ids, HttpContext.RequestAborted);
  911. }
  912. /// <summary>
  913. /// 更新知识库词库
  914. /// </summary>
  915. /// <param name="dto"></param>
  916. /// <returns></returns>
  917. [Permission(EPermission.UpdateKnowledgeWord)]
  918. [HttpPut("knowledge_word")]
  919. [LogFilter("更新知识库词库")]
  920. public async Task Update([FromBody] KnowledgeWordUpdateDto dto)
  921. {
  922. var word = await _knowledgeWrodRepository.GetAsync(dto.Id, HttpContext.RequestAborted);
  923. if (word is null)
  924. throw UserFriendlyException.SameMessage("无效知识库词库");
  925. _mapper.Map(dto, word);
  926. word.LastModificationName = _sessionContext.UserName;
  927. await _knowledgeWrodRepository.UpdateAsync(word, HttpContext.RequestAborted);
  928. }
  929. /// <summary>
  930. /// 获取知识库词库列表
  931. /// </summary>
  932. /// <param name="dto"></param>
  933. /// <returns></returns>
  934. [HttpGet("knowledge_word/list")]
  935. public async Task<PagedDto<KnowledgeWordDto>> List([FromQuery] KnowledgeWordListDto dto)
  936. {
  937. var (total, items) = await _knowledgeWrodRepository.Queryable()
  938. .WhereIF(!string.IsNullOrEmpty(dto.Tag), x => x.Tag == dto.Tag!)
  939. .WhereIF(!string.IsNullOrEmpty(dto.Classify), x => x.Classify == dto.Classify!)
  940. .WhereIF(!string.IsNullOrEmpty(dto.Synonym), x => x.Synonym != null && x.Synonym.Contains(dto.Synonym!))
  941. .OrderByDescending(x => x.CreationTime)
  942. .ToPagedListAsync(dto.PageIndex, dto.PageSize, HttpContext.RequestAborted);
  943. return new PagedDto<KnowledgeWordDto>(total, _mapper.Map<IReadOnlyList<KnowledgeWordDto>>(items));
  944. }
  945. /// <summary>
  946. /// 获取知识库词库
  947. /// </summary>
  948. /// <param name="id"></param>
  949. /// <returns></returns>
  950. [HttpGet("knowledge_word/{id}")]
  951. public async Task<KnowledgeWord> WordEntity(string id)
  952. {
  953. return await _knowledgeWrodRepository.Queryable()
  954. .FirstAsync(x => x.Id == id);
  955. }
  956. /// <summary>
  957. /// 获取知识库词库基本信息
  958. /// </summary>
  959. /// <returns></returns>
  960. [HttpGet("knowledge_word/base")]
  961. public async Task<object> Base()
  962. {
  963. var rsp = new
  964. {
  965. KnowledgeWordClassify = await _systemDomainService.GetSysDicDataByCodeAsync(SysDicTypeConsts.KnowledgeWordClassify),
  966. };
  967. return rsp;
  968. }
  969. #endregion
  970. #region 知识纠错
  971. /// <summary>
  972. /// 新增知识纠错
  973. /// </summary>
  974. /// <param name="dtos"></param>
  975. /// <returns></returns>
  976. //[Permission(EPermission.AddKnowledgeCorrection)]
  977. [HttpPost("knowledge_correction")]
  978. [LogFilter("新增知识纠错")]
  979. public async Task Add([FromBody] KnowledgeCorrectionAddDto dto)
  980. {
  981. var correction = _mapper.Map<KnowledgeCorrection>(dto);
  982. await _knowledgeCorrectionRepository.AddAsync(correction, HttpContext.RequestAborted);
  983. }
  984. /// <summary>
  985. /// 删除知识纠错
  986. /// </summary>
  987. /// <param name="dto"></param>
  988. /// <returns></returns>
  989. //[Permission(EPermission.DeleteKnowledgeCorrection)]
  990. //[HttpDelete("knowledge_correction")]
  991. //public async Task Delete([FromBody] KnowledgeCorrectionDeleteDto dto)
  992. //{
  993. // await _knowledgeCorrectionRepository.RemoveAsync(x => x.Id == dto.Id);
  994. //}
  995. /// <summary>
  996. /// 更新知识纠错
  997. /// </summary>
  998. /// <param name="dto"></param>
  999. /// <returns></returns>
  1000. //[Permission(EPermission.UpdateKnowledgeCorrection)]
  1001. //[HttpPut("knowledge_correction")]
  1002. //public async Task Update([FromBody] KnowledgeCorrectionUpdateDto dto)
  1003. //{
  1004. // var correction = await _knowledgeCorrectionRepository.GetAsync(dto.Id, HttpContext.RequestAborted);
  1005. // if (correction is null)
  1006. // throw UserFriendlyException.SameMessage("无效知识纠错");
  1007. // _mapper.Map(dto, correction);
  1008. // await _knowledgeCorrectionRepository.UpdateAsync(correction, HttpContext.RequestAborted);
  1009. //}
  1010. /// <summary>
  1011. /// 答复知识纠错
  1012. /// </summary>
  1013. /// <param name="dto"></param>
  1014. /// <returns></returns>
  1015. [Permission(EPermission.ReplyKnowledgeCorrection)]
  1016. [HttpPut("knowledge_correction/Reply")]
  1017. [LogFilter("答复知识纠错")]
  1018. public async Task Reply([FromBody] KnowledgeCorrectionUpdateDto dto)
  1019. {
  1020. var correction = await _knowledgeCorrectionRepository.GetAsync(dto.Id, HttpContext.RequestAborted);
  1021. if (correction is null)
  1022. throw UserFriendlyException.SameMessage("无效知识纠错");
  1023. _mapper.Map(dto, correction);
  1024. correction.ReplyTime = DateTime.Now;
  1025. correction.ReplyUserName = _sessionContext.UserName;
  1026. correction.State = ECorrectionState.AlreadyAnswered;
  1027. await _knowledgeCorrectionRepository.UpdateAsync(correction, HttpContext.RequestAborted);
  1028. }
  1029. /// <summary>
  1030. /// 获取知识纠错列表
  1031. /// </summary>
  1032. /// <param name="dto"></param>
  1033. /// <returns></returns>
  1034. [Permission(EPermission.KnowledgeCorrectionList)]
  1035. [HttpGet("knowledge_correction/list")]
  1036. public async Task<PagedDto<KnowledgeCorrectionDto>> List([FromQuery] KnowledgeCorrectionListDto dto)
  1037. {
  1038. var typeSpliceName = string.Empty;
  1039. if (!string.IsNullOrEmpty(dto.KnowledgeTypeId))
  1040. {
  1041. var type = await _knowledgeTypeRepository.GetAsync(x => x.Id == dto.KnowledgeTypeId);
  1042. typeSpliceName = type?.SpliceName;
  1043. }
  1044. var (total, items) = await _knowledgeCorrectionRepository.Queryable()
  1045. .Includes(x => x.Knowledge)
  1046. //.WhereIF(!string.IsNullOrEmpty(dto.KnowledgeTypeId), x => x.Knowledge.KnowledgeTypeId == dto.KnowledgeTypeId!)
  1047. .WhereIF(!string.IsNullOrEmpty(dto.CreatorName), x => x.CreatorName == dto.CreatorName!)
  1048. //.WhereIF(!string.IsNullOrEmpty(typeSpliceName), x => SqlFunc.JsonLike(x.Knowledge.KnowledgeType, typeSpliceName))
  1049. .WhereIF(!string.IsNullOrEmpty(typeSpliceName), x => x.Knowledge.KnowledgeType.Any(t => t.KnowledgeTypeSpliceName.EndsWith(typeSpliceName)))
  1050. .Where(x => !string.IsNullOrEmpty(x.Knowledge.Id))
  1051. .OrderByDescending(x => x.CreationTime)
  1052. .ToPagedListAsync(dto.PageIndex, dto.PageSize, HttpContext.RequestAborted);
  1053. return new PagedDto<KnowledgeCorrectionDto>(total, _mapper.Map<IReadOnlyList<KnowledgeCorrectionDto>>(items));
  1054. }
  1055. /// <summary>
  1056. /// 获取知识库纠错
  1057. /// </summary>
  1058. /// <param name="id"></param>
  1059. /// <returns></returns>
  1060. //[Permission(EPermission.KnowledgeCorrectionEntity)]
  1061. [HttpGet("knowledge_correction/{id}")]
  1062. public async Task<KnowledgeCorrection> CorrectionEntity(string id)
  1063. {
  1064. return await _knowledgeCorrectionRepository.Queryable()
  1065. .Includes(x => x.Knowledge)
  1066. .FirstAsync(x => x.Id == id);
  1067. }
  1068. #endregion
  1069. #region 知识提问
  1070. /// <summary>
  1071. /// 新增知识提问
  1072. /// </summary>
  1073. /// <param name="dtos"></param>
  1074. /// <returns></returns>
  1075. //[Permission(EPermission.AddKnowledgeQuestions)]
  1076. [HttpPost("knowledge_questions")]
  1077. [LogFilter("新增知识提问")]
  1078. public async Task Add([FromBody] KnowledgeQuestionsAddDto dto)
  1079. {
  1080. var questions = _mapper.Map<KnowledgeQuestions>(dto);
  1081. await _knowledgeQuestionsRepository.AddAsync(questions, HttpContext.RequestAborted);
  1082. }
  1083. /// <summary>
  1084. /// 删除知识提问
  1085. /// </summary>
  1086. /// <param name="dto"></param>
  1087. /// <returns></returns>
  1088. //[Permission(EPermission.DeleteKnowledgeQuestions)]
  1089. //[HttpDelete("knowledge_questions")]
  1090. //public async Task Delete([FromBody] KnowledgeQuestionsDeleteDto dto)
  1091. //{
  1092. // await _knowledgeQuestionsRepository.RemoveAsync(x => x.Id == dto.Id);
  1093. //}
  1094. /// <summary>
  1095. /// 更新知识提问
  1096. /// </summary>
  1097. /// <param name="dto"></param>
  1098. /// <returns></returns>
  1099. //[Permission(EPermission.UpdateKnowledgeQuestions)]
  1100. //[HttpPut("knowledge_questions")]
  1101. //public async Task Update([FromBody] KnowledgeQuestionsUpdateDto dto)
  1102. //{
  1103. // var questions = await _knowledgeQuestionsRepository.GetAsync(dto.Id, HttpContext.RequestAborted);
  1104. // if (questions is null)
  1105. // throw UserFriendlyException.SameMessage("无效知识提问");
  1106. // _mapper.Map(dto, questions);
  1107. // await _knowledgeQuestionsRepository.UpdateAsync(questions, HttpContext.RequestAborted);
  1108. //}
  1109. /// <summary>
  1110. /// 答复知识提问
  1111. /// </summary>
  1112. /// <param name="dto"></param>
  1113. /// <returns></returns>
  1114. [Permission(EPermission.ReplyKnowledgeQuestions)]
  1115. [HttpPut("knowledge_questions/Reply")]
  1116. [LogFilter("答复知识提问")]
  1117. public async Task Reply([FromBody] KnowledgeQuestionsUpdateDto dto)
  1118. {
  1119. var questions = await _knowledgeQuestionsRepository.GetAsync(dto.Id, HttpContext.RequestAborted);
  1120. if (questions is null)
  1121. throw UserFriendlyException.SameMessage("无效知识提问");
  1122. _mapper.Map(dto, questions);
  1123. questions.ReplyTime = DateTime.Now;
  1124. questions.ReplyUserName = _sessionContext.UserName;
  1125. questions.State = ECorrectionState.AlreadyAnswered;
  1126. await _knowledgeQuestionsRepository.UpdateAsync(questions, HttpContext.RequestAborted);
  1127. }
  1128. /// <summary>
  1129. /// 获取知识提问列表
  1130. /// </summary>
  1131. /// <param name="dto"></param>
  1132. /// <returns></returns>
  1133. [Permission(EPermission.KnowledgeQuestionsList)]
  1134. [HttpGet("knowledge_questions/list")]
  1135. public async Task<PagedDto<KnowledgeQuestionsDto>> List([FromQuery] KnowledgeQuestionsListDto dto)
  1136. {
  1137. var typeSpliceName = string.Empty;
  1138. if (!string.IsNullOrEmpty(dto.KnowledgeTypeId))
  1139. {
  1140. var type = await _knowledgeTypeRepository.GetAsync(x => x.Id == dto.KnowledgeTypeId);
  1141. typeSpliceName = type?.SpliceName;
  1142. }
  1143. var (total, items) = await _knowledgeQuestionsRepository.Queryable()
  1144. .Includes(x => x.Knowledge)
  1145. //.WhereIF(!string.IsNullOrEmpty(dto.KnowledgeTypeId), x => x.Knowledge.KnowledgeTypeId == dto.KnowledgeTypeId!)
  1146. //.WhereIF(!string.IsNullOrEmpty(typeSpliceName), x => SqlFunc.JsonLike(x.Knowledge.KnowledgeType, typeSpliceName))
  1147. .WhereIF(!string.IsNullOrEmpty(typeSpliceName), x => x.Knowledge.KnowledgeType.Any(t => t.KnowledgeTypeSpliceName.EndsWith(typeSpliceName)))
  1148. .WhereIF(!string.IsNullOrEmpty(dto.CreatorName), x => x.CreatorName == dto.CreatorName!)
  1149. .Where(x => !string.IsNullOrEmpty(x.Knowledge.Id))
  1150. .OrderByDescending(x => x.CreationTime)
  1151. .ToPagedListAsync(dto.PageIndex, dto.PageSize, HttpContext.RequestAborted);
  1152. return new PagedDto<KnowledgeQuestionsDto>(total, _mapper.Map<IReadOnlyList<KnowledgeQuestionsDto>>(items));
  1153. }
  1154. /// <summary>
  1155. /// 获取知识提问
  1156. /// </summary>
  1157. /// <param name="id"></param>
  1158. /// <returns></returns>
  1159. //[Permission(EPermission.KnowledgeQuestionsEntity)]
  1160. [HttpGet("knowledge_questions/{id}")]
  1161. public async Task<KnowledgeQuestions> QuestionsEntity(string id)
  1162. {
  1163. return await _knowledgeQuestionsRepository.Queryable()
  1164. .Includes(x => x.Knowledge)
  1165. .FirstAsync(x => x.Id == id);
  1166. }
  1167. #endregion
  1168. #region 知识收藏
  1169. /// <summary>
  1170. /// 知识收藏列表
  1171. /// </summary>
  1172. /// <param name="dto"></param>
  1173. /// <returns></returns>
  1174. [Permission(EPermission.KnowledgeCollectList)]
  1175. [HttpGet("knowledge_collect/list")]
  1176. public async Task<PagedDto<KnowledgeCollectDto>> List([FromQuery] KnowledgeCollectListDto dto)
  1177. {
  1178. var (total, items) = await _knowledgeCollectRepository.Queryable()
  1179. .Includes(x => x.Knowledge)
  1180. .WhereIF(!string.IsNullOrEmpty(dto.Keyword), x => x.Knowledge.Title.Contains(dto.Keyword!) || (x.Knowledge.Summary != null && x.Knowledge.Summary.Contains(dto.Keyword!)))
  1181. .WhereIF(dto.KnowledgeCollectGroupId.NotNullOrEmpty(), x => x.KnowledgeCollectGroupId == dto.KnowledgeCollectGroupId)
  1182. .Where(x => x.CreatorId == _sessionContext.UserId)
  1183. .Where(x => !string.IsNullOrEmpty(x.Knowledge.Id))
  1184. .Where(x => x.Collect!.Value)
  1185. .OrderByDescending(x => x.CreationTime)
  1186. .ToPagedListAsync(dto.PageIndex, dto.PageSize, HttpContext.RequestAborted);
  1187. return new PagedDto<KnowledgeCollectDto>(total, _mapper.Map<IReadOnlyList<KnowledgeCollectDto>>(items));
  1188. }
  1189. /// <summary>
  1190. /// 新增知识收藏
  1191. /// </summary>
  1192. /// <param name="dtos"></param>
  1193. /// <returns></returns>
  1194. //[Permission(EPermission.AddKnowledgeCollect)]
  1195. [HttpPost("knowledge_collect")]
  1196. [LogFilter("知识收藏")]
  1197. public async Task Add([FromBody] KnowledgeCollectAddDto dto)
  1198. {
  1199. var collect = await _knowledgeCollectRepository.GetAsync(x => x.KnowledgeId == dto.KnowledgeId && x.CreatorId == _sessionContext.UserId);
  1200. if (collect != null)
  1201. {
  1202. collect.Collect = dto.Collect;
  1203. await _knowledgeCollectRepository.UpdateAsync(collect, HttpContext.RequestAborted);
  1204. }
  1205. else
  1206. {
  1207. var collectNew = _mapper.Map<KnowledgeCollect>(dto);
  1208. await _knowledgeCollectRepository.AddAsync(collectNew, HttpContext.RequestAborted);
  1209. }
  1210. var count = await _knowledgeCollectRepository.Queryable()
  1211. .Where(m => m.KnowledgeId == dto.KnowledgeId && m.Collect == true).CountAsync();
  1212. await _knowledgeRepository.Updateable()
  1213. .Where(m => m.Id == dto.KnowledgeId)
  1214. .SetColumns(m => m.CollectCount == count)
  1215. .ExecuteCommandAsync();
  1216. }
  1217. /// <summary>
  1218. /// 知识评分
  1219. /// </summary>
  1220. /// <param name="dto"></param>
  1221. /// <returns></returns>
  1222. //[Permission(EPermission.AddKnowledgeScore)]
  1223. [HttpDelete("knowledge_score")]
  1224. [LogFilter("知识评分")]
  1225. public async Task Delete([FromBody] KnowledgeCollectAddDto dto)
  1226. {
  1227. var collect = await _knowledgeCollectRepository.GetAsync(x => x.KnowledgeId == dto.KnowledgeId && x.CreatorId == _sessionContext.UserId);
  1228. if (collect != null)
  1229. {
  1230. if (collect.Score > 0)
  1231. throw UserFriendlyException.SameMessage("当前知识已经评分");
  1232. collect.Score = dto.Score;
  1233. await _knowledgeCollectRepository.UpdateAsync(collect, HttpContext.RequestAborted);
  1234. }
  1235. else
  1236. {
  1237. var questions = _mapper.Map<KnowledgeCollect>(dto);
  1238. await _knowledgeCollectRepository.AddAsync(questions, HttpContext.RequestAborted);
  1239. }
  1240. //计算总分
  1241. var sugar = _knowledgeCollectRepository.Queryable().Where(x => x.KnowledgeId == dto.KnowledgeId);
  1242. var count = await sugar.CountAsync();
  1243. var collects = await sugar.SumAsync(x => x.Score);
  1244. var score = collects / count;
  1245. var knowledge = await _knowledgeRepository.GetAsync(x => x.Id == dto.KnowledgeId);
  1246. if (knowledge != null)
  1247. {
  1248. knowledge.Score = decimal.Round(score.Value, 1);
  1249. await _knowledgeRepository.UpdateAsync(knowledge, HttpContext.RequestAborted);
  1250. }
  1251. }
  1252. #endregion
  1253. #region 知识评论
  1254. /// <summary>
  1255. /// 新增知识评论
  1256. /// </summary>
  1257. /// <param name="dto"></param>
  1258. /// <returns></returns>
  1259. [HttpPost("knowledge_comment")]
  1260. public async Task Add([FromBody] KnowledgeCommentAddDto dto)
  1261. {
  1262. var model = _mapper.Map<KnowledgeComment>(dto);
  1263. await _knowledgeCommentRepository.AddAsync(model, HttpContext.RequestAborted);
  1264. if (!string.IsNullOrEmpty(dto.ReplyId))
  1265. {
  1266. var comment = await _knowledgeCommentRepository.GetAsync(dto.ReplyId);
  1267. if (comment != null)
  1268. {
  1269. comment.ReplyNum++;
  1270. await _knowledgeCommentRepository.UpdateAsync(comment, HttpContext.RequestAborted);
  1271. }
  1272. }
  1273. }
  1274. /// <summary>
  1275. /// 删除知识评论
  1276. /// </summary>
  1277. /// <param name="dto"></param>
  1278. /// <returns></returns>
  1279. [HttpDelete("knowledge_comment")]
  1280. public async Task Delete([FromBody] KnowledgeCommentDeleteDto dto)
  1281. {
  1282. var comment = await _knowledgeCommentRepository.GetAsync(dto.Id, HttpContext.RequestAborted);
  1283. if (comment is null)
  1284. throw UserFriendlyException.SameMessage("无效评论");
  1285. if (comment.CreatorId != _sessionContext.UserId)
  1286. throw UserFriendlyException.SameMessage("只有评论者可以删除当前评论");
  1287. await _knowledgeCommentRepository.RemoveAsync(x => x.Id == dto.Id);
  1288. }
  1289. /// <summary>
  1290. /// 修改知识评论
  1291. /// </summary>
  1292. /// <param name="dto"></param>
  1293. /// <returns></returns>
  1294. [HttpPut("knowledge_comment")]
  1295. public async Task Update([FromBody] KnowledgeCommentUpdateDto dto)
  1296. {
  1297. var comment = await _knowledgeCommentRepository.GetAsync(dto.Id, HttpContext.RequestAborted);
  1298. if (comment is null)
  1299. throw UserFriendlyException.SameMessage("无效评论");
  1300. _mapper.Map(dto, comment);
  1301. await _knowledgeCommentRepository.UpdateAsync(comment, HttpContext.RequestAborted);
  1302. }
  1303. /// <summary>
  1304. /// 知识评论列表
  1305. /// </summary>
  1306. /// <param name="dto"></param>
  1307. /// <returns></returns>
  1308. [HttpGet("knowledge_comment/list")]
  1309. public async Task<List<KnowledgeCommentDto>> List([FromQuery] KnowledgeCommentListDto dto)
  1310. {
  1311. var comments = await _knowledgeCommentRepository.Queryable()
  1312. .WhereIF(!string.IsNullOrEmpty(dto.KnowledgeId), x => x.KnowledgeId == dto.KnowledgeId)
  1313. .WhereIF(!string.IsNullOrEmpty(dto.ReplyId), x => x.ReplyId == dto.ReplyId)
  1314. .WhereIF(dto.All.HasValue && dto.All == false, x => x.CreatorId == _sessionContext.UserId)
  1315. .OrderByDescending(x => x.CreationTime)
  1316. .ToListAsync();
  1317. return new List<KnowledgeCommentDto>(_mapper.Map<IReadOnlyList<KnowledgeCommentDto>>(comments));
  1318. }
  1319. #endregion
  1320. #region 收藏分组
  1321. /// <summary>
  1322. /// 增加收藏分组
  1323. /// </summary>
  1324. [HttpPost("group")]
  1325. public async Task<KnowledgeCollectGroupOutDto> SaveKnowledgeCoolectGroupAsync([FromBody] KnowledgeCollectGroupInDto dto)
  1326. {
  1327. var entity = dto.Adapt<KnowledgeCollectGroup>();
  1328. if (await _knowledgeCollectGroupRepository
  1329. .Queryable()
  1330. .Where(m => m.Name == dto.Name && m.CreatorId == _sessionContext.UserId)
  1331. .AnyAsync() == true) throw UserFriendlyException.SameMessage("分组名重复, 请重新输入!");
  1332. var key = await _knowledgeCollectGroupRepository.AddAsync(entity);
  1333. if (key.IsNullOrEmpty()) throw UserFriendlyException.SameMessage("添加失败, 请重试");
  1334. var outDto = dto.Adapt<KnowledgeCollectGroupOutDto>();
  1335. outDto.Id = key;
  1336. return outDto;
  1337. }
  1338. /// <summary>
  1339. /// 删除收藏分组
  1340. /// </summary>
  1341. [HttpDelete("group")]
  1342. public async Task DeleteKnowledgeCoolectGroupAsync([FromQuery] string id)
  1343. {
  1344. var entity = await _knowledgeCollectGroupRepository.Queryable()
  1345. .Where(m => m.Id == id && m.CreatorId == _sessionContext.UserId)
  1346. .FirstAsync() ?? throw new UserFriendlyException("数据不存在");
  1347. entity.IsDeleted = true;
  1348. await _knowledgeCollectGroupRepository.UpdateAsync(entity);
  1349. await _knowledgeCollectRepository.Updateable()
  1350. .Where(m => m.KnowledgeCollectGroupId == id && m.CreatorId == _sessionContext.UserId)
  1351. .SetColumns(m => m.IsDeleted == true)
  1352. .ExecuteCommandAsync();
  1353. }
  1354. /// <summary>
  1355. /// 获取分组收藏列表
  1356. /// </summary>
  1357. /// <returns></returns>
  1358. [HttpGet("group")]
  1359. public async Task<PagedDto<KnowledgeCollectGroupOutDto>> GetKnowledgeCollectGroupListAsync([FromQuery] KnowledgeCollectGroupListInDto dto)
  1360. {
  1361. return (await _knowledgeCollectGroupRepository.Queryable()
  1362. .WhereIF(dto.Keyword.NotNullOrEmpty(), m => m.Name.Contains(dto.Keyword))
  1363. .Where(m => m.CreatorId == _sessionContext.UserId)
  1364. .Select<KnowledgeCollectGroupOutDto>()
  1365. .ToPagedListAsync(dto.PageIndex, dto.PageSize, HttpContext.RequestAborted)
  1366. ).ToPaged();
  1367. }
  1368. #endregion
  1369. #region PageView 浏览记录
  1370. /// <summary>
  1371. /// 浏览记录集合
  1372. /// </summary>
  1373. /// <param name="dto"></param>
  1374. /// <returns></returns>
  1375. [HttpGet("pageview")]
  1376. public async Task<PagedDto<PageViewOutDto>> GetPageViewListAsync([FromQuery] PageViewInDto dto)
  1377. {
  1378. return (await _knowApplication.GetPageViewListAsync(dto, HttpContext.RequestAborted))
  1379. .ToPaged();
  1380. }
  1381. /// <summary>
  1382. /// 浏览记录集合-导出
  1383. /// </summary>
  1384. /// <param name="dto"></param>
  1385. /// <returns></returns>
  1386. [HttpPost("pageview/export")]
  1387. public async Task<FileStreamResult> GetPageViewListAsync([FromBody] ExportExcelDto<PageViewInDto> dto)
  1388. {
  1389. var items = (await _knowApplication.GetPageViewListAsync(dto.QueryDto, HttpContext.RequestAborted))
  1390. .Item2;
  1391. return _exportApplication.GetExcelFile(dto, items, "浏览记录");
  1392. }
  1393. #endregion
  1394. #region 热词
  1395. /// <summary>
  1396. /// 新增热词
  1397. /// </summary>
  1398. /// <param name="dto"></param>
  1399. /// <returns></returns>
  1400. [HttpPost("hotword")]
  1401. public async Task AddKnowledgeHotWordAsync([FromBody] AddKnowledgeHotWordInDto dto)
  1402. {
  1403. await _knowApplication.AddKnowledgeHotWordAsync(dto, HttpContext.RequestAborted);
  1404. }
  1405. /// <summary>
  1406. /// 新增热词页面基础数据
  1407. /// </summary>
  1408. /// <param name="dto"></param>
  1409. /// <returns></returns>
  1410. [HttpGet("hotword/basedata")]
  1411. public async Task<Dictionary<string, dynamic>> AddKnowledgeHotWordBaseDataAsync()
  1412. {
  1413. return _baseDataApplication
  1414. .KnowledgeHotWordType()
  1415. .Build();
  1416. }
  1417. /// <summary>
  1418. /// 热词集合
  1419. /// </summary>
  1420. /// <param name="dto"></param>
  1421. /// <returns></returns>
  1422. [HttpGet("hotword")]
  1423. public async Task<PagedDto<KnowledgeHotWordOutDto>> GetKnowledgeHotWordListAsync([FromQuery] KnowledgeHotWordInDto dto)
  1424. {
  1425. return (await _knowApplication.GetKnowledgeHotWordListAsync(dto, HttpContext.RequestAborted))
  1426. .ToPaged();
  1427. }
  1428. /// <summary>
  1429. /// 修改热词
  1430. /// </summary>
  1431. /// <param name="dto"></param>
  1432. /// <returns></returns>
  1433. [HttpPut("hotword")]
  1434. public async Task UpdateKnowledgeHotWordAsync([FromBody] UpdateKnowledgeHotWordInDto dto)
  1435. {
  1436. await _knowApplication.UpdateKnowledgeHotWordAsync(dto, HttpContext.RequestAborted);
  1437. }
  1438. /// <summary>
  1439. /// 删除热词
  1440. /// </summary>
  1441. /// <param name="id"></param>
  1442. /// <returns></returns>
  1443. [HttpDelete("hotword")]
  1444. public async Task DeleteKnowledgeHotWordAsync([FromQuery] string id)
  1445. {
  1446. await _knowledgeHotWordRepository
  1447. .RemoveAsync(id);
  1448. }
  1449. #endregion
  1450. }
  1451. }