2019-02-21 01:57:43 -08:00
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
2022-05-05 18:10:34 -05:00
|
|
|
using System.IO;
|
2022-04-29 23:52:50 -05:00
|
|
|
using ConfusedPolarBear.Plugin.IntroSkipper.Configuration;
|
2019-02-21 01:57:43 -08:00
|
|
|
using MediaBrowser.Common.Configuration;
|
|
|
|
using MediaBrowser.Common.Plugins;
|
2022-06-09 14:07:40 -05:00
|
|
|
using MediaBrowser.Controller.Configuration;
|
2022-07-29 03:34:55 -05:00
|
|
|
using MediaBrowser.Controller.Entities;
|
2022-06-15 01:00:03 -05:00
|
|
|
using MediaBrowser.Controller.Library;
|
2019-02-21 01:57:43 -08:00
|
|
|
using MediaBrowser.Model.Plugins;
|
|
|
|
using MediaBrowser.Model.Serialization;
|
2022-09-27 20:31:18 -05:00
|
|
|
using Microsoft.Extensions.Logging;
|
2019-02-21 01:57:43 -08:00
|
|
|
|
2022-04-29 23:52:50 -05:00
|
|
|
namespace ConfusedPolarBear.Plugin.IntroSkipper;
|
2021-12-13 16:58:05 -07:00
|
|
|
|
|
|
|
/// <summary>
|
2022-05-01 01:24:57 -05:00
|
|
|
/// Intro skipper plugin. Uses audio analysis to find common sequences of audio shared between episodes.
|
2021-12-13 16:58:05 -07:00
|
|
|
/// </summary>
|
|
|
|
public class Plugin : BasePlugin<PluginConfiguration>, IHasWebPages
|
2019-02-21 01:57:43 -08:00
|
|
|
{
|
2022-10-28 02:25:57 -05:00
|
|
|
private readonly object _serializationLock = new();
|
|
|
|
private readonly object _introsLock = new();
|
2022-05-05 18:10:34 -05:00
|
|
|
private IXmlSerializer _xmlSerializer;
|
2022-06-15 01:00:03 -05:00
|
|
|
private ILibraryManager _libraryManager;
|
2022-09-27 20:31:18 -05:00
|
|
|
private ILogger<Plugin> _logger;
|
2022-05-05 18:10:34 -05:00
|
|
|
private string _introPath;
|
|
|
|
|
2022-05-01 01:24:57 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Initializes a new instance of the <see cref="Plugin"/> class.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="applicationPaths">Instance of the <see cref="IApplicationPaths"/> interface.</param>
|
|
|
|
/// <param name="xmlSerializer">Instance of the <see cref="IXmlSerializer"/> interface.</param>
|
2022-06-09 14:07:40 -05:00
|
|
|
/// <param name="serverConfiguration">Server configuration manager.</param>
|
2022-06-15 01:00:03 -05:00
|
|
|
/// <param name="libraryManager">Library manager.</param>
|
2022-09-27 20:31:18 -05:00
|
|
|
/// <param name="logger">Logger.</param>
|
2022-06-09 14:07:40 -05:00
|
|
|
public Plugin(
|
|
|
|
IApplicationPaths applicationPaths,
|
|
|
|
IXmlSerializer xmlSerializer,
|
2022-06-15 01:00:03 -05:00
|
|
|
IServerConfigurationManager serverConfiguration,
|
2022-09-27 20:31:18 -05:00
|
|
|
ILibraryManager libraryManager,
|
|
|
|
ILogger<Plugin> logger)
|
2022-05-01 01:24:57 -05:00
|
|
|
: base(applicationPaths, xmlSerializer)
|
|
|
|
{
|
2022-09-27 21:03:27 -05:00
|
|
|
Instance = this;
|
|
|
|
|
2022-05-05 18:10:34 -05:00
|
|
|
_xmlSerializer = xmlSerializer;
|
2022-06-15 01:00:03 -05:00
|
|
|
_libraryManager = libraryManager;
|
2022-09-27 20:31:18 -05:00
|
|
|
_logger = logger;
|
2022-05-05 18:10:34 -05:00
|
|
|
|
|
|
|
FingerprintCachePath = Path.Join(applicationPaths.PluginConfigurationsPath, "intros", "cache");
|
2022-09-27 21:03:27 -05:00
|
|
|
FFmpegPath = serverConfiguration.GetEncodingOptions().EncoderAppPathDisplay;
|
|
|
|
_introPath = Path.Join(applicationPaths.PluginConfigurationsPath, "intros", "intros.xml");
|
|
|
|
|
|
|
|
// Create the base & cache directories (if needed).
|
2022-05-05 18:10:34 -05:00
|
|
|
if (!Directory.Exists(FingerprintCachePath))
|
|
|
|
{
|
|
|
|
Directory.CreateDirectory(FingerprintCachePath);
|
|
|
|
}
|
|
|
|
|
2022-06-14 14:36:05 -05:00
|
|
|
ConfigurationChanged += OnConfigurationChanged;
|
|
|
|
|
2022-09-27 21:03:27 -05:00
|
|
|
// TODO: remove when https://github.com/jellyfin/jellyfin-meta/discussions/30 is complete
|
2022-09-27 20:31:18 -05:00
|
|
|
try
|
|
|
|
{
|
|
|
|
RestoreTimestamps();
|
|
|
|
}
|
|
|
|
catch (Exception ex)
|
|
|
|
{
|
|
|
|
_logger.LogWarning("Unable to load introduction timestamps: {Exception}", ex);
|
|
|
|
}
|
2022-05-05 18:10:34 -05:00
|
|
|
}
|
|
|
|
|
2022-06-14 14:36:05 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Fired after configuration has been saved so the auto skip timer can be stopped or started.
|
|
|
|
/// </summary>
|
|
|
|
public event EventHandler? AutoSkipChanged;
|
|
|
|
|
2022-05-09 22:56:03 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Gets the results of fingerprinting all episodes.
|
|
|
|
/// </summary>
|
2022-09-27 21:03:27 -05:00
|
|
|
public Dictionary<Guid, Intro> Intros { get; } = new();
|
2022-05-09 22:56:03 -05:00
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Gets the mapping of season ids to episodes that have been queued for fingerprinting.
|
|
|
|
/// </summary>
|
2022-09-27 21:03:27 -05:00
|
|
|
public Dictionary<Guid, List<QueuedEpisode>> AnalysisQueue { get; } = new();
|
2022-05-09 22:56:03 -05:00
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Gets or sets the total number of episodes in the queue.
|
|
|
|
/// </summary>
|
|
|
|
public int TotalQueued { get; set; }
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Gets the directory to cache fingerprints in.
|
|
|
|
/// </summary>
|
|
|
|
public string FingerprintCachePath { get; private set; }
|
|
|
|
|
2022-06-09 14:07:40 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Gets the full path to FFmpeg.
|
|
|
|
/// </summary>
|
|
|
|
public string FFmpegPath { get; private set; }
|
|
|
|
|
2022-05-09 22:56:03 -05:00
|
|
|
/// <inheritdoc />
|
|
|
|
public override string Name => "Intro Skipper";
|
|
|
|
|
|
|
|
/// <inheritdoc />
|
|
|
|
public override Guid Id => Guid.Parse("c83d86bb-a1e0-4c35-a113-e2101cf4ee6b");
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Gets the plugin instance.
|
|
|
|
/// </summary>
|
|
|
|
public static Plugin? Instance { get; private set; }
|
|
|
|
|
2022-05-05 18:10:34 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Save timestamps to disk.
|
|
|
|
/// </summary>
|
|
|
|
public void SaveTimestamps()
|
|
|
|
{
|
2022-07-17 01:54:05 -05:00
|
|
|
lock (_serializationLock)
|
2022-05-05 18:10:34 -05:00
|
|
|
{
|
2022-07-17 01:54:05 -05:00
|
|
|
var introList = new List<Intro>();
|
2022-05-05 18:10:34 -05:00
|
|
|
|
2022-07-17 01:54:05 -05:00
|
|
|
foreach (var intro in Plugin.Instance!.Intros)
|
|
|
|
{
|
|
|
|
introList.Add(intro.Value);
|
|
|
|
}
|
|
|
|
|
|
|
|
_xmlSerializer.SerializeToFile(introList, _introPath);
|
|
|
|
}
|
2022-05-05 18:10:34 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Restore previous analysis results from disk.
|
|
|
|
/// </summary>
|
|
|
|
public void RestoreTimestamps()
|
|
|
|
{
|
|
|
|
if (!File.Exists(_introPath))
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Since dictionaries can't be easily serialized, analysis results are stored on disk as a list.
|
|
|
|
var introList = (List<Intro>)_xmlSerializer.DeserializeFromFile(typeof(List<Intro>), _introPath);
|
|
|
|
|
|
|
|
foreach (var intro in introList)
|
|
|
|
{
|
|
|
|
Plugin.Instance!.Intros[intro.EpisodeId] = intro;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-29 03:34:55 -05:00
|
|
|
internal BaseItem GetItem(Guid id)
|
|
|
|
{
|
|
|
|
return _libraryManager.GetItemById(id);
|
|
|
|
}
|
|
|
|
|
2022-06-15 01:00:03 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Gets the full path for an item.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="id">Item id.</param>
|
|
|
|
/// <returns>Full path to item.</returns>
|
|
|
|
internal string GetItemPath(Guid id)
|
|
|
|
{
|
2022-07-29 03:34:55 -05:00
|
|
|
return GetItem(id).Path;
|
2022-06-15 01:00:03 -05:00
|
|
|
}
|
|
|
|
|
2022-10-28 02:25:57 -05:00
|
|
|
internal void UpdateTimestamps(Dictionary<Guid, Intro> newIntros)
|
|
|
|
{
|
|
|
|
lock (_introsLock)
|
|
|
|
{
|
|
|
|
foreach (var intro in newIntros)
|
|
|
|
{
|
|
|
|
Plugin.Instance!.Intros[intro.Key] = intro.Value;
|
|
|
|
}
|
|
|
|
|
|
|
|
Plugin.Instance!.SaveTimestamps();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-05 18:10:34 -05:00
|
|
|
/// <inheritdoc />
|
|
|
|
public IEnumerable<PluginPageInfo> GetPages()
|
|
|
|
{
|
|
|
|
return new[]
|
|
|
|
{
|
|
|
|
new PluginPageInfo
|
|
|
|
{
|
|
|
|
Name = this.Name,
|
2022-07-16 23:35:15 -05:00
|
|
|
EmbeddedResourcePath = GetType().Namespace + ".Configuration.configPage.html"
|
|
|
|
},
|
|
|
|
new PluginPageInfo
|
|
|
|
{
|
|
|
|
Name = "visualizer.js",
|
|
|
|
EmbeddedResourcePath = GetType().Namespace + ".Configuration.visualizer.js"
|
2022-05-05 18:10:34 -05:00
|
|
|
}
|
|
|
|
};
|
2022-05-01 01:24:57 -05:00
|
|
|
}
|
2022-06-14 14:36:05 -05:00
|
|
|
|
|
|
|
private void OnConfigurationChanged(object? sender, BasePluginConfiguration e)
|
|
|
|
{
|
|
|
|
AutoSkipChanged?.Invoke(this, EventArgs.Empty);
|
|
|
|
}
|
2019-03-10 08:53:30 +09:00
|
|
|
}
|