SubtitleDownloader.cs 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. #pragma warning disable CS1591
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using System.Threading;
  6. using System.Threading.Tasks;
  7. using MediaBrowser.Common.Extensions;
  8. using MediaBrowser.Controller.Entities;
  9. using MediaBrowser.Controller.Entities.Movies;
  10. using MediaBrowser.Controller.Entities.TV;
  11. using MediaBrowser.Controller.Providers;
  12. using MediaBrowser.Controller.Subtitles;
  13. using MediaBrowser.Model.Entities;
  14. using Microsoft.Extensions.Logging;
  15. namespace MediaBrowser.Providers.MediaInfo
  16. {
  17. public class SubtitleDownloader
  18. {
  19. private readonly ILogger _logger;
  20. private readonly ISubtitleManager _subtitleManager;
  21. public SubtitleDownloader(ILogger logger, ISubtitleManager subtitleManager)
  22. {
  23. _logger = logger;
  24. _subtitleManager = subtitleManager;
  25. }
  26. public async Task<List<string>> DownloadSubtitles(
  27. Video video,
  28. List<MediaStream> mediaStreams,
  29. bool skipIfEmbeddedSubtitlesPresent,
  30. bool skipIfAudioTrackMatches,
  31. bool requirePerfectMatch,
  32. IEnumerable<string> languages,
  33. string[] disabledSubtitleFetchers,
  34. string[] subtitleFetcherOrder,
  35. CancellationToken cancellationToken)
  36. {
  37. var downloadedLanguages = new List<string>();
  38. foreach (var lang in languages)
  39. {
  40. var downloaded = await DownloadSubtitles(video, mediaStreams, skipIfEmbeddedSubtitlesPresent,
  41. skipIfAudioTrackMatches, requirePerfectMatch, lang, disabledSubtitleFetchers, subtitleFetcherOrder, cancellationToken).ConfigureAwait(false);
  42. if (downloaded)
  43. {
  44. downloadedLanguages.Add(lang);
  45. }
  46. }
  47. return downloadedLanguages;
  48. }
  49. public Task<bool> DownloadSubtitles(Video video,
  50. List<MediaStream> mediaStreams,
  51. bool skipIfEmbeddedSubtitlesPresent,
  52. bool skipIfAudioTrackMatches,
  53. bool requirePerfectMatch,
  54. string lang,
  55. string[] disabledSubtitleFetchers,
  56. string[] subtitleFetcherOrder,
  57. CancellationToken cancellationToken)
  58. {
  59. if (video.VideoType != VideoType.VideoFile)
  60. {
  61. return Task.FromResult(false);
  62. }
  63. if (!video.IsCompleteMedia)
  64. {
  65. return Task.FromResult(false);
  66. }
  67. VideoContentType mediaType;
  68. if (video is Episode)
  69. {
  70. mediaType = VideoContentType.Episode;
  71. }
  72. else if (video is Movie)
  73. {
  74. mediaType = VideoContentType.Movie;
  75. }
  76. else
  77. {
  78. // These are the only supported types
  79. return Task.FromResult(false);
  80. }
  81. return DownloadSubtitles(video, mediaStreams, skipIfEmbeddedSubtitlesPresent, skipIfAudioTrackMatches,
  82. requirePerfectMatch, lang, disabledSubtitleFetchers, subtitleFetcherOrder, mediaType, cancellationToken);
  83. }
  84. private async Task<bool> DownloadSubtitles(Video video,
  85. List<MediaStream> mediaStreams,
  86. bool skipIfEmbeddedSubtitlesPresent,
  87. bool skipIfAudioTrackMatches,
  88. bool requirePerfectMatch,
  89. string language,
  90. string[] disabledSubtitleFetchers,
  91. string[] subtitleFetcherOrder,
  92. VideoContentType mediaType,
  93. CancellationToken cancellationToken)
  94. {
  95. // There's already subtitles for this language
  96. if (mediaStreams.Any(i => i.Type == MediaStreamType.Subtitle && i.IsTextSubtitleStream && string.Equals(i.Language, language, StringComparison.OrdinalIgnoreCase)))
  97. {
  98. return false;
  99. }
  100. var audioStreams = mediaStreams.Where(i => i.Type == MediaStreamType.Audio).ToList();
  101. var defaultAudioStreams = audioStreams.Where(i => i.IsDefault).ToList();
  102. // If none are marked as default, just take a guess
  103. if (defaultAudioStreams.Count == 0)
  104. {
  105. defaultAudioStreams = audioStreams.Take(1).ToList();
  106. }
  107. // There's already a default audio stream for this language
  108. if (skipIfAudioTrackMatches &&
  109. defaultAudioStreams.Any(i => string.Equals(i.Language, language, StringComparison.OrdinalIgnoreCase)))
  110. {
  111. return false;
  112. }
  113. // There's an internal subtitle stream for this language
  114. if (skipIfEmbeddedSubtitlesPresent &&
  115. mediaStreams.Any(i => i.Type == MediaStreamType.Subtitle && !i.IsExternal && string.Equals(i.Language, language, StringComparison.OrdinalIgnoreCase)))
  116. {
  117. return false;
  118. }
  119. var request = new SubtitleSearchRequest
  120. {
  121. ContentType = mediaType,
  122. IndexNumber = video.IndexNumber,
  123. Language = language,
  124. MediaPath = video.Path,
  125. Name = video.Name,
  126. ParentIndexNumber = video.ParentIndexNumber,
  127. ProductionYear = video.ProductionYear,
  128. ProviderIds = video.ProviderIds,
  129. // Stop as soon as we find something
  130. SearchAllProviders = false,
  131. IsPerfectMatch = requirePerfectMatch,
  132. DisabledSubtitleFetchers = disabledSubtitleFetchers,
  133. SubtitleFetcherOrder = subtitleFetcherOrder
  134. };
  135. var episode = video as Episode;
  136. if (episode != null)
  137. {
  138. request.IndexNumberEnd = episode.IndexNumberEnd;
  139. request.SeriesName = episode.SeriesName;
  140. }
  141. try
  142. {
  143. var searchResults = await _subtitleManager.SearchSubtitles(request, cancellationToken).ConfigureAwait(false);
  144. var result = searchResults.FirstOrDefault();
  145. if (result != null)
  146. {
  147. await _subtitleManager.DownloadSubtitles(video, result.Id, cancellationToken).ConfigureAwait(false);
  148. return true;
  149. }
  150. }
  151. catch (RateLimitExceededException)
  152. {
  153. }
  154. catch (Exception ex)
  155. {
  156. _logger.LogError(ex, "Error downloading subtitles");
  157. }
  158. return false;
  159. }
  160. }
  161. }