| | 1 | | using System; |
| | 2 | | using System.Collections.Generic; |
| | 3 | | using System.Globalization; |
| | 4 | | using System.Linq; |
| | 5 | | using System.Xml; |
| | 6 | | using Jellyfin.Data.Enums; |
| | 7 | | using MediaBrowser.Controller.Entities; |
| | 8 | |
|
| | 9 | | namespace MediaBrowser.Controller.Extensions; |
| | 10 | |
|
| | 11 | | /// <summary> |
| | 12 | | /// Provides extension methods for <see cref="XmlReader"/> to parse <see cref="BaseItem"/>'s. |
| | 13 | | /// </summary> |
| | 14 | | public static class XmlReaderExtensions |
| | 15 | | { |
| | 16 | | /// <summary> |
| | 17 | | /// Reads a trimmed string from the current node. |
| | 18 | | /// </summary> |
| | 19 | | /// <param name="reader">The <see cref="XmlReader"/>.</param> |
| | 20 | | /// <returns>The trimmed content.</returns> |
| | 21 | | public static string ReadNormalizedString(this XmlReader reader) |
| | 22 | | { |
| 237 | 23 | | ArgumentNullException.ThrowIfNull(reader); |
| | 24 | |
|
| 237 | 25 | | return reader.ReadElementContentAsString().Trim(); |
| | 26 | | } |
| | 27 | |
|
| | 28 | | /// <summary> |
| | 29 | | /// Reads an int from the current node. |
| | 30 | | /// </summary> |
| | 31 | | /// <param name="reader">The <see cref="XmlReader"/>.</param> |
| | 32 | | /// <param name="value">The parsed <c>int</c>.</param> |
| | 33 | | /// <returns>A value indicating whether the parsing succeeded.</returns> |
| | 34 | | public static bool TryReadInt(this XmlReader reader, out int value) |
| | 35 | | { |
| 88 | 36 | | ArgumentNullException.ThrowIfNull(reader); |
| | 37 | |
|
| 88 | 38 | | return int.TryParse(reader.ReadElementContentAsString(), CultureInfo.InvariantCulture, out value); |
| | 39 | | } |
| | 40 | |
|
| | 41 | | /// <summary> |
| | 42 | | /// Parses a <see cref="DateTime"/> from the current node. |
| | 43 | | /// </summary> |
| | 44 | | /// <param name="reader">The <see cref="XmlReader"/>.</param> |
| | 45 | | /// <param name="value">The parsed <see cref="DateTime"/>.</param> |
| | 46 | | /// <returns>A value indicating whether the parsing succeeded.</returns> |
| | 47 | | public static bool TryReadDateTime(this XmlReader reader, out DateTime value) |
| | 48 | | { |
| 9 | 49 | | ArgumentNullException.ThrowIfNull(reader); |
| | 50 | |
|
| 9 | 51 | | return DateTime.TryParse( |
| 9 | 52 | | reader.ReadElementContentAsString(), |
| 9 | 53 | | CultureInfo.InvariantCulture, |
| 9 | 54 | | DateTimeStyles.AssumeUniversal | DateTimeStyles.AdjustToUniversal, |
| 9 | 55 | | out value); |
| | 56 | | } |
| | 57 | |
|
| | 58 | | /// <summary> |
| | 59 | | /// Parses a <see cref="DateTime"/> from the current node. |
| | 60 | | /// </summary> |
| | 61 | | /// <param name="reader">The <see cref="XmlReader"/>.</param> |
| | 62 | | /// <param name="formatString">The date format string.</param> |
| | 63 | | /// <param name="value">The parsed <see cref="DateTime"/>.</param> |
| | 64 | | /// <returns>A value indicating whether the parsing succeeded.</returns> |
| | 65 | | public static bool TryReadDateTimeExact(this XmlReader reader, string formatString, out DateTime value) |
| | 66 | | { |
| 20 | 67 | | ArgumentNullException.ThrowIfNull(reader); |
| 20 | 68 | | ArgumentNullException.ThrowIfNull(formatString); |
| | 69 | |
|
| 20 | 70 | | return DateTime.TryParseExact( |
| 20 | 71 | | reader.ReadElementContentAsString(), |
| 20 | 72 | | formatString, |
| 20 | 73 | | CultureInfo.InvariantCulture, |
| 20 | 74 | | DateTimeStyles.AssumeUniversal | DateTimeStyles.AdjustToUniversal, |
| 20 | 75 | | out value); |
| | 76 | | } |
| | 77 | |
|
| | 78 | | /// <summary> |
| | 79 | | /// Parses a <see cref="PersonInfo"/> from the xml node. |
| | 80 | | /// </summary> |
| | 81 | | /// <param name="reader">The <see cref="XmlReader"/>.</param> |
| | 82 | | /// <returns>A <see cref="PersonInfo"/>, or <c>null</c> if none is found.</returns> |
| | 83 | | public static PersonInfo? GetPersonFromXmlNode(this XmlReader reader) |
| | 84 | | { |
| 55 | 85 | | ArgumentNullException.ThrowIfNull(reader); |
| | 86 | |
|
| 55 | 87 | | if (reader.IsEmptyElement) |
| | 88 | | { |
| 0 | 89 | | reader.Read(); |
| 0 | 90 | | return null; |
| | 91 | | } |
| | 92 | |
|
| 55 | 93 | | var name = string.Empty; |
| 55 | 94 | | var type = PersonKind.Actor; // If type is not specified assume actor |
| 55 | 95 | | var role = string.Empty; |
| 55 | 96 | | int? sortOrder = null; |
| 55 | 97 | | string? imageUrl = null; |
| | 98 | |
|
| 55 | 99 | | using var subtree = reader.ReadSubtree(); |
| 55 | 100 | | subtree.MoveToContent(); |
| 55 | 101 | | subtree.Read(); |
| | 102 | |
|
| 623 | 103 | | while (subtree is { EOF: false, ReadState: ReadState.Interactive }) |
| | 104 | | { |
| 568 | 105 | | if (subtree.NodeType != XmlNodeType.Element) |
| | 106 | | { |
| 339 | 107 | | subtree.Read(); |
| 339 | 108 | | continue; |
| | 109 | | } |
| | 110 | |
|
| 229 | 111 | | switch (subtree.Name) |
| | 112 | | { |
| | 113 | | case "name": |
| | 114 | | case "Name": |
| 55 | 115 | | name = subtree.ReadNormalizedString(); |
| 55 | 116 | | break; |
| | 117 | | case "role": |
| | 118 | | case "Role": |
| 54 | 119 | | role = subtree.ReadNormalizedString(); |
| 54 | 120 | | break; |
| | 121 | | case "type": |
| | 122 | | case "Type": |
| 11 | 123 | | Enum.TryParse(subtree.ReadElementContentAsString(), true, out type); |
| 11 | 124 | | break; |
| | 125 | | case "order": |
| | 126 | | case "sortorder": |
| | 127 | | case "SortOrder": |
| 55 | 128 | | if (subtree.TryReadInt(out var sortOrderVal)) |
| | 129 | | { |
| 55 | 130 | | sortOrder = sortOrderVal; |
| | 131 | | } |
| | 132 | |
|
| 55 | 133 | | break; |
| | 134 | | case "thumb": |
| 54 | 135 | | imageUrl = subtree.ReadNormalizedString(); |
| 54 | 136 | | break; |
| | 137 | | default: |
| 0 | 138 | | subtree.Skip(); |
| | 139 | | break; |
| | 140 | | } |
| | 141 | | } |
| | 142 | |
|
| 55 | 143 | | if (string.IsNullOrWhiteSpace(name)) |
| | 144 | | { |
| 0 | 145 | | return null; |
| | 146 | | } |
| | 147 | |
|
| 55 | 148 | | return new PersonInfo |
| 55 | 149 | | { |
| 55 | 150 | | Name = name, |
| 55 | 151 | | Role = role, |
| 55 | 152 | | Type = type, |
| 55 | 153 | | SortOrder = sortOrder, |
| 55 | 154 | | ImageUrl = imageUrl |
| 55 | 155 | | }; |
| 55 | 156 | | } |
| | 157 | |
|
| | 158 | | /// <summary> |
| | 159 | | /// Used to split names of comma or pipe delimited genres and people. |
| | 160 | | /// </summary> |
| | 161 | | /// <param name="reader">The <see cref="XmlReader"/>.</param> |
| | 162 | | /// <returns>IEnumerable{System.String}.</returns> |
| | 163 | | public static IEnumerable<string> GetStringArray(this XmlReader reader) |
| | 164 | | { |
| | 165 | | ArgumentNullException.ThrowIfNull(reader); |
| | 166 | | var value = reader.ReadElementContentAsString(); |
| | 167 | |
|
| | 168 | | // Only split by comma if there is no pipe in the string |
| | 169 | | // We have to be careful to not split names like Matthew, Jr. |
| | 170 | | var separator = !value.Contains('|', StringComparison.Ordinal) |
| | 171 | | && !value.Contains(';', StringComparison.Ordinal) |
| | 172 | | ? new[] { ',' } |
| | 173 | | : new[] { '|', ';' }; |
| | 174 | |
|
| | 175 | | foreach (var part in value.Trim().Trim(separator).Split(separator)) |
| | 176 | | { |
| | 177 | | if (!string.IsNullOrWhiteSpace(part)) |
| | 178 | | { |
| | 179 | | yield return part.Trim(); |
| | 180 | | } |
| | 181 | | } |
| | 182 | | } |
| | 183 | |
|
| | 184 | | /// <summary> |
| | 185 | | /// Parses a <see cref="PersonInfo"/> array from the xml node. |
| | 186 | | /// </summary> |
| | 187 | | /// <param name="reader">The <see cref="XmlReader"/>.</param> |
| | 188 | | /// <param name="personKind">The <see cref="PersonKind"/>.</param> |
| | 189 | | /// <returns>The <see cref="IEnumerable{PersonInfo}"/>.</returns> |
| | 190 | | public static IEnumerable<PersonInfo> GetPersonArray(this XmlReader reader, PersonKind personKind) |
| 4 | 191 | | => reader.GetStringArray() |
| 4 | 192 | | .Select(part => new PersonInfo { Name = part, Type = personKind }); |
| | 193 | | } |