WorkflowController.cs 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615
  1. using Hotline.Application.FlowEngine;
  2. using Hotline.FlowEngine.Definitions;
  3. using Hotline.FlowEngine.Workflows;
  4. using Hotline.Identity.Roles;
  5. using Hotline.Permissions;
  6. using Hotline.Repository.SqlSugar.Extensions;
  7. using Hotline.Settings;
  8. using Hotline.Share.Dtos;
  9. using Hotline.Share.Dtos.FlowEngine;
  10. using Hotline.Share.Enums.FlowEngine;
  11. using Hotline.Users;
  12. using MapsterMapper;
  13. using Microsoft.AspNetCore.Mvc;
  14. using SqlSugar;
  15. using Hotline.FlowEngine.WorkflowModules;
  16. using XF.Domain.Authentications;
  17. using XF.Domain.Exceptions;
  18. using XF.Domain.Extensions;
  19. using XF.Utility.EnumExtensions;
  20. using XF.Domain.Repository;
  21. using Hotline.Share.Dtos.FlowEngine.Workflow;
  22. using Hotline.Caching.Interfaces;
  23. using Hotline.Orders;
  24. using Hotline.Share.Dtos.Order;
  25. using Mapster;
  26. namespace Hotline.Api.Controllers;
  27. /// <summary>
  28. /// 工作流管理
  29. /// </summary>
  30. public class WorkflowController : BaseController
  31. {
  32. private readonly IDefinitionDomainService _definitionDomainService;
  33. private readonly IRepository<WorkflowDefinition> _definitionRepository;
  34. private readonly IWorkflowApplication _workflowApplication;
  35. private readonly IWorkflowDomainService _workflowDomainService;
  36. private readonly IWorkflowRepository _workflowRepository;
  37. private readonly IRepository<User> _userRepository;
  38. private readonly ISystemOrganizeRepository _organizeRepository;
  39. private readonly IRepository<Role> _roleRepository;
  40. private readonly ISystemDomainService _systemDomainService;
  41. private readonly IWfModuleDomainService _wfModuleDomainService;
  42. private readonly IRepository<WorkflowModule> _wfModuleRepository;
  43. private IRepository<WorkflowTrace> _workflowTraceRepository;
  44. private readonly IRepository<WorkflowCountersign> _workflowCountersignRepository;
  45. private readonly ISessionContext _sessionContext;
  46. private readonly IMapper _mapper;
  47. private readonly ISystemDicDataCacheManager _systemDicDataCacheManager;
  48. public WorkflowController(
  49. IDefinitionDomainService definitionDomainService,
  50. IRepository<WorkflowDefinition> definitionRepository,
  51. IWorkflowApplication workflowApplication,
  52. IWorkflowDomainService workflowDomainService,
  53. IWorkflowRepository workflowRepository,
  54. IRepository<User> userRepository,
  55. ISystemOrganizeRepository organizeRepository,
  56. IRepository<Role> roleRepository,
  57. ISystemDomainService systemDomainService,
  58. IWfModuleDomainService wfModuleDomainService,
  59. IRepository<WorkflowModule> wfModuleRepository,
  60. IRepository<WorkflowTrace> workflowTraceRepository,
  61. IRepository<WorkflowCountersign> workflowCountersignRepository,
  62. ISessionContext sessionContext,
  63. IMapper mapper,
  64. ISystemDicDataCacheManager systemDicDataCacheManager
  65. )
  66. {
  67. _definitionDomainService = definitionDomainService;
  68. _definitionRepository = definitionRepository;
  69. _workflowApplication = workflowApplication;
  70. _workflowDomainService = workflowDomainService;
  71. _workflowRepository = workflowRepository;
  72. _userRepository = userRepository;
  73. _organizeRepository = organizeRepository;
  74. _roleRepository = roleRepository;
  75. _systemDomainService = systemDomainService;
  76. _wfModuleDomainService = wfModuleDomainService;
  77. _wfModuleRepository = wfModuleRepository;
  78. _sessionContext = sessionContext;
  79. _mapper = mapper;
  80. _workflowTraceRepository = workflowTraceRepository;
  81. _workflowCountersignRepository = workflowCountersignRepository;
  82. _systemDicDataCacheManager = systemDicDataCacheManager;
  83. }
  84. /// <summary>
  85. /// 分页查询最新版本号的模板
  86. /// </summary>
  87. /// <param name="dto"></param>
  88. /// <returns></returns>
  89. [HttpGet("definition/latest")]
  90. public async Task<PagedDto<DefinitionDto>> QueryDefinitionLatest([FromQuery] QueryDefinitionDto dto)
  91. {
  92. var query2 = await _definitionRepository.Queryable()
  93. .WhereIF(dto.Status.HasValue, d => d.Status == dto.Status)
  94. .WhereIF(!string.IsNullOrEmpty(dto.Keyword),
  95. d => d.Code.Contains(dto.Keyword) || d.Name.Contains(dto.Keyword))
  96. .Select(d => new { i = SqlFunc.RowNumber($"{d.Version} desc", d.Code), d })
  97. .MergeTable()
  98. .Where(d => d.i == 1)
  99. .ToListAsync();
  100. var items = query2.Select(d => d.d).ToList();
  101. return new PagedDto<DefinitionDto>(query2.Count, _mapper.Map<IReadOnlyList<DefinitionDto>>(items));
  102. }
  103. /// <summary>
  104. /// 分页查询流程模板
  105. /// </summary>
  106. /// <param name="dto"></param>
  107. /// <returns></returns>
  108. [Permission(EPermission.FlowDefinitionQuery)]
  109. [HttpGet("definition")]
  110. public async Task<PagedDto<DefinitionDto>> QueryDefinitions([FromQuery] QueryDefinitionDto dto)
  111. {
  112. #region old version:只查询草稿、禁用以及已启用模板的最新版本
  113. ////todo 数据量大需重构
  114. //var query1 = await _definitionRepository.Queryable()
  115. // .Where(d => d.Status == EDefinitionStatus.Temporary)
  116. // .ToListAsync();
  117. //var query2 = await _definitionRepository.Queryable()
  118. // .Where(d => d.Status != EDefinitionStatus.Temporary)
  119. // .Select(d => new { i = SqlFunc.RowNumber($"{d.Version} desc", d.Code), d })
  120. // .MergeTable()
  121. // .Where(d => d.i == 1)
  122. // .ToListAsync();
  123. //var query = query1.Union(query2.Select(d => d.d));
  124. //var total = query.Count();
  125. //var items = query
  126. // .OrderBy(d => d.Status)
  127. // .ThenByDescending(d => d.CreationTime)
  128. // .Skip(dto.Skip())
  129. // .Take(dto.PageSize)
  130. // .ToList();
  131. #endregion
  132. var (total, items) = await _definitionRepository.Queryable()
  133. .WhereIF(dto.Status.HasValue, d => d.Status == dto.Status)
  134. .WhereIF(!string.IsNullOrEmpty(dto.Keyword),
  135. d => d.Code.Contains(dto.Keyword!) || d.Name.Contains(dto.Keyword!))
  136. .OrderBy(d => d.Status)
  137. .OrderBy(d => d.Code)
  138. .OrderByDescending(d => d.Version)
  139. .ToPagedListAsync(dto, HttpContext.RequestAborted);
  140. return new PagedDto<DefinitionDto>(total, _mapper.Map<IReadOnlyList<DefinitionDto>>(items));
  141. }
  142. /// <summary>
  143. /// 查询流程模板
  144. /// </summary>
  145. /// <param name="id"></param>
  146. /// <returns></returns>
  147. [Permission(EPermission.GetFlow)]
  148. [HttpGet("definition/{id}")]
  149. public async Task<DefinitionDto> GetDefinition(string id)
  150. {
  151. var definition = await _definitionRepository.GetAsync(id, HttpContext.RequestAborted);
  152. if (definition == null) return new();
  153. return _mapper.Map<DefinitionDto>(definition);
  154. }
  155. /// <summary>
  156. /// 新增流程模板草稿
  157. /// </summary>
  158. /// <param name="dto"></param>
  159. /// <returns></returns>
  160. [Permission(EPermission.FlowDefinitionAdd)]
  161. [HttpPost("definition")]
  162. public async Task<string> AddDefinition([FromBody] AddDefinitionDto dto)
  163. {
  164. return await _definitionDomainService.AddAsync(dto, HttpContext.RequestAborted);
  165. }
  166. /// <summary>
  167. /// 更新流程模板草稿
  168. /// </summary>
  169. /// <param name="dto"></param>
  170. /// <returns></returns>
  171. [Permission(EPermission.FlowDefinitionUpdate)]
  172. [HttpPut("definition")]
  173. public async Task UpdateDefinition([FromBody] UpdateDefinitionDto dto)
  174. {
  175. var definition = await _definitionRepository.GetAsync(dto.Id, HttpContext.RequestAborted);
  176. if (definition == null)
  177. throw UserFriendlyException.SameMessage("无效模板编号");
  178. if (definition.Status == EDefinitionStatus.Temporary)
  179. {
  180. _mapper.Map(dto, definition);
  181. await _definitionRepository.UpdateAsync(definition, HttpContext.RequestAborted);
  182. }
  183. else
  184. {
  185. var newDefinition = _mapper.Map<WorkflowDefinition>(dto);
  186. await _definitionRepository.AddAsync(newDefinition, HttpContext.RequestAborted);
  187. }
  188. }
  189. /// <summary>
  190. /// 删除草稿
  191. /// </summary>
  192. /// <param name="id"></param>
  193. /// <returns></returns>
  194. /// <exception cref="UserFriendlyException"></exception>
  195. [Permission(EPermission.FlowDefinitionRemove)]
  196. [HttpDelete("definition/{id}")]
  197. public async Task RemoveDefinition(string id)
  198. {
  199. var definition = await _definitionRepository.GetAsync(id, HttpContext.RequestAborted);
  200. if (definition == null) return;
  201. if (definition.Status != EDefinitionStatus.Temporary)
  202. throw new UserFriendlyException("已发布模板不能删除");
  203. await _definitionRepository.RemoveAsync(id, false, HttpContext.RequestAborted);
  204. }
  205. /// <summary>
  206. /// 发布(列表操作)
  207. /// </summary>
  208. /// <returns></returns>
  209. [Permission(EPermission.FlowDefinitionPublish)]
  210. [HttpPost("definition/{id}/publish")]
  211. public async Task Publish(string id)
  212. {
  213. await _definitionDomainService.PublishAsync(id, HttpContext.RequestAborted);
  214. }
  215. /// <summary>
  216. /// 发布(保存并发布)
  217. /// </summary>
  218. /// <returns></returns>
  219. [Obsolete]
  220. [Permission(EPermission.FlowDefinitionPublish)]
  221. [HttpPost("definition/publish")]
  222. public async Task Publish([FromBody] AddDefinitionDto dto)
  223. {
  224. await _definitionDomainService.PublishAsync(dto, HttpContext.RequestAborted);
  225. }
  226. /// <summary>
  227. /// 分页查询流程
  228. /// </summary>
  229. /// <param name="dto"></param>
  230. /// <returns></returns>
  231. [Permission(EPermission.FlowQuery)]
  232. [HttpGet]
  233. public async Task<PagedDto<WorkflowDto>> QueryPaged([FromQuery] QueryWorkflowPagedDto dto)
  234. {
  235. var (total, items) = await _workflowRepository.Queryable()
  236. .WhereIF(!string.IsNullOrEmpty(dto.ModuleCode), d => d.ModuleCode == dto.ModuleCode)
  237. .WhereIF(!string.IsNullOrEmpty(dto.Keyword),
  238. d => d.Id.Contains(dto.Keyword!) || d.Title.Contains(dto.Keyword!))
  239. .OrderByDescending(d => d.CreationTime)
  240. .ToPagedListAsync(dto, HttpContext.RequestAborted);
  241. return new PagedDto<WorkflowDto>(total, _mapper.Map<IReadOnlyList<WorkflowDto>>(items));
  242. }
  243. /// <summary>
  244. /// 查询流程办理下一步可选节点
  245. /// </summary>
  246. [HttpGet("{workflowId}/nextsteps")]
  247. public async Task<NextStepsDto> GetNextStepDefine(string workflowId)
  248. {
  249. //var workflow = await _workflowDomainService.GetWorkflowAsync(workflowId, true, true,
  250. // cancellationToken: HttpContext.RequestAborted);
  251. //var current = _workflowDomainService.FindCurrentStep(workflow);
  252. //var nextStepDefines = workflow.WorkflowDefinition.FindStepDefines(current.StepBox.NextSteps.Select(d => d.Code));
  253. //if (current.StepBox.PathPolicy is not EPathPolicy.None && current.StepBox.NextSteps.Count > 1)
  254. // _workflowDomainService.NextStepDefineFilter(current.StepBox.PathPolicy, nextStepDefines);
  255. //return new DefinedStepDto
  256. //{
  257. // DefinitionId = workflow.DefinitionId,
  258. // Steps = _mapper.Map<IReadOnlyList<StepBasicDto>>(nextStepDefines),//nextStepDefines.Select(d => new KeyValuePair<string, string>(d.Code, d.Name)).ToList(),
  259. // ExpiredTime = workflow.ExpiredTime,
  260. // Components = current.StepBox.Components,
  261. //};
  262. return await _workflowApplication.GetNextStepsAsync(workflowId, HttpContext.RequestAborted);
  263. }
  264. ///// <summary>
  265. ///// 查询流程下一节点待选配置
  266. ///// </summary>
  267. //[HttpGet("step-options")]
  268. //public async Task<NextStepOptionDto> GetNextStepOptions([FromQuery] QueryNextStepOptionDto dto)
  269. //{
  270. // var definition = await _definitionRepository.GetAsync(dto.DefineId, HttpContext.RequestAborted);
  271. // if (definition == null)
  272. // throw new UserFriendlyException("无效DefineId");
  273. // var defineStep = definition.FindStepDefine(dto.Code);
  274. // if (defineStep is null)
  275. // throw UserFriendlyException.SameMessage("未查询到对应节点配置");
  276. // return await _workflowApplication.GetNextStepOptionsAsync(defineStep, HttpContext.RequestAborted);
  277. //}
  278. /// <summary>
  279. /// 办理节点
  280. /// </summary>
  281. //[Permission(EPermission.FlowNext)]
  282. [HttpPost("next")]
  283. public async Task Next([FromBody] NextWorkflowDto dto)
  284. {
  285. await _workflowApplication.NextAsync(dto, HttpContext.RequestAborted);
  286. }
  287. /// <summary>
  288. /// 退回(返回前一节点)
  289. /// </summary>
  290. //[Permission(EPermission.FlowPrevious)]
  291. [HttpPost("previous")]
  292. public async Task Previous([FromBody] PreviousWorkflowDto dto)
  293. {
  294. var workflow = await _workflowDomainService.GetWorkflowAsync(dto.WorkflowId, withSteps: true,
  295. cancellationToken: HttpContext.RequestAborted);
  296. await _workflowDomainService.PreviousAsync(workflow, dto, HttpContext.RequestAborted);
  297. }
  298. /// <summary>
  299. /// 获取撤回可选节点
  300. /// </summary>
  301. /// <param name="workflowId"></param>
  302. /// <returns></returns>
  303. [Permission(EPermission.FlowRecall)]
  304. [HttpGet("{workflowId}/recall")]
  305. public async Task<RecallStepsDto> GetRecallSteps(string workflowId)
  306. {
  307. return await _workflowApplication.GetRecallStepsAsync(workflowId, HttpContext.RequestAborted);
  308. }
  309. /// <summary>
  310. /// 撤回至任意节点
  311. /// </summary>
  312. /// <param name="dto"></param>
  313. /// <returns></returns>
  314. [Permission(EPermission.FlowRecall)]
  315. [HttpPost("recall")]
  316. public async Task Recall([FromBody] RecallDto dto)
  317. {
  318. await _workflowApplication.RecallAsync(dto, HttpContext.RequestAborted);
  319. }
  320. /// <summary>
  321. /// 获取跳转可选节点
  322. /// </summary>
  323. /// <param name="workflowId"></param>
  324. /// <returns></returns>
  325. [Permission(EPermission.FlowJump)]
  326. [HttpGet("{workflowId}/jump")]
  327. public async Task<NextStepsDto> GetJumpSteps(string workflowId)
  328. {
  329. return await _workflowApplication.GetJumpStepsAsync(workflowId, HttpContext.RequestAborted);
  330. }
  331. /// <summary>
  332. /// 跳转至任意节点(暂无需求设计)
  333. /// </summary>
  334. /// <param name="dto"></param>
  335. /// <returns></returns>
  336. [Permission(EPermission.FlowJump)]
  337. [HttpPost("jump")]
  338. [Obsolete]
  339. public async Task Jump([FromBody] RecallDto dto)
  340. {
  341. await _workflowApplication.JumpAsync(dto, HttpContext.RequestAborted);
  342. }
  343. /// <summary>
  344. /// 获取重办可选节点
  345. /// </summary>
  346. /// <param name="workflowId"></param>
  347. /// <returns></returns>
  348. [HttpGet("{workflowId}/redo")]
  349. public async Task<NextStepsDto> GetRedoSteps(string workflowId)
  350. {
  351. return await _workflowApplication.GetRedoStepsAsync(workflowId, HttpContext.RequestAborted);
  352. }
  353. /// <summary>
  354. /// 终止流程
  355. /// </summary>
  356. [Permission(EPermission.FlowTerminate)]
  357. [HttpPost("terminate")]
  358. public async Task Terminate([FromBody] TerminateDto dto)
  359. {
  360. await _workflowDomainService.TerminateAsync(dto, HttpContext.RequestAborted);
  361. }
  362. /// <summary>
  363. /// 撤销流程
  364. /// </summary>
  365. [HttpPost("cancel")]
  366. public async Task Cancel([FromBody] CancelDto dto)
  367. {
  368. await _workflowDomainService.CancelAsync(dto, HttpContext.RequestAborted);
  369. }
  370. /// <summary>
  371. /// 否决
  372. /// </summary>
  373. [HttpPost("reject")]
  374. public async Task Reject([FromBody] RejectDto dto)
  375. {
  376. await _workflowApplication.RejectAsync(dto, HttpContext.RequestAborted);
  377. }
  378. /// <summary>
  379. /// 补充
  380. /// </summary>
  381. /// <param name="dto"></param>
  382. /// <returns></returns>
  383. //[Permission(EPermission.FlowSupplement)]
  384. [HttpPost("supplement")]
  385. public async Task Supplement([FromBody] SupplementDto dto)
  386. {
  387. var workflow = await _workflowDomainService.GetWorkflowAsync(dto.WorkflowId);
  388. await _workflowDomainService.SupplementAsync(workflow, dto, HttpContext.RequestAborted);
  389. }
  390. /// <summary>
  391. /// 查询办理类型参数
  392. /// </summary>
  393. [HttpGet("handlerclassify/{handlerType}")]
  394. public async Task<List<KeyValuePair<string, string>>> GetHandlerClassifies(EHandlerType handlerType)
  395. {
  396. switch (handlerType)
  397. {
  398. case EHandlerType.Role:
  399. var roles = await _roleRepository.QueryAsync();
  400. return roles.Select(d => new KeyValuePair<string, string>(d.Name, d.DisplayName)).ToList();
  401. case EHandlerType.OrgLevel:
  402. var orgs1 = await _systemDomainService.QueryOrgLevelStringOptionsAsync(HttpContext.RequestAborted);
  403. return orgs1.ToList();
  404. case EHandlerType.OrgType:
  405. return EnumExts.GetDescriptions<EOrgType>()
  406. .Select(d => new KeyValuePair<string, string>(d.Key.ToString(), d.Value)).ToList();
  407. case EHandlerType.AssignedOrg:
  408. var orgs = await _organizeRepository.GetOrgJson();
  409. return orgs.Select(d => new KeyValuePair<string, string>(d.Id, d.Name)).ToList();
  410. case EHandlerType.AssignedUser:
  411. default:
  412. throw new ArgumentOutOfRangeException(nameof(handlerType), handlerType, null);
  413. }
  414. }
  415. /// <summary>
  416. /// 查询流程流转记录
  417. /// </summary>
  418. /// <param name="workflowId"></param>
  419. /// <returns></returns>
  420. [HttpGet("{workflowId}/traces")]
  421. public async Task<WorkflowDto> GetWorkflowTraces(string workflowId)
  422. {
  423. var workflow =
  424. await _workflowDomainService.GetWorkflowAsync(workflowId, cancellationToken: HttpContext.RequestAborted);
  425. workflow.Traces = await _workflowTraceRepository.Queryable()
  426. .Where(d => d.WorkflowId == workflow.Id)
  427. .OrderBy(d => d.CreationTime)
  428. .ToTreeAsync(d => d.Traces, d => d.ParentId, null);
  429. return _mapper.Map<WorkflowDto>(workflow);
  430. }
  431. /// <summary>
  432. /// 查询被督办/催办部门
  433. /// </summary>
  434. /// <param name="workflowId"></param>
  435. /// <returns></returns>
  436. [HttpGet("{workflowId}/urge")]
  437. public async Task<IReadOnlyList<Kv>> GetUrgeOrgs(string workflowId)
  438. {
  439. /*
  440. * 非会签:当前部门上至一级部门
  441. * 中心会签:一级部门下至所有当前办理部门(遍历所有会签分支)
  442. */
  443. var workflow = await _workflowDomainService.GetWorkflowAsync(workflowId, withSteps: true,
  444. cancellationToken: HttpContext.RequestAborted);
  445. if (workflow.IsInCountersign && workflow.CounterSignType is ECounterSignType.Center)
  446. {
  447. var handlers = workflow.Steps
  448. .Where(d => d.BusinessType == EBusinessType.Send && !d.IsOrigin)
  449. .SelectMany(d => d.Handlers)
  450. .Distinct()
  451. .ToList();
  452. return handlers.Where(d => !d.Key.IsCenter()).ToList();
  453. }
  454. return workflow.Steps
  455. .Where(d => d.StepType != EStepType.Start
  456. && d.StepType != EStepType.End
  457. && d.IsOrigin)
  458. .SelectMany(d => d.Handlers)
  459. .Distinct()
  460. .ToList();
  461. }
  462. [HttpGet("base-data")]
  463. public async Task<dynamic> BaseData()
  464. {
  465. var levels = new List<int> { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };
  466. var orgs = levels.Select(d => new KeyValuePair<string, string>(d.ToString(), $"{d.ToChinese()}级部门办理")).ToList();
  467. var center = new KeyValuePair<string, string>("0", "中心办理");
  468. var centerIsTop = new List<KeyValuePair<string, string>> { center };
  469. centerIsTop.AddRange(orgs);
  470. return new
  471. {
  472. ModuleOptions =
  473. WorkflowModuleConsts.AllModules.Select(d => new KeyValuePair<string, string>(d.Code, d.Name)),
  474. HandlerTypeOptions = EnumExts.GetDescriptions<EHandlerType>(),
  475. BusinessTypeOptions = EnumExts.GetDescriptions<EBusinessType>(),
  476. StepPropertiesOptions = _systemDicDataCacheManager.GetSysDicDataCache(SysDicTypeConsts.WorkflowStepComponent),
  477. //PathPolicyOptions = EnumExts.GetDescriptions<EPathPolicy>(),
  478. ExecuteModeOptions = EnumExts.GetDescriptions<EExecuteMode>(),
  479. InstanceModeOptions = EnumExts.GetDescriptions<EInstanceMode>(),
  480. StepTypeOptions = EnumExts.GetDescriptions<EStepType>().Where(d => d.Key != 1 && d.Key != 2),
  481. DynamicPolicyOptions = EnumExts.GetDescriptions<EDynamicPolicy>()
  482. .Select(d => new
  483. {
  484. Key = d.Key,
  485. Value = d.Value,
  486. Items = d.Key == 0 || d.Key == 2 ? centerIsTop : orgs
  487. }),
  488. FlowTypeOptions = EnumExts.GetDescriptions<EFlowType>()
  489. };
  490. }
  491. /// <summary>
  492. /// 持久化新增工作流业务
  493. /// </summary>
  494. /// <returns></returns>
  495. [HttpGet("wfmodule/persistence")]
  496. public async Task PersistenceWfModule()
  497. {
  498. await _wfModuleDomainService.PersistenceModulesAsync(HttpContext.RequestAborted);
  499. }
  500. /// <summary>
  501. /// 查询所有工作流模块
  502. /// </summary>
  503. /// <returns></returns>
  504. [HttpGet("wfmodules")]
  505. public async Task<IReadOnlyList<WorkflowModule>> QueryWfModules()
  506. {
  507. return await _wfModuleRepository.Queryable()
  508. .Includes(d => d.Definition)
  509. .ToListAsync();
  510. }
  511. /// <summary>
  512. /// 为工作流业务匹配或取消流程模板
  513. /// </summary>
  514. /// <param name="dto"></param>
  515. /// <returns></returns>
  516. [HttpPut("wfmodule/match")]
  517. public async Task MatchDefinition([FromBody] MatchDefinitionDto dto)
  518. {
  519. if (string.IsNullOrEmpty(dto.DefinitionId))
  520. {
  521. //取消当前已配置模板
  522. await _wfModuleDomainService.MatchDefinitionAsync(dto, HttpContext.RequestAborted);
  523. }
  524. else
  525. {
  526. var definition = await _definitionRepository.GetAsync(dto.DefinitionId);
  527. if (definition == null)
  528. throw UserFriendlyException.SameMessage("无效模板编号");
  529. if (definition.Status != EDefinitionStatus.Enable)
  530. throw UserFriendlyException.SameMessage("该模板未发布");
  531. await _wfModuleDomainService.MatchDefinitionAsync(dto, HttpContext.RequestAborted);
  532. }
  533. }
  534. /// <summary>
  535. /// 查询会签信息
  536. /// </summary>
  537. /// <param name="dto"></param>
  538. /// <returns></returns>
  539. [HttpGet("countersign")]
  540. public async Task<PagedDto<WorkflowCountersignDto>> QueryWorkflowCountersigns([FromQuery] QueryWorkflowCountersignDto dto)
  541. {
  542. RefAsync<int> total = 0;
  543. var query = _workflowCountersignRepository.Queryable()
  544. .LeftJoin<Workflow>((c, w) => c.WorkflowId == w.Id)
  545. .InnerJoin<Order>((c, w, o) => w.ExternalId == o.Id)
  546. .WhereIF(dto.IsProvince.HasValue, (c, w, o) => o.IsProvince == dto.IsProvince.Value)
  547. .WhereIF(!string.IsNullOrEmpty(dto.Keyword),
  548. (c, w, o) => o.No.Contains(dto.Keyword) || o.Title.Contains(dto.Keyword));
  549. if (dto.IsOnlyStarter)
  550. query = query.Where((c, w, o) => c.StarterId == _sessionContext.RequiredUserId);
  551. var items = await query
  552. .OrderByDescending((c, w, o) => w.ExpiredTime)
  553. .Select((c, w, o) => new { c, o })
  554. .ToPageListAsync(dto.PageIndex, dto.PageSize, total, HttpContext.RequestAborted);
  555. var dtos = items.Select(d =>
  556. {
  557. var dto = _mapper.Map<WorkflowCountersignDto>(d.c);
  558. dto.Order = _mapper.Map<OrderDto>(d.o);
  559. return dto;
  560. }).ToList();
  561. return new PagedDto<WorkflowCountersignDto>(total, dtos);
  562. }
  563. }