using System; using System.Collections.Generic; using System.Threading; using System.Threading.Tasks; using System.Timers; using MediaBrowser.Common.Extensions; using MediaBrowser.Controller.Library; using MediaBrowser.Controller.Plugins; using MediaBrowser.Controller.Session; using MediaBrowser.Model.Entities; using MediaBrowser.Model.Session; using Microsoft.Extensions.Logging; namespace ConfusedPolarBear.Plugin.IntroSkipper; /// /// Automatically skip past introduction sequences. /// Commands clients to seek to the end of the intro as soon as they start playing it. /// public class AutoSkip : IServerEntryPoint { private readonly object _sentSeekCommandLock = new(); private ILogger _logger; private IUserDataManager _userDataManager; private ISessionManager _sessionManager; private System.Timers.Timer _playbackTimer = new(1000); private Dictionary _sentSeekCommand; /// /// Initializes a new instance of the class. /// /// User data manager. /// Session manager. /// Logger. public AutoSkip( IUserDataManager userDataManager, ISessionManager sessionManager, ILogger logger) { _userDataManager = userDataManager; _sessionManager = sessionManager; _logger = logger; _sentSeekCommand = new Dictionary(); } /// /// If introduction auto skipping is enabled, set it up. /// /// Task. public Task RunAsync() { _logger.LogDebug("Setting up automatic skipping"); _userDataManager.UserDataSaved += UserDataManager_UserDataSaved; Plugin.Instance!.AutoSkipChanged += AutoSkipChanged; // Make the timer restart automatically and set enabled to match the configuration value. _playbackTimer.AutoReset = true; _playbackTimer.Elapsed += PlaybackTimer_Elapsed; AutoSkipChanged(null, EventArgs.Empty); return Task.CompletedTask; } private void AutoSkipChanged(object? sender, EventArgs e) { var newState = Plugin.Instance!.Configuration.AutoSkip; _logger.LogDebug("Setting playback timer enabled to {NewState}", newState); _playbackTimer.Enabled = newState; } private void UserDataManager_UserDataSaved(object? sender, UserDataSaveEventArgs e) { var itemId = e.Item.Id; var newState = false; var episodeNumber = e.Item.IndexNumber.GetValueOrDefault(-1); // Ignore all events except playback start & end if (e.SaveReason != UserDataSaveReason.PlaybackStart && e.SaveReason != UserDataSaveReason.PlaybackFinished) { return; } // Lookup the session for this item. SessionInfo? session = null; try { foreach (var needle in _sessionManager.Sessions) { if (needle.UserId == e.UserId && needle.NowPlayingItem.Id == itemId) { session = needle; break; } } if (session == null) { _logger.LogInformation("Unable to find session for {Item}", itemId); return; } } catch (Exception ex) when (ex is NullReferenceException || ex is ResourceNotFoundException) { return; } // If this is the first episode in the season, and SkipFirstEpisode is false, pretend that we've already sent the seek command for this playback session. if (Plugin.Instance!.Configuration.SkipFirstEpisode && episodeNumber == 1) { newState = true; } // Reset the seek command state for this device. lock (_sentSeekCommandLock) { var device = session.DeviceId; _logger.LogDebug("Resetting seek command state for session {Session}", device); _sentSeekCommand[device] = newState; } } private void PlaybackTimer_Elapsed(object? sender, ElapsedEventArgs e) { foreach (var session in _sessionManager.Sessions) { var deviceId = session.DeviceId; var itemId = session.NowPlayingItem.Id; var position = session.PlayState.PositionTicks / TimeSpan.TicksPerSecond; // Don't send the seek command more than once in the same session. lock (_sentSeekCommandLock) { if (_sentSeekCommand.TryGetValue(deviceId, out var sent) && sent) { _logger.LogTrace("Already sent seek command for session {Session}", deviceId); continue; } } // Assert that an intro was detected for this item. if (!Plugin.Instance!.Intros.TryGetValue(itemId, out var intro) || !intro.Valid) { continue; } // Seek is unreliable if called at the very start of an episode. var adjustedStart = Math.Max(5, intro.IntroStart + Plugin.Instance.Configuration.SecondsOfIntroStartToPlay); var adjustedEnd = intro.IntroEnd - Plugin.Instance.Configuration.RemainingSecondsOfIntro; _logger.LogTrace( "Playback position is {Position}, intro runs from {Start} to {End}", position, adjustedStart, adjustedEnd); if (position < adjustedStart || position > adjustedEnd) { continue; } // Notify the user that an introduction is being skipped for them. var notificationText = Plugin.Instance!.Configuration.AutoSkipNotificationText; if (!string.IsNullOrWhiteSpace(notificationText)) { _sessionManager.SendMessageCommand( session.Id, session.Id, new MessageCommand() { Header = string.Empty, // some clients require header to be a string instead of null Text = notificationText, TimeoutMs = 2000, }, CancellationToken.None); } _logger.LogDebug("Sending seek command to {Session}", deviceId); _sessionManager.SendPlaystateCommand( session.Id, session.Id, new PlaystateRequest { Command = PlaystateCommand.Seek, ControllingUserId = session.UserId.ToString("N"), SeekPositionTicks = (long)adjustedEnd * TimeSpan.TicksPerSecond, }, CancellationToken.None); // Flag that we've sent the seek command so that it's not sent repeatedly lock (_sentSeekCommandLock) { _logger.LogTrace("Setting seek command state for session {Session}", deviceId); _sentSeekCommand[deviceId] = true; } } } /// /// Dispose. /// public void Dispose() { Dispose(true); GC.SuppressFinalize(this); } /// /// Protected dispose. /// /// Dispose. protected virtual void Dispose(bool disposing) { if (!disposing) { return; } _userDataManager.UserDataSaved -= UserDataManager_UserDataSaved; _playbackTimer.Stop(); _playbackTimer.Dispose(); } }