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.
388 lines
14 KiB
388 lines
14 KiB
using System;
|
|
using System.Collections.Generic;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using NLog;
|
|
using NzbDrone.Common.EnsureThat;
|
|
using NzbDrone.Common.Extensions;
|
|
using NzbDrone.Core.DataAugmentation.Scene;
|
|
using NzbDrone.Core.Messaging.Events;
|
|
using NzbDrone.Core.Organizer;
|
|
using NzbDrone.Core.Parser;
|
|
using NzbDrone.Core.Tv.Events;
|
|
using NzbDrone.Core.MediaFiles;
|
|
using NzbDrone.Core.MediaFiles.Events;
|
|
using NzbDrone.Core.Datastore;
|
|
using NzbDrone.Core.Configuration;
|
|
|
|
namespace NzbDrone.Core.Tv
|
|
{
|
|
public interface IMovieService
|
|
{
|
|
Movie GetMovie(int movieId);
|
|
List<Movie> GetMovies(IEnumerable<int> movieIds);
|
|
PagingSpec<Movie> Paged(PagingSpec<Movie> pagingSpec);
|
|
Movie AddMovie(Movie newMovie);
|
|
List<Movie> AddMovies(List<Movie> newMovies);
|
|
Movie FindByImdbId(string imdbid);
|
|
Movie FindByTitle(string title);
|
|
Movie FindByTitle(string title, int year);
|
|
Movie FindByTitleInexact(string title);
|
|
Movie FindByTitleSlug(string slug);
|
|
bool MovieExists(Movie movie);
|
|
Movie GetMovieByFileId(int fileId);
|
|
List<Movie> GetMoviesBetweenDates(DateTime start, DateTime end, bool includeUnmonitored);
|
|
PagingSpec<Movie> MoviesWithoutFiles(PagingSpec<Movie> pagingSpec);
|
|
void DeleteMovie(int movieId, bool deleteFiles);
|
|
void SetFileId(Movie movie, MovieFile movieFile);
|
|
List<Movie> GetAllMovies();
|
|
Movie UpdateMovie(Movie movie);
|
|
List<Movie> UpdateMovie(List<Movie> movie);
|
|
bool MoviePathExists(string folder);
|
|
void RemoveAddOptions(Movie movie);
|
|
List<Movie> MoviesWithFiles(int movieId);
|
|
System.Linq.Expressions.Expression<Func<Core.Tv.Movie, bool>> ConstructFilterExpression(string FilterKey, string FilterValue);
|
|
}
|
|
|
|
public class MovieService : IMovieService, IHandle<MovieFileAddedEvent>,
|
|
IHandle<MovieFileDeletedEvent>
|
|
{
|
|
private readonly IMovieRepository _movieRepository;
|
|
private readonly IConfigService _configService;
|
|
private readonly IEventAggregator _eventAggregator;
|
|
private readonly IBuildFileNames _fileNameBuilder;
|
|
private readonly Logger _logger;
|
|
|
|
public MovieService(IMovieRepository movieRepository,
|
|
IEventAggregator eventAggregator,
|
|
ISceneMappingService sceneMappingService,
|
|
IEpisodeService episodeService,
|
|
IBuildFileNames fileNameBuilder,
|
|
IConfigService configService,
|
|
Logger logger)
|
|
{
|
|
_movieRepository = movieRepository;
|
|
_eventAggregator = eventAggregator;
|
|
_fileNameBuilder = fileNameBuilder;
|
|
_configService = configService;
|
|
_logger = logger;
|
|
}
|
|
|
|
|
|
public System.Linq.Expressions.Expression<Func<Core.Tv.Movie, bool>> ConstructFilterExpression(string FilterKey, string FilterValue)
|
|
{
|
|
if (FilterKey == "all" && FilterValue == "all")
|
|
{
|
|
return v => v.Monitored == true || v.Monitored == false;
|
|
}
|
|
else if (FilterKey == "monitored" && FilterValue == "false")
|
|
{
|
|
return v => v.Monitored == false;
|
|
}
|
|
else if (FilterKey == "monitored" && FilterValue == "true")
|
|
{
|
|
return v => v.Monitored == true;
|
|
}
|
|
else if (FilterKey == "moviestatus" && FilterValue == "available")
|
|
{
|
|
//TODO: might need to handle PreDB here
|
|
return v => v.Monitored == true &&
|
|
((v.MinimumAvailability == MovieStatusType.Released && v.Status >= MovieStatusType.Released) ||
|
|
(v.MinimumAvailability == MovieStatusType.InCinemas && v.Status >= MovieStatusType.InCinemas) ||
|
|
(v.MinimumAvailability == MovieStatusType.Announced && v.Status >= MovieStatusType.Announced) ||
|
|
(v.MinimumAvailability == MovieStatusType.PreDB && v.Status >= MovieStatusType.Released));
|
|
}
|
|
else if (FilterKey == "moviestatus" && FilterValue == "announced")
|
|
{
|
|
return v => v.Status == MovieStatusType.Announced;
|
|
}
|
|
else if (FilterKey == "moviestatus" && FilterValue == "incinemas")
|
|
{
|
|
return v => v.Status == MovieStatusType.InCinemas;
|
|
}
|
|
else if (FilterKey == "moviestatus" && FilterValue == "released")
|
|
{
|
|
return v => v.Status == MovieStatusType.Released;
|
|
}
|
|
return v => v.Monitored == true;
|
|
}
|
|
|
|
public Movie GetMovie(int movieId)
|
|
{
|
|
return _movieRepository.Get(movieId);
|
|
}
|
|
|
|
public List<Movie> GetMovies(IEnumerable<int> movieIds)
|
|
{
|
|
return _movieRepository.Get(movieIds).ToList();
|
|
}
|
|
|
|
public PagingSpec<Movie> Paged(PagingSpec<Movie> pagingSpec)
|
|
{
|
|
return _movieRepository.GetPaged(pagingSpec);
|
|
}
|
|
|
|
public Movie AddMovie(Movie newMovie)
|
|
{
|
|
Ensure.That(newMovie, () => newMovie).IsNotNull();
|
|
|
|
if (string.IsNullOrWhiteSpace(newMovie.Path))
|
|
{
|
|
var folderName = _fileNameBuilder.GetMovieFolder(newMovie);
|
|
newMovie.Path = Path.Combine(newMovie.RootFolderPath, folderName);
|
|
}
|
|
|
|
_logger.Info("Adding Movie {0} Path: [{1}]", newMovie, newMovie.Path);
|
|
|
|
newMovie.CleanTitle = newMovie.Title.CleanSeriesTitle();
|
|
newMovie.SortTitle = MovieTitleNormalizer.Normalize(newMovie.Title, newMovie.TmdbId);
|
|
newMovie.Added = DateTime.UtcNow;
|
|
|
|
_movieRepository.Insert(newMovie);
|
|
_eventAggregator.PublishEvent(new MovieAddedEvent(GetMovie(newMovie.Id)));
|
|
|
|
return newMovie;
|
|
}
|
|
|
|
public List<Movie> AddMovies(List<Movie> newMovies)
|
|
{
|
|
newMovies.ForEach(m => Ensure.That(m, () => m).IsNotNull());
|
|
|
|
newMovies.ForEach(m =>
|
|
{
|
|
if (string.IsNullOrWhiteSpace(m.Path))
|
|
{
|
|
var folderName = _fileNameBuilder.GetMovieFolder(m);
|
|
m.Path = Path.Combine(m.RootFolderPath, folderName);
|
|
}
|
|
|
|
m.CleanTitle = m.Title.CleanSeriesTitle();
|
|
m.SortTitle = MovieTitleNormalizer.Normalize(m.Title, m.TmdbId);
|
|
m.Added = DateTime.UtcNow;
|
|
});
|
|
|
|
var existingMovies = GetAllMovies();
|
|
var potentialMovieCount = newMovies.Count;
|
|
|
|
newMovies = newMovies.DistinctBy(movie => movie.TmdbId).ToList(); // Ensure we don't add the same movie twice
|
|
|
|
newMovies = newMovies.ExceptBy(n => n.TmdbId, existingMovies, e => e.TmdbId, EqualityComparer<int>.Default).ToList(); // Ensure we don't add a movie that already exists
|
|
|
|
_movieRepository.InsertMany(newMovies);
|
|
|
|
_logger.Debug("Adding {0} movies, {1} duplicates detected and skipped", newMovies.Count, potentialMovieCount - newMovies.Count);
|
|
|
|
newMovies.ForEach(m =>
|
|
{
|
|
_eventAggregator.PublishEvent(new MovieAddedEvent(m));
|
|
});
|
|
|
|
return newMovies;
|
|
}
|
|
|
|
public Movie FindByTitle(string title)
|
|
{
|
|
return _movieRepository.FindByTitle(title.CleanSeriesTitle());
|
|
}
|
|
|
|
public Movie FindByImdbId(string imdbid)
|
|
{
|
|
return _movieRepository.FindByImdbId(imdbid);
|
|
}
|
|
|
|
public Movie FindByTitleInexact(string title)
|
|
{
|
|
// find any movie clean title within the provided release title
|
|
string cleanTitle = title.CleanSeriesTitle();
|
|
var list = _movieRepository.All().Where(s => cleanTitle.Contains(s.CleanTitle)).ToList();
|
|
if (!list.Any())
|
|
{
|
|
// no movie matched
|
|
return null;
|
|
}
|
|
if (list.Count == 1)
|
|
{
|
|
// return the first movie if there is only one
|
|
return list.Single();
|
|
}
|
|
// build ordered list of movie by position in the search string
|
|
var query =
|
|
list.Select(movie => new
|
|
{
|
|
position = cleanTitle.IndexOf(movie.CleanTitle),
|
|
length = movie.CleanTitle.Length,
|
|
movie = movie
|
|
})
|
|
.Where(s => (s.position>=0))
|
|
.ToList()
|
|
.OrderBy(s => s.position)
|
|
.ThenByDescending(s => s.length)
|
|
.ToList();
|
|
|
|
// get the leftmost movie that is the longest
|
|
// movie are usually the first thing in release title, so we select the leftmost and longest match
|
|
var match = query.First().movie;
|
|
|
|
_logger.Debug("Multiple movie matched {0} from title {1}", match.Title, title);
|
|
foreach (var entry in list)
|
|
{
|
|
_logger.Debug("Multiple movie match candidate: {0} cleantitle: {1}", entry.Title, entry.CleanTitle);
|
|
}
|
|
|
|
return match;
|
|
}
|
|
|
|
public Movie FindByTitle(string title, int year)
|
|
{
|
|
return _movieRepository.FindByTitle(title.CleanSeriesTitle(), year);
|
|
}
|
|
|
|
public void DeleteMovie(int movieId, bool deleteFiles)
|
|
{
|
|
var movie = _movieRepository.Get(movieId);
|
|
_movieRepository.Delete(movieId);
|
|
_eventAggregator.PublishEvent(new MovieDeletedEvent(movie, deleteFiles));
|
|
}
|
|
|
|
public List<Movie> GetAllMovies()
|
|
{
|
|
return _movieRepository.All().ToList();
|
|
}
|
|
|
|
public Movie UpdateMovie(Movie movie)
|
|
{
|
|
var storedMovie = GetMovie(movie.Id);
|
|
|
|
var updatedMovie = _movieRepository.Update(movie);
|
|
_eventAggregator.PublishEvent(new MovieEditedEvent(updatedMovie, storedMovie));
|
|
|
|
return updatedMovie;
|
|
}
|
|
|
|
public List<Movie> UpdateMovie(List<Movie> movie)
|
|
{
|
|
_logger.Debug("Updating {0} movie", movie.Count);
|
|
foreach (var s in movie)
|
|
{
|
|
_logger.Trace("Updating: {0}", s.Title);
|
|
if (!s.RootFolderPath.IsNullOrWhiteSpace())
|
|
{
|
|
var folderName = new DirectoryInfo(s.Path).Name;
|
|
s.Path = Path.Combine(s.RootFolderPath, folderName);
|
|
_logger.Trace("Changing path for {0} to {1}", s.Title, s.Path);
|
|
}
|
|
|
|
else
|
|
{
|
|
_logger.Trace("Not changing path for: {0}", s.Title);
|
|
}
|
|
}
|
|
|
|
_movieRepository.UpdateMany(movie);
|
|
_logger.Debug("{0} movie updated", movie.Count);
|
|
|
|
return movie;
|
|
}
|
|
|
|
public bool MoviePathExists(string folder)
|
|
{
|
|
return _movieRepository.MoviePathExists(folder);
|
|
}
|
|
|
|
public void RemoveAddOptions(Movie movie)
|
|
{
|
|
_movieRepository.SetFields(movie, s => s.AddOptions);
|
|
}
|
|
|
|
public void Handle(MovieFileAddedEvent message)
|
|
{
|
|
_movieRepository.SetFileId(message.MovieFile.Id, message.MovieFile.Movie.Value.Id);
|
|
_logger.Debug("Linking [{0}] > [{1}]", message.MovieFile.RelativePath, message.MovieFile.Movie.Value);
|
|
}
|
|
|
|
public void SetFileId(Movie movie, MovieFile movieFile)
|
|
{
|
|
_movieRepository.SetFileId(movieFile.Id, movie.Id);
|
|
_logger.Debug("Linking [{0}] > [{1}]", movieFile.RelativePath, movie);
|
|
}
|
|
|
|
public void Handle(MovieFileDeletedEvent message)
|
|
{
|
|
|
|
var movie = _movieRepository.GetMoviesByFileId(message.MovieFile.Id).First();
|
|
movie.MovieFileId = 0;
|
|
_logger.Debug("Detaching movie {0} from file.", movie.Id);
|
|
|
|
if (message.Reason != DeleteMediaFileReason.Upgrade && _configService.AutoUnmonitorPreviouslyDownloadedEpisodes)
|
|
{
|
|
movie.Monitored = false;
|
|
}
|
|
|
|
UpdateMovie(movie);
|
|
}
|
|
|
|
public Movie GetMovieByFileId(int fileId)
|
|
{
|
|
return _movieRepository.GetMoviesByFileId(fileId).First();
|
|
}
|
|
|
|
public Movie FindByTitleSlug(string slug)
|
|
{
|
|
return _movieRepository.FindByTitleSlug(slug);
|
|
}
|
|
|
|
public List<Movie> GetMoviesBetweenDates(DateTime start, DateTime end, bool includeUnmonitored)
|
|
{
|
|
var episodes = _movieRepository.MoviesBetweenDates(start.ToUniversalTime(), end.ToUniversalTime(), includeUnmonitored);
|
|
|
|
return episodes;
|
|
}
|
|
|
|
public List<Movie> MoviesWithFiles(int movieId)
|
|
{
|
|
return _movieRepository.MoviesWithFiles(movieId);
|
|
}
|
|
|
|
public PagingSpec<Movie> MoviesWithoutFiles(PagingSpec<Movie> pagingSpec)
|
|
{
|
|
var movieResult = _movieRepository.MoviesWithoutFiles(pagingSpec);
|
|
|
|
return movieResult;
|
|
}
|
|
|
|
public bool MovieExists(Movie movie)
|
|
{
|
|
Movie result = null;
|
|
|
|
if (movie.TmdbId != 0)
|
|
{
|
|
result = _movieRepository.FindByTmdbId(movie.TmdbId);
|
|
if (result != null)
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
if (movie.ImdbId.IsNotNullOrWhiteSpace())
|
|
{
|
|
result = _movieRepository.FindByImdbId(movie.ImdbId);
|
|
if (result != null)
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
if (movie.Year > 1850)
|
|
{
|
|
result = _movieRepository.FindByTitle(movie.Title.CleanSeriesTitle(), movie.Year);
|
|
if (result != null)
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
}
|
|
}
|