TimerManager.cs 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  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. Logger.LogInformation(
  102. "Creating recording timer for {Id}, {Name}. Timer will fire in {Minutes} minutes",
  103. item.Id,
  104. item.Name,
  105. dueTime.TotalMinutes.ToString(CultureInfo.InvariantCulture));
  106. }
  107. else
  108. {
  109. timer.Dispose();
  110. Logger.LogWarning("Timer already exists for item {Id}", item.Id);
  111. }
  112. }
  113. private void StopTimer(TimerInfo item)
  114. {
  115. if (_timers.TryRemove(item.Id, out var timer))
  116. {
  117. timer.Dispose();
  118. }
  119. }
  120. private void TimerCallback(object? state)
  121. {
  122. var timerId = (string?)state ?? throw new ArgumentNullException(nameof(state));
  123. var timer = GetAll().FirstOrDefault(i => string.Equals(i.Id, timerId, StringComparison.OrdinalIgnoreCase));
  124. if (timer != null)
  125. {
  126. TimerFired?.Invoke(this, new GenericEventArgs<TimerInfo>(timer));
  127. }
  128. }
  129. public TimerInfo? GetTimer(string id)
  130. {
  131. return GetAll().FirstOrDefault(r => string.Equals(r.Id, id, StringComparison.OrdinalIgnoreCase));
  132. }
  133. public TimerInfo? GetTimerByProgramId(string programId)
  134. {
  135. return GetAll().FirstOrDefault(r => string.Equals(r.ProgramId, programId, StringComparison.OrdinalIgnoreCase));
  136. }
  137. }
  138. }