* start to rework plex collection scanning * sync plex collections to db * sync plex collection items * update changelog
165 lines
6.0 KiB
C#
165 lines
6.0 KiB
C#
using System.Globalization;
|
|
using System.Threading.Channels;
|
|
using ErsatzTV.Core;
|
|
using ErsatzTV.Core.Domain;
|
|
using ErsatzTV.Core.Interfaces.Locking;
|
|
using ErsatzTV.Core.Interfaces.Plex;
|
|
using ErsatzTV.Core.Interfaces.Repositories;
|
|
using ErsatzTV.Core.Plex;
|
|
using Microsoft.Extensions.Logging;
|
|
|
|
namespace ErsatzTV.Application.Plex;
|
|
|
|
public class SynchronizePlexMediaSourcesHandler : IRequestHandler<SynchronizePlexMediaSources,
|
|
Either<BaseError, List<PlexMediaSource>>>
|
|
{
|
|
private const string LocalhostUri = "http://localhost:32400";
|
|
|
|
private readonly ChannelWriter<IScannerBackgroundServiceRequest> _channel;
|
|
private readonly IEntityLocker _entityLocker;
|
|
private readonly ILogger<SynchronizePlexMediaSourcesHandler> _logger;
|
|
private readonly IMediaSourceRepository _mediaSourceRepository;
|
|
private readonly IPlexSecretStore _plexSecretStore;
|
|
private readonly IPlexServerApiClient _plexServerApiClient;
|
|
private readonly IPlexTvApiClient _plexTvApiClient;
|
|
|
|
public SynchronizePlexMediaSourcesHandler(
|
|
IMediaSourceRepository mediaSourceRepository,
|
|
IPlexTvApiClient plexTvApiClient,
|
|
IPlexServerApiClient plexServerApiClient,
|
|
IPlexSecretStore plexSecretStore,
|
|
ChannelWriter<IScannerBackgroundServiceRequest> channel,
|
|
IEntityLocker entityLocker,
|
|
ILogger<SynchronizePlexMediaSourcesHandler> logger)
|
|
{
|
|
_mediaSourceRepository = mediaSourceRepository;
|
|
_plexTvApiClient = plexTvApiClient;
|
|
_plexServerApiClient = plexServerApiClient;
|
|
_plexSecretStore = plexSecretStore;
|
|
_channel = channel;
|
|
_entityLocker = entityLocker;
|
|
_logger = logger;
|
|
}
|
|
|
|
public Task<Either<BaseError, List<PlexMediaSource>>> Handle(
|
|
SynchronizePlexMediaSources request,
|
|
CancellationToken cancellationToken) => _plexTvApiClient.GetServers().BindAsync(SynchronizeAllServers);
|
|
|
|
private async Task<Either<BaseError, List<PlexMediaSource>>> SynchronizeAllServers(
|
|
List<PlexMediaSource> servers)
|
|
{
|
|
List<PlexMediaSource> allExisting = await _mediaSourceRepository.GetAllPlex();
|
|
foreach (PlexMediaSource server in servers)
|
|
{
|
|
await SynchronizeServer(allExisting, server);
|
|
}
|
|
|
|
// delete removed servers
|
|
foreach (PlexMediaSource removed in allExisting.Filter(
|
|
s => servers.All(pms => pms.ClientIdentifier != s.ClientIdentifier)))
|
|
{
|
|
_logger.LogWarning(
|
|
"Deleting removed Plex server {ServerName}!",
|
|
removed.Id.ToString(CultureInfo.InvariantCulture));
|
|
await _mediaSourceRepository.DeletePlex(removed);
|
|
}
|
|
|
|
foreach (PlexMediaSource mediaSource in await _mediaSourceRepository.GetAllPlex())
|
|
{
|
|
await _channel.WriteAsync(new SynchronizePlexLibraries(mediaSource.Id));
|
|
}
|
|
|
|
_entityLocker.UnlockPlex();
|
|
|
|
return allExisting;
|
|
}
|
|
|
|
private async Task SynchronizeServer(List<PlexMediaSource> allExisting, PlexMediaSource server)
|
|
{
|
|
if (server.Connections.All(c => c.Uri != LocalhostUri))
|
|
{
|
|
var localhost = new PlexConnection
|
|
{
|
|
PlexMediaSource = server,
|
|
PlexMediaSourceId = server.Id,
|
|
Uri = LocalhostUri
|
|
};
|
|
|
|
server.Connections.Add(localhost);
|
|
}
|
|
|
|
Option<PlexMediaSource> maybeExisting =
|
|
allExisting.Find(s => s.ClientIdentifier == server.ClientIdentifier);
|
|
|
|
foreach (PlexMediaSource existing in maybeExisting)
|
|
{
|
|
existing.Platform = server.Platform;
|
|
existing.PlatformVersion = server.PlatformVersion;
|
|
existing.ProductVersion = server.ProductVersion;
|
|
existing.ServerName = server.ServerName;
|
|
var toAdd = server.Connections
|
|
.Filter(connection => existing.Connections.All(c => c.Uri != connection.Uri)).ToList();
|
|
var toRemove = existing.Connections
|
|
.Filter(connection => server.Connections.All(c => c.Uri != connection.Uri)).ToList();
|
|
await _mediaSourceRepository.Update(existing, toAdd, toRemove);
|
|
await FindConnectionToActivate(existing);
|
|
}
|
|
|
|
if (maybeExisting.IsNone)
|
|
{
|
|
await _mediaSourceRepository.Add(server);
|
|
await FindConnectionToActivate(server);
|
|
}
|
|
}
|
|
|
|
private async Task FindConnectionToActivate(PlexMediaSource server)
|
|
{
|
|
var prioritized = server.Connections
|
|
.OrderByDescending(pc => pc.Uri == LocalhostUri)
|
|
.ThenByDescending(pc => pc.IsActive)
|
|
.ToList();
|
|
|
|
foreach (PlexConnection connection in server.Connections)
|
|
{
|
|
connection.IsActive = false;
|
|
}
|
|
|
|
Option<PlexServerAuthToken> maybeToken = await _plexSecretStore.GetServerAuthToken(server.ClientIdentifier);
|
|
foreach (PlexServerAuthToken token in maybeToken)
|
|
{
|
|
foreach (PlexConnection connection in prioritized)
|
|
{
|
|
try
|
|
{
|
|
_logger.LogDebug("Attempting to locate to Plex at {Uri}", connection.Uri);
|
|
if (await _plexServerApiClient.Ping(connection, token))
|
|
{
|
|
_logger.LogInformation("Located Plex at {Uri}", connection.Uri);
|
|
connection.IsActive = true;
|
|
break;
|
|
}
|
|
}
|
|
catch
|
|
{
|
|
// do nothing
|
|
}
|
|
}
|
|
}
|
|
|
|
if (maybeToken.IsNone)
|
|
{
|
|
_logger.LogError(
|
|
"Unable to activate Plex connection for server {Server} without auth token",
|
|
server.ServerName);
|
|
}
|
|
|
|
if (server.Connections.All(c => !c.IsActive))
|
|
{
|
|
_logger.LogError("Unable to locate Plex");
|
|
server.Connections.Head().IsActive = true;
|
|
}
|
|
|
|
await _mediaSourceRepository.Update(server, new List<PlexConnection>(), new List<PlexConnection>());
|
|
}
|
|
}
|