123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258 |
- using System.Globalization;
- using MediaBrowser.Common.Extensions;
- using MediaBrowser.Common.IO;
- using MediaBrowser.Common.MediaInfo;
- using MediaBrowser.Controller.Entities;
- using MediaBrowser.Controller.Persistence;
- using MediaBrowser.Model.Entities;
- using MediaBrowser.Model.Logging;
- using System;
- using System.Collections.Generic;
- using System.IO;
- using System.Linq;
- using System.Threading;
- using System.Threading.Tasks;
- namespace MediaBrowser.Controller.MediaInfo
- {
- /// <summary>
- /// Class FFMpegManager
- /// </summary>
- public class FFMpegManager
- {
- private readonly IServerApplicationPaths _appPaths;
- private readonly IMediaEncoder _encoder;
- private readonly ILogger _logger;
- private readonly IItemRepository _itemRepo;
- private readonly IFileSystem _fileSystem;
- /// <summary>
- /// Initializes a new instance of the <see cref="FFMpegManager" /> class.
- /// </summary>
- /// <param name="appPaths">The app paths.</param>
- /// <param name="encoder">The encoder.</param>
- /// <param name="logger">The logger.</param>
- /// <param name="itemRepo">The item repo.</param>
- /// <exception cref="System.ArgumentNullException">zipClient</exception>
- public FFMpegManager(IServerApplicationPaths appPaths, IMediaEncoder encoder, ILogger logger, IItemRepository itemRepo, IFileSystem fileSystem)
- {
- _appPaths = appPaths;
- _encoder = encoder;
- _logger = logger;
- _itemRepo = itemRepo;
- _fileSystem = fileSystem;
- }
- /// <summary>
- /// Gets the chapter images data path.
- /// </summary>
- /// <value>The chapter images data path.</value>
- public string ChapterImagesPath
- {
- get
- {
- return Path.Combine(_appPaths.DataPath, "chapter-images");
- }
- }
- /// <summary>
- /// Gets the subtitle cache path.
- /// </summary>
- /// <value>The subtitle cache path.</value>
- private string SubtitleCachePath
- {
- get
- {
- return Path.Combine(_appPaths.CachePath, "subtitles");
- }
- }
- /// <summary>
- /// The first chapter ticks
- /// </summary>
- private static readonly long FirstChapterTicks = TimeSpan.FromSeconds(15).Ticks;
- /// <summary>
- /// Extracts the chapter images.
- /// </summary>
- /// <param name="video">The video.</param>
- /// <param name="chapters">The chapters.</param>
- /// <param name="extractImages">if set to <c>true</c> [extract images].</param>
- /// <param name="saveChapters">if set to <c>true</c> [save chapters].</param>
- /// <param name="cancellationToken">The cancellation token.</param>
- /// <returns>Task.</returns>
- /// <exception cref="System.ArgumentNullException"></exception>
- public async Task<bool> PopulateChapterImages(Video video, List<ChapterInfo> chapters, bool extractImages, bool saveChapters, CancellationToken cancellationToken)
- {
- // Can't extract images if there are no video streams
- if (!video.DefaultVideoStreamIndex.HasValue)
- {
- return true;
- }
- var success = true;
- var changesMade = false;
- var runtimeTicks = video.RunTimeTicks ?? 0;
- var currentImages = GetSavedChapterImages(video);
- foreach (var chapter in chapters)
- {
- if (chapter.StartPositionTicks >= runtimeTicks)
- {
- _logger.Info("Stopping chapter extraction for {0} because a chapter was found with a position greater than the runtime.", video.Name);
- break;
- }
- var path = GetChapterImagePath(video, chapter.StartPositionTicks);
- if (!currentImages.Contains(path, StringComparer.OrdinalIgnoreCase))
- {
- if (extractImages)
- {
- if (video.VideoType == VideoType.HdDvd || video.VideoType == VideoType.Iso)
- {
- continue;
- }
- if (video.VideoType == VideoType.BluRay)
- {
- // Can only extract reliably on single file blurays
- if (video.PlayableStreamFileNames == null || video.PlayableStreamFileNames.Count != 1)
- {
- continue;
- }
- }
- // Add some time for the first chapter to make sure we don't end up with a black image
- var time = chapter.StartPositionTicks == 0 ? TimeSpan.FromTicks(Math.Min(FirstChapterTicks, video.RunTimeTicks ?? 0)) : TimeSpan.FromTicks(chapter.StartPositionTicks);
- InputType type;
- var inputPath = MediaEncoderHelpers.GetInputArgument(video, null, out type);
- try
- {
- var parentPath = Path.GetDirectoryName(path);
- Directory.CreateDirectory(parentPath);
- await _encoder.ExtractImage(inputPath, type, video.Video3DFormat, time, path, cancellationToken).ConfigureAwait(false);
- chapter.ImagePath = path;
- changesMade = true;
- }
- catch
- {
- success = false;
- break;
- }
- }
- }
- else if (!string.Equals(path, chapter.ImagePath, StringComparison.OrdinalIgnoreCase))
- {
- chapter.ImagePath = path;
- changesMade = true;
- }
- }
- if (saveChapters && changesMade)
- {
- await _itemRepo.SaveChapters(video.Id, chapters, cancellationToken).ConfigureAwait(false);
- }
- DeleteDeadImages(currentImages, chapters);
- return success;
- }
- private void DeleteDeadImages(IEnumerable<string> images, IEnumerable<ChapterInfo> chapters)
- {
- var deadImages = images
- .Except(chapters.Select(i => i.ImagePath), StringComparer.OrdinalIgnoreCase)
- .Where(i => BaseItem.SupportedImageExtensions.Contains(Path.GetExtension(i), StringComparer.OrdinalIgnoreCase))
- .ToList();
- foreach (var image in deadImages)
- {
- _logger.Debug("Deleting dead chapter image {0}", image);
- try
- {
- File.Delete(image);
- }
- catch (IOException ex)
- {
- _logger.ErrorException("Error deleting {0}.", ex, image);
- }
- }
- }
- private readonly CultureInfo UsCulture = new CultureInfo("en-US");
- /// <summary>
- /// Gets the subtitle cache path.
- /// </summary>
- /// <param name="input">The input.</param>
- /// <param name="subtitleStreamIndex">Index of the subtitle stream.</param>
- /// <param name="offset">The offset.</param>
- /// <param name="outputExtension">The output extension.</param>
- /// <returns>System.String.</returns>
- public string GetSubtitleCachePath(Video input, int subtitleStreamIndex, TimeSpan? offset, string outputExtension)
- {
- var ticksParam = offset.HasValue ? "_" + offset.Value.Ticks : "";
- var stream = _itemRepo.GetMediaStreams(new MediaStreamQuery
- {
- ItemId = input.Id,
- Index = subtitleStreamIndex
- }).FirstOrDefault();
- if (stream == null)
- {
- return null;
- }
- if (stream.IsExternal)
- {
- ticksParam += _fileSystem.GetLastWriteTimeUtc(stream.Path).Ticks;
- }
- var filename = (input.Id + "_" + subtitleStreamIndex.ToString(UsCulture) + "_" + input.DateModified.Ticks.ToString(UsCulture) + ticksParam).GetMD5() + outputExtension;
- var prefix = filename.Substring(0, 1);
- return Path.Combine(SubtitleCachePath, prefix, filename);
- }
- public string GetChapterImagePath(Video video, long chapterPositionTicks)
- {
- var filename = video.DateModified.Ticks.ToString(UsCulture) + "_" + chapterPositionTicks.ToString(UsCulture) + ".jpg";
- var videoId = video.Id.ToString();
- var prefix = videoId.Substring(0, 1);
- return Path.Combine(ChapterImagesPath, prefix, videoId, filename);
- }
- public List<string> GetSavedChapterImages(Video video)
- {
- var videoId = video.Id.ToString();
- var prefix = videoId.Substring(0, 1);
- var path = Path.Combine(ChapterImagesPath, prefix, videoId);
- try
- {
- return Directory.EnumerateFiles(path)
- .ToList();
- }
- catch (DirectoryNotFoundException)
- {
- return new List<string>();
- }
- }
- }
- }
|