SeasonPathParser.cs 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181
  1. using System;
  2. using System.Globalization;
  3. using System.IO;
  4. using System.Linq;
  5. using Emby.Naming.Common;
  6. using Emby.Naming.Extensions;
  7. namespace Emby.Naming.TV
  8. {
  9. public class SeasonPathParser
  10. {
  11. public SeasonPathParserResult Parse(string path, bool supportSpecialAliases, bool supportNumericSeasonFolders)
  12. {
  13. var result = new SeasonPathParserResult();
  14. var seasonNumberInfo = GetSeasonNumberFromPath(path, supportSpecialAliases, supportNumericSeasonFolders);
  15. result.SeasonNumber = seasonNumberInfo.seasonNumber;
  16. if (result.SeasonNumber.HasValue)
  17. {
  18. result.Success = true;
  19. result.IsSeasonFolder = seasonNumberInfo.isSeasonFolder;
  20. }
  21. return result;
  22. }
  23. /// <summary>
  24. /// A season folder must contain one of these somewhere in the name
  25. /// </summary>
  26. private static readonly string[] _seasonFolderNames =
  27. {
  28. "season",
  29. "sæson",
  30. "temporada",
  31. "saison",
  32. "staffel",
  33. "series",
  34. "сезон",
  35. "stagione"
  36. };
  37. /// <summary>
  38. /// Gets the season number from path.
  39. /// </summary>
  40. /// <param name="path">The path.</param>
  41. /// <param name="supportSpecialAliases">if set to <c>true</c> [support special aliases].</param>
  42. /// <param name="supportNumericSeasonFolders">if set to <c>true</c> [support numeric season folders].</param>
  43. /// <returns>System.Nullable{System.Int32}.</returns>
  44. private static (int? seasonNumber, bool isSeasonFolder) GetSeasonNumberFromPath(
  45. string path,
  46. bool supportSpecialAliases,
  47. bool supportNumericSeasonFolders)
  48. {
  49. var filename = Path.GetFileName(path) ?? string.Empty;
  50. if (supportSpecialAliases)
  51. {
  52. if (string.Equals(filename, "specials", StringComparison.OrdinalIgnoreCase))
  53. {
  54. return (0, true);
  55. }
  56. if (string.Equals(filename, "extras", StringComparison.OrdinalIgnoreCase))
  57. {
  58. return (0, true);
  59. }
  60. }
  61. if (supportNumericSeasonFolders)
  62. {
  63. if (int.TryParse(filename, NumberStyles.Integer, CultureInfo.InvariantCulture, out var val))
  64. {
  65. return (val, true);
  66. }
  67. }
  68. if (filename.StartsWith("s", StringComparison.OrdinalIgnoreCase))
  69. {
  70. var testFilename = filename.Substring(1);
  71. if (int.TryParse(testFilename, NumberStyles.Integer, CultureInfo.InvariantCulture, out var val))
  72. {
  73. return (val, true);
  74. }
  75. }
  76. // Look for one of the season folder names
  77. foreach (var name in _seasonFolderNames)
  78. {
  79. var index = filename.IndexOf(name, StringComparison.OrdinalIgnoreCase);
  80. if (index != -1)
  81. {
  82. var result = GetSeasonNumberFromPathSubstring(filename.Replace(name, " ", StringComparison.OrdinalIgnoreCase));
  83. if (result.Item1.HasValue)
  84. {
  85. return result;
  86. }
  87. break;
  88. }
  89. }
  90. var parts = filename.Split(new[] { '.', '_', ' ', '-' }, StringSplitOptions.RemoveEmptyEntries);
  91. var resultNumber = parts.Select(GetSeasonNumberFromPart).FirstOrDefault(i => i.HasValue);
  92. return (resultNumber, true);
  93. }
  94. private static int? GetSeasonNumberFromPart(string part)
  95. {
  96. if (part.Length < 2 || !part.StartsWith("s", StringComparison.OrdinalIgnoreCase))
  97. {
  98. return null;
  99. }
  100. part = part.Substring(1);
  101. if (int.TryParse(part, NumberStyles.Integer, CultureInfo.InvariantCulture, out var value))
  102. {
  103. return value;
  104. }
  105. return null;
  106. }
  107. /// <summary>
  108. /// Extracts the season number from the second half of the Season folder name (everything after "Season", or "Staffel")
  109. /// </summary>
  110. /// <param name="path">The path.</param>
  111. /// <returns>System.Nullable{System.Int32}.</returns>
  112. private static (int? seasonNumber, bool isSeasonFolder) GetSeasonNumberFromPathSubstring(string path)
  113. {
  114. var numericStart = -1;
  115. var length = 0;
  116. var hasOpenParenth = false;
  117. var isSeasonFolder = true;
  118. // Find out where the numbers start, and then keep going until they end
  119. for (var i = 0; i < path.Length; i++)
  120. {
  121. if (char.IsNumber(path, i))
  122. {
  123. if (!hasOpenParenth)
  124. {
  125. if (numericStart == -1)
  126. {
  127. numericStart = i;
  128. }
  129. length++;
  130. }
  131. }
  132. else if (numericStart != -1)
  133. {
  134. // There's other stuff after the season number, e.g. episode number
  135. isSeasonFolder = false;
  136. break;
  137. }
  138. var currentChar = path[i];
  139. if (currentChar.Equals('('))
  140. {
  141. hasOpenParenth = true;
  142. }
  143. else if (currentChar.Equals(')'))
  144. {
  145. hasOpenParenth = false;
  146. }
  147. }
  148. if (numericStart == -1)
  149. {
  150. return (null, isSeasonFolder);
  151. }
  152. return (int.Parse(path.Substring(numericStart, length), CultureInfo.InvariantCulture), isSeasonFolder);
  153. }
  154. }
  155. }