2011-06-20 04:59:31 +03:00
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.IO;
|
|
|
|
using System.Linq;
|
|
|
|
using NLog;
|
2011-11-13 07:07:06 +03:00
|
|
|
using NzbDrone.Common;
|
2013-03-01 10:03:41 +03:00
|
|
|
using NzbDrone.Core.MediaFiles;
|
2013-04-15 04:41:39 +03:00
|
|
|
using NzbDrone.Core.Parser;
|
2013-02-19 09:01:03 +03:00
|
|
|
using NzbDrone.Core.Tv;
|
2011-06-20 04:59:31 +03:00
|
|
|
|
|
|
|
namespace NzbDrone.Core.Providers
|
|
|
|
{
|
2011-06-20 06:04:08 +03:00
|
|
|
public class DiskScanProvider
|
2011-06-20 04:59:31 +03:00
|
|
|
{
|
|
|
|
private static readonly Logger Logger = LogManager.GetCurrentClassLogger();
|
2013-03-07 00:20:33 +03:00
|
|
|
private static readonly string[] MediaExtensions = new[] { ".mkv", ".avi", ".wmv", ".mp4", ".mpg", ".mpeg", ".xvid", ".flv", ".mov", ".rm", ".rmvb", ".divx", ".dvr-ms", ".ts", ".ogm", ".m4v", ".strm" };
|
2011-06-20 04:59:31 +03:00
|
|
|
private readonly DiskProvider _diskProvider;
|
2013-03-07 07:34:56 +03:00
|
|
|
private readonly ICleanGhostFiles _ghostFileCleaner;
|
2013-03-01 10:03:41 +03:00
|
|
|
private readonly IMediaFileService _mediaFileService;
|
2013-04-15 04:41:39 +03:00
|
|
|
private readonly IVideoFileInfoReader _videoFileInfoReader;
|
|
|
|
private readonly IParsingService _parsingService;
|
2011-06-20 04:59:31 +03:00
|
|
|
|
2013-04-15 04:41:39 +03:00
|
|
|
public DiskScanProvider(DiskProvider diskProvider, ICleanGhostFiles ghostFileCleaner, IMediaFileService mediaFileService, IVideoFileInfoReader videoFileInfoReader,
|
|
|
|
IParsingService parsingService)
|
2011-06-20 04:59:31 +03:00
|
|
|
{
|
|
|
|
_diskProvider = diskProvider;
|
2013-03-07 07:34:56 +03:00
|
|
|
_ghostFileCleaner = ghostFileCleaner;
|
2013-03-01 10:03:41 +03:00
|
|
|
_mediaFileService = mediaFileService;
|
2013-04-15 04:41:39 +03:00
|
|
|
_videoFileInfoReader = videoFileInfoReader;
|
|
|
|
_parsingService = parsingService;
|
2011-06-20 04:59:31 +03:00
|
|
|
}
|
|
|
|
|
2013-04-15 04:41:39 +03:00
|
|
|
public virtual void Scan(Series series)
|
2011-06-20 04:59:31 +03:00
|
|
|
{
|
2013-04-15 04:41:39 +03:00
|
|
|
if (!_diskProvider.FolderExists(series.Path))
|
2011-10-22 22:03:54 +03:00
|
|
|
{
|
2013-04-15 04:41:39 +03:00
|
|
|
Logger.Warn("Series folder doesn't exist: {0}", series.Path);
|
|
|
|
return;
|
2011-06-20 04:59:31 +03:00
|
|
|
}
|
|
|
|
|
2013-03-07 07:34:56 +03:00
|
|
|
_ghostFileCleaner.RemoveNonExistingFiles(series.Id);
|
2011-06-21 09:34:45 +03:00
|
|
|
|
2013-04-15 04:41:39 +03:00
|
|
|
var mediaFileList = GetVideoFiles(series.Path);
|
2011-06-20 04:59:31 +03:00
|
|
|
|
|
|
|
foreach (var filePath in mediaFileList)
|
|
|
|
{
|
2013-04-15 04:41:39 +03:00
|
|
|
ImportFile(series, filePath);
|
2011-06-20 04:59:31 +03:00
|
|
|
}
|
|
|
|
|
2013-01-20 02:15:31 +03: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 04:59:31 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
public virtual EpisodeFile ImportFile(Series series, string filePath)
|
|
|
|
{
|
|
|
|
Logger.Trace("Importing file to database [{0}]", filePath);
|
|
|
|
|
2013-03-01 10:03:41 +03:00
|
|
|
if (_mediaFileService.Exists(filePath))
|
2011-06-20 04:59:31 +03:00
|
|
|
{
|
2011-06-20 06:04:08 +03:00
|
|
|
Logger.Trace("[{0}] already exists in the database. skipping.", filePath);
|
|
|
|
return null;
|
|
|
|
}
|
2011-06-20 04:59:31 +03:00
|
|
|
|
2013-04-15 04:41:39 +03:00
|
|
|
var parsedEpisode = _parsingService.GetEpisodes(filePath, series);
|
2011-06-20 04:59:31 +03:00
|
|
|
|
2013-04-15 04:41:39 +03:00
|
|
|
if (parsedEpisode == null || !parsedEpisode.Episodes.Any())
|
|
|
|
{
|
2011-06-20 06:04:08 +03:00
|
|
|
return null;
|
2013-04-15 04:41:39 +03:00
|
|
|
}
|
2011-06-20 04:59:31 +03:00
|
|
|
|
2013-02-08 04:47:24 +03:00
|
|
|
var size = _diskProvider.GetSize(filePath);
|
|
|
|
|
2013-04-15 04:41:39 +03:00
|
|
|
if (series.SeriesType == SeriesTypes.Daily || parsedEpisode.SeasonNumber > 0)
|
2013-02-08 04:47:24 +03:00
|
|
|
{
|
2013-04-15 04:41:39 +03:00
|
|
|
var runTime = _videoFileInfoReader.GetRunTime(filePath);
|
|
|
|
if (size < Constants.IgnoreFileSize && runTime.TotalMinutes < 3)
|
2013-02-08 04:47:24 +03:00
|
|
|
{
|
|
|
|
Logger.Trace("[{0}] appears to be a sample. skipping.", filePath);
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-15 04:41:39 +03:00
|
|
|
if (parsedEpisode.Episodes.Any(e => e.EpisodeFile != null && e.EpisodeFile.Quality > parsedEpisode.Quality))
|
2011-08-26 09:23:21 +03:00
|
|
|
{
|
|
|
|
Logger.Trace("This file isn't an upgrade for all episodes. Skipping {0}", filePath);
|
2011-06-21 08:44:01 +03:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2011-06-20 06:04:08 +03:00
|
|
|
var episodeFile = new EpisodeFile();
|
|
|
|
episodeFile.DateAdded = DateTime.Now;
|
2013-02-26 06:58:57 +03:00
|
|
|
episodeFile.SeriesId = series.Id;
|
2011-11-21 03:35:29 +03:00
|
|
|
episodeFile.Path = filePath.NormalizePath();
|
2011-06-20 06:04:08 +03:00
|
|
|
episodeFile.Size = size;
|
2013-04-15 04:41:39 +03:00
|
|
|
episodeFile.Quality = parsedEpisode.Quality;
|
|
|
|
episodeFile.SeasonNumber = parsedEpisode.SeasonNumber;
|
2012-08-03 10:01:34 +03:00
|
|
|
episodeFile.SceneName = Path.GetFileNameWithoutExtension(filePath.NormalizePath());
|
2013-01-20 02:15:31 +03: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 08:33:34 +03:00
|
|
|
_mediaFileService.Add(episodeFile);
|
2011-06-20 06:04:08 +03:00
|
|
|
return episodeFile;
|
2011-06-20 04:59:31 +03:00
|
|
|
}
|
|
|
|
|
2013-04-15 04:41:39 +03:00
|
|
|
public virtual string[] GetVideoFiles(string path, bool allDirectories = true)
|
2011-06-20 04:59:31 +03:00
|
|
|
{
|
2011-11-26 02:46:29 +03:00
|
|
|
Logger.Debug("Scanning '{0}' for video files", path);
|
2011-06-20 04:59:31 +03:00
|
|
|
|
2012-08-27 13:15:22 +03:00
|
|
|
var searchOption = allDirectories ? SearchOption.AllDirectories : SearchOption.TopDirectoryOnly;
|
|
|
|
var filesOnDisk = _diskProvider.GetFiles(path, searchOption);
|
2011-06-20 04:59:31 +03:00
|
|
|
|
2013-03-07 00:20:33 +03:00
|
|
|
var mediaFileList = filesOnDisk.Where(c => MediaExtensions.Contains(Path.GetExtension(c).ToLower())).ToList();
|
2011-06-20 04:59:31 +03:00
|
|
|
|
2011-06-22 09:34:33 +03:00
|
|
|
Logger.Trace("{0} video files were found in {1}", mediaFileList.Count, path);
|
2013-04-15 04:41:39 +03:00
|
|
|
return mediaFileList.ToArray();
|
2011-06-20 04:59:31 +03:00
|
|
|
}
|
|
|
|
}
|
2011-06-20 06:25:04 +03:00
|
|
|
}
|