MovieNfoParser.cs 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160
  1. using System;
  2. using System.IO;
  3. using System.Linq;
  4. using System.Xml;
  5. using MediaBrowser.Common.Configuration;
  6. using MediaBrowser.Controller.Entities;
  7. using MediaBrowser.Controller.Entities.Movies;
  8. using MediaBrowser.Controller.Extensions;
  9. using MediaBrowser.Controller.Library;
  10. using MediaBrowser.Controller.Providers;
  11. using MediaBrowser.Model.Entities;
  12. using Microsoft.Extensions.Logging;
  13. namespace MediaBrowser.XbmcMetadata.Parsers
  14. {
  15. /// <summary>
  16. /// Nfo parser for movies.
  17. /// </summary>
  18. public class MovieNfoParser : BaseNfoParser<Video>
  19. {
  20. /// <summary>
  21. /// Initializes a new instance of the <see cref="MovieNfoParser"/> class.
  22. /// </summary>
  23. /// <param name="logger">Instance of the <see cref="ILogger"/> interface.</param>
  24. /// <param name="config">Instance of the <see cref="IConfigurationManager"/> interface.</param>
  25. /// <param name="providerManager">Instance of the <see cref="IProviderManager"/> interface.</param>
  26. /// <param name="userManager">Instance of the <see cref="IUserManager"/> interface.</param>
  27. /// <param name="userDataManager">Instance of the <see cref="IUserDataManager"/> interface.</param>
  28. /// <param name="directoryService">Instance of the <see cref="DirectoryService"/> interface.</param>
  29. public MovieNfoParser(
  30. ILogger logger,
  31. IConfigurationManager config,
  32. IProviderManager providerManager,
  33. IUserManager userManager,
  34. IUserDataManager userDataManager,
  35. IDirectoryService directoryService)
  36. : base(logger, config, providerManager, userManager, userDataManager, directoryService)
  37. {
  38. }
  39. /// <inheritdoc />
  40. protected override bool SupportsUrlAfterClosingXmlTag => true;
  41. /// <inheritdoc />
  42. protected override void FetchDataFromXmlNode(XmlReader reader, MetadataResult<Video> itemResult)
  43. {
  44. var item = itemResult.Item;
  45. switch (reader.Name)
  46. {
  47. case "id":
  48. {
  49. // Get ids from attributes
  50. item.TrySetProviderId(MetadataProvider.Tmdb, reader.GetAttribute("TMDB"));
  51. item.TrySetProviderId(MetadataProvider.Tvdb, reader.GetAttribute("TVDB"));
  52. string? imdbId = reader.GetAttribute("IMDB");
  53. // Read id from content
  54. // Content can be arbitrary according to Kodi wiki, so only parse if we are sure it matches a provider-specific schema
  55. var contentId = reader.ReadElementContentAsString();
  56. if (string.IsNullOrEmpty(imdbId) && contentId.StartsWith("tt", StringComparison.Ordinal))
  57. {
  58. imdbId = contentId;
  59. }
  60. item.TrySetProviderId(MetadataProvider.Imdb, imdbId);
  61. break;
  62. }
  63. case "set":
  64. {
  65. var movie = item as Movie;
  66. var tmdbcolid = reader.GetAttribute("tmdbcolid");
  67. movie?.TrySetProviderId(MetadataProvider.TmdbCollection, tmdbcolid);
  68. var val = reader.ReadInnerXml();
  69. if (!string.IsNullOrWhiteSpace(val) && movie is not null)
  70. {
  71. try
  72. {
  73. ParseSetXml(val, movie);
  74. }
  75. catch (Exception ex)
  76. {
  77. Logger.LogError(ex, "Error parsing set node");
  78. }
  79. }
  80. break;
  81. }
  82. case "artist":
  83. var artist = reader.ReadNormalizedString();
  84. if (!string.IsNullOrEmpty(artist) && item is MusicVideo artistVideo)
  85. {
  86. artistVideo.Artists = [..artistVideo.Artists, artist];
  87. }
  88. break;
  89. case "album":
  90. var album = reader.ReadNormalizedString();
  91. if (!string.IsNullOrEmpty(album) && item is MusicVideo albumVideo)
  92. {
  93. albumVideo.Album = album;
  94. }
  95. break;
  96. default:
  97. base.FetchDataFromXmlNode(reader, itemResult);
  98. break;
  99. }
  100. }
  101. private void ParseSetXml(string xml, Movie movie)
  102. {
  103. // These are not going to be valid xml so no sense in causing the provider to fail and spamming the log with exceptions
  104. try
  105. {
  106. using (var stringReader = new StringReader("<set>" + xml + "</set>"))
  107. using (var reader = XmlReader.Create(stringReader, GetXmlReaderSettings()))
  108. {
  109. reader.MoveToContent();
  110. reader.Read();
  111. // Loop through each element
  112. while (!reader.EOF && reader.ReadState == ReadState.Interactive)
  113. {
  114. if (reader.NodeType == XmlNodeType.Text && reader.Depth == 1)
  115. {
  116. movie.CollectionName = reader.Value;
  117. break;
  118. }
  119. else if (reader.NodeType == XmlNodeType.Element)
  120. {
  121. switch (reader.Name)
  122. {
  123. case "name":
  124. movie.CollectionName = reader.ReadElementContentAsString();
  125. break;
  126. default:
  127. reader.Skip();
  128. break;
  129. }
  130. }
  131. else
  132. {
  133. reader.Read();
  134. }
  135. }
  136. }
  137. }
  138. catch (XmlException)
  139. {
  140. }
  141. }
  142. }
  143. }