ApptaskDomainService.cs 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. using FluentValidation;
  2. using Hotline.Share.Dtos.BatchTask;
  3. using Hotline.Share.Enums.BatchTask;
  4. using Hotline.Validators.BatchTask;
  5. using MapsterMapper;
  6. using Microsoft.Extensions.Logging;
  7. using XF.Domain.Dependency;
  8. using XF.Domain.Exceptions;
  9. using XF.Domain.Repository;
  10. using static Lucene.Net.Util.Fst.Util;
  11. namespace Hotline.BatchTask;
  12. public class ApptaskDomainService : IApptaskDomainService, IScopeDependency
  13. {
  14. private readonly IRepository<Apptask> _apptaskRepository;
  15. private readonly IRepository<ApptaskItem> _apptaskItemRepository;
  16. private readonly IMapper _mapper;
  17. private readonly ILogger<ApptaskDomainService> _logger;
  18. public ApptaskDomainService(
  19. IRepository<Apptask> apptaskRepository,
  20. IRepository<ApptaskItem> apptaskItemRepository,
  21. IMapper mapper,
  22. ILogger<ApptaskDomainService> logger)
  23. {
  24. _apptaskRepository = apptaskRepository;
  25. _apptaskItemRepository = apptaskItemRepository;
  26. _mapper = mapper;
  27. _logger = logger;
  28. }
  29. /// <summary>
  30. /// 新增任务
  31. /// </summary>
  32. public async Task<string> AddAsync(AddApptaskRequest request, CancellationToken cancellation)
  33. {
  34. var validator = new ApptaskValidator();
  35. var result = validator.Validate(request);
  36. if (!result.IsValid)
  37. throw new ValidationException(result.Errors.FirstOrDefault()?.ErrorMessage);
  38. var apptask = new Apptask
  39. {
  40. Name = request.Name,
  41. Description = request.Description,
  42. TaskType = request.TaskType,
  43. ApptaskItems = request.ApptaskItems.Select(d => new ApptaskItem
  44. {
  45. BusinessId = d.BusinessId,
  46. TaskType = request.TaskType,
  47. TaskStatus = ETaskStatus.Waiting,
  48. TaskParams = d.TaskParams is null ? null : System.Text.Json.JsonSerializer.Serialize(d.TaskParams),
  49. TryLimit = request.TryLimit,
  50. Priority = request.Priority ?? 9,
  51. }).ToList()
  52. };
  53. if (string.IsNullOrEmpty(apptask.Name))
  54. apptask.CreateName();
  55. await _apptaskRepository.AddNav(apptask)
  56. .Include(d => d.ApptaskItems)
  57. .ExecuteCommandAsync();
  58. return apptask.Id;
  59. }
  60. /// <summary>
  61. /// 查询任务进度
  62. /// </summary>
  63. /// <returns></returns>
  64. public async Task<ApptaskProgressDto> GetProgressAsync(string taskId, CancellationToken cancellation)
  65. {
  66. var apptask = await _apptaskRepository.Queryable()
  67. .Includes(d => d.ApptaskItems)
  68. .FirstAsync(d => d.Id == taskId, cancellation);
  69. if (apptask is null)
  70. throw new UserFriendlyException("无效任务编号");
  71. return _mapper.Map<ApptaskProgressDto>(apptask);
  72. }
  73. /// <summary>
  74. /// 终止任务
  75. /// </summary>
  76. /// <param name="taskId"></param>
  77. /// <param name="cancellation"></param>
  78. /// <returns></returns>
  79. public async Task TerminalTaskAsync(string taskId, CancellationToken cancellation)
  80. {
  81. var apptask = await _apptaskRepository.Queryable()
  82. .Includes(d => d.ApptaskItems.Where(x => x.Tries < x.TryLimit
  83. && (x.TaskStatus == ETaskStatus.Waiting || x.TaskStatus == ETaskStatus.Failed)).ToList())
  84. .FirstAsync(d => d.Id == taskId, cancellation);
  85. if (apptask is null)
  86. throw new UserFriendlyException("无效任务编号");
  87. if (apptask.ApptaskItems.Count == 0) return;
  88. var Succeed = 0;
  89. foreach (var item in apptask.ApptaskItems)
  90. {
  91. item.TaskStatus = ETaskStatus.Terminated;
  92. var result = await _apptaskItemRepository.Updateable(item).ExecuteCommandWithOptLockAsync();
  93. if (result != 0) Succeed++;
  94. }
  95. if (Succeed != apptask.ApptaskItems.Count)
  96. await TerminalTaskAsync(taskId, cancellation);
  97. }
  98. /// <summary>
  99. /// 获取一个待执行的任务
  100. /// </summary>
  101. /// <param name="cancellation"></param>
  102. /// <returns></returns>
  103. public async Task<ApptaskItem?> GetWaitingTaskAsync(CancellationToken cancellation)
  104. {
  105. var taskItems = await _apptaskItemRepository.Queryable()
  106. .Where(d => d.Tries < d.TryLimit
  107. && (d.TaskStatus == ETaskStatus.Waiting || d.TaskStatus == ETaskStatus.Failed))
  108. .OrderBy(d => d.CreationTime)
  109. .Take(10)
  110. .ToListAsync(cancellation);
  111. if (taskItems.Count == 0) return null;
  112. foreach (var item in taskItems)
  113. {
  114. item.SetProcessing();
  115. item.Tries++;
  116. var result = await _apptaskItemRepository.Updateable(item).ExecuteCommandWithOptLockAsync();
  117. if (result != 0)
  118. return item;
  119. }
  120. return await GetWaitingTaskAsync(cancellation);
  121. }
  122. /// <summary>
  123. /// 执行任务
  124. /// </summary>
  125. /// <param name="executor"></param>
  126. /// <param name="apptaskItem"></param>
  127. /// <param name="cancellation"></param>
  128. /// <returns></returns>
  129. public async Task ExecuteAsync<TRequest>(IApptaskExecutor<TRequest> executor, ApptaskItem apptaskItem, CancellationToken cancellation)
  130. {
  131. try
  132. {
  133. TRequest request = default;
  134. if (!string.IsNullOrEmpty(apptaskItem.TaskParams))
  135. {
  136. request = System.Text.Json.JsonSerializer.Deserialize<TRequest>(apptaskItem.TaskParams);
  137. if (request is null)
  138. throw new UserFriendlyException("任务参数反序列化异常");
  139. }
  140. var result = await executor.ExecuteAsync(request, cancellation);
  141. apptaskItem.TaskStatus = result.IsSuccess ? ETaskStatus.Succeeded : ETaskStatus.Failed;
  142. apptaskItem.Message = result.Message;
  143. apptaskItem.TaskEndTime = DateTime.Now;
  144. }
  145. catch (Exception e)
  146. {
  147. _logger.LogError("批量任务执行异常:{err}", e.Message);
  148. apptaskItem.TaskStatus = ETaskStatus.Failed;
  149. apptaskItem.Message = "批量任务执行异常";
  150. apptaskItem.TaskEndTime = DateTime.Now;
  151. }
  152. finally
  153. {
  154. await _apptaskItemRepository.Updateable(apptaskItem)
  155. .UpdateColumns(d => new
  156. {
  157. d.TaskStatus,
  158. d.Message,
  159. d.TaskEndTime
  160. })
  161. .ExecuteCommandAsync(cancellation);
  162. }
  163. }
  164. }