using MediaBrowser.Common.Configuration; using MediaBrowser.Controller.Configuration; using MediaBrowser.Controller.Entities; using MediaBrowser.Controller.Entities.Movies; using MediaBrowser.Controller.Entities.TV; using MediaBrowser.Controller.Library; using MediaBrowser.Controller.Subtitles; using MediaBrowser.Model.Entities; using MediaBrowser.Model.Logging; using MediaBrowser.Model.Providers; using System; using System.Collections.Generic; using System.Linq; using System.Threading; using System.Threading.Tasks; using System.IO; using MediaBrowser.Controller.Dto; using MediaBrowser.Model.Serialization; using MediaBrowser.Model.Tasks; using MediaBrowser.Model.Extensions; namespace MediaBrowser.Providers.MediaInfo { public class SubtitleScheduledTask : IScheduledTask { private readonly ILibraryManager _libraryManager; private readonly IServerConfigurationManager _config; private readonly ISubtitleManager _subtitleManager; private readonly IMediaSourceManager _mediaSourceManager; private readonly ILogger _logger; private readonly IJsonSerializer _json; public SubtitleScheduledTask(ILibraryManager libraryManager, IJsonSerializer json, IServerConfigurationManager config, ISubtitleManager subtitleManager, ILogger logger, IMediaSourceManager mediaSourceManager) { _libraryManager = libraryManager; _config = config; _subtitleManager = subtitleManager; _logger = logger; _mediaSourceManager = mediaSourceManager; _json = json; } public string Name { get { return "Download missing subtitles"; } } public string Description { get { return "Searches the internet for missing subtitles based on metadata configuration."; } } public string Category { get { return "Library"; } } private SubtitleOptions GetOptions() { return _config.GetConfiguration("subtitles"); } public async Task Execute(CancellationToken cancellationToken, IProgress progress) { var options = GetOptions(); var types = new List(); if (options.DownloadEpisodeSubtitles) { types.Add("Episode"); } if (options.DownloadMovieSubtitles) { types.Add("Movie"); } if (types.Count == 0) { return; } var dict = new Dictionary(); foreach (var lang in options.DownloadLanguages) { var query = new InternalItemsQuery { MediaTypes = new string[] {MediaType.Video}, IsVirtualItem = false, IncludeItemTypes = types.ToArray(types.Count), DtoOptions = new DtoOptions(true), SourceTypes = new[] {SourceType.Library} }; if (options.SkipIfAudioTrackMatches) { query.HasNoAudioTrackWithLanguage = lang; } if (options.SkipIfEmbeddedSubtitlesPresent) { // Exclude if it already has any subtitles of the same language query.HasNoSubtitleTrackWithLanguage = lang; } else { // Exclude if it already has external subtitles of the same language query.HasNoExternalSubtitleTrackWithLanguage = lang; } var videosByLanguage = _libraryManager.GetItemList(query); foreach (var video in videosByLanguage) { dict[video.Id] = video; } } var videos = dict.Values.ToList(); if (videos.Count == 0) { return; } var numComplete = 0; foreach (var video in videos) { cancellationToken.ThrowIfCancellationRequested(); try { await DownloadSubtitles(video as Video, options, cancellationToken).ConfigureAwait(false); } catch (Exception ex) { _logger.ErrorException("Error downloading subtitles for {0}", ex, video.Path); } // Update progress numComplete++; double percent = numComplete; percent /= videos.Count; progress.Report(100 * percent); } } private async Task DownloadSubtitles(Video video, SubtitleOptions options, CancellationToken cancellationToken) { if ((options.DownloadEpisodeSubtitles && video is Episode) || (options.DownloadMovieSubtitles && video is Movie)) { var mediaStreams = _mediaSourceManager.GetStaticMediaSources(video, false).First().MediaStreams; var downloadedLanguages = await new SubtitleDownloader(_logger, _subtitleManager) .DownloadSubtitles(video, mediaStreams, options.SkipIfEmbeddedSubtitlesPresent, options.SkipIfAudioTrackMatches, options.RequirePerfectMatch, options.DownloadLanguages, cancellationToken).ConfigureAwait(false); // Rescan if (downloadedLanguages.Count > 0) { await video.RefreshMetadata(cancellationToken).ConfigureAwait(false); return false; } return true; } return false; } public IEnumerable GetDefaultTriggers() { return new[] { // Every so often new TaskTriggerInfo { Type = TaskTriggerInfo.TriggerInterval, IntervalTicks = TimeSpan.FromHours(24).Ticks} }; } public string Key { get { return "DownloadSubtitles"; } } } }