249 lines
7.9 KiB
C#
Raw Normal View History

2019-02-21 01:57:43 -08:00
using System;
using System.Collections.Generic;
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-10-31 01:00:39 -05:00
using MediaBrowser.Controller.Entities.TV;
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();
private IXmlSerializer _xmlSerializer;
private ILibraryManager _libraryManager;
2022-09-27 20:31:18 -05:00
private ILogger<Plugin> _logger;
private string _introPath;
2022-10-31 01:00:39 -05:00
private string _creditsPath; // TODO: FIXME: remove this
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>
/// <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,
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;
_xmlSerializer = xmlSerializer;
_libraryManager = libraryManager;
2022-09-27 20:31:18 -05:00
_logger = logger;
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");
2022-10-31 01:00:39 -05:00
// TODO: FIXME: remove this
_creditsPath = Path.Join(applicationPaths.PluginConfigurationsPath, "intros", "credits.csv");
2022-09-27 21:03:27 -05:00
// Create the base & cache directories (if needed).
if (!Directory.Exists(FingerprintCachePath))
{
Directory.CreateDirectory(FingerprintCachePath);
}
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-10-31 01:00:39 -05:00
// TODO: FIXME: remove this
if (File.Exists(_creditsPath))
{
File.Delete(_creditsPath);
}
}
/// <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 most recent media item queue.
2022-05-09 22:56:03 -05:00
/// </summary>
public Dictionary<Guid, List<QueuedEpisode>> QueuedMediaItems { 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; }
/// <summary>
/// Save timestamps to disk.
/// </summary>
public void SaveTimestamps()
{
lock (_serializationLock)
{
var introList = new List<Intro>();
foreach (var intro in Plugin.Instance!.Intros)
{
introList.Add(intro.Value);
}
_xmlSerializer.SerializeToFile(introList, _introPath);
}
}
/// <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);
}
/// <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-10-31 01:00:39 -05:00
internal void UpdateTimestamps(Dictionary<Guid, Intro> newIntros, AnalysisMode mode)
2022-10-28 02:25:57 -05:00
{
2022-10-31 01:00:39 -05:00
switch (mode)
2022-10-28 02:25:57 -05:00
{
2022-10-31 01:00:39 -05:00
case AnalysisMode.Introduction:
lock (_introsLock)
{
foreach (var intro in newIntros)
{
Plugin.Instance!.Intros[intro.Key] = intro.Value;
}
Plugin.Instance!.SaveTimestamps();
}
break;
case AnalysisMode.Credits:
// TODO: FIXME: implement properly
lock (_introsLock)
{
foreach (var credit in newIntros)
{
var item = GetItem(credit.Value.EpisodeId) as Episode;
if (item is null)
{
continue;
}
// Format: series, season number, episode number, title, start, end
var contents = string.Format(
System.Globalization.CultureInfo.InvariantCulture,
"{0},{1},{2},{3},{4},{5}\n",
item.SeriesName.Replace(",", string.Empty, StringComparison.Ordinal),
item.AiredSeasonNumber ?? 0,
item.IndexNumber ?? 0,
item.Name.Replace(",", string.Empty, StringComparison.Ordinal),
Math.Round(credit.Value.IntroStart, 2),
Math.Round(credit.Value.IntroEnd, 2));
File.AppendAllText(_creditsPath, contents);
}
}
break;
2022-10-28 02:25:57 -05:00
}
}
/// <inheritdoc />
public IEnumerable<PluginPageInfo> GetPages()
{
return new[]
{
new PluginPageInfo
{
Name = this.Name,
EmbeddedResourcePath = GetType().Namespace + ".Configuration.configPage.html"
},
new PluginPageInfo
{
Name = "visualizer.js",
EmbeddedResourcePath = GetType().Namespace + ".Configuration.visualizer.js"
}
};
2022-05-01 01:24:57 -05:00
}
private void OnConfigurationChanged(object? sender, BasePluginConfiguration e)
{
AutoSkipChanged?.Invoke(this, EventArgs.Empty);
}
2019-03-10 08:53:30 +09:00
}