2022-05-01 00:33:22 -05:00
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
2022-06-12 16:57:15 -05:00
|
|
|
using System.IO;
|
2022-05-01 00:33:22 -05:00
|
|
|
using System.Threading.Tasks;
|
2022-05-09 22:50:41 -05:00
|
|
|
using Jellyfin.Data.Enums;
|
2022-05-01 00:33:22 -05:00
|
|
|
using MediaBrowser.Controller.Entities;
|
|
|
|
using MediaBrowser.Controller.Entities.TV;
|
|
|
|
using MediaBrowser.Controller.Library;
|
|
|
|
using MediaBrowser.Controller.Plugins;
|
|
|
|
using MediaBrowser.Model.Entities;
|
|
|
|
using MediaBrowser.Model.Library;
|
|
|
|
using Microsoft.Extensions.Logging;
|
|
|
|
|
|
|
|
namespace ConfusedPolarBear.Plugin.IntroSkipper;
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Server entrypoint.
|
|
|
|
/// </summary>
|
|
|
|
public class Entrypoint : IServerEntryPoint
|
|
|
|
{
|
|
|
|
private readonly IUserManager _userManager;
|
|
|
|
private readonly IUserViewManager _userViewManager;
|
|
|
|
private readonly ILibraryManager _libraryManager;
|
|
|
|
private readonly ILogger<Entrypoint> _logger;
|
|
|
|
|
|
|
|
private readonly object _queueLock = new object();
|
|
|
|
|
|
|
|
/// <summary>
|
2022-05-09 22:50:41 -05:00
|
|
|
/// Initializes a new instance of the <see cref="Entrypoint"/> class.
|
2022-05-01 00:33:22 -05:00
|
|
|
/// </summary>
|
|
|
|
/// <param name="userManager">User manager.</param>
|
|
|
|
/// <param name="userViewManager">User view manager.</param>
|
|
|
|
/// <param name="libraryManager">Library manager.</param>
|
|
|
|
/// <param name="logger">Logger.</param>
|
|
|
|
public Entrypoint(
|
|
|
|
IUserManager userManager,
|
|
|
|
IUserViewManager userViewManager,
|
|
|
|
ILibraryManager libraryManager,
|
|
|
|
ILogger<Entrypoint> logger)
|
|
|
|
{
|
|
|
|
_userManager = userManager;
|
|
|
|
_userViewManager = userViewManager;
|
|
|
|
_libraryManager = libraryManager;
|
|
|
|
_logger = logger;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Registers event handler.
|
|
|
|
/// </summary>
|
2022-05-09 22:50:41 -05:00
|
|
|
/// <returns>Task.</returns>
|
2022-05-01 00:33:22 -05:00
|
|
|
public Task RunAsync()
|
|
|
|
{
|
2022-06-09 17:33:39 -05:00
|
|
|
Chromaprint.Logger = _logger;
|
2022-05-01 00:33:22 -05:00
|
|
|
|
2022-06-12 16:57:15 -05:00
|
|
|
#if DEBUG
|
|
|
|
LogVersion();
|
|
|
|
#endif
|
|
|
|
|
2022-06-09 14:07:40 -05:00
|
|
|
// Assert that ffmpeg with chromaprint is installed
|
2022-06-09 17:33:39 -05:00
|
|
|
if (!Chromaprint.CheckFFmpegVersion())
|
2022-05-09 22:50:41 -05:00
|
|
|
{
|
2022-06-09 14:07:40 -05:00
|
|
|
_logger.LogError("ffmpeg with chromaprint is not installed on this system - episodes will not be analyzed");
|
2022-05-01 00:33:22 -05:00
|
|
|
return Task.CompletedTask;
|
|
|
|
}
|
|
|
|
|
2022-06-12 16:08:31 -05:00
|
|
|
try
|
2022-05-09 22:50:41 -05:00
|
|
|
{
|
2022-06-12 16:08:31 -05:00
|
|
|
// As soon as a new episode is added, queue it for later analysis.
|
|
|
|
_libraryManager.ItemAdded += ItemAdded;
|
|
|
|
|
|
|
|
// For all TV show libraries, enqueue all contained items.
|
|
|
|
foreach (var folder in _libraryManager.GetVirtualFolders())
|
2022-05-09 22:50:41 -05:00
|
|
|
{
|
2022-06-12 16:08:31 -05:00
|
|
|
if (folder.CollectionType != CollectionTypeOptions.TvShows)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
2022-05-01 00:33:22 -05:00
|
|
|
|
2022-06-12 16:08:31 -05:00
|
|
|
_logger.LogInformation(
|
|
|
|
"Running startup enqueue of items in library {Name} ({ItemId})",
|
|
|
|
folder.Name,
|
|
|
|
folder.ItemId);
|
2022-05-01 00:33:22 -05:00
|
|
|
|
2022-06-12 16:08:31 -05:00
|
|
|
QueueLibraryContents(folder.ItemId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (Exception ex)
|
|
|
|
{
|
|
|
|
_logger.LogError("Unable to run startup enqueue: {Exception}", ex);
|
2022-05-01 00:33:22 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return Task.CompletedTask;
|
|
|
|
}
|
|
|
|
|
2022-05-09 22:50:41 -05:00
|
|
|
private void QueueLibraryContents(string rawId)
|
|
|
|
{
|
2022-05-10 02:10:39 -05:00
|
|
|
// FIXME: don't do this
|
2022-05-01 00:33:22 -05:00
|
|
|
|
2022-05-09 22:50:41 -05:00
|
|
|
var query = new UserViewQuery()
|
|
|
|
{
|
2022-05-01 00:33:22 -05:00
|
|
|
UserId = GetAdministrator(),
|
|
|
|
};
|
|
|
|
|
|
|
|
// Get all items from this library. Since intros may change within a season, sort the items before adding them.
|
2022-06-12 17:31:42 -05:00
|
|
|
_logger.LogDebug("Constructing user view folder");
|
2022-05-01 00:33:22 -05:00
|
|
|
var folder = _userViewManager.GetUserViews(query)[0];
|
2022-06-12 17:16:21 -05:00
|
|
|
|
|
|
|
if (folder is null)
|
|
|
|
{
|
|
|
|
_logger.LogError("Folder was null");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-06-12 17:31:42 -05:00
|
|
|
_logger.LogDebug("Getting items in folder");
|
2022-05-09 22:50:41 -05:00
|
|
|
var items = folder.GetItems(new InternalItemsQuery()
|
|
|
|
{
|
2022-05-02 01:18:31 -05:00
|
|
|
ParentId = Guid.Parse(rawId),
|
2022-05-09 22:50:41 -05:00
|
|
|
OrderBy = new[] { ("SortName", SortOrder.Ascending) },
|
2022-05-01 00:33:22 -05:00
|
|
|
IncludeItemTypes = new BaseItemKind[] { BaseItemKind.Episode },
|
|
|
|
Recursive = true,
|
|
|
|
});
|
|
|
|
|
2022-06-12 17:16:21 -05:00
|
|
|
if (items is null)
|
|
|
|
{
|
|
|
|
_logger.LogError("Folder items were null");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-05-01 00:33:22 -05:00
|
|
|
// Queue all episodes on the server for fingerprinting.
|
2022-06-12 17:31:42 -05:00
|
|
|
_logger.LogDebug("Iterating through folder contents");
|
2022-05-09 22:50:41 -05:00
|
|
|
foreach (var item in items.Items)
|
|
|
|
{
|
|
|
|
if (item is not Episode episode)
|
|
|
|
{
|
2022-05-01 00:33:22 -05:00
|
|
|
_logger.LogError("Item {Name} is not an episode", item.Name);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
QueueEpisode(episode);
|
|
|
|
}
|
2022-06-12 17:16:21 -05:00
|
|
|
|
2022-06-12 17:31:42 -05:00
|
|
|
_logger.LogDebug("Queued {Count} episodes", items.Items.Count);
|
2022-05-01 00:33:22 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Called when an item is added to the server.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="sender">Sender.</param>
|
|
|
|
/// <param name="e">ItemChangeEventArgs.</param>
|
|
|
|
private void ItemAdded(object? sender, ItemChangeEventArgs e)
|
|
|
|
{
|
2022-05-09 22:50:41 -05:00
|
|
|
if (e.Item is not Episode episode)
|
|
|
|
{
|
2022-05-01 00:33:22 -05:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
_logger.LogDebug("Queuing fingerprint of new episode {Name}", episode.Name);
|
|
|
|
|
|
|
|
QueueEpisode(episode);
|
|
|
|
}
|
|
|
|
|
2022-05-09 22:50:41 -05:00
|
|
|
private void QueueEpisode(Episode episode)
|
|
|
|
{
|
|
|
|
if (Plugin.Instance is null)
|
|
|
|
{
|
2022-05-01 00:33:22 -05:00
|
|
|
throw new InvalidOperationException("plugin instance was null");
|
|
|
|
}
|
|
|
|
|
2022-05-09 22:50:41 -05:00
|
|
|
lock (_queueLock)
|
|
|
|
{
|
2022-05-01 00:33:22 -05:00
|
|
|
var queue = Plugin.Instance.AnalysisQueue;
|
|
|
|
|
|
|
|
// Allocate a new list for each new season
|
2022-05-09 22:50:41 -05:00
|
|
|
if (!queue.ContainsKey(episode.SeasonId))
|
|
|
|
{
|
2022-05-01 00:33:22 -05:00
|
|
|
Plugin.Instance.AnalysisQueue[episode.SeasonId] = new List<QueuedEpisode>();
|
|
|
|
}
|
|
|
|
|
2022-05-03 01:09:50 -05:00
|
|
|
// Only fingerprint up to 25% of the episode and at most 10 minutes.
|
2022-05-01 00:33:22 -05:00
|
|
|
var duration = TimeSpan.FromTicks(episode.RunTimeTicks ?? 0).TotalSeconds;
|
2022-05-09 22:50:41 -05:00
|
|
|
if (duration >= 5 * 60)
|
|
|
|
{
|
2022-05-01 00:33:22 -05:00
|
|
|
duration /= 4;
|
|
|
|
}
|
|
|
|
|
2022-05-03 01:09:50 -05:00
|
|
|
duration = Math.Min(duration, 10 * 60);
|
|
|
|
|
2022-05-09 22:50:41 -05:00
|
|
|
Plugin.Instance.AnalysisQueue[episode.SeasonId].Add(new QueuedEpisode()
|
|
|
|
{
|
2022-05-01 00:33:22 -05:00
|
|
|
SeriesName = episode.SeriesName,
|
|
|
|
SeasonNumber = episode.AiredSeasonNumber ?? 0,
|
|
|
|
EpisodeId = episode.Id,
|
2022-05-30 02:23:36 -05:00
|
|
|
Name = episode.Name,
|
2022-05-01 00:33:22 -05:00
|
|
|
Path = episode.Path,
|
|
|
|
FingerprintDuration = Convert.ToInt32(duration)
|
|
|
|
});
|
|
|
|
|
|
|
|
Plugin.Instance!.TotalQueued++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// FIXME: don't do this.
|
|
|
|
/// </summary>
|
2022-05-09 22:50:41 -05:00
|
|
|
private Guid GetAdministrator()
|
|
|
|
{
|
|
|
|
foreach (var user in _userManager.Users)
|
|
|
|
{
|
2022-06-12 17:31:42 -05:00
|
|
|
_logger.LogDebug("Checking access of user {Username}", user.Username);
|
2022-06-12 17:16:21 -05:00
|
|
|
|
2022-05-09 22:50:41 -05:00
|
|
|
if (!user.HasPermission(Jellyfin.Data.Enums.PermissionKind.IsAdministrator))
|
|
|
|
{
|
2022-06-12 17:31:42 -05:00
|
|
|
_logger.LogDebug("User {Username} does not have the required access, continuing", user.Username);
|
2022-05-01 00:33:22 -05:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2022-05-02 01:18:31 -05:00
|
|
|
_logger.LogDebug("Accessing libraries as {Username}", user.Username);
|
2022-05-01 00:33:22 -05:00
|
|
|
return user.Id;
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new FingerprintException("Unable to find an administrator on this server.");
|
|
|
|
}
|
|
|
|
|
2022-06-12 16:57:15 -05:00
|
|
|
#if DEBUG
|
|
|
|
/// <summary>
|
|
|
|
/// Logs the exact commit that created this version of the plugin. Only used in unstable builds.
|
|
|
|
/// </summary>
|
|
|
|
private void LogVersion()
|
|
|
|
{
|
|
|
|
var assembly = GetType().Assembly;
|
|
|
|
var path = GetType().Namespace + ".Configuration.version.txt";
|
|
|
|
|
|
|
|
using (var stream = assembly.GetManifestResourceStream(path))
|
|
|
|
{
|
|
|
|
if (stream is null)
|
|
|
|
{
|
|
|
|
_logger.LogWarning("Unable to read embedded version information");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
var version = string.Empty;
|
|
|
|
using (var reader = new StreamReader(stream))
|
|
|
|
{
|
|
|
|
version = reader.ReadToEnd().TrimEnd();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (version == "unknown")
|
|
|
|
{
|
|
|
|
_logger.LogTrace("Embedded version information was not valid, ignoring");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
_logger.LogInformation("Unstable version built from commit {Version}", version);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2022-05-01 00:33:22 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Dispose.
|
|
|
|
/// </summary>
|
|
|
|
public void Dispose()
|
|
|
|
{
|
|
|
|
Dispose(true);
|
|
|
|
GC.SuppressFinalize(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
2022-05-09 22:56:03 -05:00
|
|
|
/// Protected dispose.
|
2022-05-01 00:33:22 -05:00
|
|
|
/// </summary>
|
2022-05-09 22:56:03 -05:00
|
|
|
/// <param name="dispose">Dispose.</param>
|
2022-05-01 00:33:22 -05:00
|
|
|
protected virtual void Dispose(bool dispose)
|
|
|
|
{
|
|
|
|
if (!dispose)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
_libraryManager.ItemAdded -= ItemAdded;
|
|
|
|
}
|
|
|
|
}
|