// Copyright (C) 2024 Intro-Skipper Contributors // SPDX-License-Identifier: GNU General Public License v3.0 only. using System; using System.Collections.Generic; using System.Threading; using System.Threading.Tasks; using IntroSkipper.Configuration; using IntroSkipper.Data; using IntroSkipper.ScheduledTasks; using MediaBrowser.Controller.Entities.TV; using MediaBrowser.Controller.Library; using MediaBrowser.Model.Entities; using MediaBrowser.Model.Plugins; using MediaBrowser.Model.Tasks; using Microsoft.Extensions.Hosting; using Microsoft.Extensions.Logging; namespace IntroSkipper; /// /// Server entrypoint. /// public sealed class Entrypoint : IHostedService, IDisposable { private readonly ITaskManager _taskManager; private readonly ILibraryManager _libraryManager; private readonly ILogger _logger; private readonly ILoggerFactory _loggerFactory; private readonly HashSet _seasonsToAnalyze = []; private readonly Timer _queueTimer; private static readonly ManualResetEventSlim _autoTaskCompletEvent = new(false); private PluginConfiguration _config; private bool _analyzeAgain; private static CancellationTokenSource? _cancellationTokenSource; /// /// Initializes a new instance of the class. /// /// Library manager. /// Task manager. /// Logger. /// Logger factory. public Entrypoint( ILibraryManager libraryManager, ITaskManager taskManager, ILogger logger, ILoggerFactory loggerFactory) { _libraryManager = libraryManager; _taskManager = taskManager; _logger = logger; _loggerFactory = loggerFactory; _config = Plugin.Instance?.Configuration ?? new PluginConfiguration(); _queueTimer = new Timer( OnTimerCallback, null, Timeout.InfiniteTimeSpan, Timeout.InfiniteTimeSpan); } /// /// Gets State of the automatic task. /// public static TaskState AutomaticTaskState { get { if (_cancellationTokenSource is not null) { return _cancellationTokenSource.IsCancellationRequested ? TaskState.Cancelling : TaskState.Running; } return TaskState.Idle; } } /// public Task StartAsync(CancellationToken cancellationToken) { _libraryManager.ItemAdded += OnItemAdded; _libraryManager.ItemUpdated += OnItemModified; _taskManager.TaskCompleted += OnLibraryRefresh; Plugin.Instance!.ConfigurationChanged += OnSettingsChanged; FFmpegWrapper.Logger = _logger; try { // Enqueue all episodes at startup to ensure any FFmpeg errors appear as early as possible _logger.LogInformation("Running startup enqueue"); var queueManager = new QueueManager(_loggerFactory.CreateLogger(), _libraryManager); queueManager?.GetMediaItems(); } catch (Exception ex) { _logger.LogError("Unable to run startup enqueue: {Exception}", ex); } return Task.CompletedTask; } /// public Task StopAsync(CancellationToken cancellationToken) { _libraryManager.ItemAdded -= OnItemAdded; _libraryManager.ItemUpdated -= OnItemModified; _taskManager.TaskCompleted -= OnLibraryRefresh; // Stop the timer _queueTimer.Change(Timeout.Infinite, 0); return Task.CompletedTask; } // Disclose source for inspiration // Implementation based on the principles of jellyfin-plugin-media-analyzer: // https://github.com/endrl/jellyfin-plugin-media-analyzer /// /// Library item was added. /// /// The sending entity. /// The . private void OnItemAdded(object? sender, ItemChangeEventArgs itemChangeEventArgs) { // Don't do anything if auto detection is disabled if (!_config.AutoDetectIntros && !_config.AutoDetectCredits) { return; } // Don't do anything if it's not a supported media type if (itemChangeEventArgs.Item is not Episode episode) { return; } if (itemChangeEventArgs.Item.LocationType == LocationType.Virtual) { return; } _seasonsToAnalyze.Add(episode.SeasonId); StartTimer(); } /// /// Library item was modified. /// /// The sending entity. /// The . private void OnItemModified(object? sender, ItemChangeEventArgs itemChangeEventArgs) { // Don't do anything if auto detection is disabled if (!_config.AutoDetectIntros && !_config.AutoDetectCredits) { return; } // Don't do anything if it's not a supported media type if (itemChangeEventArgs.Item is not Episode episode) { return; } if (itemChangeEventArgs.Item.LocationType == LocationType.Virtual) { return; } _seasonsToAnalyze.Add(episode.SeasonId); StartTimer(); } /// /// TaskManager task ended. /// /// The sending entity. /// The . private void OnLibraryRefresh(object? sender, TaskCompletionEventArgs eventArgs) { // Don't do anything if auto detection is disabled if (!_config.AutoDetectIntros && !_config.AutoDetectCredits) { return; } var result = eventArgs.Result; if (result.Key != "RefreshLibrary") { return; } if (result.Status != TaskCompletionStatus.Completed) { return; } // Unless user initiated, this is likely an overlap if (AutomaticTaskState == TaskState.Running) { return; } StartTimer(); } private void OnSettingsChanged(object? sender, BasePluginConfiguration e) => _config = (PluginConfiguration)e; /// /// Start timer to debounce analyzing. /// private void StartTimer() { if (AutomaticTaskState == TaskState.Running) { _analyzeAgain = true; } else if (AutomaticTaskState == TaskState.Idle) { _logger.LogDebug("Media Library changed, analyzis will start soon!"); _queueTimer.Change(TimeSpan.FromMilliseconds(20000), Timeout.InfiniteTimeSpan); } } /// /// Wait for timer callback to be completed. /// private void OnTimerCallback(object? state) { try { PerformAnalysis(); } catch (Exception ex) { _logger.LogError(ex, "Error in PerformAnalysis"); } // Clean up _cancellationTokenSource = null; _autoTaskCompletEvent.Set(); } /// /// Wait for timer to be completed. /// private void PerformAnalysis() { _logger.LogInformation("Initiate automatic analysis task."); _autoTaskCompletEvent.Reset(); using (_cancellationTokenSource = new CancellationTokenSource()) using (ScheduledTaskSemaphore.Acquire(_cancellationTokenSource.Token)) { var seasonIds = new HashSet(_seasonsToAnalyze); _seasonsToAnalyze.Clear(); _analyzeAgain = false; var progress = new Progress(); var modes = new List(); var tasklogger = _loggerFactory.CreateLogger("DefaultLogger"); if (_config.AutoDetectIntros) { modes.Add(AnalysisMode.Introduction); tasklogger = _loggerFactory.CreateLogger(); } if (_config.AutoDetectCredits) { modes.Add(AnalysisMode.Credits); tasklogger = modes.Count == 2 ? _loggerFactory.CreateLogger() : _loggerFactory.CreateLogger(); } var baseCreditAnalyzer = new BaseItemAnalyzerTask( modes, tasklogger, _loggerFactory, _libraryManager); baseCreditAnalyzer.AnalyzeItems(progress, _cancellationTokenSource.Token, seasonIds); // New item detected, start timer again if (_analyzeAgain && !_cancellationTokenSource.IsCancellationRequested) { _logger.LogInformation("Analyzing ended, but we need to analyze again!"); StartTimer(); } } } /// /// Method to cancel the automatic task. /// /// Cancellation token. public static void CancelAutomaticTask(CancellationToken cancellationToken) { if (_cancellationTokenSource != null && !_cancellationTokenSource.IsCancellationRequested) { try { _cancellationTokenSource.Cancel(); } catch (ObjectDisposedException) { _cancellationTokenSource = null; } } _autoTaskCompletEvent.Wait(TimeSpan.FromSeconds(60), cancellationToken); // Wait for the signal } /// public void Dispose() { _queueTimer.Dispose(); _cancellationTokenSource?.Dispose(); _autoTaskCompletEvent.Dispose(); } }