< Summary - Jellyfin

Information
Class: Jellyfin.Extensions.StringExtensions
Assembly: Jellyfin.Extensions
File(s): /srv/git/jellyfin/src/Jellyfin.Extensions/StringExtensions.cs
Line coverage
60%
Covered lines: 23
Uncovered lines: 15
Coverable lines: 38
Total lines: 139
Line coverage: 60.5%
Branch coverage
88%
Covered branches: 16
Total branches: 18
Branch coverage: 88.8%
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
.cctor()100%210%
RemoveDiacritics(...)100%11100%
HasDiacritics(...)100%22100%
Count(...)100%44100%
LeftPart(...)100%44100%
RightPart(...)100%66100%
Transliterated(...)0%620%
Trimmed(...)100%11100%

File(s)

/srv/git/jellyfin/src/Jellyfin.Extensions/StringExtensions.cs

#LineLine coverage
 1using System;
 2using System.Collections.Generic;
 3using System.Linq;
 4using System.Text.RegularExpressions;
 5using ICU4N.Text;
 6
 7namespace Jellyfin.Extensions
 8{
 9    /// <summary>
 10    /// Provides extensions methods for <see cref="string" />.
 11    /// </summary>
 12    public static partial class StringExtensions
 13    {
 014        private static readonly Lazy<string> _transliteratorId = new(() =>
 015            Environment.GetEnvironmentVariable("JELLYFIN_TRANSLITERATOR_ID")
 016            ?? "Any-Latin; Latin-Ascii; Lower; NFD; [:Nonspacing Mark:] Remove; [:Punctuation:] Remove;");
 17
 018        private static readonly Lazy<Transliterator?> _transliterator = new(() =>
 019        {
 020            try
 021            {
 022                return Transliterator.GetInstance(_transliteratorId.Value);
 023            }
 024            catch (ArgumentException)
 025            {
 026                return null;
 027            }
 028        });
 29
 30        // Matches non-conforming unicode chars
 31        // https://mnaoumov.wordpress.com/2014/06/14/stripping-invalid-characters-from-utf-16-strings/
 32
 33        [GeneratedRegex("([\ud800-\udbff](?![\udc00-\udfff]))|((?<![\ud800-\udbff])[\udc00-\udfff])|(�)")]
 34        private static partial Regex NonConformingUnicodeRegex();
 35
 36        /// <summary>
 37        /// Removes the diacritics character from the strings.
 38        /// </summary>
 39        /// <param name="text">The string to act on.</param>
 40        /// <returns>The string without diacritics character.</returns>
 41        public static string RemoveDiacritics(this string text)
 20842            => Diacritics.Extensions.StringExtensions.RemoveDiacritics(
 20843                NonConformingUnicodeRegex().Replace(text, string.Empty));
 44
 45        /// <summary>
 46        /// Checks whether or not the specified string has diacritics in it.
 47        /// </summary>
 48        /// <param name="text">The string to check.</param>
 49        /// <returns>True if the string has diacritics, false otherwise.</returns>
 50        public static bool HasDiacritics(this string text)
 1251            => Diacritics.Extensions.StringExtensions.HasDiacritics(text)
 1252                || NonConformingUnicodeRegex().IsMatch(text);
 53
 54        /// <summary>
 55        /// Counts the number of occurrences of [needle] in the string.
 56        /// </summary>
 57        /// <param name="value">The haystack to search in.</param>
 58        /// <param name="needle">The character to search for.</param>
 59        /// <returns>The number of occurrences of the [needle] character.</returns>
 60        public static int Count(this ReadOnlySpan<char> value, char needle)
 61        {
 462            var count = 0;
 463            var length = value.Length;
 10864            for (var i = 0; i < length; i++)
 65            {
 5066                if (value[i] == needle)
 67                {
 668                    count++;
 69                }
 70            }
 71
 472            return count;
 73        }
 74
 75        /// <summary>
 76        /// Returns the part on the left of the <c>needle</c>.
 77        /// </summary>
 78        /// <param name="haystack">The string to seek.</param>
 79        /// <param name="needle">The needle to find.</param>
 80        /// <returns>The part left of the <paramref name="needle" />.</returns>
 81        public static ReadOnlySpan<char> LeftPart(this ReadOnlySpan<char> haystack, char needle)
 82        {
 37783            if (haystack.IsEmpty)
 84            {
 285                return ReadOnlySpan<char>.Empty;
 86            }
 87
 37588            var pos = haystack.IndexOf(needle);
 37589            return pos == -1 ? haystack : haystack[..pos];
 90        }
 91
 92        /// <summary>
 93        /// Returns the part on the right of the <c>needle</c>.
 94        /// </summary>
 95        /// <param name="haystack">The string to seek.</param>
 96        /// <param name="needle">The needle to find.</param>
 97        /// <returns>The part right of the <paramref name="needle" />.</returns>
 98        public static ReadOnlySpan<char> RightPart(this ReadOnlySpan<char> haystack, char needle)
 99        {
 9100            if (haystack.IsEmpty)
 101            {
 1102                return ReadOnlySpan<char>.Empty;
 103            }
 104
 8105            var pos = haystack.LastIndexOf(needle);
 8106            if (pos == -1)
 107            {
 1108                return haystack;
 109            }
 110
 7111            if (pos == haystack.Length - 1)
 112            {
 3113                return ReadOnlySpan<char>.Empty;
 114            }
 115
 4116            return haystack[(pos + 1)..];
 117        }
 118
 119        /// <summary>
 120        /// Returns a transliterated string which only contain ascii characters.
 121        /// </summary>
 122        /// <param name="text">The string to act on.</param>
 123        /// <returns>The transliterated string.</returns>
 124        public static string Transliterated(this string text)
 125        {
 0126            return (_transliterator.Value is null) ? text : _transliterator.Value.Transliterate(text);
 127        }
 128
 129        /// <summary>
 130        /// Ensures all strings are non-null and trimmed of leading an trailing blanks.
 131        /// </summary>
 132        /// <param name="values">The enumerable of strings to trim.</param>
 133        /// <returns>The enumeration of trimmed strings.</returns>
 134        public static IEnumerable<string> Trimmed(this IEnumerable<string> values)
 135        {
 5136            return values.Select(i => (i ?? string.Empty).Trim());
 137        }
 138    }
 139}