| | 1 | | using System; |
| | 2 | | using System.Collections.Generic; |
| | 3 | | using System.Globalization; |
| | 4 | | using System.IO; |
| | 5 | | using System.Linq; |
| | 6 | | using System.Text; |
| | 7 | | using System.Threading; |
| | 8 | | using System.Xml; |
| | 9 | | using Jellyfin.Data.Enums; |
| | 10 | | using Jellyfin.Extensions; |
| | 11 | | using MediaBrowser.Common.Configuration; |
| | 12 | | using MediaBrowser.Common.Providers; |
| | 13 | | using MediaBrowser.Controller.Entities; |
| | 14 | | using MediaBrowser.Controller.Entities.Movies; |
| | 15 | | using MediaBrowser.Controller.Entities.TV; |
| | 16 | | using MediaBrowser.Controller.Extensions; |
| | 17 | | using MediaBrowser.Controller.Library; |
| | 18 | | using MediaBrowser.Controller.Providers; |
| | 19 | | using MediaBrowser.Model.Entities; |
| | 20 | | using MediaBrowser.XbmcMetadata.Configuration; |
| | 21 | | using MediaBrowser.XbmcMetadata.Savers; |
| | 22 | | using Microsoft.Extensions.Logging; |
| | 23 | |
|
| | 24 | | namespace MediaBrowser.XbmcMetadata.Parsers |
| | 25 | | { |
| | 26 | | /// <summary> |
| | 27 | | /// The BaseNfoParser class. |
| | 28 | | /// </summary> |
| | 29 | | /// <typeparam name="T">The type.</typeparam> |
| | 30 | | public class BaseNfoParser<T> |
| | 31 | | where T : BaseItem |
| | 32 | | { |
| | 33 | | private readonly IConfigurationManager _config; |
| | 34 | | private readonly IUserManager _userManager; |
| | 35 | | private readonly IUserDataManager _userDataManager; |
| | 36 | | private readonly IDirectoryService _directoryService; |
| | 37 | | private Dictionary<string, string> _validProviderIds; |
| | 38 | |
|
| | 39 | | /// <summary> |
| | 40 | | /// Initializes a new instance of the <see cref="BaseNfoParser{T}" /> class. |
| | 41 | | /// </summary> |
| | 42 | | /// <param name="logger">Instance of the <see cref="ILogger"/> interface.</param> |
| | 43 | | /// <param name="config">Instance of the <see cref="IConfigurationManager"/> interface.</param> |
| | 44 | | /// <param name="providerManager">Instance of the <see cref="IProviderManager"/> interface.</param> |
| | 45 | | /// <param name="userManager">Instance of the <see cref="IUserManager"/> interface.</param> |
| | 46 | | /// <param name="userDataManager">Instance of the <see cref="IUserDataManager"/> interface.</param> |
| | 47 | | /// <param name="directoryService">Instance of the <see cref="IDirectoryService"/> interface.</param> |
| | 48 | | public BaseNfoParser( |
| | 49 | | ILogger logger, |
| | 50 | | IConfigurationManager config, |
| | 51 | | IProviderManager providerManager, |
| | 52 | | IUserManager userManager, |
| | 53 | | IUserDataManager userDataManager, |
| | 54 | | IDirectoryService directoryService) |
| | 55 | | { |
| 29 | 56 | | Logger = logger; |
| 29 | 57 | | _config = config; |
| 29 | 58 | | ProviderManager = providerManager; |
| 29 | 59 | | _validProviderIds = new Dictionary<string, string>(); |
| 29 | 60 | | _userManager = userManager; |
| 29 | 61 | | _userDataManager = userDataManager; |
| 29 | 62 | | _directoryService = directoryService; |
| 29 | 63 | | } |
| | 64 | |
|
| | 65 | | /// <summary> |
| | 66 | | /// Gets the logger. |
| | 67 | | /// </summary> |
| | 68 | | protected ILogger Logger { get; } |
| | 69 | |
|
| | 70 | | /// <summary> |
| | 71 | | /// Gets the provider manager. |
| | 72 | | /// </summary> |
| | 73 | | protected IProviderManager ProviderManager { get; } |
| | 74 | |
|
| | 75 | | /// <summary> |
| | 76 | | /// Gets a value indicating whether URLs after a closing XML tag are supporrted. |
| | 77 | | /// </summary> |
| 3 | 78 | | protected virtual bool SupportsUrlAfterClosingXmlTag => false; |
| | 79 | |
|
| | 80 | | /// <summary> |
| | 81 | | /// Fetches metadata for an item from one xml file. |
| | 82 | | /// </summary> |
| | 83 | | /// <param name="item">The <see cref="MetadataResult{T}"/>.</param> |
| | 84 | | /// <param name="metadataFile">The metadata file.</param> |
| | 85 | | /// <param name="cancellationToken">The <see cref="CancellationToken"/>.</param> |
| | 86 | | /// <exception cref="ArgumentNullException"><c>item</c> is <c>null</c>.</exception> |
| | 87 | | /// <exception cref="ArgumentException"><c>metadataFile</c> is <c>null</c> or empty.</exception> |
| | 88 | | public void Fetch(MetadataResult<T> item, string metadataFile, CancellationToken cancellationToken) |
| | 89 | | { |
| 29 | 90 | | if (item.Item is null) |
| | 91 | | { |
| 7 | 92 | | throw new ArgumentException("Item can't be null.", nameof(item)); |
| | 93 | | } |
| | 94 | |
|
| 22 | 95 | | ArgumentException.ThrowIfNullOrEmpty(metadataFile); |
| | 96 | |
|
| 15 | 97 | | _validProviderIds = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase); |
| | 98 | |
|
| 15 | 99 | | var idInfos = ProviderManager.GetExternalIdInfos(item.Item); |
| | 100 | |
|
| 52 | 101 | | foreach (var info in idInfos) |
| | 102 | | { |
| 11 | 103 | | var id = info.Key + "Id"; |
| 11 | 104 | | _validProviderIds.TryAdd(id, info.Key); |
| | 105 | | } |
| | 106 | |
|
| | 107 | | // Additional Mappings |
| 15 | 108 | | _validProviderIds.Add("collectionnumber", "TmdbCollection"); |
| 15 | 109 | | _validProviderIds.Add("tmdbcolid", "TmdbCollection"); |
| 15 | 110 | | _validProviderIds.Add("imdb_id", "Imdb"); |
| | 111 | |
|
| 15 | 112 | | Fetch(item, metadataFile, GetXmlReaderSettings(), cancellationToken); |
| 15 | 113 | | } |
| | 114 | |
|
| | 115 | | /// <summary> |
| | 116 | | /// Fetches the specified item. |
| | 117 | | /// </summary> |
| | 118 | | /// <param name="item">The <see cref="MetadataResult{T}"/>.</param> |
| | 119 | | /// <param name="metadataFile">The metadata file.</param> |
| | 120 | | /// <param name="settings">The <see cref="XmlReaderSettings"/>.</param> |
| | 121 | | /// <param name="cancellationToken">The <see cref="CancellationToken"/>.</param> |
| | 122 | | protected virtual void Fetch(MetadataResult<T> item, string metadataFile, XmlReaderSettings settings, Cancellati |
| | 123 | | { |
| 11 | 124 | | if (!SupportsUrlAfterClosingXmlTag) |
| | 125 | | { |
| 3 | 126 | | using (var fileStream = File.OpenRead(metadataFile)) |
| 3 | 127 | | using (var streamReader = new StreamReader(fileStream, Encoding.UTF8)) |
| 3 | 128 | | using (var reader = XmlReader.Create(streamReader, settings)) |
| | 129 | | { |
| 3 | 130 | | item.ResetPeople(); |
| | 131 | |
|
| 3 | 132 | | reader.MoveToContent(); |
| 3 | 133 | | reader.Read(); |
| | 134 | |
|
| | 135 | | // Loop through each element |
| 187 | 136 | | while (!reader.EOF && reader.ReadState == ReadState.Interactive) |
| | 137 | | { |
| 184 | 138 | | cancellationToken.ThrowIfCancellationRequested(); |
| | 139 | |
|
| 184 | 140 | | if (reader.NodeType == XmlNodeType.Element) |
| | 141 | | { |
| 82 | 142 | | FetchDataFromXmlNode(reader, item); |
| | 143 | | } |
| | 144 | | else |
| | 145 | | { |
| 102 | 146 | | reader.Read(); |
| | 147 | | } |
| | 148 | | } |
| 3 | 149 | | } |
| | 150 | |
|
| 3 | 151 | | return; |
| | 152 | | } |
| | 153 | |
|
| 8 | 154 | | item.ResetPeople(); |
| | 155 | |
|
| | 156 | | // Need to handle a url after the xml data |
| | 157 | | // http://kodi.wiki/view/NFO_files/movies |
| | 158 | |
|
| 8 | 159 | | var xml = File.ReadAllText(metadataFile); |
| | 160 | |
|
| | 161 | | // Find last closing Tag |
| | 162 | | // Need to do this in two steps to account for random > characters after the closing xml |
| 8 | 163 | | var index = xml.LastIndexOf("</", StringComparison.Ordinal); |
| | 164 | |
|
| | 165 | | // If closing tag exists, move to end of Tag |
| 8 | 166 | | if (index != -1) |
| | 167 | | { |
| 4 | 168 | | index = xml.IndexOf('>', index); |
| | 169 | | } |
| | 170 | |
|
| 8 | 171 | | if (index != -1) |
| | 172 | | { |
| 4 | 173 | | var endingXml = xml.AsSpan().Slice(index); |
| | 174 | |
|
| 4 | 175 | | ParseProviderLinks(item.Item, endingXml); |
| | 176 | |
|
| | 177 | | // If the file is just an IMDb url, don't go any further |
| 4 | 178 | | if (index == 0) |
| | 179 | | { |
| 0 | 180 | | return; |
| | 181 | | } |
| | 182 | |
|
| 4 | 183 | | xml = xml.Substring(0, index + 1); |
| | 184 | | } |
| | 185 | | else |
| | 186 | | { |
| | 187 | | // If the file is just provider urls, handle that |
| 4 | 188 | | ParseProviderLinks(item.Item, xml); |
| | 189 | |
|
| 4 | 190 | | return; |
| | 191 | | } |
| | 192 | |
|
| | 193 | | // These are not going to be valid xml so no sense in causing the provider to fail and spamming the log with |
| | 194 | | try |
| | 195 | | { |
| 4 | 196 | | using (var stringReader = new StringReader(xml)) |
| 4 | 197 | | using (var reader = XmlReader.Create(stringReader, settings)) |
| | 198 | | { |
| 4 | 199 | | reader.MoveToContent(); |
| 4 | 200 | | reader.Read(); |
| | 201 | |
|
| | 202 | | // Loop through each element |
| 591 | 203 | | while (!reader.EOF && reader.ReadState == ReadState.Interactive) |
| | 204 | | { |
| 587 | 205 | | cancellationToken.ThrowIfCancellationRequested(); |
| | 206 | |
|
| 587 | 207 | | if (reader.NodeType == XmlNodeType.Element) |
| | 208 | | { |
| 275 | 209 | | FetchDataFromXmlNode(reader, item); |
| | 210 | | } |
| | 211 | | else |
| | 212 | | { |
| 312 | 213 | | reader.Read(); |
| | 214 | | } |
| | 215 | | } |
| 4 | 216 | | } |
| 4 | 217 | | } |
| 0 | 218 | | catch (XmlException) |
| | 219 | | { |
| 0 | 220 | | } |
| 4 | 221 | | } |
| | 222 | |
|
| | 223 | | /// <summary> |
| | 224 | | /// Parses a XML tag to a provider id. |
| | 225 | | /// </summary> |
| | 226 | | /// <param name="item">The item.</param> |
| | 227 | | /// <param name="xml">The xml tag.</param> |
| | 228 | | protected void ParseProviderLinks(T item, ReadOnlySpan<char> xml) |
| | 229 | | { |
| 8 | 230 | | if (ProviderIdParsers.TryFindImdbId(xml, out var imdbId)) |
| | 231 | | { |
| 2 | 232 | | item.SetProviderId(MetadataProvider.Imdb, imdbId.ToString()); |
| | 233 | | } |
| | 234 | |
|
| 8 | 235 | | if (item is Movie) |
| | 236 | | { |
| 5 | 237 | | if (ProviderIdParsers.TryFindTmdbMovieId(xml, out var tmdbId)) |
| | 238 | | { |
| 2 | 239 | | item.SetProviderId(MetadataProvider.Tmdb, tmdbId.ToString()); |
| | 240 | | } |
| | 241 | | } |
| | 242 | |
|
| 8 | 243 | | if (item is Series) |
| | 244 | | { |
| 2 | 245 | | if (ProviderIdParsers.TryFindTmdbSeriesId(xml, out var tmdbId)) |
| | 246 | | { |
| 0 | 247 | | item.SetProviderId(MetadataProvider.Tmdb, tmdbId.ToString()); |
| | 248 | | } |
| | 249 | |
|
| 2 | 250 | | if (ProviderIdParsers.TryFindTvdbId(xml, out var tvdbId)) |
| | 251 | | { |
| 1 | 252 | | item.SetProviderId(MetadataProvider.Tvdb, tvdbId.ToString()); |
| | 253 | | } |
| | 254 | | } |
| 8 | 255 | | } |
| | 256 | |
|
| | 257 | | /// <summary> |
| | 258 | | /// Fetches metadata from an XML node. |
| | 259 | | /// </summary> |
| | 260 | | /// <param name="reader">The <see cref="XmlReader"/>.</param> |
| | 261 | | /// <param name="itemResult">The <see cref="MetadataResult{T}"/>.</param> |
| | 262 | | protected virtual void FetchDataFromXmlNode(XmlReader reader, MetadataResult<T> itemResult) |
| | 263 | | { |
| 441 | 264 | | var item = itemResult.Item; |
| 441 | 265 | | var nfoConfiguration = _config.GetNfoConfiguration(); |
| | 266 | | UserItemData? userData; |
| | 267 | |
|
| 441 | 268 | | switch (reader.Name) |
| | 269 | | { |
| | 270 | | case "dateadded": |
| 5 | 271 | | if (reader.TryReadDateTime(out var dateCreated)) |
| | 272 | | { |
| 5 | 273 | | item.DateCreated = dateCreated; |
| | 274 | | } |
| | 275 | |
|
| 5 | 276 | | break; |
| | 277 | | case "originaltitle": |
| 6 | 278 | | item.OriginalTitle = reader.ReadNormalizedString(); |
| 6 | 279 | | break; |
| | 280 | | case "name": |
| | 281 | | case "title": |
| | 282 | | case "localtitle": |
| 13 | 283 | | item.Name = reader.ReadNormalizedString(); |
| 13 | 284 | | break; |
| | 285 | | case "sortname": |
| 1 | 286 | | item.SortName = reader.ReadNormalizedString(); |
| 1 | 287 | | break; |
| | 288 | | case "criticrating": |
| 1 | 289 | | var criticRatingText = reader.ReadElementContentAsString(); |
| 1 | 290 | | if (float.TryParse(criticRatingText, CultureInfo.InvariantCulture, out var value)) |
| | 291 | | { |
| 1 | 292 | | item.CriticRating = value; |
| | 293 | | } |
| | 294 | |
|
| 1 | 295 | | break; |
| | 296 | | case "sorttitle": |
| 1 | 297 | | item.ForcedSortName = reader.ReadNormalizedString(); |
| 1 | 298 | | break; |
| | 299 | | case "biography": |
| | 300 | | case "plot": |
| | 301 | | case "review": |
| 11 | 302 | | item.Overview = reader.ReadNormalizedString(); |
| 11 | 303 | | break; |
| | 304 | | case "language": |
| 1 | 305 | | item.PreferredMetadataLanguage = reader.ReadNormalizedString(); |
| 1 | 306 | | break; |
| | 307 | | case "watched": |
| 8 | 308 | | var played = reader.ReadElementContentAsBoolean(); |
| 8 | 309 | | if (Guid.TryParse(nfoConfiguration.UserId, out var userId)) |
| | 310 | | { |
| 1 | 311 | | var user = _userManager.GetUserById(userId); |
| 1 | 312 | | if (user is not null) |
| | 313 | | { |
| 1 | 314 | | userData = _userDataManager.GetUserData(user, item); |
| 1 | 315 | | userData.Played = played; |
| 1 | 316 | | _userDataManager.SaveUserData(user, item, userData, UserDataSaveReason.Import, CancellationT |
| | 317 | | } |
| | 318 | | } |
| | 319 | |
|
| 1 | 320 | | break; |
| | 321 | | case "playcount": |
| 4 | 322 | | if (reader.TryReadInt(out var count) |
| 4 | 323 | | && Guid.TryParse(nfoConfiguration.UserId, out var playCountUserId)) |
| | 324 | | { |
| 1 | 325 | | var user = _userManager.GetUserById(playCountUserId); |
| 1 | 326 | | if (user is not null) |
| | 327 | | { |
| 1 | 328 | | userData = _userDataManager.GetUserData(user, item); |
| 1 | 329 | | userData.PlayCount = count; |
| 1 | 330 | | _userDataManager.SaveUserData(user, item, userData, UserDataSaveReason.Import, CancellationT |
| | 331 | | } |
| | 332 | | } |
| | 333 | |
|
| 1 | 334 | | break; |
| | 335 | | case "lastplayed": |
| 4 | 336 | | if (reader.TryReadDateTime(out var lastPlayed) |
| 4 | 337 | | && Guid.TryParse(nfoConfiguration.UserId, out var lastPlayedUserId)) |
| | 338 | | { |
| 1 | 339 | | var user = _userManager.GetUserById(lastPlayedUserId); |
| 1 | 340 | | if (user is not null) |
| | 341 | | { |
| 1 | 342 | | userData = _userDataManager.GetUserData(user, item); |
| 1 | 343 | | userData.LastPlayedDate = lastPlayed; |
| 1 | 344 | | _userDataManager.SaveUserData(user, item, userData, UserDataSaveReason.Import, CancellationT |
| | 345 | | } |
| | 346 | | } |
| | 347 | |
|
| 1 | 348 | | break; |
| | 349 | | case "countrycode": |
| 1 | 350 | | item.PreferredMetadataCountryCode = reader.ReadNormalizedString(); |
| 1 | 351 | | break; |
| | 352 | | case "lockedfields": |
| | 353 | | { |
| 0 | 354 | | var val = reader.ReadElementContentAsString(); |
| | 355 | |
|
| 0 | 356 | | if (!string.IsNullOrWhiteSpace(val)) |
| | 357 | | { |
| 0 | 358 | | item.LockedFields = val.Split('|').Select(i => |
| 0 | 359 | | { |
| 0 | 360 | | if (Enum.TryParse(i, true, out MetadataField field)) |
| 0 | 361 | | { |
| 0 | 362 | | return (MetadataField?)field; |
| 0 | 363 | | } |
| 0 | 364 | |
|
| 0 | 365 | | return null; |
| 0 | 366 | | }).OfType<MetadataField>().ToArray(); |
| | 367 | | } |
| | 368 | |
|
| 0 | 369 | | break; |
| | 370 | | } |
| | 371 | |
|
| | 372 | | case "tagline": |
| 4 | 373 | | item.Tagline = reader.ReadNormalizedString(); |
| 4 | 374 | | break; |
| | 375 | | case "country": |
| | 376 | | { |
| 3 | 377 | | var val = reader.ReadElementContentAsString(); |
| | 378 | |
|
| 3 | 379 | | if (!string.IsNullOrWhiteSpace(val)) |
| | 380 | | { |
| 3 | 381 | | item.ProductionLocations = val.Split('/') |
| 3 | 382 | | .Select(i => i.Trim()) |
| 3 | 383 | | .Where(i => !string.IsNullOrWhiteSpace(i)) |
| 3 | 384 | | .ToArray(); |
| | 385 | | } |
| | 386 | |
|
| 3 | 387 | | break; |
| | 388 | | } |
| | 389 | |
|
| | 390 | | case "mpaa": |
| 4 | 391 | | item.OfficialRating = reader.ReadNormalizedString(); |
| 4 | 392 | | break; |
| | 393 | | case "customrating": |
| 1 | 394 | | item.CustomRating = reader.ReadNormalizedString(); |
| 1 | 395 | | break; |
| | 396 | | case "runtime": |
| 4 | 397 | | var runtimeText = reader.ReadElementContentAsString(); |
| 4 | 398 | | if (int.TryParse(runtimeText.AsSpan().LeftPart(' '), NumberStyles.Integer, CultureInfo.InvariantCult |
| | 399 | | { |
| 4 | 400 | | item.RunTimeTicks = TimeSpan.FromMinutes(runtime).Ticks; |
| | 401 | | } |
| | 402 | |
|
| 4 | 403 | | break; |
| | 404 | | case "aspectratio": |
| 1 | 405 | | var aspectRatio = reader.ReadNormalizedString(); |
| 1 | 406 | | if (!string.IsNullOrEmpty(aspectRatio) && item is IHasAspectRatio hasAspectRatio) |
| | 407 | | { |
| 1 | 408 | | hasAspectRatio.AspectRatio = aspectRatio; |
| | 409 | | } |
| | 410 | |
|
| 1 | 411 | | break; |
| | 412 | | case "lockdata": |
| 1 | 413 | | item.IsLocked = string.Equals(reader.ReadElementContentAsString(), "true", StringComparison.OrdinalI |
| 1 | 414 | | break; |
| | 415 | | case "studio": |
| 4 | 416 | | var studio = reader.ReadNormalizedString(); |
| 4 | 417 | | if (!string.IsNullOrEmpty(studio)) |
| | 418 | | { |
| 4 | 419 | | item.AddStudio(studio); |
| | 420 | | } |
| | 421 | |
|
| 4 | 422 | | break; |
| | 423 | | case "director": |
| 12 | 424 | | foreach (var director in reader.GetPersonArray(PersonKind.Director)) |
| | 425 | | { |
| 3 | 426 | | itemResult.AddPerson(director); |
| | 427 | | } |
| | 428 | |
|
| | 429 | | break; |
| | 430 | | case "credits": |
| | 431 | | { |
| 4 | 432 | | var val = reader.ReadElementContentAsString(); |
| | 433 | |
|
| 4 | 434 | | if (!string.IsNullOrWhiteSpace(val)) |
| | 435 | | { |
| 4 | 436 | | var parts = val.Split('/').Select(i => i.Trim()) |
| 4 | 437 | | .Where(i => !string.IsNullOrEmpty(i)); |
| | 438 | |
|
| 16 | 439 | | foreach (var p in parts.Select(v => new PersonInfo { Name = v.Trim(), Type = PersonKind.Writ |
| | 440 | | { |
| 4 | 441 | | if (string.IsNullOrWhiteSpace(p.Name)) |
| | 442 | | { |
| | 443 | | continue; |
| | 444 | | } |
| | 445 | |
|
| 4 | 446 | | itemResult.AddPerson(p); |
| | 447 | | } |
| | 448 | | } |
| | 449 | |
|
| | 450 | | break; |
| | 451 | | } |
| | 452 | |
|
| | 453 | | case "writer": |
| 4 | 454 | | foreach (var writer in reader.GetPersonArray(PersonKind.Writer)) |
| | 455 | | { |
| 1 | 456 | | itemResult.AddPerson(writer); |
| | 457 | | } |
| | 458 | |
|
| | 459 | | break; |
| | 460 | | case "actor": |
| 55 | 461 | | var person = reader.GetPersonFromXmlNode(); |
| 55 | 462 | | if (person is not null) |
| | 463 | | { |
| 55 | 464 | | itemResult.AddPerson(person); |
| | 465 | | } |
| | 466 | |
|
| 55 | 467 | | break; |
| | 468 | | case "trailer": |
| 4 | 469 | | var trailer = reader.ReadNormalizedString(); |
| 4 | 470 | | if (!string.IsNullOrEmpty(trailer)) |
| | 471 | | { |
| 1 | 472 | | if (trailer.StartsWith("plugin://plugin.video.youtube/?action=play_video&videoid=", StringCompar |
| | 473 | | { |
| | 474 | | // Deprecated format |
| 1 | 475 | | item.AddTrailerUrl(trailer.Replace( |
| 1 | 476 | | "plugin://plugin.video.youtube/?action=play_video&videoid=", |
| 1 | 477 | | BaseNfoSaver.YouTubeWatchUrl, |
| 1 | 478 | | StringComparison.OrdinalIgnoreCase)); |
| | 479 | |
|
| 1 | 480 | | var suggestedUrl = trailer.Replace( |
| 1 | 481 | | "plugin://plugin.video.youtube/?action=play_video&videoid=", |
| 1 | 482 | | "plugin://plugin.video.youtube/play/?video_id=", |
| 1 | 483 | | StringComparison.OrdinalIgnoreCase); |
| 1 | 484 | | Logger.LogWarning("Trailer URL uses a deprecated format : {Url}. Using {NewUrl} instead is a |
| | 485 | | } |
| 0 | 486 | | else if (trailer.StartsWith("plugin://plugin.video.youtube/play/?video_id=", StringComparison.Or |
| | 487 | | { |
| | 488 | | // Proper format |
| 0 | 489 | | item.AddTrailerUrl(trailer.Replace( |
| 0 | 490 | | "plugin://plugin.video.youtube/play/?video_id=", |
| 0 | 491 | | BaseNfoSaver.YouTubeWatchUrl, |
| 0 | 492 | | StringComparison.OrdinalIgnoreCase)); |
| | 493 | | } |
| | 494 | | } |
| | 495 | |
|
| 0 | 496 | | break; |
| | 497 | | case "displayorder": |
| 0 | 498 | | var displayOrder = reader.ReadNormalizedString(); |
| 0 | 499 | | if (!string.IsNullOrEmpty(displayOrder) && item is IHasDisplayOrder hasDisplayOrder) |
| | 500 | | { |
| 0 | 501 | | hasDisplayOrder.DisplayOrder = displayOrder; |
| | 502 | | } |
| | 503 | |
|
| 0 | 504 | | break; |
| | 505 | | case "year": |
| 6 | 506 | | if (reader.TryReadInt(out var productionYear) && productionYear > 1850) |
| | 507 | | { |
| 6 | 508 | | item.ProductionYear = productionYear; |
| | 509 | | } |
| | 510 | |
|
| 6 | 511 | | break; |
| | 512 | | case "rating": |
| 7 | 513 | | var rating = reader.ReadElementContentAsString().Replace(',', '.'); |
| | 514 | | // All external meta is saving this as '.' for decimal I believe...but just to be sure |
| 7 | 515 | | if (float.TryParse(rating, NumberStyles.AllowDecimalPoint, CultureInfo.InvariantCulture, out var com |
| | 516 | | { |
| 6 | 517 | | item.CommunityRating = communityRating; |
| | 518 | | } |
| | 519 | |
|
| 6 | 520 | | break; |
| | 521 | | case "ratings": |
| 3 | 522 | | FetchFromRatingsNode(reader, item); |
| 3 | 523 | | break; |
| | 524 | | case "aired": |
| | 525 | | case "formed": |
| | 526 | | case "premiered": |
| | 527 | | case "releasedate": |
| 19 | 528 | | if (reader.TryReadDateTimeExact(nfoConfiguration.ReleaseDateFormat, out var releaseDate)) |
| | 529 | | { |
| 14 | 530 | | item.PremiereDate = releaseDate; |
| | 531 | |
|
| | 532 | | // Production year can already be set by the year tag |
| 14 | 533 | | item.ProductionYear ??= releaseDate.Year; |
| | 534 | | } |
| | 535 | |
|
| | 536 | | break; |
| | 537 | | case "enddate": |
| 1 | 538 | | if (reader.TryReadDateTimeExact(nfoConfiguration.ReleaseDateFormat, out var endDate)) |
| | 539 | | { |
| 1 | 540 | | item.EndDate = endDate; |
| | 541 | | } |
| | 542 | |
|
| 1 | 543 | | break; |
| | 544 | | case "genre": |
| | 545 | | { |
| 13 | 546 | | var val = reader.ReadElementContentAsString(); |
| | 547 | |
|
| 13 | 548 | | if (!string.IsNullOrWhiteSpace(val)) |
| | 549 | | { |
| 13 | 550 | | var parts = val.Split('/') |
| 13 | 551 | | .Select(i => i.Trim()) |
| 13 | 552 | | .Where(i => !string.IsNullOrWhiteSpace(i)); |
| | 553 | |
|
| 52 | 554 | | foreach (var p in parts) |
| | 555 | | { |
| 13 | 556 | | item.AddGenre(p); |
| | 557 | | } |
| | 558 | | } |
| | 559 | |
|
| | 560 | | break; |
| | 561 | | } |
| | 562 | |
|
| | 563 | | case "style": |
| | 564 | | case "tag": |
| 2 | 565 | | var tag = reader.ReadNormalizedString(); |
| 2 | 566 | | if (!string.IsNullOrEmpty(tag)) |
| | 567 | | { |
| 2 | 568 | | item.AddTag(tag); |
| | 569 | | } |
| | 570 | |
|
| 2 | 571 | | break; |
| | 572 | | case "fileinfo": |
| 3 | 573 | | FetchFromFileInfoNode(reader, item); |
| 3 | 574 | | break; |
| | 575 | | case "uniqueid": |
| 6 | 576 | | if (reader.IsEmptyElement) |
| | 577 | | { |
| 0 | 578 | | reader.Read(); |
| 0 | 579 | | break; |
| | 580 | | } |
| | 581 | |
|
| 6 | 582 | | var provider = reader.GetAttribute("type"); |
| 6 | 583 | | var providerId = reader.ReadElementContentAsString(); |
| 6 | 584 | | item.TrySetProviderId(provider, providerId); |
| | 585 | |
|
| 6 | 586 | | break; |
| | 587 | | case "thumb": |
| 166 | 588 | | FetchThumbNode(reader, itemResult, "thumb"); |
| 166 | 589 | | break; |
| | 590 | | case "fanart": |
| | 591 | | { |
| 4 | 592 | | if (reader.IsEmptyElement) |
| | 593 | | { |
| 0 | 594 | | reader.Read(); |
| 0 | 595 | | break; |
| | 596 | | } |
| | 597 | |
|
| 4 | 598 | | using var subtree = reader.ReadSubtree(); |
| 4 | 599 | | if (!subtree.ReadToDescendant("thumb")) |
| | 600 | | { |
| 0 | 601 | | break; |
| | 602 | | } |
| | 603 | |
|
| 4 | 604 | | FetchThumbNode(subtree, itemResult, "fanart"); |
| 4 | 605 | | break; |
| | 606 | | } |
| | 607 | |
|
| | 608 | | default: |
| 66 | 609 | | string readerName = reader.Name; |
| 66 | 610 | | if (_validProviderIds.TryGetValue(readerName, out string? providerIdValue)) |
| | 611 | | { |
| 3 | 612 | | var id = reader.ReadElementContentAsString(); |
| 3 | 613 | | item.TrySetProviderId(providerIdValue, id); |
| | 614 | | } |
| | 615 | | else |
| | 616 | | { |
| 63 | 617 | | reader.Skip(); |
| | 618 | | } |
| | 619 | |
|
| | 620 | | break; |
| | 621 | | } |
| 113 | 622 | | } |
| | 623 | |
|
| | 624 | | private void FetchThumbNode(XmlReader reader, MetadataResult<T> itemResult, string parentNode) |
| | 625 | | { |
| 170 | 626 | | var artType = reader.GetAttribute("aspect"); |
| 170 | 627 | | var val = reader.ReadElementContentAsString(); |
| | 628 | |
|
| | 629 | | // artType is null if the thumb node is a child of the fanart tag |
| | 630 | | // -> set image type to fanart |
| 170 | 631 | | if (string.IsNullOrWhiteSpace(artType) && parentNode.Equals("fanart", StringComparison.Ordinal)) |
| | 632 | | { |
| 4 | 633 | | artType = "fanart"; |
| | 634 | | } |
| 166 | 635 | | else if (string.IsNullOrWhiteSpace(artType)) |
| | 636 | | { |
| | 637 | | // Sonarr writes thumb tags for posters without aspect property |
| 14 | 638 | | artType = "poster"; |
| | 639 | | } |
| | 640 | |
|
| | 641 | | // skip: |
| | 642 | | // - empty uri |
| | 643 | | // - tag containing '.' because we can't set images for seasons, episodes or movie sets within series or mov |
| 170 | 644 | | if (string.IsNullOrEmpty(val) || artType.Contains('.', StringComparison.Ordinal)) |
| | 645 | | { |
| 11 | 646 | | return; |
| | 647 | | } |
| | 648 | |
|
| 159 | 649 | | ImageType imageType = GetImageType(artType); |
| | 650 | |
|
| 159 | 651 | | if (!Uri.TryCreate(val, UriKind.Absolute, out var uri)) |
| | 652 | | { |
| 1 | 653 | | Logger.LogError("Image location {Path} specified in nfo file for {ItemName} is not a valid URL or file p |
| 1 | 654 | | return; |
| | 655 | | } |
| | 656 | |
|
| 158 | 657 | | if (uri.IsFile) |
| | 658 | | { |
| | 659 | | // only allow one item of each type |
| 2 | 660 | | if (itemResult.Images.Any(x => x.Type == imageType)) |
| | 661 | | { |
| 0 | 662 | | return; |
| | 663 | | } |
| | 664 | |
|
| 2 | 665 | | var fileSystemMetadata = _directoryService.GetFile(val); |
| | 666 | | // non existing file returns null |
| 2 | 667 | | if (fileSystemMetadata is null || !fileSystemMetadata.Exists) |
| | 668 | | { |
| 1 | 669 | | Logger.LogWarning("Artwork file {Path} specified in nfo file for {ItemName} does not exist.", uri, i |
| 1 | 670 | | return; |
| | 671 | | } |
| | 672 | |
|
| 1 | 673 | | itemResult.Images.Add(new LocalImageInfo() |
| 1 | 674 | | { |
| 1 | 675 | | FileInfo = fileSystemMetadata, |
| 1 | 676 | | Type = imageType |
| 1 | 677 | | }); |
| | 678 | | } |
| | 679 | | else |
| | 680 | | { |
| | 681 | | // only allow one item of each type |
| 156 | 682 | | if (itemResult.RemoteImages.Any(x => x.Type == imageType)) |
| | 683 | | { |
| 121 | 684 | | return; |
| | 685 | | } |
| | 686 | |
|
| 35 | 687 | | itemResult.RemoteImages.Add((uri.ToString(), imageType)); |
| | 688 | | } |
| 35 | 689 | | } |
| | 690 | |
|
| | 691 | | private void FetchFromFileInfoNode(XmlReader parentReader, T item) |
| | 692 | | { |
| 3 | 693 | | if (parentReader.IsEmptyElement) |
| | 694 | | { |
| 0 | 695 | | parentReader.Read(); |
| 0 | 696 | | return; |
| | 697 | | } |
| | 698 | |
|
| 3 | 699 | | using var reader = parentReader.ReadSubtree(); |
| 3 | 700 | | reader.MoveToContent(); |
| 3 | 701 | | reader.Read(); |
| | 702 | |
|
| | 703 | | // Loop through each element |
| 18 | 704 | | while (!reader.EOF && reader.ReadState == ReadState.Interactive) |
| | 705 | | { |
| 15 | 706 | | if (reader.NodeType != XmlNodeType.Element) |
| | 707 | | { |
| 12 | 708 | | reader.Read(); |
| 12 | 709 | | continue; |
| | 710 | | } |
| | 711 | |
|
| 3 | 712 | | switch (reader.Name) |
| | 713 | | { |
| | 714 | | case "streamdetails": |
| 3 | 715 | | FetchFromStreamDetailsNode(reader, item); |
| 3 | 716 | | break; |
| | 717 | | default: |
| 0 | 718 | | reader.Skip(); |
| | 719 | | break; |
| | 720 | | } |
| | 721 | | } |
| 6 | 722 | | } |
| | 723 | |
|
| | 724 | | private void FetchFromStreamDetailsNode(XmlReader parentReader, T item) |
| | 725 | | { |
| 3 | 726 | | if (parentReader.IsEmptyElement) |
| | 727 | | { |
| 0 | 728 | | parentReader.Read(); |
| 0 | 729 | | return; |
| | 730 | | } |
| | 731 | |
|
| 3 | 732 | | using var reader = parentReader.ReadSubtree(); |
| 3 | 733 | | reader.MoveToContent(); |
| 3 | 734 | | reader.Read(); |
| | 735 | |
|
| | 736 | | // Loop through each element |
| 32 | 737 | | while (!reader.EOF && reader.ReadState == ReadState.Interactive) |
| | 738 | | { |
| 29 | 739 | | if (reader.NodeType != XmlNodeType.Element) |
| | 740 | | { |
| 20 | 741 | | reader.Read(); |
| 20 | 742 | | continue; |
| | 743 | | } |
| | 744 | |
|
| 9 | 745 | | switch (reader.Name) |
| | 746 | | { |
| | 747 | | case "video": |
| 3 | 748 | | FetchFromVideoNode(reader, item); |
| 3 | 749 | | break; |
| | 750 | | case "subtitle": |
| 2 | 751 | | FetchFromSubtitleNode(reader, item); |
| 2 | 752 | | break; |
| | 753 | | default: |
| 4 | 754 | | reader.Skip(); |
| | 755 | | break; |
| | 756 | | } |
| | 757 | | } |
| 6 | 758 | | } |
| | 759 | |
|
| | 760 | | private void FetchFromVideoNode(XmlReader parentReader, T item) |
| | 761 | | { |
| 3 | 762 | | if (parentReader.IsEmptyElement) |
| | 763 | | { |
| 0 | 764 | | parentReader.Read(); |
| 0 | 765 | | return; |
| | 766 | | } |
| | 767 | |
|
| 3 | 768 | | using var reader = parentReader.ReadSubtree(); |
| 3 | 769 | | reader.MoveToContent(); |
| 3 | 770 | | reader.Read(); |
| | 771 | |
|
| | 772 | | // Loop through each element |
| 53 | 773 | | while (!reader.EOF && reader.ReadState == ReadState.Interactive) |
| | 774 | | { |
| 50 | 775 | | if (reader.NodeType != XmlNodeType.Element || item is not Video video) |
| | 776 | | { |
| 28 | 777 | | reader.Read(); |
| 28 | 778 | | continue; |
| | 779 | | } |
| | 780 | |
|
| 22 | 781 | | switch (reader.Name) |
| | 782 | | { |
| | 783 | | case "format3d": |
| 1 | 784 | | var format = reader.ReadElementContentAsString(); |
| 1 | 785 | | if (string.Equals("HSBS", format, StringComparison.OrdinalIgnoreCase)) |
| | 786 | | { |
| 1 | 787 | | video.Video3DFormat = Video3DFormat.HalfSideBySide; |
| | 788 | | } |
| 0 | 789 | | else if (string.Equals("HTAB", format, StringComparison.OrdinalIgnoreCase)) |
| | 790 | | { |
| 0 | 791 | | video.Video3DFormat = Video3DFormat.HalfTopAndBottom; |
| | 792 | | } |
| 0 | 793 | | else if (string.Equals("FTAB", format, StringComparison.OrdinalIgnoreCase)) |
| | 794 | | { |
| 0 | 795 | | video.Video3DFormat = Video3DFormat.FullTopAndBottom; |
| | 796 | | } |
| 0 | 797 | | else if (string.Equals("FSBS", format, StringComparison.OrdinalIgnoreCase)) |
| | 798 | | { |
| 0 | 799 | | video.Video3DFormat = Video3DFormat.FullSideBySide; |
| | 800 | | } |
| 0 | 801 | | else if (string.Equals("MVC", format, StringComparison.OrdinalIgnoreCase)) |
| | 802 | | { |
| 0 | 803 | | video.Video3DFormat = Video3DFormat.MVC; |
| | 804 | | } |
| | 805 | |
|
| 0 | 806 | | break; |
| | 807 | | case "aspect": |
| 3 | 808 | | video.AspectRatio = reader.ReadNormalizedString(); |
| 3 | 809 | | break; |
| | 810 | | case "width": |
| 3 | 811 | | video.Width = reader.ReadElementContentAsInt(); |
| 3 | 812 | | break; |
| | 813 | | case "height": |
| 3 | 814 | | video.Height = reader.ReadElementContentAsInt(); |
| 3 | 815 | | break; |
| | 816 | | case "durationinseconds": |
| 3 | 817 | | video.RunTimeTicks = new TimeSpan(0, 0, reader.ReadElementContentAsInt()).Ticks; |
| 3 | 818 | | break; |
| | 819 | | default: |
| 9 | 820 | | reader.Skip(); |
| | 821 | | break; |
| | 822 | | } |
| | 823 | | } |
| 6 | 824 | | } |
| | 825 | |
|
| | 826 | | private void FetchFromSubtitleNode(XmlReader parentReader, T item) |
| | 827 | | { |
| 2 | 828 | | if (parentReader.IsEmptyElement) |
| | 829 | | { |
| 0 | 830 | | parentReader.Read(); |
| 0 | 831 | | return; |
| | 832 | | } |
| | 833 | |
|
| 2 | 834 | | using var reader = parentReader.ReadSubtree(); |
| 2 | 835 | | reader.MoveToContent(); |
| 2 | 836 | | reader.Read(); |
| | 837 | |
|
| | 838 | | // Loop through each element |
| 10 | 839 | | while (!reader.EOF && reader.ReadState == ReadState.Interactive) |
| | 840 | | { |
| 8 | 841 | | if (reader.NodeType != XmlNodeType.Element) |
| | 842 | | { |
| 6 | 843 | | reader.Read(); |
| 6 | 844 | | continue; |
| | 845 | | } |
| | 846 | |
|
| 2 | 847 | | switch (reader.Name) |
| | 848 | | { |
| | 849 | | case "language": |
| 2 | 850 | | _ = reader.ReadElementContentAsString(); |
| 2 | 851 | | if (item is Video video) |
| | 852 | | { |
| 2 | 853 | | video.HasSubtitles = true; |
| | 854 | | } |
| | 855 | |
|
| 2 | 856 | | break; |
| | 857 | | default: |
| 0 | 858 | | reader.Skip(); |
| | 859 | | break; |
| | 860 | | } |
| | 861 | | } |
| 4 | 862 | | } |
| | 863 | |
|
| | 864 | | private void FetchFromRatingsNode(XmlReader parentReader, T item) |
| | 865 | | { |
| 3 | 866 | | if (parentReader.IsEmptyElement) |
| | 867 | | { |
| 0 | 868 | | parentReader.Read(); |
| 0 | 869 | | return; |
| | 870 | | } |
| | 871 | |
|
| 3 | 872 | | using var reader = parentReader.ReadSubtree(); |
| 3 | 873 | | reader.MoveToContent(); |
| 3 | 874 | | reader.Read(); |
| | 875 | |
|
| | 876 | | // Loop through each element |
| 42 | 877 | | while (!reader.EOF && reader.ReadState == ReadState.Interactive) |
| | 878 | | { |
| 39 | 879 | | if (reader.NodeType == XmlNodeType.Element) |
| | 880 | | { |
| 11 | 881 | | switch (reader.Name) |
| | 882 | | { |
| | 883 | | case "rating": |
| | 884 | | { |
| 11 | 885 | | if (reader.IsEmptyElement) |
| | 886 | | { |
| 0 | 887 | | reader.Read(); |
| 0 | 888 | | continue; |
| | 889 | | } |
| | 890 | |
|
| 11 | 891 | | var ratingName = reader.GetAttribute("name"); |
| | 892 | |
|
| 11 | 893 | | using var subtree = reader.ReadSubtree(); |
| 11 | 894 | | FetchFromRatingNode(subtree, item, ratingName); |
| | 895 | |
|
| 11 | 896 | | break; |
| | 897 | | } |
| | 898 | |
|
| | 899 | | default: |
| 0 | 900 | | reader.Skip(); |
| 0 | 901 | | break; |
| | 902 | | } |
| | 903 | | } |
| | 904 | | else |
| | 905 | | { |
| 28 | 906 | | reader.Read(); |
| | 907 | | } |
| | 908 | | } |
| 6 | 909 | | } |
| | 910 | |
|
| | 911 | | private void FetchFromRatingNode(XmlReader reader, T item, string? ratingName) |
| | 912 | | { |
| 11 | 913 | | reader.MoveToContent(); |
| 11 | 914 | | reader.Read(); |
| | 915 | |
|
| | 916 | | // Loop through each element |
| 77 | 917 | | while (!reader.EOF && reader.ReadState == ReadState.Interactive) |
| | 918 | | { |
| 66 | 919 | | if (reader.NodeType == XmlNodeType.Element) |
| | 920 | | { |
| 22 | 921 | | switch (reader.Name) |
| | 922 | | { |
| | 923 | | case "value": |
| 11 | 924 | | var val = reader.ReadElementContentAsString(); |
| | 925 | |
|
| 11 | 926 | | if (float.TryParse(val, NumberStyles.AllowDecimalPoint, CultureInfo.InvariantCulture, out va |
| | 927 | | { |
| | 928 | | // if ratingName contains tomato --> assume critic rating |
| 11 | 929 | | if (ratingName is not null |
| 11 | 930 | | && ratingName.Contains("tomato", StringComparison.OrdinalIgnoreCase) |
| 11 | 931 | | && !ratingName.Contains("audience", StringComparison.OrdinalIgnoreCase)) |
| | 932 | | { |
| 2 | 933 | | if (!ratingName.Contains("avg", StringComparison.OrdinalIgnoreCase)) |
| | 934 | | { |
| 2 | 935 | | item.CriticRating = ratingValue; |
| | 936 | | } |
| | 937 | | } |
| | 938 | | else |
| | 939 | | { |
| 9 | 940 | | item.CommunityRating = ratingValue; |
| | 941 | | } |
| | 942 | | } |
| | 943 | |
|
| 9 | 944 | | break; |
| | 945 | | default: |
| 11 | 946 | | reader.Skip(); |
| 11 | 947 | | break; |
| | 948 | | } |
| | 949 | | } |
| | 950 | | else |
| | 951 | | { |
| 44 | 952 | | reader.Read(); |
| | 953 | | } |
| | 954 | | } |
| 11 | 955 | | } |
| | 956 | |
|
| | 957 | | internal XmlReaderSettings GetXmlReaderSettings() |
| 16 | 958 | | => new XmlReaderSettings() |
| 16 | 959 | | { |
| 16 | 960 | | ValidationType = ValidationType.None, |
| 16 | 961 | | CheckCharacters = false, |
| 16 | 962 | | IgnoreProcessingInstructions = true, |
| 16 | 963 | | IgnoreComments = true |
| 16 | 964 | | }; |
| | 965 | |
|
| | 966 | | /// <summary> |
| | 967 | | /// Parses the <see cref="ImageType"/> from the NFO aspect property. |
| | 968 | | /// </summary> |
| | 969 | | /// <param name="aspect">The NFO aspect property.</param> |
| | 970 | | /// <returns>The <see cref="ImageType"/>.</returns> |
| | 971 | | private static ImageType GetImageType(string aspect) |
| | 972 | | { |
| 159 | 973 | | return aspect switch |
| 159 | 974 | | { |
| 23 | 975 | | "banner" => ImageType.Banner, |
| 23 | 976 | | "clearlogo" => ImageType.Logo, |
| 12 | 977 | | "discart" => ImageType.Disc, |
| 18 | 978 | | "landscape" => ImageType.Thumb, |
| 8 | 979 | | "clearart" => ImageType.Art, |
| 6 | 980 | | "fanart" => ImageType.Backdrop, |
| 159 | 981 | | // unknown type (including "poster") --> primary |
| 69 | 982 | | _ => ImageType.Primary, |
| 159 | 983 | | }; |
| | 984 | | } |
| | 985 | | } |
| | 986 | | } |