jellyfin-server/Emby.Server.Implementations/LiveTv/EmbyTV/TimerManager.cs

196 lines
5.8 KiB
C#
Raw Normal View History

2015-07-20 18:32:55 +00:00
using MediaBrowser.Common.Events;
using MediaBrowser.Controller.LiveTv;
using MediaBrowser.Model.Events;
using MediaBrowser.Model.Logging;
using MediaBrowser.Model.Serialization;
using System;
using System.Collections.Concurrent;
2016-01-28 19:10:56 +00:00
using System.Globalization;
2015-07-20 18:32:55 +00:00
using System.Linq;
using System.Threading;
2017-05-26 06:48:54 +00:00
2016-10-25 19:02:04 +00:00
using MediaBrowser.Controller.IO;
using MediaBrowser.Model.IO;
using MediaBrowser.Model.LiveTv;
2016-11-03 23:35:19 +00:00
using MediaBrowser.Model.Threading;
2018-09-12 17:26:21 +00:00
using MediaBrowser.Model.System;
2015-07-20 18:32:55 +00:00
2016-11-03 23:35:19 +00:00
namespace Emby.Server.Implementations.LiveTv.EmbyTV
2015-07-20 18:32:55 +00:00
{
public class TimerManager : ItemDataProvider<TimerInfo>
{
2016-11-03 23:35:19 +00:00
private readonly ConcurrentDictionary<string, ITimer> _timers = new ConcurrentDictionary<string, ITimer>(StringComparer.OrdinalIgnoreCase);
private readonly ILogger _logger;
2015-07-20 18:32:55 +00:00
public event EventHandler<GenericEventArgs<TimerInfo>> TimerFired;
2016-11-03 23:35:19 +00:00
private readonly ITimerFactory _timerFactory;
2018-09-12 17:26:21 +00:00
private readonly IPowerManagement _powerManagement;
2015-07-20 18:32:55 +00:00
2018-09-12 17:26:21 +00:00
public TimerManager(IFileSystem fileSystem, IJsonSerializer jsonSerializer, ILogger logger, string dataPath, ILogger logger1, ITimerFactory timerFactory, IPowerManagement powerManagement)
2015-09-13 23:07:54 +00:00
: base(fileSystem, jsonSerializer, logger, dataPath, (r1, r2) => string.Equals(r1.Id, r2.Id, StringComparison.OrdinalIgnoreCase))
2015-07-20 18:32:55 +00:00
{
_logger = logger1;
2016-11-03 23:35:19 +00:00
_timerFactory = timerFactory;
2018-09-12 17:26:21 +00:00
_powerManagement = powerManagement;
2015-07-20 18:32:55 +00:00
}
public void RestartTimers()
{
StopTimers();
2015-07-29 03:42:03 +00:00
foreach (var item in GetAll().ToList())
{
2018-09-12 17:26:21 +00:00
AddOrUpdateSystemTimer(item, false);
2015-07-29 03:42:03 +00:00
}
2015-07-20 18:32:55 +00:00
}
public void StopTimers()
{
foreach (var pair in _timers.ToList())
{
pair.Value.Dispose();
}
_timers.Clear();
}
public override void Delete(TimerInfo item)
{
base.Delete(item);
2015-08-22 18:29:12 +00:00
StopTimer(item);
2015-07-20 18:32:55 +00:00
}
public override void Update(TimerInfo item)
{
base.Update(item);
2018-09-12 17:26:21 +00:00
AddOrUpdateSystemTimer(item, false);
2015-07-20 18:32:55 +00:00
}
2016-06-23 05:26:34 +00:00
public void AddOrUpdate(TimerInfo item, bool resetTimer)
{
if (resetTimer)
{
AddOrUpdate(item);
return;
}
var list = GetAll().ToList();
if (!list.Any(i => EqualityComparer(i, item)))
{
base.Add(item);
}
else
{
base.Update(item);
}
}
2015-07-20 18:32:55 +00:00
public override void Add(TimerInfo item)
{
2018-09-12 17:26:21 +00:00
if (string.IsNullOrEmpty(item.Id))
2015-07-20 18:32:55 +00:00
{
throw new ArgumentException("TimerInfo.Id cannot be null or empty.");
}
base.Add(item);
2018-09-12 17:26:21 +00:00
AddOrUpdateSystemTimer(item, true);
2016-09-26 18:59:18 +00:00
}
private bool ShouldStartTimer(TimerInfo item)
{
if (item.Status == RecordingStatus.Completed ||
item.Status == RecordingStatus.Cancelled)
{
return false;
}
return true;
2015-07-20 18:32:55 +00:00
}
2018-09-12 17:26:21 +00:00
private void AddOrUpdateSystemTimer(TimerInfo item, bool scheduleSystemWakeTimer)
2015-07-20 18:32:55 +00:00
{
2016-09-26 18:59:18 +00:00
StopTimer(item);
if (!ShouldStartTimer(item))
{
return;
}
2015-07-29 03:42:03 +00:00
var startDate = RecordingHelper.GetStartTime(item);
var now = DateTime.UtcNow;
if (startDate < now)
{
EventHelper.FireEventIfNotNull(TimerFired, this, new GenericEventArgs<TimerInfo> { Argument = item }, Logger);
return;
}
2016-09-26 18:59:18 +00:00
var dueTime = startDate - now;
StartTimer(item, dueTime);
2018-09-12 17:26:21 +00:00
if (scheduleSystemWakeTimer && dueTime >= TimeSpan.FromMinutes(15))
{
ScheduleSystemWakeTimer(startDate, item.Name);
}
}
private void ScheduleSystemWakeTimer(DateTime startDate, string displayName)
{
try
{
_powerManagement.ScheduleWake(startDate.AddMinutes(-5), displayName);
}
catch (Exception ex)
{
_logger.ErrorException("Error scheduling wake timer", ex);
}
2015-08-22 18:29:12 +00:00
}
2015-07-20 18:32:55 +00:00
2016-09-26 18:59:18 +00:00
private void StartTimer(TimerInfo item, TimeSpan dueTime)
2015-08-22 18:29:12 +00:00
{
2016-11-03 23:35:19 +00:00
var timer = _timerFactory.Create(TimerCallback, item.Id, dueTime, TimeSpan.Zero);
2015-07-20 18:32:55 +00:00
2016-01-28 19:10:56 +00:00
if (_timers.TryAdd(item.Id, timer))
{
_logger.Info("Creating recording timer for {0}, {1}. Timer will fire in {2} minutes", item.Id, item.Name, dueTime.TotalMinutes.ToString(CultureInfo.InvariantCulture));
2016-01-28 19:10:56 +00:00
}
else
2015-07-20 18:32:55 +00:00
{
timer.Dispose();
2016-01-28 19:10:56 +00:00
_logger.Warn("Timer already exists for item {0}", item.Id);
2015-07-20 18:32:55 +00:00
}
}
2015-08-22 18:29:12 +00:00
private void StopTimer(TimerInfo item)
{
2016-11-03 23:35:19 +00:00
ITimer timer;
2015-08-22 18:29:12 +00:00
if (_timers.TryRemove(item.Id, out timer))
{
timer.Dispose();
}
}
2015-07-20 18:32:55 +00:00
private void TimerCallback(object state)
{
var timerId = (string)state;
var timer = GetAll().FirstOrDefault(i => string.Equals(i.Id, timerId, StringComparison.OrdinalIgnoreCase));
if (timer != null)
{
EventHelper.FireEventIfNotNull(TimerFired, this, new GenericEventArgs<TimerInfo> { Argument = timer }, Logger);
}
}
2016-09-27 05:13:56 +00:00
public TimerInfo GetTimer(string id)
{
return GetAll().FirstOrDefault(r => string.Equals(r.Id, id, StringComparison.OrdinalIgnoreCase));
}
2017-03-26 04:21:32 +00:00
public TimerInfo GetTimerByProgramId(string programId)
{
return GetAll().FirstOrDefault(r => string.Equals(r.ProgramId, programId, StringComparison.OrdinalIgnoreCase));
}
2015-07-20 18:32:55 +00:00
}
}