KnowledgeController.cs 59 KB

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