ApptaskDomainService.cs 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166
  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(AddApptaskDto dto, CancellationToken cancellation)
  33. {
  34. var validator = new ApptaskValidator();
  35. var result = validator.Validate(dto);
  36. if (!result.IsValid)
  37. throw new ValidationException(result.Errors.FirstOrDefault()?.ErrorMessage);
  38. var apptask = _mapper.Map<Apptask>(dto);
  39. if (string.IsNullOrEmpty(apptask.Name))
  40. apptask.CreateName();
  41. await _apptaskRepository.AddNav(apptask)
  42. .Include(d => d.ApptaskItems)
  43. .ExecuteCommandAsync();
  44. return apptask.Id;
  45. }
  46. /// <summary>
  47. /// 查询任务进度
  48. /// </summary>
  49. /// <returns></returns>
  50. public async Task<ApptaskProgressDto> GetProgressAsync(string taskId, CancellationToken cancellation)
  51. {
  52. var apptask = await _apptaskRepository.Queryable()
  53. .Includes(d => d.ApptaskItems)
  54. .FirstAsync(d => d.Id == taskId, cancellation);
  55. if (apptask is null)
  56. throw new UserFriendlyException("无效任务编号");
  57. return _mapper.Map<ApptaskProgressDto>(apptask);
  58. }
  59. /// <summary>
  60. /// 终止任务
  61. /// </summary>
  62. /// <param name="taskId"></param>
  63. /// <param name="cancellation"></param>
  64. /// <returns></returns>
  65. public async Task TerminalTaskAsync(string taskId, CancellationToken cancellation)
  66. {
  67. var apptask = await _apptaskRepository.Queryable()
  68. .Includes(d => d.ApptaskItems.Where(x => x.Tries < x.TryLimit
  69. && (x.TaskStatus == ETaskStatus.Waiting || x.TaskStatus == ETaskStatus.Failed)).ToList())
  70. .FirstAsync(d => d.Id == taskId, cancellation);
  71. if (apptask is null)
  72. throw new UserFriendlyException("无效任务编号");
  73. if (apptask.ApptaskItems.Count == 0) return;
  74. var Succeed = 0;
  75. foreach (var item in apptask.ApptaskItems)
  76. {
  77. item.TaskStatus = ETaskStatus.Terminated;
  78. var result = await _apptaskItemRepository.Updateable(item).ExecuteCommandWithOptLockAsync();
  79. if (result != 0) Succeed++;
  80. }
  81. if (Succeed != apptask.ApptaskItems.Count)
  82. await TerminalTaskAsync(taskId, cancellation);
  83. }
  84. /// <summary>
  85. /// 获取一个待执行的任务
  86. /// </summary>
  87. /// <param name="cancellation"></param>
  88. /// <returns></returns>
  89. public async Task<ApptaskItem?> GetWaitingTaskAsync(CancellationToken cancellation)
  90. {
  91. var taskItems = await _apptaskItemRepository.Queryable()
  92. .Where(d => d.Tries < d.TryLimit
  93. && (d.TaskStatus == ETaskStatus.Waiting || d.TaskStatus == ETaskStatus.Failed))
  94. .OrderBy(d => d.CreationTime)
  95. .Take(10)
  96. .ToListAsync(cancellation);
  97. if (taskItems.Count == 0) return null;
  98. foreach (var item in taskItems)
  99. {
  100. item.SetProcessing();
  101. item.Tries++;
  102. var result = await _apptaskItemRepository.Updateable(item).ExecuteCommandWithOptLockAsync();
  103. if (result != 0)
  104. return item;
  105. }
  106. return await GetWaitingTaskAsync(cancellation);
  107. }
  108. /// <summary>
  109. /// 执行任务
  110. /// </summary>
  111. /// <param name="executor"></param>
  112. /// <param name="apptaskItem"></param>
  113. /// <param name="cancellation"></param>
  114. /// <returns></returns>
  115. public async Task ExecuteAsync<TRequest>(IApptaskExecutor<TRequest> executor, ApptaskItem apptaskItem, CancellationToken cancellation)
  116. {
  117. try
  118. {
  119. TRequest request = default;
  120. if (apptaskItem.TaskParams is not null)
  121. {
  122. request = System.Text.Json.JsonSerializer.Deserialize<TRequest>(apptaskItem.TaskParams);
  123. if (request is null)
  124. throw new UserFriendlyException("任务参数反序列化异常");
  125. }
  126. var result = await executor.ExecuteAsync(request, cancellation);
  127. apptaskItem.TaskStatus = result.IsSuccess ? ETaskStatus.Succeeded : ETaskStatus.Failed;
  128. apptaskItem.Message = result.Message;
  129. apptaskItem.TaskEndTime = DateTime.Now;
  130. }
  131. catch (Exception e)
  132. {
  133. _logger.LogError("批量任务执行异常:{err}", e.Message);
  134. apptaskItem.TaskStatus = ETaskStatus.Failed;
  135. apptaskItem.Message = "批量任务执行异常";
  136. apptaskItem.TaskEndTime = DateTime.Now;
  137. }
  138. finally
  139. {
  140. await _apptaskItemRepository.Updateable(apptaskItem)
  141. .UpdateColumns(d => new
  142. {
  143. d.TaskStatus,
  144. d.Message,
  145. d.TaskEndTime
  146. })
  147. .ExecuteCommandAsync(cancellation);
  148. }
  149. }
  150. }