EpisodeNfoParser.cs 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202
  1. using System;
  2. using System.IO;
  3. using System.Text;
  4. using System.Threading;
  5. using System.Xml;
  6. using MediaBrowser.Common.Configuration;
  7. using MediaBrowser.Controller.Entities.TV;
  8. using MediaBrowser.Controller.Extensions;
  9. using MediaBrowser.Controller.Library;
  10. using MediaBrowser.Controller.Providers;
  11. using Microsoft.Extensions.Logging;
  12. namespace MediaBrowser.XbmcMetadata.Parsers
  13. {
  14. /// <summary>
  15. /// Nfo parser for episodes.
  16. /// </summary>
  17. public class EpisodeNfoParser : BaseNfoParser<Episode>
  18. {
  19. /// <summary>
  20. /// Initializes a new instance of the <see cref="EpisodeNfoParser"/> class.
  21. /// </summary>
  22. /// <param name="logger">Instance of the <see cref="ILogger{BaseNfoParser}"/> interface.</param>
  23. /// <param name="config">Instance of the <see cref="IConfigurationManager"/> interface.</param>
  24. /// <param name="providerManager">Instance of the <see cref="IProviderManager"/> interface.</param>
  25. /// <param name="userManager">Instance of the <see cref="IUserManager"/> interface.</param>
  26. /// <param name="userDataManager">Instance of the <see cref="IUserDataManager"/> interface.</param>
  27. /// <param name="directoryService">Instance of the <see cref="IDirectoryService"/> interface.</param>
  28. public EpisodeNfoParser(
  29. ILogger logger,
  30. IConfigurationManager config,
  31. IProviderManager providerManager,
  32. IUserManager userManager,
  33. IUserDataManager userDataManager,
  34. IDirectoryService directoryService)
  35. : base(logger, config, providerManager, userManager, userDataManager, directoryService)
  36. {
  37. }
  38. /// <inheritdoc />
  39. protected override void Fetch(MetadataResult<Episode> item, string metadataFile, XmlReaderSettings settings, CancellationToken cancellationToken)
  40. {
  41. item.ResetPeople();
  42. var xmlFile = File.ReadAllText(metadataFile);
  43. var srch = "</episodedetails>";
  44. var index = xmlFile.IndexOf(srch, StringComparison.OrdinalIgnoreCase);
  45. var xml = xmlFile;
  46. if (index != -1)
  47. {
  48. xml = xmlFile.Substring(0, index + srch.Length);
  49. xmlFile = xmlFile.Substring(index + srch.Length);
  50. }
  51. // These are not going to be valid xml so no sense in causing the provider to fail and spamming the log with exceptions
  52. try
  53. {
  54. // Extract episode details from the first episodedetails block
  55. ReadEpisodeDetailsFromXml(item, xml, settings, cancellationToken);
  56. // Extract the last episode number from nfo
  57. // Retrieves all additional episodedetails blocks from the rest of the nfo and concatenates the name, originalTitle and overview tags with the first episode
  58. // This is needed because XBMC metadata uses multiple episodedetails blocks instead of episodenumberend tag
  59. var name = new StringBuilder(item.Item.Name);
  60. var originalTitle = new StringBuilder(item.Item.OriginalTitle);
  61. var overview = new StringBuilder(item.Item.Overview);
  62. while ((index = xmlFile.IndexOf(srch, StringComparison.OrdinalIgnoreCase)) != -1)
  63. {
  64. xml = xmlFile.Substring(0, index + srch.Length);
  65. xmlFile = xmlFile.Substring(index + srch.Length);
  66. var additionalEpisode = new MetadataResult<Episode>()
  67. {
  68. Item = new Episode()
  69. };
  70. // Extract episode details from additional episodedetails block
  71. ReadEpisodeDetailsFromXml(additionalEpisode, xml, settings, cancellationToken);
  72. if (!string.IsNullOrEmpty(additionalEpisode.Item.Name))
  73. {
  74. name.Append(" / ").Append(additionalEpisode.Item.Name);
  75. }
  76. if (!string.IsNullOrEmpty(additionalEpisode.Item.Overview))
  77. {
  78. overview.Append(" / ").Append(additionalEpisode.Item.Overview);
  79. }
  80. if (!string.IsNullOrEmpty(additionalEpisode.Item.OriginalTitle))
  81. {
  82. originalTitle.Append(" / ").Append(additionalEpisode.Item.OriginalTitle);
  83. }
  84. if (additionalEpisode.Item.IndexNumber is not null)
  85. {
  86. item.Item.IndexNumberEnd = Math.Max((int)additionalEpisode.Item.IndexNumber, item.Item.IndexNumberEnd ?? (int)additionalEpisode.Item.IndexNumber);
  87. }
  88. }
  89. item.Item.Name = name.ToString();
  90. item.Item.OriginalTitle = originalTitle.ToString();
  91. item.Item.Overview = overview.ToString();
  92. }
  93. catch (XmlException)
  94. {
  95. }
  96. }
  97. /// <inheritdoc />
  98. protected override void FetchDataFromXmlNode(XmlReader reader, MetadataResult<Episode> itemResult)
  99. {
  100. var item = itemResult.Item;
  101. switch (reader.Name)
  102. {
  103. case "season":
  104. if (reader.TryReadInt(out var seasonNumber))
  105. {
  106. item.ParentIndexNumber = seasonNumber;
  107. }
  108. break;
  109. case "episode":
  110. if (reader.TryReadInt(out var episodeNumber))
  111. {
  112. item.IndexNumber = episodeNumber;
  113. }
  114. break;
  115. case "episodenumberend":
  116. if (reader.TryReadInt(out var episodeNumberEnd))
  117. {
  118. item.IndexNumberEnd = episodeNumberEnd;
  119. }
  120. break;
  121. case "airsbefore_episode":
  122. case "displayepisode":
  123. if (reader.TryReadInt(out var airsBeforeEpisode))
  124. {
  125. item.AirsBeforeEpisodeNumber = airsBeforeEpisode;
  126. }
  127. break;
  128. case "airsafter_season":
  129. case "displayafterseason":
  130. if (reader.TryReadInt(out var airsAfterSeason))
  131. {
  132. item.AirsAfterSeasonNumber = airsAfterSeason;
  133. }
  134. break;
  135. case "airsbefore_season":
  136. case "displayseason":
  137. if (reader.TryReadInt(out var airsBeforeSeason))
  138. {
  139. item.AirsBeforeSeasonNumber = airsBeforeSeason;
  140. }
  141. break;
  142. case "showtitle":
  143. item.SeriesName = reader.ReadNormalizedString();
  144. break;
  145. default:
  146. base.FetchDataFromXmlNode(reader, itemResult);
  147. break;
  148. }
  149. }
  150. /// <summary>
  151. /// Reads the episode details from the given xml and saves the result in the provided result item.
  152. /// </summary>
  153. private void ReadEpisodeDetailsFromXml(MetadataResult<Episode> item, string xml, XmlReaderSettings settings, CancellationToken cancellationToken)
  154. {
  155. using (var stringReader = new StringReader(xml))
  156. using (var reader = XmlReader.Create(stringReader, settings))
  157. {
  158. reader.MoveToContent();
  159. reader.Read();
  160. // Loop through each element
  161. while (!reader.EOF && reader.ReadState == ReadState.Interactive)
  162. {
  163. cancellationToken.ThrowIfCancellationRequested();
  164. if (reader.NodeType == XmlNodeType.Element)
  165. {
  166. FetchDataFromXmlNode(reader, item);
  167. }
  168. else
  169. {
  170. reader.Read();
  171. }
  172. }
  173. }
  174. }
  175. }
  176. }