Radarr/NzbDrone.Core/MediaFiles/DiskScanService.cs

167 lines
6.2 KiB
C#
Raw Normal View History

2011-06-20 01:59:31 +00:00
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using NLog;
2011-11-13 04:07:06 +00:00
using NzbDrone.Common;
using NzbDrone.Common.Messaging;
using NzbDrone.Core.MediaFiles.Commands;
using NzbDrone.Core.Parser;
using NzbDrone.Core.Providers;
using NzbDrone.Core.Tv;
using NzbDrone.Core.Tv.Events;
2011-06-20 01:59:31 +00:00
namespace NzbDrone.Core.MediaFiles
2011-06-20 01:59:31 +00:00
{
2013-04-15 05:27:32 +00:00
public interface IDiskScanService
{
EpisodeFile ImportFile(Series series, string filePath);
string[] GetVideoFiles(string path, bool allDirectories = true);
}
public class DiskScanService : IDiskScanService, IExecute<DiskScanCommand>, IHandle<EpisodeInfoAddedEvent>
2011-06-20 01:59:31 +00:00
{
private static readonly Logger Logger = LogManager.GetCurrentClassLogger();
private static readonly string[] MediaExtensions = new[] { ".mkv", ".avi", ".wmv", ".mp4", ".mpg", ".mpeg", ".xvid", ".flv", ".mov", ".rm", ".rmvb", ".divx", ".dvr-ms", ".ts", ".ogm", ".m4v", ".strm" };
2013-05-10 23:53:50 +00:00
private readonly IDiskProvider _diskProvider;
private readonly ISeriesService _seriesService;
2013-03-01 07:03:41 +00:00
private readonly IMediaFileService _mediaFileService;
private readonly IVideoFileInfoReader _videoFileInfoReader;
private readonly IParsingService _parsingService;
private readonly IMessageAggregator _messageAggregator;
2011-06-20 01:59:31 +00:00
public DiskScanService(IDiskProvider diskProvider, ISeriesService seriesService, IMediaFileService mediaFileService, IVideoFileInfoReader videoFileInfoReader,
IParsingService parsingService, IMessageAggregator messageAggregator)
2011-06-20 01:59:31 +00:00
{
_diskProvider = diskProvider;
_seriesService = seriesService;
2013-03-01 07:03:41 +00:00
_mediaFileService = mediaFileService;
_videoFileInfoReader = videoFileInfoReader;
_parsingService = parsingService;
_messageAggregator = messageAggregator;
2011-06-20 01:59:31 +00:00
}
private void Scan(Series series)
2011-06-20 01:59:31 +00:00
{
if (!_diskProvider.FolderExists(series.Path))
{
Logger.Warn("Series folder doesn't exist: {0}", series.Path);
return;
2011-06-20 01:59:31 +00:00
}
_messageAggregator.PublishCommand(new CleanMediaFileDb(series.Id));
var mediaFileList = GetVideoFiles(series.Path);
2011-06-20 01:59:31 +00:00
foreach (var filePath in mediaFileList)
{
try
{
ImportFile(series, filePath);
}
catch (Exception e)
{
Logger.ErrorException("Couldn't import file " + filePath, e);
}
2011-06-20 01:59:31 +00:00
}
2013-01-19 23:15:31 +00:00
//Todo: Find the "best" episode file for all found episodes and import that one
//Todo: Move the episode linking to here, instead of import (or rename import)
2011-06-20 01:59:31 +00:00
}
public EpisodeFile ImportFile(Series series, string filePath)
2011-06-20 01:59:31 +00:00
{
Logger.Trace("Importing file to database [{0}]", filePath);
2013-03-01 07:03:41 +00:00
if (_mediaFileService.Exists(filePath))
2011-06-20 01:59:31 +00:00
{
2011-06-20 03:04:08 +00:00
Logger.Trace("[{0}] already exists in the database. skipping.", filePath);
return null;
}
2011-06-20 01:59:31 +00:00
var parsedEpisode = _parsingService.GetEpisodes(filePath, series);
2011-06-20 01:59:31 +00:00
if (parsedEpisode == null || !parsedEpisode.Episodes.Any())
{
2011-06-20 03:04:08 +00:00
return null;
}
2011-06-20 01:59:31 +00:00
var size = _diskProvider.GetFileSize(filePath);
if (series.SeriesType == SeriesTypes.Daily || parsedEpisode.SeasonNumber > 0)
{
var runTime = _videoFileInfoReader.GetRunTime(filePath);
if (size < Constants.IgnoreFileSize && runTime.TotalMinutes < 3)
{
Logger.Trace("[{0}] appears to be a sample. skipping.", filePath);
return null;
}
}
if (parsedEpisode.Episodes.Any(e => e.EpisodeFileId != 0 && e.EpisodeFile.Value.Quality > parsedEpisode.Quality))
{
Logger.Trace("This file isn't an upgrade for all episodes. Skipping {0}", filePath);
2011-06-21 05:44:01 +00:00
return null;
}
2011-06-20 03:04:08 +00:00
var episodeFile = new EpisodeFile();
episodeFile.DateAdded = DateTime.Now;
episodeFile.SeriesId = series.Id;
2013-04-30 00:04:14 +00:00
episodeFile.Path = filePath.CleanPath();
2011-06-20 03:04:08 +00:00
episodeFile.Size = size;
episodeFile.Quality = parsedEpisode.Quality;
episodeFile.SeasonNumber = parsedEpisode.SeasonNumber;
2013-04-30 00:04:14 +00:00
episodeFile.SceneName = Path.GetFileNameWithoutExtension(filePath.CleanPath());
episodeFile.Episodes = parsedEpisode.Episodes;
2013-01-19 23:15:31 +00:00
//Todo: We shouldn't actually import the file until we confirm its the only one we want.
//Todo: Separate episodeFile creation from importing (pass file to import to import)
2013-03-05 05:33:34 +00:00
_mediaFileService.Add(episodeFile);
2011-06-20 03:04:08 +00:00
return episodeFile;
2011-06-20 01:59:31 +00:00
}
public string[] GetVideoFiles(string path, bool allDirectories = true)
2011-06-20 01:59:31 +00:00
{
Logger.Debug("Scanning '{0}' for video files", path);
2011-06-20 01:59:31 +00:00
var searchOption = allDirectories ? SearchOption.AllDirectories : SearchOption.TopDirectoryOnly;
var filesOnDisk = _diskProvider.GetFiles(path, searchOption);
2011-06-20 01:59:31 +00:00
var mediaFileList = filesOnDisk.Where(c => MediaExtensions.Contains(Path.GetExtension(c).ToLower())).ToList();
2011-06-20 01:59:31 +00:00
2011-06-22 06:34:33 +00:00
Logger.Trace("{0} video files were found in {1}", mediaFileList.Count, path);
return mediaFileList.ToArray();
2011-06-20 01:59:31 +00:00
}
public void Execute(DiskScanCommand message)
{
var seriesToScan = new List<Series>();
if (message.SeriesId.HasValue)
{
seriesToScan.Add(_seriesService.GetSeries(message.SeriesId.Value));
}
else
{
seriesToScan.AddRange(_seriesService.GetAllSeries());
}
foreach (var series in seriesToScan)
{
try
{
Scan(series);
}
catch (Exception e)
{
Logger.ErrorException("Diskscan failed for " + series.Title, e);
}
}
}
public void Handle(EpisodeInfoAddedEvent message)
{
Scan(message.Series);
}
2011-06-20 01:59:31 +00:00
}
2011-06-20 03:25:04 +00:00
}