diff --git a/src/NzbDrone.Common/Extensions/RegexExtensions.cs b/src/NzbDrone.Common/Extensions/RegexExtensions.cs
new file mode 100644
index 000000000..5e99c8400
--- /dev/null
+++ b/src/NzbDrone.Common/Extensions/RegexExtensions.cs
@@ -0,0 +1,13 @@
+using System;
+using System.Text.RegularExpressions;
+
+namespace NzbDrone.Common.Extensions
+{
+ public static class RegexExtensions
+ {
+ public static int EndIndex(this Capture regexMatch)
+ {
+ return regexMatch.Index + regexMatch.Length;
+ }
+ }
+}
diff --git a/src/NzbDrone.Common/NzbDrone.Common.csproj b/src/NzbDrone.Common/NzbDrone.Common.csproj
index fad4a7e93..7378e855d 100644
--- a/src/NzbDrone.Common/NzbDrone.Common.csproj
+++ b/src/NzbDrone.Common/NzbDrone.Common.csproj
@@ -102,6 +102,7 @@
+
diff --git a/src/NzbDrone.Core.Test/ParserTests/IsPossibleSpecialEpisodeFixture.cs b/src/NzbDrone.Core.Test/ParserTests/IsPossibleSpecialEpisodeFixture.cs
index 0db07c03f..747884b07 100644
--- a/src/NzbDrone.Core.Test/ParserTests/IsPossibleSpecialEpisodeFixture.cs
+++ b/src/NzbDrone.Core.Test/ParserTests/IsPossibleSpecialEpisodeFixture.cs
@@ -46,5 +46,11 @@ namespace NzbDrone.Core.Test.ParserTests
{
Parser.Parser.ParseTitle(title).IsPossibleSpecialEpisode.Should().BeTrue();
}
+
+ [TestCase("Big.Special.Show.S05.HDTV.x264-2HD")]
+ public void IsPossibleSpecialEpisode_should_be_false_for_Special_in_title(string title)
+ {
+ Parser.Parser.ParseTitle(title).IsPossibleSpecialEpisode.Should().BeFalse();
+ }
}
}
diff --git a/src/NzbDrone.Core/Parser/Model/ParsedEpisodeInfo.cs b/src/NzbDrone.Core/Parser/Model/ParsedEpisodeInfo.cs
index 7e5977afa..5b2128160 100644
--- a/src/NzbDrone.Core/Parser/Model/ParsedEpisodeInfo.cs
+++ b/src/NzbDrone.Core/Parser/Model/ParsedEpisodeInfo.cs
@@ -24,6 +24,7 @@ namespace NzbDrone.Core.Parser.Model
public string ReleaseGroup { get; set; }
public string ReleaseHash { get; set; }
public int SeasonPart { get; set; }
+ public string ReleaseTokens { get; set; }
public ParsedEpisodeInfo()
{
@@ -104,6 +105,13 @@ namespace NzbDrone.Core.Parser.Model
{
episodeString = string.Format("{0}", string.Join("-", AbsoluteEpisodeNumbers.Select(c => c.ToString("000"))));
}
+ else if (Special)
+ {
+ if (SeasonNumber != 0)
+ episodeString = string.Format("[Unknown Season {0:00} Special]", SeasonNumber);
+ else
+ episodeString = "[Unknown Special]";
+ }
return string.Format("{0} - {1} {2}", SeriesTitle, episodeString, Quality);
}
diff --git a/src/NzbDrone.Core/Parser/Parser.cs b/src/NzbDrone.Core/Parser/Parser.cs
index 7fac3c646..faba1172e 100644
--- a/src/NzbDrone.Core/Parser/Parser.cs
+++ b/src/NzbDrone.Core/Parser/Parser.cs
@@ -450,7 +450,7 @@ namespace NzbDrone.Core.Parser
if (result != null)
{
- if (result.FullSeason && title.ContainsIgnoreCase("Special"))
+ if (result.FullSeason && result.ReleaseTokens.ContainsIgnoreCase("Special"))
{
result.FullSeason = false;
result.Special = true;
@@ -643,6 +643,8 @@ namespace NzbDrone.Core.Parser
int airYear;
int.TryParse(matchCollection[0].Groups["airyear"].Value, out airYear);
+ int lastSeasonEpisodeStringIndex = matchCollection[0].Groups["title"].EndIndex();
+
ParsedEpisodeInfo result;
if (airYear < 1900)
@@ -653,7 +655,11 @@ namespace NzbDrone.Core.Parser
{
int parsedSeason;
if (int.TryParse(seasonCapture.Value, out parsedSeason))
+ {
seasons.Add(parsedSeason);
+
+ lastSeasonEpisodeStringIndex = Math.Max(lastSeasonEpisodeStringIndex, seasonCapture.EndIndex());
+ }
}
//If no season was found it should be treated as a mini series and season 1
@@ -688,6 +694,8 @@ namespace NzbDrone.Core.Parser
var count = last - first + 1;
result.EpisodeNumbers = Enumerable.Range(first, count).ToArray();
+
+ lastSeasonEpisodeStringIndex = Math.Max(lastSeasonEpisodeStringIndex, episodeCaptures.Last().EndIndex());
}
if (absoluteEpisodeCaptures.Any())
@@ -707,6 +715,8 @@ namespace NzbDrone.Core.Parser
result.SpecialAbsoluteEpisodeNumbers = new decimal[] { first };
result.Special = true;
+
+ lastSeasonEpisodeStringIndex = Math.Max(lastSeasonEpisodeStringIndex, absoluteEpisodeCaptures.First().EndIndex());
}
else
{
@@ -717,6 +727,8 @@ namespace NzbDrone.Core.Parser
{
result.Special = true;
}
+
+ lastSeasonEpisodeStringIndex = Math.Max(lastSeasonEpisodeStringIndex, absoluteEpisodeCaptures.Last().EndIndex());
}
}
@@ -782,6 +794,10 @@ namespace NzbDrone.Core.Parser
throw new InvalidDateException("Invalid date found: {0}", airDate);
}
+ lastSeasonEpisodeStringIndex = Math.Max(lastSeasonEpisodeStringIndex, matchCollection[0].Groups["airyear"].EndIndex());
+ lastSeasonEpisodeStringIndex = Math.Max(lastSeasonEpisodeStringIndex, matchCollection[0].Groups["airmonth"].EndIndex());
+ lastSeasonEpisodeStringIndex = Math.Max(lastSeasonEpisodeStringIndex, matchCollection[0].Groups["airday"].EndIndex());
+
result = new ParsedEpisodeInfo
{
ReleaseTitle = releaseTitle,
@@ -789,6 +805,11 @@ namespace NzbDrone.Core.Parser
};
}
+ if (lastSeasonEpisodeStringIndex != releaseTitle.Length)
+ result.ReleaseTokens = releaseTitle.Substring(lastSeasonEpisodeStringIndex);
+ else
+ result.ReleaseTokens = releaseTitle;
+
result.SeriesTitle = seriesName;
result.SeriesTitleInfo = GetSeriesTitleInfo(result.SeriesTitle);