OrderDomainService.cs 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. using Hotline.FlowEngine;
  2. using Hotline.Share.Enums.Order;
  3. using Microsoft.Extensions.Logging;
  4. using XF.Domain.Authentications;
  5. using XF.Domain.Cache;
  6. using XF.Domain.Dependency;
  7. using XF.Domain.Entities;
  8. using XF.Domain.Exceptions;
  9. using XF.Domain.Extensions;
  10. namespace Hotline.Orders;
  11. public class OrderDomainService : IOrderDomainService, IScopeDependency
  12. {
  13. private const string OrderNoPrefix = "OrderNo:";
  14. private readonly IOrderRepository _orderRepository;
  15. private readonly ITypedCache<CacheOrderNO> _cacheOrderNo;
  16. private readonly ISessionContext _sessionContext;
  17. private readonly ILogger<OrderDomainService> _logger;
  18. public OrderDomainService(
  19. IOrderRepository orderRepository,
  20. ITypedCache<CacheOrderNO> cacheOrderNo,
  21. ISessionContext sessionContext,
  22. ILogger<OrderDomainService> logger)
  23. {
  24. _orderRepository = orderRepository;
  25. _cacheOrderNo = cacheOrderNo;
  26. _sessionContext = sessionContext;
  27. _logger = logger;
  28. }
  29. public async Task<string> AddAsync(Order order, CancellationToken cancellationToken)
  30. {
  31. if (order.OrderType == EOrderType.MarketSupervisionBy12315 && order.AcceptType == EAcceptType.Complain && order.OrderComplain == null)
  32. throw UserFriendlyException.SameMessage("非法投诉参数");
  33. if (order.OrderType == EOrderType.MarketSupervisionBy12315 && order.AcceptType == EAcceptType.Report && order.OrderReport == null)
  34. throw UserFriendlyException.SameMessage("非法举报参数");
  35. order.Init(_sessionContext.RequiredUserId);
  36. order.No = GenerateNewOrderNo();
  37. return await _orderRepository.AddOrderNavAsync(order, cancellationToken);
  38. }
  39. /// <summary>
  40. /// 归档
  41. /// </summary>
  42. public async Task FileAsync(string? orderId, CancellationToken cancellationToken)
  43. {
  44. var order = await GetOrderAsync(orderId, cancellationToken);
  45. if (order.Status is EOrderStatus.Filed) return;
  46. order.Status = EOrderStatus.Filed;
  47. await _orderRepository.UpdateAsync(order, cancellationToken);
  48. }
  49. /// <summary>
  50. /// 接办工单(查看详情视为接办)
  51. /// </summary>
  52. public async Task AcceptAsync(string? orderId, CancellationToken cancellationToken)
  53. {
  54. var order = await GetOrderAsync(orderId, cancellationToken);
  55. CheckOrderIfFiled(order);
  56. order.Accept();
  57. await _orderRepository.UpdateAsync(order, cancellationToken);
  58. }
  59. /// <summary>
  60. /// 工单办理(每个节点都会触发)
  61. /// </summary>
  62. public async Task ManageFlowNextAsync(FlowAssignMode assignMode,
  63. bool isCountersignStart, bool isCountersignEnd,
  64. string? orderId, DateTime? currentStepTime, string? CurrentStepName, DateTime expiredTime,
  65. CancellationToken cancellationToken)
  66. {
  67. _logger.LogInformation($"{nameof(ManageFlowNextAsync)}");
  68. var order = await GetOrderAsync(orderId, cancellationToken);
  69. CheckOrderIfFiled(order);
  70. _logger.LogInformation($"{nameof(ManageFlowNextAsync)}, before: order.AssignUserIds: {string.Join(',', order.AssignUserIds)}, order.AssignOrgs: {string.Join(',', order.AssignOrgCodes)}");
  71. //更新指派信息
  72. order.Assign(assignMode.FlowAssignType, assignMode.Handlers);
  73. _logger.LogInformation($"{nameof(ManageFlowNextAsync)}, after: order.AssignUserIds: {string.Join(',', order.AssignUserIds)}, order.AssignOrgs: {string.Join(',', order.AssignOrgCodes)}");
  74. //更新流转信息
  75. order.ManageFlow(isCountersignStart, isCountersignEnd, currentStepTime, CurrentStepName, expiredTime);
  76. _logger.LogInformation($"更新流转信息 over, order.AssignUserIds: {string.Join(',', order.AssignUserIds)}");
  77. await _orderRepository.UpdateAsync(order, cancellationToken);
  78. _logger.LogInformation($"update db over, order.AssignUserIds: {string.Join(',', order.AssignUserIds)}");
  79. //todo push msg to next step handler
  80. }
  81. /// <summary>
  82. /// 工单最终办理(此完结流程并未结束)
  83. /// </summary>
  84. /// <remarks>
  85. /// 需求:工单实际办理部门办理完成视为工单最终办理完结,
  86. /// 实际办理部门指:1.第一个汇总节点的前一个非汇总节点代表的办理部门 2.汇总以后如果再次开启会签需将工单回退到未办理完结的状态重新等待下一个汇总节点
  87. /// </remarks>
  88. /// <returns></returns>
  89. public async Task FinalManageAsync(string? orderId, CancellationToken cancellationToken)
  90. {
  91. var order = await GetOrderAsync(orderId, cancellationToken);
  92. CheckOrderIfFiled(order);
  93. order.FinalManage();
  94. await _orderRepository.UpdateAsync(order, cancellationToken);
  95. }
  96. /// <summary>
  97. /// 取消最终办理(汇总以后又重新指派到非汇总汇总节点办理的场景)
  98. /// </summary>
  99. /// <returns></returns>
  100. public async Task RecallFinalManageAsync(string? orderId, CancellationToken cancellationToken)
  101. {
  102. var order = await GetOrderAsync(orderId, cancellationToken);
  103. CheckOrderIfFiled(order);
  104. order.RecallFinalManage();
  105. await _orderRepository.UpdateAsync(order, cancellationToken);
  106. }
  107. /// <summary>
  108. /// 工单办理流程流转到结束节点
  109. /// </summary>
  110. public async Task ManageFlowEndAsync(string? orderId, CancellationToken cancellationToken)
  111. {
  112. var order = await GetOrderAsync(orderId, cancellationToken);
  113. CheckOrderIfFiled(order);
  114. order.Filed();
  115. await _orderRepository.UpdateAsync(order, cancellationToken);
  116. }
  117. #region private
  118. private async Task<Order> GetOrderAsync(string? orderId, CancellationToken cancellationToken)
  119. {
  120. if (string.IsNullOrEmpty(orderId))
  121. throw UserFriendlyException.SameMessage("无效工单编号");
  122. var order = await _orderRepository.GetAsync(orderId, cancellationToken);
  123. if (order == null)
  124. throw new UserFriendlyException($"无效工单编号, orderId: {orderId}", "无效工单编号");
  125. return order;
  126. }
  127. private void CheckOrderIfFiled(Order order)
  128. {
  129. if (order.Status is EOrderStatus.Filed)
  130. throw UserFriendlyException.SameMessage("工单已归档");
  131. }
  132. private string GenerateNewOrderNo()
  133. {
  134. var today = DateTime.Today;
  135. var cacheKey = $"{OrderNoPrefix}{today:yyyyMMdd}";
  136. var cacheOrderNo = _cacheOrderNo.GetOrAdd(cacheKey, f =>
  137. {
  138. var todayOrderCount = _orderRepository.Queryable(true)
  139. .CountAsync(d => d.CreationTime.Date == today.Date)
  140. .GetAwaiter()
  141. .GetResult();
  142. return new CacheOrderNO { TotalCount = todayOrderCount };
  143. }, ExpireMode.Absolute, TimeSpan.FromDays(1));
  144. cacheOrderNo.TotalCount += 1;
  145. var no = GenerateOrderNo(today, cacheOrderNo.TotalCount);
  146. _cacheOrderNo.Update(cacheKey, d => cacheOrderNo);
  147. return no;
  148. }
  149. private string GenerateOrderNo(DateTime today, int count)
  150. {
  151. return $"{today:yyyyMMdd}{count.ToString("000000")}";
  152. }
  153. #endregion
  154. }
  155. public class CacheOrderNO
  156. {
  157. public int TotalCount { get; set; }
  158. }