MissingEpisodeProvider.cs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388
  1. #pragma warning disable CS1591
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Globalization;
  5. using System.Linq;
  6. using System.Threading;
  7. using System.Threading.Tasks;
  8. using MediaBrowser.Controller.Configuration;
  9. using MediaBrowser.Controller.Entities;
  10. using MediaBrowser.Controller.Entities.TV;
  11. using MediaBrowser.Controller.Library;
  12. using MediaBrowser.Controller.Providers;
  13. using MediaBrowser.Model.Entities;
  14. using MediaBrowser.Model.Globalization;
  15. using MediaBrowser.Model.IO;
  16. using MediaBrowser.Providers.Plugins.TheTvdb;
  17. using Microsoft.Extensions.Logging;
  18. namespace MediaBrowser.Providers.TV
  19. {
  20. public class MissingEpisodeProvider
  21. {
  22. private const double UnairedEpisodeThresholdDays = 2;
  23. private readonly IServerConfigurationManager _config;
  24. private readonly ILogger _logger;
  25. private readonly ILibraryManager _libraryManager;
  26. private readonly ILocalizationManager _localization;
  27. private readonly IFileSystem _fileSystem;
  28. private readonly TvdbClientManager _tvdbClientManager;
  29. public MissingEpisodeProvider(
  30. ILogger logger,
  31. IServerConfigurationManager config,
  32. ILibraryManager libraryManager,
  33. ILocalizationManager localization,
  34. IFileSystem fileSystem,
  35. TvdbClientManager tvdbClientManager)
  36. {
  37. _logger = logger;
  38. _config = config;
  39. _libraryManager = libraryManager;
  40. _localization = localization;
  41. _fileSystem = fileSystem;
  42. _tvdbClientManager = tvdbClientManager;
  43. }
  44. public async Task<bool> Run(Series series, bool addNewItems, CancellationToken cancellationToken)
  45. {
  46. var tvdbId = series.GetProviderId(MetadataProvider.Tvdb);
  47. if (string.IsNullOrEmpty(tvdbId))
  48. {
  49. return false;
  50. }
  51. var episodes = await _tvdbClientManager.GetAllEpisodesAsync(Convert.ToInt32(tvdbId), series.GetPreferredMetadataLanguage(), cancellationToken);
  52. var episodeLookup = episodes
  53. .Select(i =>
  54. {
  55. DateTime.TryParse(i.FirstAired, out var firstAired);
  56. var seasonNumber = i.AiredSeason.GetValueOrDefault(-1);
  57. var episodeNumber = i.AiredEpisodeNumber.GetValueOrDefault(-1);
  58. return (seasonNumber, episodeNumber, firstAired);
  59. })
  60. .Where(i => i.seasonNumber != -1 && i.episodeNumber != -1)
  61. .OrderBy(i => i.seasonNumber)
  62. .ThenBy(i => i.episodeNumber)
  63. .ToList();
  64. var allRecursiveChildren = series.GetRecursiveChildren();
  65. var hasBadData = HasInvalidContent(allRecursiveChildren);
  66. // Be conservative here to avoid creating missing episodes for ones they already have
  67. var addMissingEpisodes = !hasBadData && _libraryManager.GetLibraryOptions(series).ImportMissingEpisodes;
  68. var anySeasonsRemoved = RemoveObsoleteOrMissingSeasons(allRecursiveChildren, episodeLookup);
  69. if (anySeasonsRemoved)
  70. {
  71. // refresh this
  72. allRecursiveChildren = series.GetRecursiveChildren();
  73. }
  74. var anyEpisodesRemoved = RemoveObsoleteOrMissingEpisodes(allRecursiveChildren, episodeLookup, addMissingEpisodes);
  75. if (anyEpisodesRemoved)
  76. {
  77. // refresh this
  78. allRecursiveChildren = series.GetRecursiveChildren();
  79. }
  80. var hasNewEpisodes = false;
  81. if (addNewItems && series.IsMetadataFetcherEnabled(_libraryManager.GetLibraryOptions(series), TvdbSeriesProvider.Current.Name))
  82. {
  83. hasNewEpisodes = await AddMissingEpisodes(series, allRecursiveChildren, addMissingEpisodes, episodeLookup, cancellationToken)
  84. .ConfigureAwait(false);
  85. }
  86. if (hasNewEpisodes || anySeasonsRemoved || anyEpisodesRemoved)
  87. {
  88. return true;
  89. }
  90. return false;
  91. }
  92. /// <summary>
  93. /// Returns true if a series has any seasons or episodes without season or episode numbers
  94. /// If this data is missing no virtual items will be added in order to prevent possible duplicates.
  95. /// </summary>
  96. private bool HasInvalidContent(IList<BaseItem> allItems)
  97. {
  98. return allItems.OfType<Season>().Any(i => !i.IndexNumber.HasValue) ||
  99. allItems.OfType<Episode>().Any(i =>
  100. {
  101. if (!i.ParentIndexNumber.HasValue)
  102. {
  103. return true;
  104. }
  105. // You could have episodes under season 0 with no number
  106. return false;
  107. });
  108. }
  109. private async Task<bool> AddMissingEpisodes(
  110. Series series,
  111. IEnumerable<BaseItem> allItems,
  112. bool addMissingEpisodes,
  113. IReadOnlyCollection<(int seasonNumber, int episodenumber, DateTime firstAired)> episodeLookup,
  114. CancellationToken cancellationToken)
  115. {
  116. var existingEpisodes = allItems.OfType<Episode>().ToList();
  117. var seasonCounts = episodeLookup.GroupBy(e => e.seasonNumber).ToDictionary(g => g.Key, g => g.Count());
  118. var hasChanges = false;
  119. foreach (var tuple in episodeLookup)
  120. {
  121. if (tuple.seasonNumber <= 0 || tuple.episodenumber <= 0)
  122. {
  123. // Ignore episode/season zeros
  124. continue;
  125. }
  126. var existingEpisode = GetExistingEpisode(existingEpisodes, seasonCounts, tuple);
  127. if (existingEpisode != null)
  128. {
  129. continue;
  130. }
  131. var airDate = tuple.firstAired;
  132. var now = DateTime.UtcNow.AddDays(-UnairedEpisodeThresholdDays);
  133. if (airDate < now && addMissingEpisodes || airDate > now)
  134. {
  135. // tvdb has a lot of nearly blank episodes
  136. _logger.LogInformation("Creating virtual missing/unaired episode {0} {1}x{2}", series.Name, tuple.seasonNumber, tuple.episodenumber);
  137. await AddEpisode(series, tuple.seasonNumber, tuple.episodenumber, cancellationToken).ConfigureAwait(false);
  138. hasChanges = true;
  139. }
  140. }
  141. return hasChanges;
  142. }
  143. /// <summary>
  144. /// Removes the virtual entry after a corresponding physical version has been added.
  145. /// </summary>
  146. private bool RemoveObsoleteOrMissingEpisodes(
  147. IEnumerable<BaseItem> allRecursiveChildren,
  148. IEnumerable<(int seasonNumber, int episodeNumber, DateTime firstAired)> episodeLookup,
  149. bool allowMissingEpisodes)
  150. {
  151. var existingEpisodes = allRecursiveChildren.OfType<Episode>();
  152. var physicalEpisodes = new List<Episode>();
  153. var virtualEpisodes = new List<Episode>();
  154. foreach (var episode in existingEpisodes)
  155. {
  156. if (episode.LocationType == LocationType.Virtual)
  157. {
  158. virtualEpisodes.Add(episode);
  159. }
  160. else
  161. {
  162. physicalEpisodes.Add(episode);
  163. }
  164. }
  165. var episodesToRemove = virtualEpisodes
  166. .Where(i =>
  167. {
  168. if (!i.IndexNumber.HasValue || !i.ParentIndexNumber.HasValue)
  169. {
  170. return true;
  171. }
  172. var seasonNumber = i.ParentIndexNumber.Value;
  173. var episodeNumber = i.IndexNumber.Value;
  174. // If there's a physical episode with the same season and episode number, delete it
  175. if (physicalEpisodes.Any(p =>
  176. p.ParentIndexNumber.HasValue && p.ParentIndexNumber.Value == seasonNumber &&
  177. p.ContainsEpisodeNumber(episodeNumber)))
  178. {
  179. return true;
  180. }
  181. // If the episode no longer exists in the remote lookup, delete it
  182. if (!episodeLookup.Any(e => e.seasonNumber == seasonNumber && e.episodeNumber == episodeNumber))
  183. {
  184. return true;
  185. }
  186. // If it's missing, but not unaired, remove it
  187. return !allowMissingEpisodes && i.IsMissingEpisode &&
  188. (!i.PremiereDate.HasValue ||
  189. i.PremiereDate.Value.ToLocalTime().Date.AddDays(UnairedEpisodeThresholdDays) <
  190. DateTime.Now.Date);
  191. });
  192. var hasChanges = false;
  193. foreach (var episodeToRemove in episodesToRemove)
  194. {
  195. _libraryManager.DeleteItem(episodeToRemove, new DeleteOptions
  196. {
  197. DeleteFileLocation = true
  198. }, false);
  199. hasChanges = true;
  200. }
  201. return hasChanges;
  202. }
  203. /// <summary>
  204. /// Removes the obsolete or missing seasons.
  205. /// </summary>
  206. /// <param name="allRecursiveChildren"></param>
  207. /// <param name="episodeLookup">The episode lookup.</param>
  208. /// <returns><see cref="bool" />.</returns>
  209. private bool RemoveObsoleteOrMissingSeasons(
  210. IList<BaseItem> allRecursiveChildren,
  211. IEnumerable<(int seasonNumber, int episodeNumber, DateTime firstAired)> episodeLookup)
  212. {
  213. var existingSeasons = allRecursiveChildren.OfType<Season>().ToList();
  214. var physicalSeasons = new List<Season>();
  215. var virtualSeasons = new List<Season>();
  216. foreach (var season in existingSeasons)
  217. {
  218. if (season.LocationType == LocationType.Virtual)
  219. {
  220. virtualSeasons.Add(season);
  221. }
  222. else
  223. {
  224. physicalSeasons.Add(season);
  225. }
  226. }
  227. var allEpisodes = allRecursiveChildren.OfType<Episode>().ToList();
  228. var seasonsToRemove = virtualSeasons
  229. .Where(i =>
  230. {
  231. if (i.IndexNumber.HasValue)
  232. {
  233. var seasonNumber = i.IndexNumber.Value;
  234. // If there's a physical season with the same number, delete it
  235. if (physicalSeasons.Any(p => p.IndexNumber.HasValue && p.IndexNumber.Value == seasonNumber && string.Equals(p.Series.PresentationUniqueKey, i.Series.PresentationUniqueKey, StringComparison.Ordinal)))
  236. {
  237. return true;
  238. }
  239. // If the season no longer exists in the remote lookup, delete it, but only if an existing episode doesn't require it
  240. return episodeLookup.All(e => e.seasonNumber != seasonNumber) && allEpisodes.All(s => s.ParentIndexNumber != seasonNumber || s.IsInSeasonFolder);
  241. }
  242. // Season does not have a number
  243. // Remove if there are no episodes directly in series without a season number
  244. return allEpisodes.All(s => s.ParentIndexNumber.HasValue || s.IsInSeasonFolder);
  245. });
  246. var hasChanges = false;
  247. foreach (var seasonToRemove in seasonsToRemove)
  248. {
  249. _libraryManager.DeleteItem(seasonToRemove, new DeleteOptions
  250. {
  251. DeleteFileLocation = true
  252. }, false);
  253. hasChanges = true;
  254. }
  255. return hasChanges;
  256. }
  257. /// <summary>
  258. /// Adds the episode.
  259. /// </summary>
  260. /// <param name="series">The series.</param>
  261. /// <param name="seasonNumber">The season number.</param>
  262. /// <param name="episodeNumber">The episode number.</param>
  263. /// <param name="cancellationToken">The cancellation token.</param>
  264. /// <returns>Task.</returns>
  265. private async Task AddEpisode(Series series, int seasonNumber, int episodeNumber, CancellationToken cancellationToken)
  266. {
  267. var season = series.Children.OfType<Season>()
  268. .FirstOrDefault(i => i.IndexNumber.HasValue && i.IndexNumber.Value == seasonNumber);
  269. if (season == null)
  270. {
  271. var provider = new DummySeasonProvider(_logger, _localization, _libraryManager, _fileSystem);
  272. season = await provider.AddSeason(series, seasonNumber, true, cancellationToken).ConfigureAwait(false);
  273. }
  274. var name = "Episode " + episodeNumber.ToString(CultureInfo.InvariantCulture);
  275. var episode = new Episode
  276. {
  277. Name = name,
  278. IndexNumber = episodeNumber,
  279. ParentIndexNumber = seasonNumber,
  280. Id = _libraryManager.GetNewItemId(
  281. series.Id + seasonNumber.ToString(CultureInfo.InvariantCulture) + name,
  282. typeof(Episode)),
  283. IsVirtualItem = true,
  284. SeasonId = season?.Id ?? Guid.Empty,
  285. SeriesId = series.Id
  286. };
  287. season.AddChild(episode, cancellationToken);
  288. await episode.RefreshMetadata(new MetadataRefreshOptions(new DirectoryService(_fileSystem)), cancellationToken).ConfigureAwait(false);
  289. }
  290. /// <summary>
  291. /// Gets the existing episode.
  292. /// </summary>
  293. /// <param name="existingEpisodes">The existing episodes.</param>
  294. /// <param name="seasonCounts"></param>
  295. /// <param name="episodeTuple"></param>
  296. /// <returns>Episode.</returns>
  297. private Episode GetExistingEpisode(IEnumerable<Episode> existingEpisodes, IReadOnlyDictionary<int, int> seasonCounts, (int seasonNumber, int episodeNumber, DateTime firstAired) episodeTuple)
  298. {
  299. var seasonNumber = episodeTuple.seasonNumber;
  300. var episodeNumber = episodeTuple.episodeNumber;
  301. while (true)
  302. {
  303. var episode = GetExistingEpisode(existingEpisodes, seasonNumber, episodeNumber);
  304. if (episode != null)
  305. {
  306. return episode;
  307. }
  308. seasonNumber--;
  309. if (seasonCounts.ContainsKey(seasonNumber))
  310. {
  311. episodeNumber += seasonCounts[seasonNumber];
  312. }
  313. else
  314. {
  315. break;
  316. }
  317. }
  318. return null;
  319. }
  320. private Episode GetExistingEpisode(IEnumerable<Episode> existingEpisodes, int season, int episode)
  321. => existingEpisodes.FirstOrDefault(i => i.ParentIndexNumber == season && i.ContainsEpisodeNumber(episode));
  322. }
  323. }