2011-06-17 04:48:24 +02:00
|
|
|
using System;
|
2010-10-21 03:49:23 +02:00
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.IO;
|
|
|
|
using System.Linq;
|
2011-06-14 03:23:04 +02:00
|
|
|
using Ninject;
|
2010-10-21 03:49:23 +02:00
|
|
|
using NLog;
|
2011-06-07 08:29:07 +02:00
|
|
|
using NzbDrone.Core.Helpers;
|
2011-04-04 05:50:12 +02:00
|
|
|
using NzbDrone.Core.Providers.Core;
|
2010-10-21 03:49:23 +02:00
|
|
|
using NzbDrone.Core.Repository;
|
2011-06-07 08:29:07 +02:00
|
|
|
using NzbDrone.Core.Repository.Quality;
|
2011-06-17 05:29:39 +02:00
|
|
|
using PetaPoco;
|
2011-11-21 01:35:29 +01:00
|
|
|
using NzbDrone.Common;
|
2010-10-21 03:49:23 +02:00
|
|
|
|
|
|
|
namespace NzbDrone.Core.Providers
|
|
|
|
{
|
2011-04-10 03:34:36 +02:00
|
|
|
public class MediaFileProvider
|
2010-10-21 03:49:23 +02:00
|
|
|
{
|
|
|
|
private static readonly Logger Logger = LogManager.GetCurrentClassLogger();
|
2011-06-07 08:29:07 +02:00
|
|
|
private readonly ConfigProvider _configProvider;
|
2011-06-17 05:29:39 +02:00
|
|
|
private readonly IDatabase _database;
|
2011-06-20 05:25:04 +02:00
|
|
|
private readonly EpisodeProvider _episodeProvider;
|
2010-10-21 03:49:23 +02:00
|
|
|
|
2011-06-14 03:23:04 +02:00
|
|
|
[Inject]
|
2011-06-20 05:04:08 +02:00
|
|
|
public MediaFileProvider(EpisodeProvider episodeProvider, ConfigProvider configProvider, IDatabase database)
|
2010-10-21 03:49:23 +02:00
|
|
|
{
|
|
|
|
_episodeProvider = episodeProvider;
|
2011-06-07 08:29:07 +02:00
|
|
|
_configProvider = configProvider;
|
2011-06-17 05:29:39 +02:00
|
|
|
_database = database;
|
2010-10-21 03:49:23 +02:00
|
|
|
}
|
|
|
|
|
2011-06-20 05:25:04 +02:00
|
|
|
public MediaFileProvider()
|
|
|
|
{
|
|
|
|
}
|
2010-10-24 09:46:58 +02:00
|
|
|
|
2011-06-22 03:12:20 +02:00
|
|
|
public virtual int Add(EpisodeFile episodeFile)
|
|
|
|
{
|
|
|
|
return Convert.ToInt32(_database.Insert(episodeFile));
|
|
|
|
}
|
|
|
|
|
2011-05-18 07:32:23 +02:00
|
|
|
public virtual void Update(EpisodeFile episodeFile)
|
2011-02-18 07:49:23 +01:00
|
|
|
{
|
2011-06-17 05:29:39 +02:00
|
|
|
_database.Update(episodeFile);
|
2011-06-20 03:59:31 +02:00
|
|
|
}
|
2011-02-18 07:49:23 +01:00
|
|
|
|
2011-06-22 03:12:20 +02:00
|
|
|
public virtual void Delete(int episodeFileId)
|
|
|
|
{
|
2011-06-22 09:26:36 +02:00
|
|
|
_database.Delete<EpisodeFile>(episodeFileId);
|
2011-06-22 03:12:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public virtual bool Exists(string path)
|
|
|
|
{
|
2011-11-21 01:35:29 +01:00
|
|
|
return _database.Exists<EpisodeFile>("WHERE Path =@0", path.NormalizePath());
|
2011-06-22 03:12:20 +02:00
|
|
|
}
|
|
|
|
|
2011-05-18 07:32:23 +02:00
|
|
|
public virtual EpisodeFile GetEpisodeFile(int episodeFileId)
|
2011-02-18 07:49:23 +01:00
|
|
|
{
|
2011-06-19 01:03:58 +02:00
|
|
|
return _database.Single<EpisodeFile>(episodeFileId);
|
2011-02-18 07:49:23 +01:00
|
|
|
}
|
|
|
|
|
2011-05-18 07:32:23 +02:00
|
|
|
public virtual List<EpisodeFile> GetEpisodeFiles()
|
2011-02-22 07:22:40 +01:00
|
|
|
{
|
2011-06-17 05:29:39 +02:00
|
|
|
return _database.Fetch<EpisodeFile>();
|
2011-02-22 07:22:40 +01:00
|
|
|
}
|
|
|
|
|
2011-06-19 22:43:33 +02:00
|
|
|
public virtual IList<EpisodeFile> GetSeriesFiles(int seriesId)
|
2011-02-24 01:40:11 +01:00
|
|
|
{
|
2011-08-22 02:48:37 +02:00
|
|
|
return _database.Fetch<EpisodeFile>("WHERE SeriesId= @0", seriesId);
|
|
|
|
}
|
|
|
|
|
|
|
|
public virtual IList<EpisodeFile> GetSeasonFiles(int seriesId, int seasonNumber)
|
|
|
|
{
|
|
|
|
return _database.Fetch<EpisodeFile>("WHERE SeriesId= @0 AND SeasonNumber = @1", seriesId, seasonNumber);
|
2011-02-24 01:40:11 +01:00
|
|
|
}
|
|
|
|
|
2011-06-03 05:04:39 +02:00
|
|
|
public virtual Tuple<int, int> GetEpisodeFilesCount(int seriesId)
|
|
|
|
{
|
2011-06-15 01:39:09 +02:00
|
|
|
var allEpisodes = _episodeProvider.GetEpisodeBySeries(seriesId).ToList();
|
2011-06-03 05:04:39 +02:00
|
|
|
|
2011-06-23 08:56:17 +02:00
|
|
|
var episodeTotal = allEpisodes.Where(e => !e.Ignored && e.AirDate != null && e.AirDate <= DateTime.Today).ToList();
|
2011-06-15 01:39:09 +02:00
|
|
|
var avilableEpisodes = episodeTotal.Where(e => e.EpisodeFileId > 0).ToList();
|
2011-06-03 05:04:39 +02:00
|
|
|
|
2011-06-15 01:39:09 +02:00
|
|
|
return new Tuple<int, int>(avilableEpisodes.Count, episodeTotal.Count);
|
2011-06-03 05:04:39 +02:00
|
|
|
}
|
|
|
|
|
2011-06-20 05:04:08 +02:00
|
|
|
public virtual FileInfo CalculateFilePath(Series series, int seasonNumber, string fileName, string extention)
|
2011-06-07 08:29:07 +02:00
|
|
|
{
|
2011-06-20 05:25:04 +02:00
|
|
|
string path = series.Path;
|
2011-06-20 05:04:08 +02:00
|
|
|
if (series.SeasonFolder)
|
2011-06-07 08:29:07 +02:00
|
|
|
{
|
2011-07-08 05:36:02 +02:00
|
|
|
var seasonFolder = _configProvider.SortingSeasonFolderFormat
|
2011-06-22 05:40:24 +02:00
|
|
|
.Replace("%0s", seasonNumber.ToString("00"))
|
|
|
|
.Replace("%s", seasonNumber.ToString());
|
|
|
|
|
|
|
|
path = Path.Combine(path, seasonFolder);
|
2011-06-07 08:29:07 +02:00
|
|
|
}
|
2011-06-07 23:19:11 +02:00
|
|
|
|
2011-06-20 05:04:08 +02:00
|
|
|
path = Path.Combine(path, fileName + extention);
|
2011-06-07 23:19:11 +02:00
|
|
|
|
2011-06-20 05:04:08 +02:00
|
|
|
return new FileInfo(path);
|
2011-06-07 08:29:07 +02:00
|
|
|
}
|
|
|
|
|
2012-01-17 05:05:36 +01:00
|
|
|
public virtual void CleanUpDatabase()
|
2011-06-21 03:49:16 +02:00
|
|
|
{
|
2012-01-17 05:05:36 +01:00
|
|
|
Logger.Trace("Verifying Episode > Episode file relationships.");
|
|
|
|
|
|
|
|
string updateString = "UPDATE Episodes SET EpisodeFileId = 0, GrabDate = NULL, PostDownloadStatus = 0";
|
|
|
|
|
|
|
|
if (_configProvider.AutoIgnorePreviouslyDownloadedEpisodes)
|
|
|
|
{
|
|
|
|
updateString += ", Ignored = 1";
|
|
|
|
}
|
|
|
|
|
|
|
|
var updated = _database.Execute(updateString +
|
|
|
|
@"WHERE EpisodeFileId IN
|
|
|
|
(SELECT Episodes.EpisodeFileId FROM Episodes
|
|
|
|
LEFT OUTER JOIN EpisodeFiles
|
|
|
|
ON Episodes.EpisodeFileId = EpisodeFiles.EpisodeFileId
|
|
|
|
WHERE Episodes.EpisodeFileId > 0 AND EpisodeFiles.EpisodeFileId IS NULL)");
|
2011-06-21 08:34:45 +02:00
|
|
|
|
|
|
|
if (updated > 0)
|
|
|
|
{
|
2011-06-22 08:34:33 +02:00
|
|
|
Logger.Debug("Removed {0} invalid links to episode files.", updated);
|
2011-06-21 08:34:45 +02:00
|
|
|
}
|
|
|
|
|
2011-07-06 09:36:49 +02:00
|
|
|
Logger.Trace("Deleting orphan files.");
|
2011-06-21 08:34:45 +02:00
|
|
|
|
2012-01-17 05:05:36 +01:00
|
|
|
updated = _database.Execute(@"DELETE FROM EpisodeFiles
|
2011-06-21 03:49:16 +02:00
|
|
|
WHERE EpisodeFileId IN
|
|
|
|
(SELECT EpisodeFiles.EpisodeFileId FROM EpisodeFiles
|
|
|
|
LEFT OUTER JOIN Episodes
|
|
|
|
ON EpisodeFiles.EpisodeFileId = Episodes.EpisodeFileId
|
2012-01-17 05:05:36 +01:00
|
|
|
WHERE Episodes.EpisodeFileId IS NULL)");
|
2011-06-21 08:34:45 +02:00
|
|
|
|
|
|
|
if (updated > 0)
|
|
|
|
{
|
2011-07-06 09:36:49 +02:00
|
|
|
Logger.Debug("Removed {0} orphan file(s) from database.S", updated);
|
2011-06-21 08:34:45 +02:00
|
|
|
}
|
2011-06-21 03:49:16 +02:00
|
|
|
}
|
|
|
|
|
2012-01-22 20:25:59 +01:00
|
|
|
public virtual string GetNewFilename(IList<Episode> episodes, string seriesTitle, QualityTypes quality, bool proper)
|
2011-06-07 08:29:07 +02:00
|
|
|
{
|
2012-02-12 23:11:08 +01:00
|
|
|
var sortedEpisodes = episodes.OrderBy(e => e.EpisodeNumber);
|
|
|
|
|
2011-07-08 05:36:02 +02:00
|
|
|
var separatorStyle = EpisodeSortingHelper.GetSeparatorStyle(_configProvider.SortingSeparatorStyle);
|
|
|
|
var numberStyle = EpisodeSortingHelper.GetNumberStyle(_configProvider.SortingNumberStyle);
|
2011-06-07 08:29:07 +02:00
|
|
|
|
2012-02-12 23:11:08 +01:00
|
|
|
string episodeNames = sortedEpisodes.First().Title;
|
2011-06-07 08:29:07 +02:00
|
|
|
|
2011-06-20 05:25:04 +02:00
|
|
|
string result = String.Empty;
|
2011-06-07 08:29:07 +02:00
|
|
|
|
2011-07-08 05:36:02 +02:00
|
|
|
if (_configProvider.SortingIncludeSeriesName)
|
2011-06-07 08:29:07 +02:00
|
|
|
{
|
2011-06-20 04:33:43 +02:00
|
|
|
result += seriesTitle + separatorStyle.Pattern;
|
2011-06-07 08:29:07 +02:00
|
|
|
}
|
|
|
|
|
2012-02-12 23:11:08 +01:00
|
|
|
result += numberStyle.Pattern.Replace("%0e", String.Format("{0:00}", sortedEpisodes.First().EpisodeNumber));
|
2011-06-07 08:29:07 +02:00
|
|
|
|
2011-06-20 03:59:31 +02:00
|
|
|
if (episodes.Count > 1)
|
2011-06-07 08:29:07 +02:00
|
|
|
{
|
2011-07-08 05:36:02 +02:00
|
|
|
var multiEpisodeStyle = EpisodeSortingHelper.GetMultiEpisodeStyle(_configProvider.SortingMultiEpisodeStyle);
|
2011-06-07 08:29:07 +02:00
|
|
|
|
2012-02-12 23:11:08 +01:00
|
|
|
foreach (var episode in sortedEpisodes.Skip(1))
|
2011-06-07 08:29:07 +02:00
|
|
|
{
|
2011-06-20 03:59:31 +02:00
|
|
|
if (multiEpisodeStyle.Name == "Duplicate")
|
|
|
|
{
|
|
|
|
result += separatorStyle.Pattern + numberStyle.Pattern;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
result += multiEpisodeStyle.Pattern;
|
|
|
|
}
|
2011-06-07 08:29:07 +02:00
|
|
|
|
2011-06-20 03:59:31 +02:00
|
|
|
result = result.Replace("%0e", String.Format("{0:00}", episode.EpisodeNumber));
|
|
|
|
episodeNames += String.Format(" + {0}", episode.Title);
|
|
|
|
}
|
2011-06-07 08:29:07 +02:00
|
|
|
}
|
|
|
|
|
2011-06-20 03:59:31 +02:00
|
|
|
result = result
|
|
|
|
.Replace("%s", String.Format("{0}", episodes.First().SeasonNumber))
|
|
|
|
.Replace("%0s", String.Format("{0:00}", episodes.First().SeasonNumber))
|
|
|
|
.Replace("%x", numberStyle.EpisodeSeparator)
|
|
|
|
.Replace("%p", separatorStyle.Pattern);
|
2011-06-07 08:29:07 +02:00
|
|
|
|
2011-07-08 05:57:44 +02:00
|
|
|
if (_configProvider.SortingIncludeEpisodeTitle)
|
2011-06-07 08:29:07 +02:00
|
|
|
{
|
|
|
|
episodeNames = episodeNames.TrimEnd(' ', '+');
|
2011-06-20 03:59:31 +02:00
|
|
|
result += separatorStyle.Pattern + episodeNames;
|
2011-06-07 08:29:07 +02:00
|
|
|
}
|
|
|
|
|
2011-07-08 05:36:02 +02:00
|
|
|
if (_configProvider.SortingAppendQuality)
|
2012-01-22 20:25:59 +01:00
|
|
|
{
|
2011-06-20 03:59:31 +02:00
|
|
|
result += String.Format(" [{0}]", quality);
|
2011-06-07 08:29:07 +02:00
|
|
|
|
2012-01-22 20:25:59 +01:00
|
|
|
if (proper)
|
|
|
|
result += " [Proper]";
|
|
|
|
}
|
|
|
|
|
2011-07-08 05:36:02 +02:00
|
|
|
if (_configProvider.SortingReplaceSpaces)
|
2011-06-20 03:59:31 +02:00
|
|
|
result = result.Replace(' ', '.');
|
2011-06-07 08:29:07 +02:00
|
|
|
|
2011-07-06 09:36:49 +02:00
|
|
|
Logger.Trace("New File Name is: [{0}]", result.Trim());
|
2011-06-21 01:46:54 +02:00
|
|
|
return CleanFilename(result.Trim());
|
|
|
|
}
|
|
|
|
|
|
|
|
public static string CleanFilename(string name)
|
|
|
|
{
|
|
|
|
string result = name;
|
|
|
|
string[] badCharacters = { "\\", "/", "<", ">", "?", "*", ":", "|", "\"" };
|
|
|
|
string[] goodCharacters = { "+", "+", "{", "}", "!", "@", "-", "#", "`" };
|
|
|
|
|
|
|
|
for (int i = 0; i < badCharacters.Length; i++)
|
|
|
|
result = result.Replace(badCharacters[i], goodCharacters[i]);
|
|
|
|
|
2011-06-20 03:59:31 +02:00
|
|
|
return result.Trim();
|
2011-06-07 08:29:07 +02:00
|
|
|
}
|
2010-10-21 03:49:23 +02:00
|
|
|
}
|
2011-04-10 04:44:01 +02:00
|
|
|
}
|