SeasonPathParser.cs 5.9 KB

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