mirror of https://github.com/Ombi-app/Ombi
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
209 lines
7.7 KiB
209 lines
7.7 KiB
#region Copyright
|
|
// /************************************************************************
|
|
// Copyright (c) 2017 Jamie Rees
|
|
// File: JellyfinAvaliabilityCheker.cs
|
|
// Created By: Jamie Rees
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining
|
|
// a copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to
|
|
// permit persons to whom the Software is furnished to do so, subject to
|
|
// the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be
|
|
// included in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
// ************************************************************************/
|
|
#endregion
|
|
|
|
using System;
|
|
using System.Linq;
|
|
using System.Threading.Tasks;
|
|
using Microsoft.EntityFrameworkCore;
|
|
using Microsoft.Extensions.Logging;
|
|
using Ombi.Core;
|
|
using Ombi.Core.Services;
|
|
using Ombi.Helpers;
|
|
using Ombi.Hubs;
|
|
using Ombi.Notifications.Models;
|
|
using Ombi.Settings.Settings.Models;
|
|
using Ombi.Store.Entities;
|
|
using Ombi.Store.Repository;
|
|
using Ombi.Store.Repository.Requests;
|
|
using Quartz;
|
|
|
|
namespace Ombi.Schedule.Jobs.Jellyfin
|
|
{
|
|
public class JellyfinAvaliabilityChecker : AvailabilityChecker, IJellyfinAvaliabilityChecker
|
|
{
|
|
public JellyfinAvaliabilityChecker(IJellyfinContentRepository repo, ITvRequestRepository t, IMovieRequestRepository m,
|
|
INotificationHelper n, ILogger<JellyfinAvaliabilityChecker> log, INotificationHubService notification, IFeatureService featureService)
|
|
: base(t, n, log, notification)
|
|
{
|
|
_repo = repo;
|
|
_movieRepo = m;
|
|
_featureService = featureService;
|
|
}
|
|
|
|
private readonly IMovieRequestRepository _movieRepo;
|
|
private readonly IJellyfinContentRepository _repo;
|
|
private readonly IFeatureService _featureService;
|
|
|
|
public async Task Execute(IJobExecutionContext job)
|
|
{
|
|
_log.LogInformation("Starting Jellyfin Availability Check");
|
|
await NotificationHubService.SendNotificationToAdmins("Jellyfin Availability Checker Started");
|
|
await ProcessMovies();
|
|
await ProcessTv();
|
|
|
|
_log.LogInformation("Finished Jellyfin Availability Check");
|
|
await NotificationHubService.SendNotificationToAdmins("Jellyfin Availability Checker Finished");
|
|
}
|
|
|
|
private async Task ProcessMovies()
|
|
{
|
|
var feature4kEnabled = await _featureService.FeatureEnabled(FeatureNames.Movie4KRequests);
|
|
var movies = _movieRepo.GetAll().Include(x => x.RequestedUser).Where(x => !x.Available || (!x.Available4K && x.Has4KRequest));
|
|
|
|
foreach (var movie in movies)
|
|
{
|
|
var has4kRequest = movie.Has4KRequest;
|
|
JellyfinContent jellyfinContent = null;
|
|
if (movie.TheMovieDbId > 0)
|
|
{
|
|
jellyfinContent = await _repo.GetByTheMovieDbId(movie.TheMovieDbId.ToString());
|
|
}
|
|
else if(movie.ImdbId.HasValue())
|
|
{
|
|
jellyfinContent = await _repo.GetByImdbId(movie.ImdbId);
|
|
}
|
|
|
|
if (jellyfinContent == null)
|
|
{
|
|
// We don't have this yet
|
|
continue;
|
|
}
|
|
|
|
_log.LogInformation("We have found the request {0} on Jellyfin, sending the notification", movie?.Title ?? string.Empty);
|
|
|
|
var notify = false;
|
|
|
|
if (has4kRequest && jellyfinContent.Has4K && !movie.Available4K)
|
|
{
|
|
movie.Available4K = true;
|
|
movie.MarkedAsAvailable4K = DateTime.Now;
|
|
notify = true;
|
|
}
|
|
|
|
// If we have a non-4k version or we don't care about versions, then mark as available
|
|
if (!movie.Available && ( !feature4kEnabled || jellyfinContent.Quality != null ))
|
|
{
|
|
movie.Available = true;
|
|
movie.MarkedAsAvailable = DateTime.Now;
|
|
notify = true;
|
|
}
|
|
|
|
if (notify)
|
|
{
|
|
var recipient = movie.RequestedUser.Email.HasValue() ? movie.RequestedUser.Email : string.Empty;
|
|
|
|
_log.LogDebug("MovieId: {0}, RequestUser: {1}", movie.Id, recipient);
|
|
|
|
await _notificationService.Notify(new NotificationOptions
|
|
{
|
|
DateTime = DateTime.Now,
|
|
NotificationType = NotificationType.RequestAvailable,
|
|
RequestId = movie.Id,
|
|
RequestType = RequestType.Movie,
|
|
Recipient = recipient,
|
|
});
|
|
}
|
|
}
|
|
await _movieRepo.Save();
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
/// TODO This is EXCATLY the same as the PlexAvailabilityChecker. Refactor Please future Jamie
|
|
/// </summary>
|
|
/// <returns></returns>
|
|
private async Task ProcessTv()
|
|
{
|
|
var tv = _tvRepo.GetChild().Where(x => !x.Available);
|
|
var jellyfinEpisodes = _repo.GetAllEpisodes().Include(x => x.Series);
|
|
|
|
foreach (var child in tv)
|
|
{
|
|
|
|
var useImdb = false;
|
|
var useTvDb = false;
|
|
if (child.ParentRequest.ImdbId.HasValue())
|
|
{
|
|
useImdb = true;
|
|
}
|
|
|
|
if (child.ParentRequest.TvDbId.ToString().HasValue())
|
|
{
|
|
useTvDb = true;
|
|
}
|
|
|
|
var tvDbId = child.ParentRequest.TvDbId;
|
|
var imdbId = child.ParentRequest.ImdbId;
|
|
IQueryable<IMediaServerEpisode> seriesEpisodes = null;
|
|
if (useImdb)
|
|
{
|
|
seriesEpisodes = jellyfinEpisodes.Where(x => x.Series.ImdbId == imdbId.ToString());
|
|
}
|
|
|
|
if (useTvDb && (seriesEpisodes == null || !seriesEpisodes.Any()))
|
|
{
|
|
seriesEpisodes = jellyfinEpisodes.Where(x => x.Series.TvDbId == tvDbId.ToString());
|
|
}
|
|
|
|
if (seriesEpisodes == null)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (!seriesEpisodes.Any())
|
|
{
|
|
// Let's try and match the series by name
|
|
seriesEpisodes = jellyfinEpisodes.Where(x =>
|
|
x.Series.Title == child.Title);
|
|
}
|
|
|
|
await ProcessTvShow(seriesEpisodes, child);
|
|
}
|
|
|
|
await _tvRepo.Save();
|
|
}
|
|
|
|
private bool _disposed;
|
|
protected virtual void Dispose(bool disposing)
|
|
{
|
|
if (_disposed)
|
|
return;
|
|
|
|
if (disposing)
|
|
{
|
|
}
|
|
_disposed = true;
|
|
}
|
|
|
|
public void Dispose()
|
|
{
|
|
Dispose(true);
|
|
GC.SuppressFinalize(this);
|
|
}
|
|
}
|
|
} |