PurchaseRequestExternalPushService.cs 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108
  1. using System.Globalization;
  2. using Microsoft.Extensions.Options;
  3. using Yitter.IdGenerator;
  4. namespace Admin.NET.Plugin.AiDOP.Supply;
  5. /// <summary>
  6. /// 采购申请外部事务推送服务。对齐旧 PrSendSAP:写 QadTracking;真实 SAP/QAD 默认关闭。
  7. /// </summary>
  8. public class PurchaseRequestExternalPushService : ITransient
  9. {
  10. private readonly ISqlSugarClient _db;
  11. private readonly AidopProcurementOptions _options;
  12. public PurchaseRequestExternalPushService(ISqlSugarClient db, IOptions<AidopProcurementOptions> options)
  13. {
  14. _db = db;
  15. _options = options.Value ?? new AidopProcurementOptions();
  16. }
  17. public async Task<PurchaseRequestExternalPushResult> CreateQadTrackingForGeneratedRequestsAsync(List<PurchaseRequestMain> requests, string account)
  18. {
  19. var result = new PurchaseRequestExternalPushResult();
  20. var candidates = SelectPushCandidates(requests);
  21. if (candidates.Count == 0) return result;
  22. if (_options.EnableRealSapPush)
  23. result.Warnings.Add("EnableRealSapPush=true,但真实 SAP/QAD 接口未接入,仍仅写 QadTracking 事务记录。");
  24. if (!await QadTrackingTableExistsAsync())
  25. {
  26. result.Warnings.Add("QadTracking 表不存在,已跳过外部事务写入(主事务不回滚)。");
  27. result.SkippedCandidateCount = candidates.Count;
  28. return result;
  29. }
  30. var pushedIds = new List<long>();
  31. foreach (var group in candidates.GroupBy(x => x.TenantId.ToString()))
  32. {
  33. var seqId = GenerateSeqId();
  34. foreach (var pr in group)
  35. {
  36. await _db.Ado.ExecuteCommandAsync(
  37. """
  38. INSERT INTO QadTracking
  39. (Ufld1, Domain, SeqID, TransType, Subject, `Order`, `Int1`, CreateTime, CreateUser, UpdateTime, UpdateUser)
  40. VALUES
  41. ('C', @Domain, @SeqID, 'nbr-pr-mes', '请购单', @Order, @Int1, @Now, @User, @Now, @User)
  42. """,
  43. new SugarParameter("@Domain", group.Key),
  44. new SugarParameter("@SeqID", seqId),
  45. new SugarParameter("@Order", pr.PrBillNo),
  46. new SugarParameter("@Int1", pr.Id),
  47. new SugarParameter("@Now", DateTime.Now),
  48. new SugarParameter("@User", account ?? string.Empty));
  49. pr.State = 2;
  50. pr.UpdateByName = account;
  51. pr.UpdateTime = DateTime.Now;
  52. pushedIds.Add(pr.Id);
  53. seqId++;
  54. }
  55. }
  56. result.TrackingCount = pushedIds.Count;
  57. result.PushedPrIds = pushedIds;
  58. return result;
  59. }
  60. public static List<PurchaseRequestMain> SelectPushCandidates(IEnumerable<PurchaseRequestMain> requests)
  61. {
  62. return requests
  63. .Where(x => x.IsRequireGoods == 0
  64. && (x.State ?? 0) == 1
  65. && !string.IsNullOrWhiteSpace(x.PrBillNo))
  66. .ToList();
  67. }
  68. private async Task<bool> QadTrackingTableExistsAsync()
  69. {
  70. var cnt = await _db.Ado.GetIntAsync(
  71. """
  72. SELECT COUNT(*) FROM information_schema.tables
  73. WHERE table_schema = DATABASE() AND table_name = 'QadTracking'
  74. """);
  75. return cnt > 0;
  76. }
  77. private static long GenerateSeqId()
  78. {
  79. var now = DateTime.Now;
  80. var value = string.Concat(
  81. now.ToString("yyMMddHHmmss", CultureInfo.InvariantCulture),
  82. now.Millisecond.ToString("D3", CultureInfo.InvariantCulture),
  83. Random.Shared.Next(0, 1000).ToString("D3", CultureInfo.InvariantCulture));
  84. return long.TryParse(value, System.Globalization.NumberStyles.Integer, CultureInfo.InvariantCulture, out var seqId)
  85. ? seqId + 1
  86. : YitIdHelper.NextId();
  87. }
  88. }
  89. public sealed class PurchaseRequestExternalPushResult
  90. {
  91. public int TrackingCount { get; set; }
  92. public int SkippedCandidateCount { get; set; }
  93. public List<long> PushedPrIds { get; set; } = new();
  94. public List<string> Warnings { get; set; } = new();
  95. }