TimerManager.cs 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181
  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. ArgumentException.ThrowIfNullOrEmpty(item.Id);
  64. base.Add(item);
  65. AddOrUpdateSystemTimer(item);
  66. }
  67. private static bool ShouldStartTimer(TimerInfo item)
  68. {
  69. if (item.Status == RecordingStatus.Completed
  70. || item.Status == RecordingStatus.Cancelled)
  71. {
  72. return false;
  73. }
  74. return true;
  75. }
  76. private void AddOrUpdateSystemTimer(TimerInfo item)
  77. {
  78. StopTimer(item);
  79. if (!ShouldStartTimer(item))
  80. {
  81. return;
  82. }
  83. var startDate = RecordingHelper.GetStartTime(item);
  84. var now = DateTime.UtcNow;
  85. if (startDate < now)
  86. {
  87. TimerFired?.Invoke(this, new GenericEventArgs<TimerInfo>(item));
  88. return;
  89. }
  90. var dueTime = startDate - now;
  91. StartTimer(item, dueTime);
  92. }
  93. private void StartTimer(TimerInfo item, TimeSpan dueTime)
  94. {
  95. var timer = new Timer(TimerCallback, item.Id, dueTime, TimeSpan.Zero);
  96. if (_timers.TryAdd(item.Id, timer))
  97. {
  98. if (item.IsSeries)
  99. {
  100. Logger.LogInformation(
  101. "Creating recording timer for {Id}, {Name} {SeasonNumber}x{EpisodeNumber:D2} on channel {ChannelId}. Timer will fire in {Minutes} minutes at {StartDate}",
  102. item.Id,
  103. item.Name,
  104. item.SeasonNumber,
  105. item.EpisodeNumber,
  106. item.ChannelId,
  107. dueTime.TotalMinutes.ToString(CultureInfo.InvariantCulture),
  108. item.StartDate);
  109. }
  110. else
  111. {
  112. Logger.LogInformation(
  113. "Creating recording timer for {Id}, {Name} on channel {ChannelId}. Timer will fire in {Minutes} minutes at {StartDate}",
  114. item.Id,
  115. item.Name,
  116. item.ChannelId,
  117. dueTime.TotalMinutes.ToString(CultureInfo.InvariantCulture),
  118. item.StartDate);
  119. }
  120. }
  121. else
  122. {
  123. timer.Dispose();
  124. Logger.LogWarning("Timer already exists for item {Id}", item.Id);
  125. }
  126. }
  127. private void StopTimer(TimerInfo item)
  128. {
  129. if (_timers.TryRemove(item.Id, out var timer))
  130. {
  131. timer.Dispose();
  132. }
  133. }
  134. private void TimerCallback(object? state)
  135. {
  136. var timerId = (string?)state ?? throw new ArgumentNullException(nameof(state));
  137. var timer = GetAll().FirstOrDefault(i => string.Equals(i.Id, timerId, StringComparison.OrdinalIgnoreCase));
  138. if (timer is not null)
  139. {
  140. TimerFired?.Invoke(this, new GenericEventArgs<TimerInfo>(timer));
  141. }
  142. }
  143. public TimerInfo? GetTimer(string id)
  144. {
  145. return GetAll().FirstOrDefault(r => string.Equals(r.Id, id, StringComparison.OrdinalIgnoreCase));
  146. }
  147. public TimerInfo? GetTimerByProgramId(string programId)
  148. {
  149. return GetAll().FirstOrDefault(r => string.Equals(r.ProgramId, programId, StringComparison.OrdinalIgnoreCase));
  150. }
  151. }
  152. }