aboutsummaryrefslogtreecommitdiff
path: root/MediaBrowser.Plugins.Trailers/ScheduledTasks/CurrentTrailerDownloadTask.cs
diff options
context:
space:
mode:
authorLukePulverenti <luke.pulverenti@gmail.com>2013-02-20 20:33:05 -0500
committerLukePulverenti <luke.pulverenti@gmail.com>2013-02-20 20:33:05 -0500
commit767cdc1f6f6a63ce997fc9476911e2c361f9d402 (patch)
tree49add55976f895441167c66cfa95e5c7688d18ce /MediaBrowser.Plugins.Trailers/ScheduledTasks/CurrentTrailerDownloadTask.cs
parent845554722efaed872948a9e0f7202e3ef52f1b6e (diff)
Pushing missing changes
Diffstat (limited to 'MediaBrowser.Plugins.Trailers/ScheduledTasks/CurrentTrailerDownloadTask.cs')
-rw-r--r--MediaBrowser.Plugins.Trailers/ScheduledTasks/CurrentTrailerDownloadTask.cs311
1 files changed, 311 insertions, 0 deletions
diff --git a/MediaBrowser.Plugins.Trailers/ScheduledTasks/CurrentTrailerDownloadTask.cs b/MediaBrowser.Plugins.Trailers/ScheduledTasks/CurrentTrailerDownloadTask.cs
new file mode 100644
index 000000000..90bd80caf
--- /dev/null
+++ b/MediaBrowser.Plugins.Trailers/ScheduledTasks/CurrentTrailerDownloadTask.cs
@@ -0,0 +1,311 @@
+using MediaBrowser.Common.IO;
+using MediaBrowser.Common.ScheduledTasks;
+using MediaBrowser.Common.Serialization;
+using MediaBrowser.Controller;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Model.Net;
+using MediaBrowser.Model.Tasks;
+using MediaBrowser.Plugins.Trailers.Entities;
+using System;
+using System.Collections.Generic;
+using System.ComponentModel.Composition;
+using System.IO;
+using System.Linq;
+using System.Threading;
+using System.Threading.Tasks;
+
+namespace MediaBrowser.Plugins.Trailers.ScheduledTasks
+{
+ /// <summary>
+ /// Downloads trailers from the web at scheduled times
+ /// </summary>
+ [Export(typeof(IScheduledTask))]
+ public class CurrentTrailerDownloadTask : BaseScheduledTask<Kernel>
+ {
+ /// <summary>
+ /// Creates the triggers that define when the task will run
+ /// </summary>
+ /// <returns>IEnumerable{BaseTaskTrigger}.</returns>
+ protected override IEnumerable<BaseTaskTrigger> GetDefaultTriggers()
+ {
+ var trigger = new DailyTrigger { TimeOfDay = TimeSpan.FromHours(2) }; //2am
+
+ return new[] { trigger };
+ }
+
+ /// <summary>
+ /// Returns the task to be executed
+ /// </summary>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <param name="progress">The progress.</param>
+ /// <returns>Task.</returns>
+ protected override async Task ExecuteInternal(CancellationToken cancellationToken, IProgress<TaskProgress> progress)
+ {
+ // Get the list of trailers
+ var trailers = await AppleTrailerListingDownloader.GetTrailerList(cancellationToken).ConfigureAwait(false);
+
+ progress.Report(new TaskProgress { PercentComplete = 1 });
+
+ var trailersToDownload = trailers.Where(t => !IsOldTrailer(t.Video)).ToList();
+
+ cancellationToken.ThrowIfCancellationRequested();
+
+ var numComplete = 0;
+
+ // Fetch them all in parallel
+ var tasks = trailersToDownload.Select(t => Task.Run(async () =>
+ {
+ cancellationToken.ThrowIfCancellationRequested();
+
+ try
+ {
+ await DownloadTrailer(t, cancellationToken).ConfigureAwait(false);
+ }
+ catch (Exception ex)
+ {
+ Logger.ErrorException("Error downloading {0}", ex, t.TrailerUrl);
+ }
+
+ // Update progress
+ lock (progress)
+ {
+ numComplete++;
+ double percent = numComplete;
+ percent /= trailersToDownload.Count;
+
+ // Leave 1% for DeleteOldTrailers
+ progress.Report(new TaskProgress { PercentComplete = (99 * percent) + 1 });
+ }
+ }));
+
+ cancellationToken.ThrowIfCancellationRequested();
+
+ await Task.WhenAll(tasks).ConfigureAwait(false);
+
+ cancellationToken.ThrowIfCancellationRequested();
+
+ if (Plugin.Instance.Configuration.DeleteOldTrailers)
+ {
+ // Enforce MaxTrailerAge
+ DeleteOldTrailers();
+ }
+
+ progress.Report(new TaskProgress { PercentComplete = 100 });
+ }
+
+ /// <summary>
+ /// Downloads a single trailer into the trailers directory
+ /// </summary>
+ /// <param name="trailer">The trailer.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task.</returns>
+ private async Task DownloadTrailer(TrailerInfo trailer, CancellationToken cancellationToken)
+ {
+ // Construct the trailer foldername
+ var folderName = FileSystem.GetValidFilename(trailer.Video.Name);
+
+ if (trailer.Video.ProductionYear.HasValue)
+ {
+ folderName += string.Format(" ({0})", trailer.Video.ProductionYear);
+ }
+
+ var folderPath = Path.Combine(Plugin.Instance.DownloadPath, folderName);
+
+ // Figure out which image we're going to download
+ var imageUrl = trailer.HdImageUrl ?? trailer.ImageUrl;
+
+ // Construct the video filename (to match the folder name)
+ var videoFileName = Path.ChangeExtension(folderName, Path.GetExtension(trailer.TrailerUrl));
+
+ // Construct the image filename (folder + original extension)
+ var imageFileName = Path.ChangeExtension("folder", Path.GetExtension(imageUrl));
+
+ // Construct full paths
+ var videoFilePath = Path.Combine(folderPath, videoFileName);
+ var imageFilePath = Path.Combine(folderPath, imageFileName);
+
+ // Create tasks to download each of them, if we don't already have them
+ Task<string> videoTask = null;
+ Task<MemoryStream> imageTask = null;
+
+ var tasks = new List<Task>();
+
+ if (!File.Exists(videoFilePath))
+ {
+ Logger.Info("Downloading trailer: " + trailer.TrailerUrl);
+
+ // Fetch the video to a temp file because it's too big to put into a MemoryStream
+ videoTask = Kernel.HttpManager.FetchToTempFile(trailer.TrailerUrl, Kernel.ResourcePools.AppleTrailerVideos, cancellationToken, new Progress<double> { }, "QuickTime/7.6.2");
+ tasks.Add(videoTask);
+ }
+
+ if (!string.IsNullOrWhiteSpace(imageUrl) && !File.Exists(imageFilePath))
+ {
+ // Fetch the image to a memory stream
+ Logger.Info("Downloading trailer image: " + imageUrl);
+ imageTask = Kernel.HttpManager.FetchToMemoryStream(imageUrl, Kernel.ResourcePools.AppleTrailerImages, cancellationToken);
+ tasks.Add(imageTask);
+ }
+
+ try
+ {
+ // Wait for both downloads to finish
+ await Task.WhenAll(tasks).ConfigureAwait(false);
+ }
+ catch (HttpException ex)
+ {
+ Logger.ErrorException("Error downloading trailer file or image", ex);
+ }
+
+ var videoFailed = false;
+ var directoryEnsured = false;
+
+ // Proces the video file task result
+ if (videoTask != null)
+ {
+ if (videoTask.Status == TaskStatus.RanToCompletion)
+ {
+ EnsureDirectory(folderPath);
+
+ directoryEnsured = true;
+
+ // Move the temp file to the final destination
+ try
+ {
+ File.Move(videoTask.Result, videoFilePath);
+ }
+ catch (IOException ex)
+ {
+ Logger.ErrorException("Error moving temp file", ex);
+ File.Delete(videoTask.Result);
+ videoFailed = true;
+ }
+ }
+ else
+ {
+ Logger.Info("Trailer download failed: " + trailer.TrailerUrl);
+
+ // Don't bother with the image if the video download failed
+ videoFailed = true;
+ }
+ }
+
+ // Process the image file task result
+ if (imageTask != null && !videoFailed && imageTask.Status == TaskStatus.RanToCompletion)
+ {
+ if (!directoryEnsured)
+ {
+ EnsureDirectory(folderPath);
+ }
+
+ try
+ {
+ // Save the image to the file system
+ using (var fs = new FileStream(imageFilePath, FileMode.Create, FileAccess.Write, FileShare.Read, StreamDefaults.DefaultFileStreamBufferSize, FileOptions.Asynchronous))
+ {
+ using (var sourceStream = imageTask.Result)
+ {
+ await sourceStream.CopyToAsync(fs).ConfigureAwait(false);
+ }
+ }
+ }
+ catch (IOException ex)
+ {
+ Logger.ErrorException("Error saving image to file system", ex);
+ }
+ }
+
+ // Save metadata only if the video was downloaded
+ if (!videoFailed && videoTask != null)
+ {
+ JsonSerializer.SerializeToFile(trailer.Video, Path.Combine(folderPath, "trailer.json"));
+ }
+ }
+
+ /// <summary>
+ /// Determines whether [is old trailer] [the specified trailer].
+ /// </summary>
+ /// <param name="trailer">The trailer.</param>
+ /// <returns><c>true</c> if [is old trailer] [the specified trailer]; otherwise, <c>false</c>.</returns>
+ private bool IsOldTrailer(Trailer trailer)
+ {
+ if (!Plugin.Instance.Configuration.MaxTrailerAge.HasValue)
+ {
+ return false;
+ }
+
+ if (!trailer.PremiereDate.HasValue)
+ {
+ return false;
+ }
+
+ var now = DateTime.UtcNow;
+
+ // Not old if it still hasn't premiered.
+ if (now < trailer.PremiereDate.Value)
+ {
+ return false;
+ }
+
+ return (DateTime.UtcNow - trailer.PremiereDate.Value).TotalDays >
+ Plugin.Instance.Configuration.MaxTrailerAge.Value;
+ }
+
+ /// <summary>
+ /// Deletes trailers that are older than the supplied date
+ /// </summary>
+ private void DeleteOldTrailers()
+ {
+ var collectionFolder = (Folder)Kernel.RootFolder.Children.First(c => c.GetType().Name.Equals(typeof(TrailerCollectionFolder).Name));
+
+ foreach (var trailer in collectionFolder.RecursiveChildren.OfType<Trailer>().Where(IsOldTrailer))
+ {
+ Logger.Info("Deleting old trailer: " + trailer.Name);
+
+ Directory.Delete(Path.GetDirectoryName(trailer.Path), true);
+ }
+ }
+
+ /// <summary>
+ /// Ensures the directory.
+ /// </summary>
+ /// <param name="path">The path.</param>
+ private void EnsureDirectory(string path)
+ {
+ if (!Directory.Exists(path))
+ {
+ Directory.CreateDirectory(path);
+ }
+ }
+
+ /// <summary>
+ /// Gets the name of the task
+ /// </summary>
+ /// <value>The name.</value>
+ public override string Name
+ {
+ get { return "Find current trailers"; }
+ }
+
+ /// <summary>
+ /// Gets the category.
+ /// </summary>
+ /// <value>The category.</value>
+ public override string Category
+ {
+ get
+ {
+ return "Trailers";
+ }
+ }
+
+ /// <summary>
+ /// Gets the description.
+ /// </summary>
+ /// <value>The description.</value>
+ public override string Description
+ {
+ get { return "Searches the web for upcoming movie trailers, and downloads them based on your Trailer plugin settings."; }
+ }
+ }
+}