WorkflowEndHandler.cs 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346
  1. using DotNetCore.CAP;
  2. using Hotline.Application.CallCenter;
  3. using Hotline.Application.JudicialManagement;
  4. using Hotline.Application.Orders;
  5. using Hotline.Application.Quality;
  6. using Hotline.Article;
  7. using Hotline.CallCenter.Calls;
  8. using Hotline.CallCenter.Configs;
  9. using Hotline.CallCenter.Tels;
  10. using Hotline.FlowEngine.Notifications;
  11. using Hotline.FlowEngine.WorkflowModules;
  12. using Hotline.KnowledgeBase;
  13. using Hotline.Orders;
  14. using Hotline.Settings.TimeLimits;
  15. using Hotline.Share.Dtos;
  16. using Hotline.Share.Dtos.FlowEngine.Workflow;
  17. using Hotline.Share.Dtos.Order;
  18. using Hotline.Share.Dtos.TrCallCenter;
  19. using Hotline.Share.Enums.Order;
  20. using Hotline.Share.Enums.Quality;
  21. using MapsterMapper;
  22. using MediatR;
  23. using Microsoft.Extensions.Logging;
  24. using Microsoft.Extensions.Options;
  25. using StackExchange.Redis;
  26. using XF.Domain.Authentications;
  27. using XF.Domain.Repository;
  28. namespace Hotline.Application.Handlers.FlowEngine;
  29. public class WorkflowEndHandler : INotificationHandler<EndWorkflowNotify>
  30. {
  31. private readonly IKnowledgeDomainService _knowledgeDomainService;
  32. private readonly IOrderDomainService _orderDomainService;
  33. private readonly IOrderApplication _orderApplication;
  34. private readonly ITelDomainService _telDomainService;
  35. private readonly IOrderRepository _orderRepository;
  36. private readonly ICapPublisher _capPublisher;
  37. private readonly IMapper _mapper;
  38. private readonly IRepository<OrderVisitDetail> _orderVisitedDetailRepository;
  39. private readonly IRepository<OrderScreen> _orderScreenRepository;
  40. private readonly IRepository<Circular> _circularRepository;
  41. private readonly IRepositorySqlSugar<CircularReadGroup> _circularReadGroupRepository;
  42. private readonly IRepositorySqlSugar<CircularRecord> _circularRecordRepository;
  43. private readonly ICircularRecordDomainService _circularRecordDomainService;
  44. private readonly IRepository<OrderVisit> _orderVisitRepository;
  45. private readonly IRepository<OrderDelay> _orderDelayRepository;
  46. private readonly IRepository<OrderVisitApply> _orderVisitApplyRepository;
  47. private readonly ITimeLimitDomainService _timeLimitDomainService;
  48. private readonly IMediator _mediator;
  49. private readonly ILogger<WorkflowEndHandler> _logger;
  50. private readonly IKnowledgeRepository _knowledgeRepository;
  51. private readonly IKnowledgeWorkFlowRepository _knowledgeWorkFlowRepository;
  52. // private readonly IRepository<TrCallRecord> _trCallRecordRepository;
  53. private readonly IQualityApplication _qualityApplication;
  54. private readonly IEnforcementApplication _enforcementApplication;
  55. private readonly ICallApplication _callApplication;
  56. private readonly IOptionsSnapshot<CallCenterConfiguration> _callcenterOptions;
  57. private readonly ISessionContext _sessionContext;
  58. public WorkflowEndHandler(
  59. IKnowledgeDomainService knowledgeDomainService,
  60. IOrderDomainService orderDomainService,
  61. IOrderApplication orderApplication,
  62. ITelDomainService telDomainService,
  63. IOrderRepository orderRepository,
  64. ICapPublisher capPublisher,
  65. IMapper mapper,
  66. IRepository<OrderVisitDetail> orderVisitedDetailRepository,
  67. IRepository<OrderScreen> orderScreenRepository,
  68. IRepository<Bulletin> bulletinRepository,
  69. IRepository<Circular> circularRepository,
  70. IRepository<CircularReadGroup> circularReadGroupRepository,
  71. IRepository<CircularRecord> circularRecordRepository,
  72. ICircularRecordDomainService circularRecordDomainService,
  73. IRepository<OrderDelay> orderDelayRepository,
  74. IRepository<OrderVisit> orderVisitRepository,
  75. IRepository<OrderVisitApply> orderVisitApplyRepository,
  76. ITimeLimitDomainService timeLimitDomainService,
  77. IMediator mediator,
  78. IKnowledgeRepository knowledgeRepository,
  79. IKnowledgeWorkFlowRepository knowledgeWorkFlowRepository,
  80. ILogger<WorkflowEndHandler> logger,
  81. // IRepository<TrCallRecord> trCallRecordRepository,
  82. IQualityApplication qualityApplication,
  83. IEnforcementApplication enforcementApplication,
  84. ICallApplication callApplication,
  85. IOptionsSnapshot<CallCenterConfiguration> callcenterOptions,
  86. ISessionContext sessionContext)
  87. {
  88. _knowledgeDomainService = knowledgeDomainService;
  89. _orderDomainService = orderDomainService;
  90. _orderApplication = orderApplication;
  91. _telDomainService = telDomainService;
  92. _orderRepository = orderRepository;
  93. _capPublisher = capPublisher;
  94. _mapper = mapper;
  95. _orderScreenRepository = orderScreenRepository;
  96. _orderVisitedDetailRepository = orderVisitedDetailRepository;
  97. _circularRepository = circularRepository;
  98. _circularReadGroupRepository = circularReadGroupRepository;
  99. _circularRecordRepository = circularRecordRepository;
  100. _circularRecordDomainService = circularRecordDomainService;
  101. _orderDelayRepository = orderDelayRepository;
  102. _orderVisitRepository = orderVisitRepository;
  103. _orderVisitApplyRepository = orderVisitApplyRepository;
  104. _timeLimitDomainService = timeLimitDomainService;
  105. _mediator = mediator;
  106. _logger = logger;
  107. _knowledgeRepository = knowledgeRepository;
  108. _knowledgeWorkFlowRepository = knowledgeWorkFlowRepository;
  109. // _trCallRecordRepository = trCallRecordRepository;
  110. _qualityApplication = qualityApplication;
  111. _enforcementApplication = enforcementApplication;
  112. _callApplication = callApplication;
  113. _callcenterOptions = callcenterOptions;
  114. _sessionContext = sessionContext;
  115. }
  116. /// <summary>Handles a notification</summary>
  117. /// <param name="notification">The notification</param>
  118. /// <param name="cancellationToken">Cancellation token</param>
  119. public async Task Handle(EndWorkflowNotify notification, CancellationToken cancellationToken)
  120. {
  121. var workflow = notification.Workflow;
  122. //审批是否通过
  123. var isReviewPass = workflow.IsReviewPass();
  124. switch (workflow.ModuleCode)
  125. {
  126. case WorkflowModuleConsts.KnowledgeAdd://新增知识库
  127. case WorkflowModuleConsts.KnowledgeUpdate://修改知识库
  128. case WorkflowModuleConsts.KnowledgeDelete://删除知识库
  129. var knowledgeWork = await _knowledgeWorkFlowRepository.Queryable().Where(x => x.Id == workflow.ExternalId).FirstAsync(cancellationToken);
  130. var knowledge = await _knowledgeRepository.Queryable().Where(x => x.Id == knowledgeWork.KnowledgeId).FirstAsync(cancellationToken);
  131. knowledge.Flowed(workflow.FlowedUserIds, workflow.FlowedOrgIds, workflow.HandlerUsers, workflow.HandlerOrgs);
  132. await _knowledgeRepository.UpdateAsync(knowledge, cancellationToken);
  133. if (isReviewPass)
  134. {
  135. await _knowledgeDomainService.EndWorkKnowledge(workflow, cancellationToken);
  136. }
  137. else
  138. {
  139. await _knowledgeDomainService.TerminateWorkKnowledge(workflow, cancellationToken);
  140. }
  141. break;
  142. case WorkflowModuleConsts.TelRestApply:
  143. await _telDomainService.TelRestApplyPassAsync(workflow.ExternalId, cancellationToken);
  144. break;
  145. case WorkflowModuleConsts.OrderHandle:
  146. var order = await _orderDomainService.GetOrderAsync(workflow.ExternalId,
  147. withExtension: true, cancellationToken: cancellationToken);
  148. //order.CheckIfFiled();
  149. order.UpdateHandlingStatus(workflow.IsInCountersign);
  150. _mapper.Map(workflow, order);
  151. var now = DateTime.Now;
  152. var handleDuration = order.CenterToOrgTime.HasValue && order.ActualHandleTime.HasValue
  153. ? _timeLimitDomainService.CalcWorkTime(order.CenterToOrgTime.Value,
  154. order.ActualHandleTime.Value, order.ProcessType is EProcessType.Zhiban)
  155. : 0;
  156. var fileDuration = order.CenterToOrgTime.HasValue
  157. ? _timeLimitDomainService.CalcWorkTime(order.CenterToOrgTime.Value,
  158. now, order.ProcessType is EProcessType.Zhiban)
  159. : 0;
  160. var allDuration = order.StartTime.HasValue
  161. ? _timeLimitDomainService.CalcWorkTime(order.StartTime.Value, now,
  162. order.ProcessType is EProcessType.Zhiban)
  163. : 0;
  164. var creationTimeHandleDurationWorkday = order.ActualHandleTime.HasValue
  165. ? _timeLimitDomainService.CalcWorkTime(order.CreationTime, now,
  166. order.ProcessType is EProcessType.Zhiban)
  167. : 0;
  168. var centerToOrgHandleDurationWorkday = order.ActualHandleTime.HasValue && order.CenterToOrgTime.HasValue
  169. ? _timeLimitDomainService.CalcWorkTime(order.CenterToOrgTime.Value, now,
  170. order.ProcessType is EProcessType.Zhiban)
  171. : 0;
  172. order.File(now, handleDuration, fileDuration, allDuration, creationTimeHandleDurationWorkday, centerToOrgHandleDurationWorkday);
  173. order.FileUserId = notification.Trace.HandlerId;
  174. order.FileUserName = notification.Trace.HandlerName;
  175. order.FileUserOrgId = notification.Trace.HandlerOrgId;
  176. order.FileUserOrgName = notification.Trace.HandlerOrgName;
  177. order.FileOrgIsCenter = notification.Trace.HandlerOrgIsCenter;
  178. order.FileOpinion = notification.Dto.Opinion;
  179. //记录冗余归档数据
  180. if (notification.Workflow.Steps.Any(x => x.BusinessType == Share.Enums.FlowEngine.EBusinessType.Send))
  181. {
  182. order.FileUserRole = EFileUserType.Dispatch;
  183. }
  184. else
  185. {
  186. order.FileUserRole = EFileUserType.Seat;
  187. }
  188. if (order.ProcessType == EProcessType.Jiaoban)
  189. {
  190. order.FileUserRole = EFileUserType.Org;
  191. }
  192. //是否已解决
  193. order.IsResolved = notification.Dto.External == null ? false : notification.Dto.External.IsResolved;
  194. await _orderRepository.UpdateAsync(order, cancellationToken);
  195. //var callRecord = await _trCallRecordRepository.GetAsync(p => p.CallAccept == order.CallId, cancellationToken); //由CallAccept改为OtherAccept
  196. //var callRecord = await _trCallRecordRepository.GetAsync(p => p.OtherAccept == order.CallId, cancellationToken);
  197. var orderFlowDto = new OrderFlowDto
  198. {
  199. Order = _mapper.Map<OrderDto>(order),
  200. WorkflowTrace = _mapper.Map<WorkflowTraceDto>(notification.Trace)
  201. };
  202. // if (callRecord != null)
  203. // {
  204. // orderFlowDto.TrCallRecordDto = _mapper.Map<TrCallDto>(callRecord);
  205. // }
  206. if (order.SourceChannelCode == AppDefaults.SourceChannel.DianHua &&
  207. !string.IsNullOrEmpty(order.CallId))
  208. {
  209. if (_callcenterOptions.Value.CallCenterType == AppDefaults.CallCenterType.TianRun)
  210. {
  211. // var callRecord = await _trCallRecordRepository.GetAsync(p => p.OtherAccept == order.CallId, cancellationToken);
  212. var callRecord = await _callApplication.GetTianrunCallAsync(order.CallId, cancellationToken);
  213. if (callRecord != null)
  214. {
  215. orderFlowDto.TrCallRecordDto = _mapper.Map<TrCallDto>(callRecord);
  216. }
  217. }
  218. else if (_callcenterOptions.Value.CallCenterType == AppDefaults.CallCenterType.XingTang)
  219. {
  220. var call = await _callApplication.GetCallAsync(order.CallId, cancellationToken);
  221. if (call is not null)
  222. orderFlowDto.TrCallRecordDto = _mapper.Map<TrCallDto>(call);
  223. }
  224. }
  225. //这里需要判断是否是警情退回
  226. orderFlowDto.IsNonPoliceReturn = notification.Dto.External == null ? false : notification.Dto.External.IsPoliceReturn;
  227. await _capPublisher.PublishAsync(Hotline.Share.Mq.EventNames.HotlineOrderFiled, orderFlowDto, cancellationToken: cancellationToken);
  228. try
  229. {
  230. //写入质检 针对受理之后直接结束的工单
  231. await _qualityApplication.AddQualityAsync(EQualitySource.Accepted, order.Id, cancellationToken);
  232. }
  233. catch (Exception e)
  234. {
  235. _logger.LogError($"写入质检异常!orderId: {order.Id}, \r\n{e.Message}");
  236. }
  237. //司法行政监督管理-工单处理
  238. await _enforcementApplication.AddEnforcementOrderAsync(order, cancellationToken);
  239. //推诿工单
  240. // await _enforcementApplication.AddPassTheBuckOrderAsync(order, _sessionContext.OrgId, _sessionContext.OrgName, cancellationToken);
  241. break;
  242. case WorkflowModuleConsts.OrderScreen:
  243. var screen = await _orderScreenRepository.GetAsync(workflow.ExternalId, cancellationToken);
  244. if (screen != null)
  245. {
  246. screen.Flowed(workflow.FlowedUserIds, workflow.FlowedOrgIds, workflow.HandlerUsers, workflow.HandlerOrgs);
  247. if (isReviewPass)
  248. {
  249. screen.Status = EScreenStatus.End;
  250. screen.ReplyContent = workflow.ActualOpinion;
  251. await _orderScreenRepository.UpdateAsync(screen, cancellationToken);
  252. var visitDetail =
  253. await _orderVisitedDetailRepository.GetAsync(screen.VisitDetailId, cancellationToken);
  254. if (visitDetail != null)
  255. {
  256. var screenSatisfy = new Kv() { Key = "-1", Value = "视为满意" };
  257. visitDetail.OrgProcessingResults = screenSatisfy;
  258. //visitDetail.OrgHandledAttitude = screenSatisfy;
  259. await _orderVisitedDetailRepository.UpdateAsync(visitDetail, cancellationToken);
  260. // 修改主表当前评价结果
  261. await _orderVisitRepository.Updateable().SetColumns(v => new OrderVisit() { NowEvaluate = screenSatisfy }).Where(v => v.Id == visitDetail.VisitId).ExecuteCommandAsync(cancellationToken);
  262. //获取回访信息
  263. var visit = await _orderVisitRepository.Queryable().Includes(x => x.Order)
  264. .Includes(x => x.OrderVisitDetails)
  265. .Where(x => x.Id == screen.VisitId).FirstAsync(cancellationToken);
  266. if (visit != null)
  267. {
  268. //获取回访明细
  269. var visitDe = visit.OrderVisitDetails.First(x => x.Id == screen.VisitDetailId);
  270. //推省上
  271. await _capPublisher.PublishAsync(Hotline.Share.Mq.EventNames.HotlineOrderScreenApplyed,
  272. new PublishVisitDto()
  273. {
  274. Order = _mapper.Map<OrderDto>(visit.Order),
  275. No = visit.No,
  276. VisitType = visit.VisitType,
  277. VisitName = visit.CreatorName,
  278. VisitTime = visit.VisitTime,
  279. VisitRemark = string.IsNullOrEmpty(visitDe.VisitContent) ? screenSatisfy.Value : visitDe.VisitContent,
  280. AreaCode = visit.Order.AreaCode!,
  281. SubjectResultSatifyCode = visitDe.OrgProcessingResults?.Key,
  282. FirstSatisfactionCode = visit.Order.FirstVisitResultCode!,
  283. ClientGuid = ""
  284. });
  285. //推门户
  286. await _capPublisher.PublishAsync(Hotline.Share.Mq.EventNames.HotlineOrderVisitedWeb, new PublishVisitAllDto()
  287. {
  288. Id = visit.Id,
  289. Order = _mapper.Map<OrderDto>(visit.Order),
  290. OrderVisitDetails = _mapper.Map<List<VisitDetailDto>>(visit.OrderVisitDetails),
  291. VisitName = visit.CreatorName,
  292. VisitTime = visit.VisitTime,
  293. VisitType = visit.VisitType,
  294. VisitState = visit.VisitState,
  295. PublishTime = visit.PublishTime,
  296. }, cancellationToken: cancellationToken);
  297. }
  298. }
  299. }
  300. else
  301. {
  302. await _orderRepository.OrderScreenRevisionVisit(screen.VisitId, true, cancellationToken);
  303. screen.Status = EScreenStatus.Refuse;
  304. screen.ReplyContent = workflow.ActualOpinion;
  305. await _orderScreenRepository.UpdateAsync(screen, cancellationToken);
  306. }
  307. }
  308. break;
  309. case WorkflowModuleConsts.OrderDelay:
  310. var delay = await _orderDelayRepository.GetAsync(workflow.ExternalId, cancellationToken);
  311. if (delay != null)
  312. {
  313. delay.Flowed(workflow.FlowedUserIds, workflow.FlowedOrgIds, workflow.HandlerUsers, workflow.HandlerOrgs);
  314. if (isReviewPass)
  315. {
  316. delay.DelayState = isReviewPass ? EDelayState.Pass : EDelayState.NoPass;
  317. await _orderDelayRepository.UpdateAsync(delay, cancellationToken);
  318. //处理工单延期
  319. await _orderApplication.DelayOrderExpiredTimeAsync(delay.OrderId, delay.DelayNum,
  320. delay.DelayUnit,delay.IsProDelay, cancellationToken);
  321. }
  322. else
  323. {
  324. delay.DelayState = EDelayState.NoPass;
  325. await _orderDelayRepository.UpdateAsync(delay, cancellationToken);
  326. }
  327. }
  328. break;
  329. }
  330. }
  331. }