TimerManager.cs 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  1. #pragma warning disable CS1591
  2. using System;
  3. using System.Collections.Concurrent;
  4. using System.Globalization;
  5. using System.Linq;
  6. using System.Threading;
  7. using Jellyfin.Data.Events;
  8. using MediaBrowser.Controller.LiveTv;
  9. using MediaBrowser.Model.LiveTv;
  10. using Microsoft.Extensions.Logging;
  11. namespace Emby.Server.Implementations.LiveTv.EmbyTV
  12. {
  13. public class TimerManager : ItemDataProvider<TimerInfo>
  14. {
  15. private readonly ConcurrentDictionary<string, Timer> _timers = new ConcurrentDictionary<string, Timer>(StringComparer.OrdinalIgnoreCase);
  16. public TimerManager(ILogger logger, string dataPath)
  17. : base(logger, dataPath, (r1, r2) => string.Equals(r1.Id, r2.Id, StringComparison.OrdinalIgnoreCase))
  18. {
  19. }
  20. public event EventHandler<GenericEventArgs<TimerInfo>>? TimerFired;
  21. public void RestartTimers()
  22. {
  23. StopTimers();
  24. foreach (var item in GetAll())
  25. {
  26. AddOrUpdateSystemTimer(item);
  27. }
  28. }
  29. public void StopTimers()
  30. {
  31. foreach (var pair in _timers.ToList())
  32. {
  33. pair.Value.Dispose();
  34. }
  35. _timers.Clear();
  36. }
  37. public override void Delete(TimerInfo item)
  38. {
  39. base.Delete(item);
  40. StopTimer(item);
  41. }
  42. public override void Update(TimerInfo item)
  43. {
  44. base.Update(item);
  45. AddOrUpdateSystemTimer(item);
  46. }
  47. public void AddOrUpdate(TimerInfo item, bool resetTimer)
  48. {
  49. if (resetTimer)
  50. {
  51. AddOrUpdate(item);
  52. return;
  53. }
  54. base.AddOrUpdate(item);
  55. }
  56. public override void AddOrUpdate(TimerInfo item)
  57. {
  58. base.AddOrUpdate(item);
  59. AddOrUpdateSystemTimer(item);
  60. }
  61. public override void Add(TimerInfo item)
  62. {
  63. if (string.IsNullOrEmpty(item.Id))
  64. {
  65. throw new ArgumentException("TimerInfo.Id cannot be null or empty.");
  66. }
  67. base.Add(item);
  68. AddOrUpdateSystemTimer(item);
  69. }
  70. private static bool ShouldStartTimer(TimerInfo item)
  71. {
  72. if (item.Status == RecordingStatus.Completed
  73. || item.Status == RecordingStatus.Cancelled)
  74. {
  75. return false;
  76. }
  77. return true;
  78. }
  79. private void AddOrUpdateSystemTimer(TimerInfo item)
  80. {
  81. StopTimer(item);
  82. if (!ShouldStartTimer(item))
  83. {
  84. return;
  85. }
  86. var startDate = RecordingHelper.GetStartTime(item);
  87. var now = DateTime.UtcNow;
  88. if (startDate < now)
  89. {
  90. TimerFired?.Invoke(this, new GenericEventArgs<TimerInfo>(item));
  91. return;
  92. }
  93. var dueTime = startDate - now;
  94. StartTimer(item, dueTime);
  95. }
  96. private void StartTimer(TimerInfo item, TimeSpan dueTime)
  97. {
  98. var timer = new Timer(TimerCallback, item.Id, dueTime, TimeSpan.Zero);
  99. if (_timers.TryAdd(item.Id, timer))
  100. {
  101. if (item.IsSeries)
  102. {
  103. Logger.LogInformation(
  104. "Creating recording timer for {Id}, {Name} {SeasonNumber}x{EpisodeNumber:D2} on channel {ChannelId}. Timer will fire in {Minutes} minutes at {StartDate}",
  105. item.Id,
  106. item.Name,
  107. item.SeasonNumber,
  108. item.EpisodeNumber,
  109. item.ChannelId,
  110. dueTime.TotalMinutes.ToString(CultureInfo.InvariantCulture),
  111. item.StartDate);
  112. }
  113. else
  114. {
  115. Logger.LogInformation(
  116. "Creating recording timer for {Id}, {Name} on channel {ChannelId}. Timer will fire in {Minutes} minutes at {StartDate}",
  117. item.Id,
  118. item.Name,
  119. item.ChannelId,
  120. dueTime.TotalMinutes.ToString(CultureInfo.InvariantCulture),
  121. item.StartDate);
  122. }
  123. }
  124. else
  125. {
  126. timer.Dispose();
  127. Logger.LogWarning("Timer already exists for item {Id}", item.Id);
  128. }
  129. }
  130. private void StopTimer(TimerInfo item)
  131. {
  132. if (_timers.TryRemove(item.Id, out var timer))
  133. {
  134. timer.Dispose();
  135. }
  136. }
  137. private void TimerCallback(object? state)
  138. {
  139. var timerId = (string?)state ?? throw new ArgumentNullException(nameof(state));
  140. var timer = GetAll().FirstOrDefault(i => string.Equals(i.Id, timerId, StringComparison.OrdinalIgnoreCase));
  141. if (timer is not null)
  142. {
  143. TimerFired?.Invoke(this, new GenericEventArgs<TimerInfo>(timer));
  144. }
  145. }
  146. public TimerInfo? GetTimer(string id)
  147. {
  148. return GetAll().FirstOrDefault(r => string.Equals(r.Id, id, StringComparison.OrdinalIgnoreCase));
  149. }
  150. public TimerInfo? GetTimerByProgramId(string programId)
  151. {
  152. return GetAll().FirstOrDefault(r => string.Equals(r.ProgramId, programId, StringComparison.OrdinalIgnoreCase));
  153. }
  154. }
  155. }