using ErsatzTV.Core; using ErsatzTV.Core.Domain; using ErsatzTV.Core.Extensions; using ErsatzTV.Core.Interfaces.Jellyfin; using ErsatzTV.Core.Interfaces.Metadata; using ErsatzTV.Core.Interfaces.Repositories; using ErsatzTV.Core.Jellyfin; using ErsatzTV.Core.Metadata; using ErsatzTV.Scanner.Core.Metadata; using Microsoft.Extensions.Logging; namespace ErsatzTV.Scanner.Core.Jellyfin; public class JellyfinTelevisionLibraryScanner : MediaServerTelevisionLibraryScanner, IJellyfinTelevisionLibraryScanner { private readonly IJellyfinApiClient _jellyfinApiClient; private readonly ILogger _logger; private readonly IMediaSourceRepository _mediaSourceRepository; private readonly IJellyfinPathReplacementService _pathReplacementService; private readonly IJellyfinTelevisionRepository _televisionRepository; public JellyfinTelevisionLibraryScanner( IJellyfinApiClient jellyfinApiClient, IMediaSourceRepository mediaSourceRepository, IJellyfinTelevisionRepository televisionRepository, IJellyfinPathReplacementService pathReplacementService, ILocalFileSystem localFileSystem, IMetadataRepository metadataRepository, IMediator mediator, ILogger logger) : base( localFileSystem, metadataRepository, mediator, logger) { _jellyfinApiClient = jellyfinApiClient; _mediaSourceRepository = mediaSourceRepository; _televisionRepository = televisionRepository; _pathReplacementService = pathReplacementService; _logger = logger; } protected override bool ServerSupportsRemoteStreaming => true; public async Task> ScanLibrary( string address, string apiKey, JellyfinLibrary library, bool deepScan, CancellationToken cancellationToken) { List pathReplacements = await _mediaSourceRepository.GetJellyfinPathReplacements(library.MediaSourceId); string GetLocalPath(JellyfinEpisode episode) { return _pathReplacementService.GetReplacementJellyfinPath( pathReplacements, episode.GetHeadVersion().MediaFiles.Head().Path, false); } return await ScanLibrary( _televisionRepository, new JellyfinConnectionParameters(address, apiKey, library.MediaSourceId), library, GetLocalPath, deepScan, cancellationToken); } protected override IAsyncEnumerable> GetShowLibraryItems( JellyfinConnectionParameters connectionParameters, JellyfinLibrary library) => _jellyfinApiClient.GetShowLibraryItems(connectionParameters.Address, connectionParameters.ApiKey, library); protected override string MediaServerItemId(JellyfinShow show) => show.ItemId; protected override string MediaServerItemId(JellyfinSeason season) => season.ItemId; protected override string MediaServerItemId(JellyfinEpisode episode) => episode.ItemId; protected override string MediaServerEtag(JellyfinShow show) => show.Etag; protected override string MediaServerEtag(JellyfinSeason season) => season.Etag; protected override string MediaServerEtag(JellyfinEpisode episode) => episode.Etag; protected override IAsyncEnumerable> GetSeasonLibraryItems( JellyfinLibrary library, JellyfinConnectionParameters connectionParameters, JellyfinShow show) => _jellyfinApiClient.GetSeasonLibraryItems( connectionParameters.Address, connectionParameters.ApiKey, library, show.ItemId); protected override IAsyncEnumerable> GetEpisodeLibraryItems( JellyfinLibrary library, JellyfinConnectionParameters connectionParameters, JellyfinShow show, JellyfinSeason season) => _jellyfinApiClient.GetEpisodeLibraryItems( connectionParameters.Address, connectionParameters.ApiKey, library, season.ItemId); protected override Task> GetFullMetadata( JellyfinConnectionParameters connectionParameters, JellyfinLibrary library, MediaItemScanResult result, JellyfinShow incoming, bool deepScan) => Task.FromResult(Option.None); protected override Task> GetFullMetadata( JellyfinConnectionParameters connectionParameters, JellyfinLibrary library, MediaItemScanResult result, JellyfinSeason incoming, bool deepScan) => Task.FromResult(Option.None); protected override Task> GetFullMetadata( JellyfinConnectionParameters connectionParameters, JellyfinLibrary library, MediaItemScanResult result, JellyfinEpisode incoming, bool deepScan) => Task.FromResult(Option.None); protected override Task>> GetFullMetadataAndStatistics( JellyfinConnectionParameters connectionParameters, JellyfinLibrary library, MediaItemScanResult result, JellyfinEpisode incoming) => Task.FromResult(Option>.None); protected override async Task> GetMediaServerStatistics( JellyfinConnectionParameters connectionParameters, JellyfinLibrary library, MediaItemScanResult result, JellyfinEpisode incoming) { _logger.LogDebug("Refreshing {Attribute} for {Path}", "Jellyfin Statistics", result.LocalPath); Either maybeVersion = await _jellyfinApiClient.GetPlaybackInfo( connectionParameters.Address, connectionParameters.ApiKey, library, incoming.ItemId); foreach (BaseError error in maybeVersion.LeftToSeq()) { _logger.LogWarning("Failed to get episode statistics from Jellyfin: {Error}", error.ToString()); } // chapters are pulled with metadata, not with statistics, but we need to save them here foreach (MediaVersion version in maybeVersion.RightToSeq()) { version.Chapters = result.Item.GetHeadVersion().Chapters; } return maybeVersion.ToOption(); } protected override Task>> UpdateMetadata( MediaItemScanResult result, ShowMetadata fullMetadata) => Task.FromResult>>(result); protected override Task>> UpdateMetadata( MediaItemScanResult result, SeasonMetadata fullMetadata) => Task.FromResult>>(result); protected override Task>> UpdateMetadata( MediaItemScanResult result, EpisodeMetadata fullMetadata) => Task.FromResult>>(result); }