mirror of
https://github.com/Sonarr/Sonarr.git
synced 2024-12-23 02:05:27 +02:00
193672b652
New: Anime support New: pull alternate names from thexem.de New: Search using all alternate names (if rage ID is unavailable) New: Show scene mapping information when hovering over episode number New: Full season searching for anime (searches for each episode) New: animezb.com anime indexer New: Treat BD as bluray Fixed: Parsing of 2 digit absolute episode numbers Fixed: Loading series details page for series that start with period Fixed: Return 0 results when manual search fails, instead of an error Fixed: animezb URL
107 lines
3.4 KiB
C#
107 lines
3.4 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using FluentValidation.Results;
|
|
using NzbDrone.Core.Parser.Model;
|
|
using NzbDrone.Core.Tv;
|
|
|
|
namespace NzbDrone.Core.Organizer
|
|
{
|
|
public interface IFilenameValidationService
|
|
{
|
|
ValidationFailure ValidateStandardFilename(SampleResult sampleResult);
|
|
ValidationFailure ValidateDailyFilename(SampleResult sampleResult);
|
|
ValidationFailure ValidateAnimeFilename(SampleResult sampleResult);
|
|
}
|
|
|
|
public class FilenameValidationService : IFilenameValidationService
|
|
{
|
|
private const string ERROR_MESSAGE = "Produces invalid file names";
|
|
|
|
public ValidationFailure ValidateStandardFilename(SampleResult sampleResult)
|
|
{
|
|
var validationFailure = new ValidationFailure("StandardEpisodeFormat", ERROR_MESSAGE);
|
|
var parsedEpisodeInfo = Parser.Parser.ParseTitle(sampleResult.Filename);
|
|
|
|
if (parsedEpisodeInfo == null)
|
|
{
|
|
return validationFailure;
|
|
}
|
|
|
|
if (!ValidateSeasonAndEpisodeNumbers(sampleResult.Episodes, parsedEpisodeInfo))
|
|
{
|
|
return validationFailure;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
public ValidationFailure ValidateDailyFilename(SampleResult sampleResult)
|
|
{
|
|
var validationFailure = new ValidationFailure("DailyEpisodeFormat", ERROR_MESSAGE);
|
|
var parsedEpisodeInfo = Parser.Parser.ParseTitle(sampleResult.Filename);
|
|
|
|
if (parsedEpisodeInfo == null)
|
|
{
|
|
return validationFailure;
|
|
}
|
|
|
|
if (parsedEpisodeInfo.IsDaily())
|
|
{
|
|
if (!parsedEpisodeInfo.AirDate.Equals(sampleResult.Episodes.Single().AirDate))
|
|
{
|
|
return validationFailure;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
if (!ValidateSeasonAndEpisodeNumbers(sampleResult.Episodes, parsedEpisodeInfo))
|
|
{
|
|
return validationFailure;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
public ValidationFailure ValidateAnimeFilename(SampleResult sampleResult)
|
|
{
|
|
var validationFailure = new ValidationFailure("AnimeEpisodeFormat", ERROR_MESSAGE);
|
|
var parsedEpisodeInfo = Parser.Parser.ParseTitle(sampleResult.Filename);
|
|
|
|
if (parsedEpisodeInfo == null)
|
|
{
|
|
return validationFailure;
|
|
}
|
|
|
|
if (parsedEpisodeInfo.AbsoluteEpisodeNumbers.Any())
|
|
{
|
|
if (!parsedEpisodeInfo.AbsoluteEpisodeNumbers.First().Equals(sampleResult.Episodes.First().AbsoluteEpisodeNumber))
|
|
{
|
|
return validationFailure;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
if (!ValidateSeasonAndEpisodeNumbers(sampleResult.Episodes, parsedEpisodeInfo))
|
|
{
|
|
return validationFailure;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
private bool ValidateSeasonAndEpisodeNumbers(List<Episode> episodes, ParsedEpisodeInfo parsedEpisodeInfo)
|
|
{
|
|
if (parsedEpisodeInfo.SeasonNumber != episodes.First().SeasonNumber ||
|
|
!parsedEpisodeInfo.EpisodeNumbers.OrderBy(e => e).SequenceEqual(episodes.Select(e => e.EpisodeNumber).OrderBy(e => e)))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
}
|
|
}
|