ApprovalFlowService.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311
  1. // Admin.NET 项目的版权、商标、专利和其他相关权利均受相应法律法规的保护。使用本项目应遵守相关法律法规和许可证的要求。
  2. //
  3. // 本项目主要遵循 MIT 许可证和 Apache 许可证(版本 2.0)进行分发和使用。许可证位于源代码树根目录中的 LICENSE-MIT 和 LICENSE-APACHE 文件。
  4. //
  5. // 不得利用本项目从事危害国家安全、扰乱社会秩序、侵犯他人合法权益等法律法规禁止的活动!任何基于本项目二次开发而产生的一切法律纠纷和责任,我们不承担任何责任!
  6. using System.Text.Json;
  7. namespace Admin.NET.Plugin.ApprovalFlow.Service;
  8. /// <summary>
  9. /// 审批流程服务
  10. /// </summary>
  11. [ApiDescriptionSettings(ApprovalFlowConst.GroupName, Order = 100)]
  12. public class ApprovalFlowService : IDynamicApiController, ITransient
  13. {
  14. private readonly SqlSugarRepository<ApprovalFlow> _approvalFlowRep;
  15. private readonly SqlSugarRepository<ApprovalFlowVersion> _versionRep;
  16. private readonly SqlSugarRepository<SysUser> _sysUserRep;
  17. private readonly UserManager _userManager;
  18. public ApprovalFlowService(
  19. SqlSugarRepository<ApprovalFlow> approvalFlowRep,
  20. SqlSugarRepository<ApprovalFlowVersion> versionRep,
  21. SqlSugarRepository<SysUser> sysUserRep,
  22. UserManager userManager)
  23. {
  24. _approvalFlowRep = approvalFlowRep;
  25. _versionRep = versionRep;
  26. _sysUserRep = sysUserRep;
  27. _userManager = userManager;
  28. }
  29. /// <summary>
  30. /// 分页查询审批流
  31. /// </summary>
  32. /// <param name="input"></param>
  33. /// <returns></returns>
  34. [HttpPost]
  35. [ApiDescriptionSettings(Name = "Page")]
  36. public async Task<SqlSugarPagedList<ApprovalFlowOutput>> Page(ApprovalFlowInput input)
  37. {
  38. return await _approvalFlowRep.AsQueryable()
  39. .WhereIF(!string.IsNullOrWhiteSpace(input.Keyword), u => u.Code.Contains(input.Keyword.Trim()) || u.Name.Contains(input.Keyword.Trim()) || u.Remark.Contains(input.Keyword.Trim()))
  40. .WhereIF(!string.IsNullOrWhiteSpace(input.Code), u => u.Code.Contains(input.Code.Trim()))
  41. .WhereIF(!string.IsNullOrWhiteSpace(input.Name), u => u.Name.Contains(input.Name.Trim()))
  42. .WhereIF(!string.IsNullOrWhiteSpace(input.Remark), u => u.Remark.Contains(input.Remark.Trim()))
  43. .Select<ApprovalFlowOutput>()
  44. .ToPagedListAsync(input.Page, input.PageSize);
  45. }
  46. /// <summary>
  47. /// 增加审批流
  48. /// </summary>
  49. /// <param name="input"></param>
  50. /// <returns></returns>
  51. [ApiDescriptionSettings(Name = "Add"), HttpPost]
  52. public async Task<long> Add(AddApprovalFlowInput input)
  53. {
  54. var entity = input.Adapt<ApprovalFlow>();
  55. if (input.Code == null)
  56. {
  57. entity.Code = await LastCode("");
  58. }
  59. await _approvalFlowRep.InsertAsync(entity);
  60. return entity.Id;
  61. }
  62. /// <summary>
  63. /// 更新审批流
  64. /// </summary>
  65. /// <param name="input"></param>
  66. /// <returns></returns>
  67. [ApiDescriptionSettings(Name = "Update"), HttpPost]
  68. public async Task Update(UpdateApprovalFlowInput input)
  69. {
  70. var entity = input.Adapt<ApprovalFlow>();
  71. await _approvalFlowRep.AsUpdateable(entity).IgnoreColumns(ignoreAllNullColumns: true).ExecuteCommandAsync();
  72. }
  73. /// <summary>
  74. /// 删除审批流
  75. /// </summary>
  76. /// <param name="input"></param>
  77. /// <returns></returns>
  78. [ApiDescriptionSettings(Name = "Delete"), HttpPost]
  79. public async Task Delete(DeleteApprovalFlowInput input)
  80. {
  81. var entity = await _approvalFlowRep.GetByIdAsync(input.Id) ?? throw Oops.Oh(ErrorCodeEnum.D1002);
  82. await _approvalFlowRep.FakeDeleteAsync(entity); // 假删除
  83. }
  84. /// <summary>
  85. /// 获取审批流
  86. /// </summary>
  87. /// <param name="input"></param>
  88. /// <returns></returns>
  89. public async Task<ApprovalFlow> GetDetail([FromQuery] QueryByIdApprovalFlowInput input)
  90. {
  91. return await _approvalFlowRep.GetByIdAsync(input.Id);
  92. }
  93. /// <summary>
  94. /// 根据编码获取审批流信息
  95. /// </summary>
  96. /// <param name="code"></param>
  97. /// <returns></returns>
  98. public async Task<ApprovalFlow> GetInfo([FromQuery] string code)
  99. {
  100. return await _approvalFlowRep.GetFirstAsync(u => u.Code == code);
  101. }
  102. /// <summary>
  103. /// 获取审批流列表
  104. /// </summary>
  105. /// <param name="input"></param>
  106. /// <returns></returns>
  107. public async Task<List<ApprovalFlowOutput>> GetList([FromQuery] ApprovalFlowInput input)
  108. {
  109. return await _approvalFlowRep.AsQueryable().Select<ApprovalFlowOutput>().ToListAsync();
  110. }
  111. /// <summary>
  112. /// 搜索审批人候选用户
  113. /// </summary>
  114. [HttpGet]
  115. [ApiDescriptionSettings(Name = "CandidateUsers")]
  116. [DisplayName("搜索审批人候选用户")]
  117. public async Task<List<ApprovalCandidateUserOutput>> CandidateUsers([FromQuery] ApprovalCandidateUserInput input)
  118. {
  119. var keyword = input.Keyword?.Trim();
  120. var pageSize = input.PageSize is > 0 and <= 100 ? input.PageSize : 20;
  121. var keywordId = long.TryParse(keyword, out var parsedId) ? parsedId : 0;
  122. return await _sysUserRep.AsQueryable()
  123. .Where(u => u.TenantId == _userManager.TenantId)
  124. .Where(u => u.AccountType != AccountTypeEnum.SuperAdmin)
  125. .WhereIF(!string.IsNullOrWhiteSpace(keyword), u =>
  126. u.Id == keywordId ||
  127. u.Account.Contains(keyword!) ||
  128. u.RealName.Contains(keyword!))
  129. .OrderBy(u => new { u.OrderNo, u.Id })
  130. .Select(u => new ApprovalCandidateUserOutput
  131. {
  132. Id = u.Id,
  133. Account = u.Account,
  134. RealName = u.RealName,
  135. })
  136. .Take(pageSize)
  137. .ToListAsync();
  138. }
  139. /// <summary>
  140. /// 获取今天创建的最大编号
  141. /// </summary>
  142. /// <param name="prefix"></param>
  143. /// <returns></returns>
  144. private async Task<string> LastCode(string prefix)
  145. {
  146. var today = DateTime.Now.Date;
  147. var count = await _approvalFlowRep.AsQueryable().Where(u => u.CreateTime >= today).CountAsync();
  148. return prefix + DateTime.Now.ToString("yyMMdd") + string.Format("{0:d2}", count + 1);
  149. }
  150. /// <summary>
  151. /// 发布流程(Version+1, IsPublished=true)
  152. /// </summary>
  153. [ApiDescriptionSettings(Name = "Publish"), HttpPost]
  154. [DisplayName("发布审批流")]
  155. public async Task Publish([FromBody] BaseIdInput input)
  156. {
  157. var entity = await _approvalFlowRep.GetByIdAsync(input.Id) ?? throw Oops.Oh(ErrorCodeEnum.D1002);
  158. if (string.IsNullOrWhiteSpace(entity.FlowJson))
  159. throw Oops.Oh("请先设计流程图再发布");
  160. if (string.IsNullOrWhiteSpace(entity.BizType))
  161. throw Oops.Oh("请先设置业务类型再发布");
  162. entity.Version += 1;
  163. entity.IsPublished = true;
  164. await _approvalFlowRep.AsUpdateable(entity)
  165. .UpdateColumns(u => new { u.Version, u.IsPublished })
  166. .ExecuteCommandAsync();
  167. await _versionRep.InsertAsync(new ApprovalFlowVersion
  168. {
  169. FlowId = entity.Id,
  170. Version = entity.Version,
  171. FlowJson = entity.FlowJson,
  172. FormJson = entity.FormJson,
  173. PublishTime = DateTime.Now,
  174. PublisherName = _userManager.RealName,
  175. });
  176. }
  177. /// <summary>
  178. /// 获取流程版本历史
  179. /// </summary>
  180. [HttpGet]
  181. [ApiDescriptionSettings(Name = "VersionHistory")]
  182. [DisplayName("获取版本历史")]
  183. public async Task<List<FlowVersionOutput>> VersionHistory([FromQuery] long flowId)
  184. {
  185. return await _versionRep.AsQueryable()
  186. .Where(v => v.FlowId == flowId)
  187. .OrderByDescending(v => v.Version)
  188. .Select(v => new FlowVersionOutput
  189. {
  190. Id = v.Id,
  191. Version = v.Version,
  192. PublishTime = v.PublishTime,
  193. PublisherName = v.PublisherName,
  194. })
  195. .ToListAsync();
  196. }
  197. /// <summary>
  198. /// 获取指定版本详情(含 FlowJson)
  199. /// </summary>
  200. [HttpGet]
  201. [ApiDescriptionSettings(Name = "VersionDetail")]
  202. [DisplayName("获取版本详情")]
  203. public async Task<ApprovalFlowVersion?> VersionDetail([FromQuery] long versionId)
  204. {
  205. return await _versionRep.GetByIdAsync(versionId);
  206. }
  207. /// <summary>
  208. /// 回退到指定版本(将该版本的 FlowJson/FormJson 复制回主记录)
  209. /// </summary>
  210. [HttpPost]
  211. [ApiDescriptionSettings(Name = "RevertVersion")]
  212. [DisplayName("回退版本")]
  213. public async Task RevertVersion([FromBody] BaseIdInput input)
  214. {
  215. var version = await _versionRep.GetByIdAsync(input.Id)
  216. ?? throw Oops.Oh("版本记录不存在");
  217. var flow = await _approvalFlowRep.GetByIdAsync(version.FlowId)
  218. ?? throw Oops.Oh("流程不存在");
  219. flow.FlowJson = version.FlowJson;
  220. flow.FormJson = version.FormJson;
  221. flow.IsPublished = false;
  222. await _approvalFlowRep.AsUpdateable(flow)
  223. .UpdateColumns(u => new { u.FlowJson, u.FormJson, u.IsPublished })
  224. .ExecuteCommandAsync();
  225. }
  226. /// <summary>
  227. /// 获取业务类型列表
  228. /// </summary>
  229. [HttpGet]
  230. [ApiDescriptionSettings(Name = "BizTypes")]
  231. [DisplayName("获取业务类型列表")]
  232. public async Task<List<string>> GetBizTypes()
  233. {
  234. return await _approvalFlowRep.AsQueryable()
  235. .Where(u => !string.IsNullOrWhiteSpace(u.BizType) && !u.IsDelete)
  236. .Select(u => u.BizType!)
  237. .Distinct()
  238. .ToListAsync();
  239. }
  240. /// <summary>
  241. /// 获取指定业务类型的最新已发布流程定义
  242. /// </summary>
  243. [HttpGet]
  244. [ApiDescriptionSettings(Name = "PublishedByBizType")]
  245. [DisplayName("获取已发布流程")]
  246. public async Task<ApprovalFlowOutput?> GetPublishedByBizType([FromQuery] string bizType)
  247. {
  248. return await _approvalFlowRep.AsQueryable()
  249. .Where(u => u.BizType == bizType && u.IsPublished && !u.IsDelete)
  250. .OrderByDescending(u => u.Version)
  251. .Select<ApprovalFlowOutput>()
  252. .FirstAsync();
  253. }
  254. [HttpGet]
  255. [ApiDescriptionSettings(Name = "FlowList")]
  256. [DisplayName("获取审批流结构")]
  257. public async Task<dynamic> FlowList([FromQuery] string code)
  258. {
  259. var result = await _approvalFlowRep.AsQueryable().Where(u => u.Code == code).Select<ApprovalFlowOutput>().FirstAsync();
  260. var FlowJson = result.FlowJson != null ? JsonSerializer.Deserialize<ApprovalFlowItem>(result.FlowJson) : new ApprovalFlowItem();
  261. var FormJson = result.FormJson != null ? JsonSerializer.Deserialize<ApprovalFormItem>(result.FormJson) : new ApprovalFormItem();
  262. return new
  263. {
  264. FlowJson,
  265. FormJson
  266. };
  267. }
  268. [HttpGet]
  269. [ApiDescriptionSettings(Name = "FormRoutes")]
  270. [DisplayName("获取审批流规则")]
  271. public async Task<List<string>> FormRoutes()
  272. {
  273. var results = await _approvalFlowRep.AsQueryable().Select<ApprovalFlowOutput>().ToListAsync();
  274. var list = new List<string>();
  275. foreach (var item in results)
  276. {
  277. if (string.IsNullOrWhiteSpace(item.FormJson))
  278. continue;
  279. var formJson = JsonSerializer.Deserialize<ApprovalFormItem>(item.FormJson);
  280. if (formJson != null && !string.IsNullOrEmpty(formJson.Route))
  281. list.Add(formJson.Route);
  282. }
  283. return list;
  284. }
  285. }