2022-06-22 22:03:34 -05:00
namespace ConfusedPolarBear.Plugin.IntroSkipper ;
using System ;
using System.Collections.Generic ;
2022-06-27 00:21:30 -05:00
using System.Linq ;
2022-06-22 22:03:34 -05:00
using Jellyfin.Data.Enums ;
using MediaBrowser.Controller.Entities ;
using MediaBrowser.Controller.Entities.TV ;
using MediaBrowser.Controller.Library ;
using MediaBrowser.Model.Entities ;
using Microsoft.Extensions.Logging ;
/// <summary>
/// Manages enqueuing library items for analysis.
/// </summary>
public class QueueManager
{
private ILibraryManager _libraryManager ;
private ILogger < QueueManager > _logger ;
/// <summary>
/// Initializes a new instance of the <see cref="QueueManager"/> class.
/// </summary>
/// <param name="logger">Logger.</param>
/// <param name="libraryManager">Library manager.</param>
public QueueManager ( ILogger < QueueManager > logger , ILibraryManager libraryManager )
{
_logger = logger ;
_libraryManager = libraryManager ;
}
/// <summary>
/// Iterates through all libraries on the server and queues all episodes for analysis.
/// </summary>
public void EnqueueAllEpisodes ( )
{
// Assert that ffmpeg with chromaprint is installed
if ( ! Chromaprint . CheckFFmpegVersion ( ) )
{
throw new FingerprintException (
2022-06-26 19:02:50 -05:00
"ffmpeg with chromaprint is not installed on this system - episodes will not be analyzed. If Jellyfin is running natively, install jellyfin-ffmpeg5. If Jellyfin is running in a container, upgrade it to the latest version of 10.8.0." ) ;
2022-06-22 22:03:34 -05:00
}
Plugin . Instance ! . AnalysisQueue . Clear ( ) ;
2022-06-27 00:34:57 -05:00
Plugin . Instance ! . TotalQueued = 0 ;
2022-06-22 22:03:34 -05:00
2022-06-27 00:21:30 -05:00
// Get the list of library names which have been selected for analysis, ignoring whitespace and empty entries.
var selected = Plugin . Instance ! . Configuration . SelectedLibraries
. Split ( ',' , StringSplitOptions . RemoveEmptyEntries | StringSplitOptions . TrimEntries )
. ToList ( ) ;
if ( selected . Count > 0 )
{
_logger . LogInformation ( "Limiting analysis to the following libraries: {Selected}" , selected ) ;
}
else
{
_logger . LogDebug ( "Not limiting analysis by library name" ) ;
}
// For all selected TV show libraries, enqueue all contained items.
2022-06-22 22:03:34 -05:00
foreach ( var folder in _libraryManager . GetVirtualFolders ( ) )
{
if ( folder . CollectionType ! = CollectionTypeOptions . TvShows )
{
continue ;
}
2022-06-27 00:21:30 -05:00
// If libraries have been selected for analysis, ensure this library was selected.
if ( selected . Count > 0 & & ! selected . Contains ( folder . Name ) )
{
_logger . LogDebug ( "Not analyzing library \"{Name}\"" , folder . Name ) ;
continue ;
}
2022-06-22 22:03:34 -05:00
_logger . LogInformation (
"Running enqueue of items in library {Name} ({ItemId})" ,
folder . Name ,
folder . ItemId ) ;
try
{
QueueLibraryContents ( folder . ItemId ) ;
}
catch ( Exception ex )
{
_logger . LogError ( "Failed to enqueue items from library {Name}: {Exception}" , folder . Name , ex ) ;
}
}
}
private void QueueLibraryContents ( string rawId )
{
_logger . LogDebug ( "Constructing anonymous internal query" ) ;
var query = new InternalItemsQuery ( )
{
// Order by series name, season, and then episode number so that status updates are logged in order
ParentId = Guid . Parse ( rawId ) ,
OrderBy = new [ ]
{
( "SeriesSortName" , SortOrder . Ascending ) ,
( "ParentIndexNumber" , SortOrder . Ascending ) ,
( "IndexNumber" , SortOrder . Ascending ) ,
} ,
IncludeItemTypes = new BaseItemKind [ ] { BaseItemKind . Episode } ,
Recursive = true ,
} ;
_logger . LogDebug ( "Getting items" ) ;
var items = _libraryManager . GetItemList ( query , false ) ;
if ( items is null )
{
_logger . LogError ( "Library query result is null" ) ;
return ;
}
// Queue all episodes on the server for fingerprinting.
_logger . LogDebug ( "Iterating through library items" ) ;
foreach ( var item in items )
{
if ( item is not Episode episode )
{
_logger . LogError ( "Item {Name} is not an episode" , item . Name ) ;
continue ;
}
QueueEpisode ( episode ) ;
}
_logger . LogDebug ( "Queued {Count} episodes" , items . Count ) ;
}
private void QueueEpisode ( Episode episode )
{
if ( Plugin . Instance is null )
{
throw new InvalidOperationException ( "plugin instance was null" ) ;
}
if ( string . IsNullOrEmpty ( episode . Path ) )
{
_logger . LogWarning ( "Not queuing episode {Id} as no path was provided by Jellyfin" , episode . Id ) ;
return ;
}
// Only fingerprint up to 25% of the episode and at most 10 minutes.
var duration = TimeSpan . FromTicks ( episode . RunTimeTicks ? ? 0 ) . TotalSeconds ;
if ( duration > = 5 * 60 )
{
duration / = 4 ;
}
duration = Math . Min ( duration , 10 * 60 ) ;
2022-07-03 02:47:48 -05:00
// Allocate a new list for each new season
Plugin . Instance ! . AnalysisQueue . TryAdd ( episode . SeasonId , new List < QueuedEpisode > ( ) ) ;
// Queue the episode for analysis
2022-06-22 22:03:34 -05:00
Plugin . Instance . AnalysisQueue [ episode . SeasonId ] . Add ( new QueuedEpisode ( )
{
SeriesName = episode . SeriesName ,
SeasonNumber = episode . AiredSeasonNumber ? ? 0 ,
EpisodeId = episode . Id ,
Name = episode . Name ,
Path = episode . Path ,
FingerprintDuration = Convert . ToInt32 ( duration )
} ) ;
Plugin . Instance ! . TotalQueued + + ;
}
}