< Summary - Jellyfin

Information
Class: Emby.Naming.TV.SeasonPathParser
Assembly: Emby.Naming
File(s): /srv/git/jellyfin/Emby.Naming/TV/SeasonPathParser.cs
Line coverage
62%
Covered lines: 35
Uncovered lines: 21
Coverable lines: 56
Total lines: 177
Line coverage: 62.5%
Branch coverage
57%
Covered branches: 23
Total branches: 40
Branch coverage: 57.5%
Method coverage

Feature is only available for sponsors

Upgrade to PRO version

Coverage history

Coverage history 0 25 50 75 100

Metrics

MethodBranch coverage Crap Score Cyclomatic complexity Line coverage
Parse(...)75%44100%
GetSeasonNumberFromPath(...)100%1818100%
CheckMatch(...)100%22100%
GetSeasonNumberFromPathSubstring(...)0%272160%

File(s)

/srv/git/jellyfin/Emby.Naming/TV/SeasonPathParser.cs

#LineLine coverage
 1using System;
 2using System.Globalization;
 3using System.IO;
 4using System.Text.RegularExpressions;
 5
 6namespace Emby.Naming.TV
 7{
 8    /// <summary>
 9    /// Class to parse season paths.
 10    /// </summary>
 11    public static partial class SeasonPathParser
 12    {
 13        [GeneratedRegex(@"^\s*((?<seasonnumber>(?>\d+))(?:st|nd|rd|th|\.)*(?!\s*[Ee]\d+))\s*(?:[[시즌]*|[シーズン]*|[sS](?:eas
 14        private static partial Regex ProcessPre();
 15
 16        [GeneratedRegex(@"^\s*(?:[[시즌]*|[シーズン]*|[sS](?:eason|æson|aison|taffel|eries|tagione|äsong|eizoen|easong|ezon|ez
 17        private static partial Regex ProcessPost();
 18
 19        /// <summary>
 20        /// Attempts to parse season number from path.
 21        /// </summary>
 22        /// <param name="path">Path to season.</param>
 23        /// <param name="parentPath">Folder name of the parent.</param>
 24        /// <param name="supportSpecialAliases">Support special aliases when parsing.</param>
 25        /// <param name="supportNumericSeasonFolders">Support numeric season folders when parsing.</param>
 26        /// <returns>Returns <see cref="SeasonPathParserResult"/> object.</returns>
 27        public static SeasonPathParserResult Parse(string path, string? parentPath, bool supportSpecialAliases, bool sup
 28        {
 4229            var result = new SeasonPathParserResult();
 4230            var parentFolderName = parentPath is null ? null : new DirectoryInfo(parentPath).Name;
 31
 4232            var (seasonNumber, isSeasonFolder) = GetSeasonNumberFromPath(path, parentFolderName, supportSpecialAliases, 
 33
 4234            result.SeasonNumber = seasonNumber;
 35
 4236            if (result.SeasonNumber.HasValue)
 37            {
 3738                result.Success = true;
 3739                result.IsSeasonFolder = isSeasonFolder;
 40            }
 41
 4242            return result;
 43        }
 44
 45        /// <summary>
 46        /// Gets the season number from path.
 47        /// </summary>
 48        /// <param name="path">The path.</param>
 49        /// <param name="parentFolderName">The parent folder name.</param>
 50        /// <param name="supportSpecialAliases">if set to <c>true</c> [support special aliases].</param>
 51        /// <param name="supportNumericSeasonFolders">if set to <c>true</c> [support numeric season folders].</param>
 52        /// <returns>System.Nullable{System.Int32}.</returns>
 53        private static (int? SeasonNumber, bool IsSeasonFolder) GetSeasonNumberFromPath(
 54            string path,
 55            string? parentFolderName,
 56            bool supportSpecialAliases,
 57            bool supportNumericSeasonFolders)
 58        {
 4259            string filename = Path.GetFileName(path);
 4260            filename = Regex.Replace(filename, "[ ._-]", string.Empty);
 61
 4262            if (parentFolderName is not null)
 63            {
 4264                parentFolderName = Regex.Replace(parentFolderName, "[ ._-]", string.Empty);
 4265                filename = filename.Replace(parentFolderName, string.Empty, StringComparison.OrdinalIgnoreCase);
 66            }
 67
 4268            if (supportSpecialAliases)
 69            {
 4270                if (string.Equals(filename, "specials", StringComparison.OrdinalIgnoreCase))
 71                {
 172                    return (0, true);
 73                }
 74
 4175                if (string.Equals(filename, "extras", StringComparison.OrdinalIgnoreCase))
 76                {
 177                    return (0, true);
 78                }
 79            }
 80
 4081            if (supportNumericSeasonFolders)
 82            {
 4083                if (int.TryParse(filename, NumberStyles.Integer, CultureInfo.InvariantCulture, out var val))
 84                {
 185                    return (val, true);
 86                }
 87            }
 88
 3989            if (filename.StartsWith('s'))
 90            {
 691                var testFilename = filename.AsSpan()[1..];
 92
 693                if (int.TryParse(testFilename, NumberStyles.Integer, CultureInfo.InvariantCulture, out var val))
 94                {
 195                    return (val, true);
 96                }
 97            }
 98
 3899            var preMatch = ProcessPre().Match(filename);
 38100            if (preMatch.Success)
 101            {
 1102                return CheckMatch(preMatch);
 103            }
 104            else
 105            {
 37106                var postMatch = ProcessPost().Match(filename);
 37107                return CheckMatch(postMatch);
 108            }
 109        }
 110
 111        private static (int? SeasonNumber, bool IsSeasonFolder) CheckMatch(Match match)
 112        {
 38113            var numberString = match.Groups["seasonnumber"];
 38114            if (numberString.Success)
 115            {
 33116                var seasonNumber = int.Parse(numberString.Value, CultureInfo.InvariantCulture);
 33117                return (seasonNumber, true);
 118            }
 119
 5120            return (null, false);
 121        }
 122
 123        /// <summary>
 124        /// Extracts the season number from the second half of the Season folder name (everything after "Season", or "St
 125        /// </summary>
 126        /// <param name="path">The path.</param>
 127        /// <returns>System.Nullable{System.Int32}.</returns>
 128        private static (int? SeasonNumber, bool IsSeasonFolder) GetSeasonNumberFromPathSubstring(ReadOnlySpan<char> path
 129        {
 0130            var numericStart = -1;
 0131            var length = 0;
 132
 0133            var hasOpenParenthesis = false;
 0134            var isSeasonFolder = true;
 135
 136            // Find out where the numbers start, and then keep going until they end
 0137            for (var i = 0; i < path.Length; i++)
 138            {
 0139                if (char.IsNumber(path[i]))
 140                {
 0141                    if (!hasOpenParenthesis)
 142                    {
 0143                        if (numericStart == -1)
 144                        {
 0145                            numericStart = i;
 146                        }
 147
 0148                        length++;
 149                    }
 150                }
 0151                else if (numericStart != -1)
 152                {
 153                    // There's other stuff after the season number, e.g. episode number
 0154                    isSeasonFolder = false;
 0155                    break;
 156                }
 157
 0158                var currentChar = path[i];
 0159                if (currentChar == '(')
 160                {
 0161                    hasOpenParenthesis = true;
 162                }
 0163                else if (currentChar == ')')
 164                {
 0165                    hasOpenParenthesis = false;
 166                }
 167            }
 168
 0169            if (numericStart == -1)
 170            {
 0171                return (null, isSeasonFolder);
 172            }
 173
 0174            return (int.Parse(path.Slice(numericStart, length), provider: CultureInfo.InvariantCulture), isSeasonFolder)
 175        }
 176    }
 177}