| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181 | #pragma warning disable CS1591using System;using System.Collections.Concurrent;using System.Globalization;using System.Linq;using System.Threading;using Jellyfin.Data.Events;using MediaBrowser.Controller.LiveTv;using MediaBrowser.Model.LiveTv;using Microsoft.Extensions.Logging;namespace Emby.Server.Implementations.LiveTv.EmbyTV{    public class TimerManager : ItemDataProvider<TimerInfo>    {        private readonly ConcurrentDictionary<string, Timer> _timers = new ConcurrentDictionary<string, Timer>(StringComparer.OrdinalIgnoreCase);        public TimerManager(ILogger logger, string dataPath)            : base(logger, dataPath, (r1, r2) => string.Equals(r1.Id, r2.Id, StringComparison.OrdinalIgnoreCase))        {        }        public event EventHandler<GenericEventArgs<TimerInfo>>? TimerFired;        public void RestartTimers()        {            StopTimers();            foreach (var item in GetAll())            {                AddOrUpdateSystemTimer(item);            }        }        public void StopTimers()        {            foreach (var pair in _timers.ToList())            {                pair.Value.Dispose();            }            _timers.Clear();        }        public override void Delete(TimerInfo item)        {            base.Delete(item);            StopTimer(item);        }        public override void Update(TimerInfo item)        {            base.Update(item);            AddOrUpdateSystemTimer(item);        }        public void AddOrUpdate(TimerInfo item, bool resetTimer)        {            if (resetTimer)            {                AddOrUpdate(item);                return;            }            base.AddOrUpdate(item);        }        public override void AddOrUpdate(TimerInfo item)        {            base.AddOrUpdate(item);            AddOrUpdateSystemTimer(item);        }        public override void Add(TimerInfo item)        {            ArgumentException.ThrowIfNullOrEmpty(item.Id);            base.Add(item);            AddOrUpdateSystemTimer(item);        }        private static bool ShouldStartTimer(TimerInfo item)        {            if (item.Status == RecordingStatus.Completed                || item.Status == RecordingStatus.Cancelled)            {                return false;            }            return true;        }        private void AddOrUpdateSystemTimer(TimerInfo item)        {            StopTimer(item);            if (!ShouldStartTimer(item))            {                return;            }            var startDate = RecordingHelper.GetStartTime(item);            var now = DateTime.UtcNow;            if (startDate < now)            {                TimerFired?.Invoke(this, new GenericEventArgs<TimerInfo>(item));                return;            }            var dueTime = startDate - now;            StartTimer(item, dueTime);        }        private void StartTimer(TimerInfo item, TimeSpan dueTime)        {            var timer = new Timer(TimerCallback, item.Id, dueTime, TimeSpan.Zero);            if (_timers.TryAdd(item.Id, timer))            {                if (item.IsSeries)                {                    Logger.LogInformation(                    "Creating recording timer for {Id}, {Name} {SeasonNumber}x{EpisodeNumber:D2} on channel {ChannelId}. Timer will fire in {Minutes} minutes at {StartDate}",                    item.Id,                    item.Name,                    item.SeasonNumber,                    item.EpisodeNumber,                    item.ChannelId,                    dueTime.TotalMinutes.ToString(CultureInfo.InvariantCulture),                    item.StartDate);                }                else                {                    Logger.LogInformation(                    "Creating recording timer for {Id}, {Name} on channel {ChannelId}. Timer will fire in {Minutes} minutes at {StartDate}",                    item.Id,                    item.Name,                    item.ChannelId,                    dueTime.TotalMinutes.ToString(CultureInfo.InvariantCulture),                    item.StartDate);                }            }            else            {                timer.Dispose();                Logger.LogWarning("Timer already exists for item {Id}", item.Id);            }        }        private void StopTimer(TimerInfo item)        {            if (_timers.TryRemove(item.Id, out var timer))            {                timer.Dispose();            }        }        private void TimerCallback(object? state)        {            var timerId = (string?)state ?? throw new ArgumentNullException(nameof(state));            var timer = GetAll().FirstOrDefault(i => string.Equals(i.Id, timerId, StringComparison.OrdinalIgnoreCase));            if (timer is not null)            {                TimerFired?.Invoke(this, new GenericEventArgs<TimerInfo>(timer));            }        }        public TimerInfo? GetTimer(string id)        {            return GetAll().FirstOrDefault(r => string.Equals(r.Id, id, StringComparison.OrdinalIgnoreCase));        }        public TimerInfo? GetTimerByProgramId(string programId)        {            return GetAll().FirstOrDefault(r => string.Equals(r.ProgramId, programId, StringComparison.OrdinalIgnoreCase));        }    }}
 |