jellyfin/MediaBrowser.Providers/TV/SeriesMetadataService.cs

288 lines
11 KiB
C#
Raw Normal View History

#pragma warning disable CS1591
using System.Collections.Generic;
using System.Globalization;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
using MediaBrowser.Controller.Configuration;
using MediaBrowser.Controller.Dto;
2014-02-03 13:51:28 -07:00
using MediaBrowser.Controller.Entities.TV;
using MediaBrowser.Controller.Library;
using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.Globalization;
using MediaBrowser.Model.IO;
using MediaBrowser.Providers.Manager;
using Microsoft.Extensions.Logging;
2014-02-03 13:51:28 -07:00
namespace MediaBrowser.Providers.TV
{
2014-02-09 00:27:44 -07:00
public class SeriesMetadataService : MetadataService<Series, SeriesInfo>
2014-02-03 13:51:28 -07:00
{
private readonly ILocalizationManager _localizationManager;
2019-02-15 12:11:27 -07:00
public SeriesMetadataService(
IServerConfigurationManager serverConfigurationManager,
ILogger<SeriesMetadataService> logger,
2019-02-15 12:11:27 -07:00
IProviderManager providerManager,
IFileSystem fileSystem,
ILibraryManager libraryManager,
ILocalizationManager localizationManager)
2019-09-10 13:37:53 -07:00
: base(serverConfigurationManager, logger, providerManager, fileSystem, libraryManager)
2014-02-03 13:51:28 -07:00
{
_localizationManager = localizationManager;
}
/// <inheritdoc />
protected override async Task AfterMetadataRefresh(Series item, MetadataRefreshOptions refreshOptions, CancellationToken cancellationToken)
{
await base.AfterMetadataRefresh(item, refreshOptions, cancellationToken).ConfigureAwait(false);
RemoveObsoleteEpisodes(item);
RemoveObsoleteSeasons(item);
2022-08-02 08:46:38 -07:00
await UpdateAndCreateSeasonsAsync(item, cancellationToken).ConfigureAwait(false);
2015-06-28 18:10:45 -07:00
}
2019-09-10 13:37:53 -07:00
/// <inheritdoc />
2015-06-28 18:10:45 -07:00
protected override bool IsFullLocalMetadata(Series item)
{
if (string.IsNullOrWhiteSpace(item.Overview))
{
2015-06-28 18:10:45 -07:00
return false;
}
2020-06-15 14:43:52 -07:00
2015-06-28 18:10:45 -07:00
if (!item.ProductionYear.HasValue)
{
2015-06-28 18:10:45 -07:00
return false;
}
2020-06-15 14:43:52 -07:00
2015-06-28 18:10:45 -07:00
return base.IsFullLocalMetadata(item);
}
2019-09-10 13:37:53 -07:00
/// <inheritdoc />
2020-06-09 15:12:53 -07:00
protected override void MergeData(MetadataResult<Series> source, MetadataResult<Series> target, MetadataField[] lockedFields, bool replaceData, bool mergeMetadataSettings)
2015-06-28 18:10:45 -07:00
{
base.MergeData(source, target, lockedFields, replaceData, mergeMetadataSettings);
2015-06-28 18:10:45 -07:00
var sourceItem = source.Item;
var targetItem = target.Item;
2022-08-02 08:46:38 -07:00
var sourceSeasonNames = sourceItem.SeasonNames;
var targetSeasonNames = targetItem.SeasonNames;
if (replaceData || targetSeasonNames.Count == 0)
{
targetItem.SeasonNames = sourceSeasonNames;
}
else if (targetSeasonNames.Count != sourceSeasonNames.Count || !sourceSeasonNames.Keys.All(targetSeasonNames.ContainsKey))
{
foreach (var (number, name) in sourceSeasonNames)
2022-08-02 08:46:38 -07:00
{
targetSeasonNames.TryAdd(number, name);
2022-08-02 08:46:38 -07:00
}
}
2017-08-13 13:15:07 -07:00
if (replaceData || string.IsNullOrEmpty(targetItem.AirTime))
{
targetItem.AirTime = sourceItem.AirTime;
}
2015-06-28 18:10:45 -07:00
if (replaceData || !targetItem.Status.HasValue)
2015-01-28 14:29:02 -07:00
{
2015-06-28 18:10:45 -07:00
targetItem.Status = sourceItem.Status;
2015-01-28 14:29:02 -07:00
}
2017-08-13 13:15:07 -07:00
2022-12-05 07:00:20 -07:00
if (replaceData || targetItem.AirDays is null || targetItem.AirDays.Length == 0)
2017-08-13 13:15:07 -07:00
{
targetItem.AirDays = sourceItem.AirDays;
}
2015-02-25 11:11:49 -07:00
}
private void RemoveObsoleteSeasons(Series series)
{
2022-08-02 08:46:38 -07:00
// TODO Legacy. It's not really "physical" seasons as any virtual seasons are always converted to non-virtual in UpdateAndCreateSeasonsAsync.
var physicalSeasonNumbers = new HashSet<int>();
var virtualSeasons = new List<Season>();
foreach (var existingSeason in series.Children.OfType<Season>())
{
if (existingSeason.LocationType != LocationType.Virtual && existingSeason.IndexNumber.HasValue)
{
physicalSeasonNumbers.Add(existingSeason.IndexNumber.Value);
}
else if (existingSeason.LocationType == LocationType.Virtual)
{
virtualSeasons.Add(existingSeason);
}
}
foreach (var virtualSeason in virtualSeasons)
{
var seasonNumber = virtualSeason.IndexNumber;
// If there's a physical season with the same number or no episodes in the season, delete it
if ((seasonNumber.HasValue && physicalSeasonNumbers.Contains(seasonNumber.Value))
|| !virtualSeason.GetEpisodes().Any())
{
Logger.LogInformation("Removing virtual season {SeasonNumber} in series {SeriesName}", virtualSeason.IndexNumber, series.Name);
LibraryManager.DeleteItem(
virtualSeason,
new DeleteOptions
{
DeleteFileLocation = true
},
false);
}
}
}
private void RemoveObsoleteEpisodes(Series series)
{
var episodes = series.GetEpisodes(null, new DtoOptions()).OfType<Episode>().ToList();
var numberOfEpisodes = episodes.Count;
// TODO: O(n^2), but can it be done faster without overcomplicating it?
for (var i = 0; i < numberOfEpisodes; i++)
{
var currentEpisode = episodes[i];
// The outer loop only examines virtual episodes
if (!currentEpisode.IsVirtualItem)
{
continue;
}
// Virtual episodes without an episode number are practically orphaned and should be deleted
if (!currentEpisode.IndexNumber.HasValue)
{
DeleteEpisode(currentEpisode);
continue;
}
for (var j = i + 1; j < numberOfEpisodes; j++)
{
var comparisonEpisode = episodes[j];
// The inner loop is only for "physical" episodes
if (comparisonEpisode.IsVirtualItem
|| currentEpisode.ParentIndexNumber != comparisonEpisode.ParentIndexNumber
|| !comparisonEpisode.ContainsEpisodeNumber(currentEpisode.IndexNumber.Value))
{
continue;
}
DeleteEpisode(currentEpisode);
break;
}
}
}
private void DeleteEpisode(Episode episode)
{
Logger.LogInformation(
"Removing virtual episode S{SeasonNumber}E{EpisodeNumber} in series {SeriesName}",
episode.ParentIndexNumber,
episode.IndexNumber,
episode.SeriesName);
LibraryManager.DeleteItem(
episode,
new DeleteOptions
{
DeleteFileLocation = true
},
false);
}
/// <summary>
2022-08-02 08:46:38 -07:00
/// Creates seasons for all episodes if they don't exist.
/// If no season number can be determined, a dummy season will be created.
2022-08-02 08:46:38 -07:00
/// Updates seasons names.
/// </summary>
/// <param name="series">The series.</param>
/// <param name="cancellationToken">The cancellation token.</param>
/// <returns>The async task.</returns>
2022-08-02 08:46:38 -07:00
private async Task UpdateAndCreateSeasonsAsync(Series series, CancellationToken cancellationToken)
{
2022-08-02 08:46:38 -07:00
var seasonNames = series.SeasonNames;
2021-11-28 11:56:31 -07:00
var seriesChildren = series.GetRecursiveChildren(i => i is Episode || i is Season);
2022-08-02 08:46:38 -07:00
var seasons = seriesChildren.OfType<Season>().ToList();
var uniqueSeasonNumbers = seriesChildren
2021-11-28 11:56:31 -07:00
.OfType<Episode>()
2022-08-02 08:46:38 -07:00
.Select(e => e.ParentIndexNumber >= 0 ? e.ParentIndexNumber : null)
.Distinct();
// Loop through the unique season numbers
2022-08-02 08:46:38 -07:00
foreach (var seasonNumber in uniqueSeasonNumbers)
{
// Null season numbers will have a 'dummy' season created because seasons are always required.
var existingSeason = seasons.FirstOrDefault(i => i.IndexNumber == seasonNumber);
2022-08-02 08:46:38 -07:00
string? seasonName = null;
if (seasonNumber.HasValue && seasonNames.TryGetValue(seasonNumber.Value, out var tmp))
2022-08-02 08:46:38 -07:00
{
seasonName = tmp;
2022-08-02 08:46:38 -07:00
}
2022-12-05 07:00:20 -07:00
if (existingSeason is null)
{
2022-08-02 08:46:38 -07:00
var season = await CreateSeasonAsync(series, seasonName, seasonNumber, cancellationToken).ConfigureAwait(false);
series.AddChild(season);
}
2022-08-02 08:46:38 -07:00
else
{
2022-08-02 08:46:38 -07:00
existingSeason.Name = GetValidSeasonNameForSeries(series, seasonName, seasonNumber);
await existingSeason.UpdateToRepositoryAsync(ItemUpdateType.MetadataEdit, cancellationToken).ConfigureAwait(false);
}
}
}
/// <summary>
/// Creates a new season, adds it to the database by linking it to the [series] and refreshes the metadata.
/// </summary>
/// <param name="series">The series.</param>
2022-08-02 08:46:38 -07:00
/// <param name="seasonName">The season name.</param>
/// <param name="seasonNumber">The season number.</param>
/// <param name="cancellationToken">The cancellation token.</param>
/// <returns>The newly created season.</returns>
private async Task<Season> CreateSeasonAsync(
Series series,
2022-08-02 08:46:38 -07:00
string? seasonName,
int? seasonNumber,
CancellationToken cancellationToken)
{
2022-08-02 08:46:38 -07:00
seasonName = GetValidSeasonNameForSeries(series, seasonName, seasonNumber);
Logger.LogInformation("Creating Season {SeasonName} entry for {SeriesName}", seasonName, series.Name);
var season = new Season
{
Name = seasonName,
IndexNumber = seasonNumber,
Id = LibraryManager.GetNewItemId(
series.Id + (seasonNumber ?? -1).ToString(CultureInfo.InvariantCulture) + seasonName,
typeof(Season)),
IsVirtualItem = false,
SeriesId = series.Id,
SeriesName = series.Name
};
series.AddChild(season);
await season.RefreshMetadata(new MetadataRefreshOptions(new DirectoryService(FileSystem)), cancellationToken).ConfigureAwait(false);
return season;
}
2022-08-02 08:46:38 -07:00
private string GetValidSeasonNameForSeries(Series series, string? seasonName, int? seasonNumber)
{
if (string.IsNullOrEmpty(seasonName))
{
seasonName = seasonNumber switch
{
null => _localizationManager.GetLocalizedString("NameSeasonUnknown"),
0 => LibraryManager.GetLibraryOptions(series).SeasonZeroDisplayName,
_ => string.Format(CultureInfo.InvariantCulture, _localizationManager.GetLocalizedString("NameSeasonNumber"), seasonNumber.Value)
};
}
return seasonName;
}
2014-02-03 13:51:28 -07:00
}
}