WorkflowEndHandler.cs 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376
  1. using DotNetCore.CAP;
  2. using Hotline.Application.CallCenter;
  3. using Hotline.Application.OrderApp;
  4. using Hotline.Caching.Interfaces;
  5. using Hotline.CallCenter.Tels;
  6. using Hotline.Configurations;
  7. using Hotline.EventBus;
  8. using Hotline.FlowEngine.Notifications;
  9. using Hotline.FlowEngine.WorkflowModules;
  10. using Hotline.JudicialManagement.Notifies;
  11. using Hotline.KnowledgeBase;
  12. using Hotline.Orders;
  13. using Hotline.Schedulings;
  14. using Hotline.Settings;
  15. using Hotline.Settings.TimeLimitDomain;
  16. using Hotline.Share.Dtos.FlowEngine.Workflow;
  17. using Hotline.Share.Dtos.Order;
  18. using Hotline.Share.Dtos.TrCallCenter;
  19. using Hotline.Share.Enums.CallCenter;
  20. using Hotline.Share.Enums.FlowEngine;
  21. using Hotline.Share.Enums.Order;
  22. using Hotline.Share.Tools;
  23. using Hotline.Snapshot.Notifications;
  24. using MapsterMapper;
  25. using MediatR;
  26. using Microsoft.Extensions.Logging;
  27. using Microsoft.Extensions.Options;
  28. using SqlSugar.Extensions;
  29. using XF.Domain.Exceptions;
  30. using XF.Domain.Repository;
  31. namespace Hotline.Application.Handlers.FlowEngine;
  32. public class WorkflowEndHandler : INotificationHandler<EndWorkflowNotify>
  33. {
  34. private readonly IKnowledgeDomainService _knowledgeDomainService;
  35. private readonly IOrderDomainService _orderDomainService;
  36. private readonly IOrderApplication _orderApplication;
  37. private readonly ITelDomainService _telDomainService;
  38. private readonly IOrderRepository _orderRepository;
  39. private readonly ICapPublisher _capPublisher;
  40. private readonly IMapper _mapper;
  41. private readonly IRepository<OrderDelay> _orderDelayRepository;
  42. private readonly ILogger<WorkflowEndHandler> _logger;
  43. private readonly IKnowledgeRepository _knowledgeRepository;
  44. private readonly ICallApplication _callApplication;
  45. private readonly IOptionsSnapshot<AppConfiguration> _appOptions;
  46. private readonly ISystemSettingCacheManager _systemSettingCacheManager;
  47. private readonly Publisher _publisher;
  48. private readonly ICalcExpireTime _expireTime;
  49. private readonly IRepository<OrderTerminate> _orderTerminateRepository;
  50. private readonly IRepository<OrderSecondaryHandling> _orderSecondaryHandlingRepository;
  51. private readonly IRepository<Scheduling> _schedulingRepository;
  52. public WorkflowEndHandler(
  53. IMapper mapper,
  54. IKnowledgeDomainService knowledgeDomainService,
  55. IOrderDomainService orderDomainService,
  56. IOrderApplication orderApplication,
  57. ITelDomainService telDomainService,
  58. IOrderRepository orderRepository,
  59. IKnowledgeRepository knowledgeRepository,
  60. IRepository<OrderDelay> orderDelayRepository,
  61. ICapPublisher capPublisher,
  62. ICallApplication callApplication,
  63. IOptionsSnapshot<AppConfiguration> appOptions,
  64. ISystemSettingCacheManager systemSettingCacheManager,
  65. Publisher publisher,
  66. ILogger<WorkflowEndHandler> logger,
  67. ICalcExpireTime expireTime,
  68. IRepository<OrderTerminate> orderTerminateRepository,
  69. IRepository<OrderSecondaryHandling> orderSecondaryHandlingRepository,
  70. IRepository<Scheduling> schedulingRepository)
  71. {
  72. _mapper = mapper;
  73. _knowledgeDomainService = knowledgeDomainService;
  74. _orderDomainService = orderDomainService;
  75. _orderApplication = orderApplication;
  76. _telDomainService = telDomainService;
  77. _orderRepository = orderRepository;
  78. _knowledgeRepository = knowledgeRepository;
  79. _orderDelayRepository = orderDelayRepository;
  80. _capPublisher = capPublisher;
  81. _callApplication = callApplication;
  82. _appOptions = appOptions;
  83. _systemSettingCacheManager = systemSettingCacheManager;
  84. _publisher = publisher;
  85. _logger = logger;
  86. _expireTime = expireTime;
  87. _orderTerminateRepository = orderTerminateRepository;
  88. _orderSecondaryHandlingRepository = orderSecondaryHandlingRepository;
  89. _schedulingRepository = schedulingRepository;
  90. }
  91. /// <summary>Handles a notification</summary>
  92. /// <param name="notification">The notification</param>
  93. /// <param name="cancellationToken">Cancellation token</param>
  94. public async Task Handle(EndWorkflowNotify notification, CancellationToken cancellationToken)
  95. {
  96. try
  97. {
  98. var workflow = notification.Workflow;
  99. //审批是否通过
  100. var isReviewPass = workflow.IsReviewPass();
  101. switch (workflow.ModuleCode)
  102. {
  103. case WorkflowModuleConsts.KnowledgeAdd://新增知识库
  104. case WorkflowModuleConsts.KnowledgeUpdate://修改知识库
  105. case WorkflowModuleConsts.KnowledgeDelete://删除知识库
  106. case WorkflowModuleConsts.KnowledgeOffshelf: //下架知识库
  107. //var knowledgeWork = await _knowledgeWorkFlowRepository.Queryable().Where(x => x.Id == workflow.ExternalId).FirstAsync(cancellationToken);
  108. var knowledge = await _knowledgeRepository.Queryable().Where(x => x.Id == workflow.ExternalId).FirstAsync(cancellationToken);
  109. knowledge.Flowed(workflow.FlowedUserIds, workflow.FlowedOrgIds, workflow.HandlerUsers, workflow.HandlerOrgs);
  110. await _knowledgeRepository.UpdateAsync(knowledge, cancellationToken);
  111. if (isReviewPass)
  112. {
  113. await _knowledgeDomainService.EndWorkKnowledge(workflow, cancellationToken);
  114. }
  115. else
  116. {
  117. await _knowledgeDomainService.TerminateWorkKnowledge(workflow, cancellationToken);
  118. }
  119. break;
  120. case WorkflowModuleConsts.TelRestApply:
  121. await _telDomainService.TelRestApplyPassAsync(workflow.ExternalId, cancellationToken);
  122. break;
  123. case WorkflowModuleConsts.OrderHandle:
  124. case WorkflowModuleConsts.OrderHandleSnapshot:
  125. var order = await _orderDomainService.GetOrderAsync(workflow.ExternalId,
  126. withExtension: true, cancellationToken: cancellationToken);
  127. //order.CheckIfFiled();
  128. //order.UpdateHandlingStatus(workflow.IsInCountersign);
  129. _mapper.Map(workflow, order);
  130. var now = DateTime.Now;
  131. var handleDuration = order.CenterToOrgTime.HasValue && order.ActualHandleTime.HasValue
  132. ? // _timeLimitDomainService.CalcWorkTime(
  133. await _expireTime.CalcWorkTime(
  134. order.CenterToOrgTime.Value,
  135. order.ActualHandleTime.Value, order.ProcessType is EProcessType.Zhiban)
  136. : 0;
  137. var fileDuration = order.CenterToOrgTime.HasValue
  138. ? //_timeLimitDomainService.CalcWorkTime(
  139. await _expireTime.CalcWorkTime(
  140. order.CenterToOrgTime.Value,
  141. now, order.ProcessType is EProcessType.Zhiban)
  142. : 0;
  143. var allDuration = order.StartTime.HasValue
  144. ? // _timeLimitDomainService.CalcWorkTime(
  145. await _expireTime.CalcWorkTime(
  146. order.StartTime.Value, now,
  147. order.ProcessType is EProcessType.Zhiban)
  148. : 0;
  149. var creationTimeHandleDurationWorkday = order.ActualHandleTime.HasValue
  150. ? //_timeLimitDomainService.CalcWorkTimeEx(
  151. await _expireTime.CalcWorkTimeEx(
  152. order.CreationTime, now,
  153. order.ProcessType is EProcessType.Zhiban)
  154. : 0;
  155. var centerToOrgHandleDurationWorkday = order.ActualHandleTime.HasValue && order.CenterToOrgTime.HasValue
  156. ? //_timeLimitDomainService.CalcWorkTimeEx(
  157. await _expireTime.CalcWorkTimeEx(
  158. order.CenterToOrgTime.Value, now,
  159. order.ProcessType is EProcessType.Zhiban)
  160. : 0;
  161. creationTimeHandleDurationWorkday = creationTimeHandleDurationWorkday <= 0 ? 10 : creationTimeHandleDurationWorkday;
  162. centerToOrgHandleDurationWorkday = centerToOrgHandleDurationWorkday <= 0 ? 10 : centerToOrgHandleDurationWorkday;
  163. double? secondaryHandlingDurationWorkday = null;
  164. DateTime? secondaryHandlingAuditTime = null;
  165. //计算二次办理时长
  166. if (_appOptions.Value.IsZiGong)
  167. {
  168. //查询是否有二次办理申请通过的
  169. var orderSecondaryData = await _orderSecondaryHandlingRepository.GetAsync(p => p.OrderId == order.Id &&
  170. p.IsHandel == false && p.State == ESecondaryHandlingState.End, cancellationToken);
  171. if (orderSecondaryData != null)
  172. {
  173. //处理办理时长
  174. orderSecondaryData.IsHandel = true;
  175. await _orderSecondaryHandlingRepository.UpdateAsync(orderSecondaryData, cancellationToken);
  176. if (orderSecondaryData.AuditTime.HasValue)
  177. {
  178. secondaryHandlingAuditTime = orderSecondaryData.AuditTime;
  179. secondaryHandlingDurationWorkday = orderSecondaryData.AuditTime.HasValue
  180. ?
  181. await _expireTime.CalcWorkTimeEx(
  182. orderSecondaryData.AuditTime.Value, now,
  183. false)
  184. : null;
  185. }
  186. }
  187. }
  188. order.File(now, handleDuration, fileDuration, allDuration, creationTimeHandleDurationWorkday, centerToOrgHandleDurationWorkday,
  189. secondaryHandlingDurationWorkday, secondaryHandlingAuditTime);
  190. order.FileUserId = notification.Trace.HandlerId;
  191. order.FileUserName = notification.Trace.HandlerName;
  192. order.FileUserOrgId = notification.Trace.HandlerOrgId;
  193. order.FileUserOrgName = notification.Trace.HandlerOrgName;
  194. order.FileOrgIsCenter = notification.Trace.HandlerOrgIsCenter;
  195. order.FileOpinion = notification.Dto.Opinion;
  196. /*
  197. *需求:
  198. *1、判断工单属于哪种归档类型,需由谁归档来判断,热线中心归档的就叫中心归档件,部门归档就叫做部门归档件
  199. 2、若工单发起过部门会签,最后由热线中心归档,应属于叫中心归档件和非会签件
  200. 3、若工单发起过中心会签,最后由部门归档,应属于部门归档件和非会签件
  201. */
  202. if (_appOptions.Value.IsYiBin)
  203. {
  204. if (order.FileOrgIsCenter.Value)
  205. {
  206. if (order.CounterSignType is ECounterSignType.Department)
  207. order.CounterSignType = null;
  208. }
  209. else
  210. {
  211. if (order.CounterSignType is ECounterSignType.Center)
  212. order.CounterSignType = null;
  213. }
  214. //随机派单员(市中心综合办)
  215. if (order.FileUserOrgId == "001175")
  216. {
  217. try
  218. {
  219. var schedulingList = await _schedulingRepository.Queryable().Includes(x => x.SchedulingUser).Where(x => x.SchedulingTime!.Value.Date == DateTime.Now.Date).ToListAsync();
  220. if (schedulingList != null && schedulingList.Count > 0)
  221. {
  222. int randomNum = Random.Shared.Next(schedulingList.Count);
  223. var userId = schedulingList[randomNum].SchedulingUser.UserId;
  224. order.WaitForPublisherId = userId;
  225. }
  226. }
  227. catch
  228. {
  229. }
  230. }
  231. }
  232. //记录冗余归档数据
  233. if (notification.Workflow.Steps.Any(x => x.BusinessType == Share.Enums.FlowEngine.EBusinessType.Send))
  234. {
  235. order.FileUserRole = EFileUserType.Dispatch;
  236. }
  237. else
  238. {
  239. order.FileUserRole = EFileUserType.Seat;
  240. }
  241. if (order.ProcessType == EProcessType.Jiaoban)
  242. {
  243. order.FileUserRole = EFileUserType.Org;
  244. }
  245. //是否已解决
  246. order.IsResolved = notification.Dto.External == null ? false : notification.Dto.External.IsResolved;
  247. //await _orderRepository.UpdateAsync(order, cancellationToken);
  248. await _orderRepository.Updateable(order)
  249. .ExecuteCommandAsync(cancellationToken);
  250. //var callRecord = await _trCallRecordRepository.GetAsync(p => p.CallAccept == order.CallId, cancellationToken); //由CallAccept改为OtherAccept
  251. //var callRecord = await _trCallRecordRepository.GetAsync(p => p.OtherAccept == order.CallId, cancellationToken);
  252. var orderFlowDto = new OrderFlowDto
  253. {
  254. Order = _mapper.Map<OrderDto>(order),
  255. WorkflowTrace = _mapper.Map<WorkflowTraceDto>(notification.Trace)
  256. };
  257. // if (callRecord != null)
  258. // {
  259. // orderFlowDto.TrCallRecordDto = _mapper.Map<TrCallDto>(callRecord);
  260. // }
  261. if (order.SourceChannelCode == AppDefaults.SourceChannel.DianHua &&
  262. !string.IsNullOrEmpty(order.CallId))
  263. {
  264. if (_appOptions.Value.GetDefaultAppScopeConfiguration().CallCenterType == AppDefaults.CallCenterType.TianRun)
  265. {
  266. // var callRecord = await _trCallRecordRepository.GetAsync(p => p.OtherAccept == order.CallId, cancellationToken);
  267. var callRecord = await _callApplication.GetTianrunCallAsync(order.CallId, cancellationToken);
  268. if (callRecord != null)
  269. {
  270. orderFlowDto.TrCallRecordDto = _mapper.Map<TrCallDto>(callRecord);
  271. }
  272. }
  273. else if (_appOptions.Value.GetDefaultAppScopeConfiguration().CallCenterType == AppDefaults.CallCenterType.XingTang)
  274. {
  275. var call = await _callApplication.GetCallAsync(order.CallId, cancellationToken);
  276. if (call is not null)
  277. {
  278. orderFlowDto.TrCallRecordDto = _mapper.Map<TrCallDto>(call);
  279. // 工单开始办理如果获取的通话记录是呼出并且录音文件是空就不推送通话记录
  280. // 如果 通话记录是呼入, 并且没有录音文件
  281. if (_systemSettingCacheManager.OrderStartHandlerPushCallIsNull && call.Direction == ECallDirection.Out && call.AudioFile.IsNullOrEmpty())
  282. {
  283. orderFlowDto.TrCallRecordDto = null;
  284. }
  285. }
  286. }
  287. }
  288. //这里需要判断是否是警情退回
  289. orderFlowDto.IsNonPoliceReturn = notification.Dto.External == null ? false : notification.Dto.External.IsPoliceReturn;
  290. //如果是泸州,这里需要查询是否有工单延期
  291. if (_appOptions.Value.IsLuZhou)
  292. {
  293. var orderDelay = await _orderDelayRepository.Queryable().Where(p => p.OrderId == order.Id && p.DelayState == EDelayState.Pass)
  294. .OrderByDescending(p => p.ApplyDelayTime)
  295. .FirstAsync();
  296. orderFlowDto.OrderSearchDelay = _mapper.Map<OrderSearchDelayDto>(orderDelay);
  297. }
  298. await _capPublisher.PublishAsync(Hotline.Share.Mq.EventNames.HotlineOrderFiled, orderFlowDto, cancellationToken: cancellationToken);
  299. if (_systemSettingCacheManager.Snapshot)
  300. {
  301. await _publisher.PublishAsync(new SnapshotOrderFiledNotification(order.Id), PublishStrategy.ParallelWhenAll, cancellationToken);
  302. }
  303. await _orderDomainService.OrderAutomaticPublishAsync(order, cancellationToken);
  304. //try
  305. //{
  306. // //写入质检 针对受理之后直接结束的工单
  307. // await _qualityApplication.AddQualityAsync(EQualitySource.Accepted, order.Id, cancellationToken);
  308. //}
  309. //catch (Exception e)
  310. //{
  311. // _logger.LogError($"写入质检异常!orderId: {order.Id}, \r\n{e.Message}");
  312. //}
  313. //司法行政监督管理-工单处理
  314. //如果没开启则不处理
  315. var isOpenJudicialManagement = _systemSettingCacheManager.GetSetting(SettingConstants.IsOpenJudicialManagement)?.SettingValue[0];
  316. if (isOpenJudicialManagement == "true")
  317. await _publisher.PublishAsync(new JudicialManagementOrderNotify(order), PublishStrategy.ParallelWhenAll, cancellationToken);
  318. //推诿工单
  319. // await _enforcementApplication.AddPassTheBuckOrderAsync(order, _sessionContext.OrgId, _sessionContext.OrgName, cancellationToken);
  320. break;
  321. case WorkflowModuleConsts.OrderDelay:
  322. var delay = await _orderDelayRepository.GetAsync(workflow.ExternalId, cancellationToken);
  323. if (delay != null)
  324. {
  325. //delay.Flowed(workflow.FlowedUserIds, workflow.FlowedOrgIds, workflow.HandlerUsers, workflow.HandlerOrgs);
  326. delay.DelayState = isReviewPass ? EDelayState.Pass : EDelayState.NoPass;
  327. await _orderDelayRepository.Updateable(delay)
  328. .UpdateColumns(d => d.DelayState)
  329. .ExecuteCommandAsync(cancellationToken);
  330. if (isReviewPass)
  331. {
  332. //处理工单延期
  333. await _orderApplication.DelayOrderExpiredTimeAsync(delay.OrderId, delay.DelayNum,
  334. delay.DelayUnit, delay.IsProDelay, cancellationToken);
  335. }
  336. }
  337. break;
  338. case WorkflowModuleConsts.OrderTerminate:
  339. var orderTerminate = await _orderTerminateRepository.Queryable()
  340. .Where(x => x.Id == workflow.ExternalId).FirstAsync(cancellationToken);
  341. if (orderTerminate != null)
  342. {
  343. orderTerminate.Status = isReviewPass ? ETerminateStatus.End : ETerminateStatus.Refuse;
  344. await _orderTerminateRepository.UpdateAsync(orderTerminate, cancellationToken);
  345. }
  346. break;
  347. }
  348. }
  349. catch (Exception e)
  350. {
  351. _logger.LogError($"{nameof(WorkflowEndHandler)}异常,{e}");
  352. throw;
  353. }
  354. }
  355. }