RedisQueue.cs 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182
  1. // Admin.NET 项目的版权、商标、专利和其他相关权利均受相应法律法规的保护。使用本项目应遵守相关法律法规和许可证的要求。
  2. //
  3. // 本项目主要遵循 MIT 许可证和 Apache 许可证(版本 2.0)进行分发和使用。许可证位于源代码树根目录中的 LICENSE-MIT 和 LICENSE-APACHE 文件。
  4. //
  5. // 不得利用本项目从事危害国家安全、扰乱社会秩序、侵犯他人合法权益等法律法规禁止的活动!任何基于本项目二次开发而产生的一切法律纠纷和责任,我们不承担任何责任!
  6. using NewLife.Caching.Queues;
  7. namespace Admin.NET.Core;
  8. /// <summary>
  9. /// Redis 消息队列
  10. /// </summary>
  11. public static class RedisQueue
  12. {
  13. private static ICacheProvider _cacheProvider = App.GetRequiredService<ICacheProvider>();
  14. /// <summary>创建Redis消息队列。默认消费一次,指定消费者group时使用STREAM结构,支持多消费组共享消息</summary>
  15. /// <remarks>
  16. /// 使用队列时,可根据是否设置消费组来决定使用简单队列还是完整队列。 简单队列(如RedisQueue)可用作命令队列,Topic很多,但几乎没有消息。 完整队列(如RedisStream)可用作消息队列,Topic很少,但消息很多,并且支持多消费组。
  17. /// </remarks>
  18. /// <typeparam name="T"></typeparam>
  19. /// <param name="topic">主题</param>
  20. /// <param name="group">消费组。未指定消费组时使用简单队列(如RedisQueue),指定消费组时使用完整队列(如RedisStream)</param>
  21. /// <returns></returns>
  22. public static IProducerConsumer<T> GetQueue<T>(String topic, String group = null)
  23. {
  24. // 队列需要单列
  25. var key = $"myStream:{topic}";
  26. if (_cacheProvider.InnerCache.TryGetValue<IProducerConsumer<T>>(key, out var queue)) return queue;
  27. queue = _cacheProvider.GetQueue<T>(topic, group);
  28. _cacheProvider.Cache.Set(key, queue);
  29. return queue;
  30. }
  31. /// <summary>
  32. /// 获取可信队列,需要确认
  33. /// </summary>
  34. /// <typeparam name="T"></typeparam>
  35. /// <param name="topic"></param>
  36. /// <returns></returns>
  37. public static RedisReliableQueue<T> GetRedisReliableQueue<T>(string topic)
  38. {
  39. // 队列需要单列
  40. var key = $"myQueue:{topic}";
  41. if (_cacheProvider.InnerCache.TryGetValue<RedisReliableQueue<T>>(key, out var queue)) return queue;
  42. queue = (_cacheProvider.Cache as FullRedis).GetReliableQueue<T>(topic);
  43. _cacheProvider.Cache.Set(key, queue);
  44. return queue;
  45. }
  46. /// <summary>
  47. /// 可信队列回滚
  48. /// </summary>
  49. /// <param name="topic"></param>
  50. /// <param name="retryInterval"></param>
  51. /// <returns></returns>
  52. public static int RollbackAllAck(string topic, int retryInterval = 60)
  53. {
  54. var queue = GetRedisReliableQueue<string>(topic);
  55. queue.RetryInterval = retryInterval;
  56. return queue.RollbackAllAck();
  57. }
  58. /// <summary>
  59. /// 发送一个数据列表到可信队列
  60. /// </summary>
  61. /// <param name="topic"></param>
  62. /// <param name="value"></param>
  63. /// <typeparam name="T"></typeparam>
  64. /// <returns></returns>
  65. public static int AddReliableQueueList<T>(string topic, List<T> value)
  66. {
  67. var queue = GetRedisReliableQueue<T>(topic);
  68. var count = queue.Count;
  69. var result = queue.Add(value.ToArray());
  70. return result - count;
  71. }
  72. /// <summary>
  73. /// 发送一条数据到可信队列
  74. /// </summary>
  75. /// <param name="topic"></param>
  76. /// <param name="value"></param>
  77. /// <typeparam name="T"></typeparam>
  78. /// <returns></returns>
  79. public static int AddReliableQueue<T>(string topic, T value)
  80. {
  81. var queue = GetRedisReliableQueue<T>(topic);
  82. var count = queue.Count;
  83. var result = queue.Add(value);
  84. return result - count;
  85. }
  86. /// <summary>
  87. /// 获取延迟队列
  88. /// </summary>
  89. /// <param name="topic"></param>
  90. /// <typeparam name="T"></typeparam>
  91. /// <returns></returns>
  92. public static RedisDelayQueue<T> GetDelayQueue<T>(string topic)
  93. {
  94. // 队列需要单列
  95. var key = $"myDelay:{topic}";
  96. if (_cacheProvider.InnerCache.TryGetValue<RedisDelayQueue<T>>(key, out var queue)) return queue;
  97. queue = (_cacheProvider.Cache as FullRedis).GetDelayQueue<T>(topic);
  98. _cacheProvider.Cache.Set(key, queue);
  99. return queue;
  100. }
  101. /// <summary>
  102. /// 发送一条数据到延迟队列
  103. /// </summary>
  104. /// <param name="topic"></param>
  105. /// <param name="value"></param>
  106. /// <param name="delay">延迟时间。单位秒</param>
  107. /// <typeparam name="T"></typeparam>
  108. /// <returns></returns>
  109. public static int AddDelayQueue<T>(string topic, T value, int delay)
  110. {
  111. var queue = GetDelayQueue<T>(topic);
  112. return queue.Add(value, delay);
  113. }
  114. /// <summary>
  115. /// 发送数据列表到延迟队列
  116. /// </summary>
  117. /// <param name="topic"></param>
  118. /// <param name="value"></param>
  119. /// <param name="delay"></param>
  120. /// <typeparam name="T">延迟时间。单位秒</typeparam>
  121. /// <returns></returns>
  122. public static int AddDelayQueue<T>(string topic, List<T> value, int delay)
  123. {
  124. var queue = GetDelayQueue<T>(topic);
  125. queue.Delay = delay;
  126. return queue.Add(value.ToArray());
  127. }
  128. /// <summary>
  129. /// 在可信队列获取一条数据
  130. /// </summary>
  131. /// <param name="topic"></param>
  132. /// <typeparam name="T"></typeparam>
  133. /// <returns></returns>
  134. public static T ReliableTakeOne<T>(string topic)
  135. {
  136. var queue = GetRedisReliableQueue<T>(topic);
  137. return queue.TakeOne(1);
  138. }
  139. /// <summary>
  140. /// 异步在可信队列获取一条数据
  141. /// </summary>
  142. /// <param name="topic"></param>
  143. /// <typeparam name="T"></typeparam>
  144. /// <returns></returns>
  145. public static async Task<T> ReliableTakeOneAsync<T>(string topic)
  146. {
  147. var queue = GetRedisReliableQueue<T>(topic);
  148. return await queue.TakeOneAsync(1);
  149. }
  150. /// <summary>
  151. /// 在可信队列获取多条数据
  152. /// </summary>
  153. /// <param name="topic"></param>
  154. /// <param name="count"></param>
  155. /// <typeparam name="T"></typeparam>
  156. /// <returns></returns>
  157. public static List<T> ReliableTake<T>(string topic, int count)
  158. {
  159. var queue = GetRedisReliableQueue<T>(topic);
  160. return queue.Take(count).ToList();
  161. }
  162. }