using ErsatzTV.Core; using ErsatzTV.Core.Domain; using ErsatzTV.Core.Emby; using ErsatzTV.Core.Extensions; using ErsatzTV.Core.Interfaces.Emby; using ErsatzTV.Core.Interfaces.Metadata; using ErsatzTV.Core.Interfaces.Repositories; using ErsatzTV.Core.Metadata; using ErsatzTV.Scanner.Core.Metadata; using Microsoft.Extensions.Logging; namespace ErsatzTV.Scanner.Core.Emby; public class EmbyTelevisionLibraryScanner : MediaServerTelevisionLibraryScanner, IEmbyTelevisionLibraryScanner { private readonly IEmbyApiClient _embyApiClient; private readonly ILogger _logger; private readonly IMediaSourceRepository _mediaSourceRepository; private readonly IEmbyPathReplacementService _pathReplacementService; private readonly IEmbyTelevisionRepository _televisionRepository; public EmbyTelevisionLibraryScanner( IEmbyApiClient embyApiClient, IMediaSourceRepository mediaSourceRepository, IEmbyTelevisionRepository televisionRepository, IEmbyPathReplacementService pathReplacementService, ILocalFileSystem localFileSystem, IMetadataRepository metadataRepository, IMediator mediator, ILogger logger) : base( localFileSystem, metadataRepository, mediator, logger) { _embyApiClient = embyApiClient; _mediaSourceRepository = mediaSourceRepository; _televisionRepository = televisionRepository; _pathReplacementService = pathReplacementService; _logger = logger; } protected override bool ServerSupportsRemoteStreaming => true; public async Task> ScanLibrary( string address, string apiKey, EmbyLibrary library, bool deepScan, CancellationToken cancellationToken) { List pathReplacements = await _mediaSourceRepository.GetEmbyPathReplacements(library.MediaSourceId); string GetLocalPath(EmbyEpisode episode) { return _pathReplacementService.GetReplacementEmbyPath( pathReplacements, episode.GetHeadVersion().MediaFiles.Head().Path, false); } return await ScanLibrary( _televisionRepository, new EmbyConnectionParameters(address, apiKey), library, GetLocalPath, deepScan, cancellationToken); } protected override IAsyncEnumerable> GetShowLibraryItems( EmbyConnectionParameters connectionParameters, EmbyLibrary library) => _embyApiClient.GetShowLibraryItems(connectionParameters.Address, connectionParameters.ApiKey, library); protected override string MediaServerItemId(EmbyShow show) => show.ItemId; protected override string MediaServerItemId(EmbySeason season) => season.ItemId; protected override string MediaServerItemId(EmbyEpisode episode) => episode.ItemId; protected override string MediaServerEtag(EmbyShow show) => show.Etag; protected override string MediaServerEtag(EmbySeason season) => season.Etag; protected override string MediaServerEtag(EmbyEpisode episode) => episode.Etag; protected override IAsyncEnumerable> GetSeasonLibraryItems( EmbyLibrary library, EmbyConnectionParameters connectionParameters, EmbyShow show) => _embyApiClient.GetSeasonLibraryItems( connectionParameters.Address, connectionParameters.ApiKey, library, show.ItemId); protected override IAsyncEnumerable> GetEpisodeLibraryItems( EmbyLibrary library, EmbyConnectionParameters connectionParameters, EmbyShow show, EmbySeason season) => _embyApiClient.GetEpisodeLibraryItems( connectionParameters.Address, connectionParameters.ApiKey, library, show.ItemId, season.ItemId); protected override Task> GetFullMetadata( EmbyConnectionParameters connectionParameters, EmbyLibrary library, MediaItemScanResult result, EmbyShow incoming, bool deepScan) => Task.FromResult(Option.None); protected override Task> GetFullMetadata( EmbyConnectionParameters connectionParameters, EmbyLibrary library, MediaItemScanResult result, EmbySeason incoming, bool deepScan) => Task.FromResult(Option.None); protected override Task> GetFullMetadata( EmbyConnectionParameters connectionParameters, EmbyLibrary library, MediaItemScanResult result, EmbyEpisode incoming, bool deepScan) => Task.FromResult(Option.None); protected override Task>> GetFullMetadataAndStatistics( EmbyConnectionParameters connectionParameters, EmbyLibrary library, MediaItemScanResult result, EmbyEpisode incoming) => Task.FromResult(Option>.None); protected override async Task> GetMediaServerStatistics( EmbyConnectionParameters connectionParameters, EmbyLibrary library, MediaItemScanResult result, EmbyEpisode incoming) { _logger.LogDebug("Refreshing {Attribute} for {Path}", "Emby Statistics", result.LocalPath); Either maybeVersion = await _embyApiClient.GetPlaybackInfo( connectionParameters.Address, connectionParameters.ApiKey, library, incoming.ItemId); foreach (BaseError error in maybeVersion.LeftToSeq()) { _logger.LogWarning("Failed to get episode statistics from Emby: {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); }