2015-06-28 18:10:45 -07:00
|
|
|
|
using MediaBrowser.Controller.Entities.TV;
|
|
|
|
|
using MediaBrowser.Controller.Providers;
|
|
|
|
|
using MediaBrowser.Model.Entities;
|
|
|
|
|
using MediaBrowser.Model.Logging;
|
|
|
|
|
using System;
|
2014-02-08 21:52:52 -07:00
|
|
|
|
using System.Collections.Generic;
|
2013-11-17 08:27:48 -07:00
|
|
|
|
using System.Globalization;
|
2013-02-20 18:33:05 -07:00
|
|
|
|
using System.IO;
|
2013-08-12 12:18:31 -07:00
|
|
|
|
using System.Threading;
|
2013-02-20 18:33:05 -07:00
|
|
|
|
using System.Xml;
|
2015-10-03 21:23:11 -07:00
|
|
|
|
using CommonIO;
|
2013-02-20 18:33:05 -07:00
|
|
|
|
|
2014-06-29 20:04:50 -07:00
|
|
|
|
namespace MediaBrowser.LocalMetadata.Parsers
|
2013-02-20 18:33:05 -07:00
|
|
|
|
{
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Class EpisodeXmlParser
|
|
|
|
|
/// </summary>
|
|
|
|
|
public class EpisodeXmlParser : BaseItemXmlParser<Episode>
|
|
|
|
|
{
|
2014-02-08 21:52:52 -07:00
|
|
|
|
private List<LocalImageInfo> _imagesFound;
|
2015-10-03 20:38:46 -07:00
|
|
|
|
private readonly IFileSystem _fileSystem;
|
2014-02-08 21:52:52 -07:00
|
|
|
|
|
2016-08-26 21:33:18 -07:00
|
|
|
|
public EpisodeXmlParser(ILogger logger, IFileSystem fileSystem, IProviderManager providerManager)
|
|
|
|
|
: base(logger, providerManager)
|
2013-02-21 13:26:35 -07:00
|
|
|
|
{
|
2015-10-03 20:38:46 -07:00
|
|
|
|
_fileSystem = fileSystem;
|
2013-08-12 12:18:31 -07:00
|
|
|
|
}
|
|
|
|
|
|
2014-02-20 22:04:11 -07:00
|
|
|
|
private string _xmlPath;
|
|
|
|
|
|
2015-06-28 18:10:45 -07:00
|
|
|
|
public void Fetch(MetadataResult<Episode> item,
|
2014-05-14 13:55:16 -07:00
|
|
|
|
List<LocalImageInfo> images,
|
|
|
|
|
string metadataFile,
|
|
|
|
|
CancellationToken cancellationToken)
|
2013-08-12 12:18:31 -07:00
|
|
|
|
{
|
2014-02-08 21:52:52 -07:00
|
|
|
|
_imagesFound = images;
|
2014-02-20 22:04:11 -07:00
|
|
|
|
_xmlPath = metadataFile;
|
2014-02-08 21:52:52 -07:00
|
|
|
|
|
|
|
|
|
Fetch(item, metadataFile, cancellationToken);
|
2013-02-21 13:26:35 -07:00
|
|
|
|
}
|
|
|
|
|
|
2013-11-17 08:27:48 -07:00
|
|
|
|
private static readonly CultureInfo UsCulture = new CultureInfo("en-US");
|
2013-12-08 13:33:24 -07:00
|
|
|
|
|
2013-02-20 18:33:05 -07:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Fetches the data from XML node.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="reader">The reader.</param>
|
2015-06-28 18:10:45 -07:00
|
|
|
|
/// <param name="result">The result.</param>
|
|
|
|
|
protected override void FetchDataFromXmlNode(XmlReader reader, MetadataResult<Episode> result)
|
2013-02-20 18:33:05 -07:00
|
|
|
|
{
|
2015-06-28 18:10:45 -07:00
|
|
|
|
var item = result.Item;
|
|
|
|
|
|
2013-02-20 18:33:05 -07:00
|
|
|
|
switch (reader.Name)
|
|
|
|
|
{
|
|
|
|
|
case "Episode":
|
2013-08-12 12:18:31 -07:00
|
|
|
|
|
2013-02-20 18:33:05 -07:00
|
|
|
|
//MB generated metadata is within an "Episode" node
|
|
|
|
|
using (var subTree = reader.ReadSubtree())
|
|
|
|
|
{
|
|
|
|
|
subTree.MoveToContent();
|
|
|
|
|
|
|
|
|
|
// Loop through each element
|
|
|
|
|
while (subTree.Read())
|
|
|
|
|
{
|
|
|
|
|
if (subTree.NodeType == XmlNodeType.Element)
|
|
|
|
|
{
|
2015-06-28 18:10:45 -07:00
|
|
|
|
FetchDataFromXmlNode(subTree, result);
|
2013-02-20 18:33:05 -07:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case "filename":
|
|
|
|
|
{
|
2013-04-18 08:56:50 -07:00
|
|
|
|
var filename = reader.ReadElementContentAsString();
|
2013-02-20 18:33:05 -07:00
|
|
|
|
|
|
|
|
|
if (!string.IsNullOrWhiteSpace(filename))
|
|
|
|
|
{
|
2014-02-08 21:52:52 -07:00
|
|
|
|
// Strip off everything but the filename. Some metadata tools like MetaBrowser v1.0 will have an 'episodes' prefix
|
|
|
|
|
// even though it's actually using the metadata folder.
|
|
|
|
|
filename = Path.GetFileName(filename);
|
2013-02-20 18:33:05 -07:00
|
|
|
|
|
2014-02-20 22:04:11 -07:00
|
|
|
|
var parentFolder = Path.GetDirectoryName(_xmlPath);
|
|
|
|
|
filename = Path.Combine(parentFolder, filename);
|
2015-10-03 20:38:46 -07:00
|
|
|
|
var file = _fileSystem.GetFileInfo(filename);
|
2013-04-18 08:56:50 -07:00
|
|
|
|
|
2014-02-08 21:52:52 -07:00
|
|
|
|
if (file.Exists)
|
|
|
|
|
{
|
|
|
|
|
_imagesFound.Add(new LocalImageInfo
|
|
|
|
|
{
|
|
|
|
|
Type = ImageType.Primary,
|
|
|
|
|
FileInfo = file
|
|
|
|
|
});
|
|
|
|
|
}
|
2013-02-20 18:33:05 -07:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case "SeasonNumber":
|
|
|
|
|
{
|
|
|
|
|
var number = reader.ReadElementContentAsString();
|
|
|
|
|
|
|
|
|
|
if (!string.IsNullOrWhiteSpace(number))
|
|
|
|
|
{
|
|
|
|
|
int num;
|
|
|
|
|
|
|
|
|
|
if (int.TryParse(number, out num))
|
|
|
|
|
{
|
|
|
|
|
item.ParentIndexNumber = num;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case "EpisodeNumber":
|
|
|
|
|
{
|
|
|
|
|
var number = reader.ReadElementContentAsString();
|
|
|
|
|
|
|
|
|
|
if (!string.IsNullOrWhiteSpace(number))
|
|
|
|
|
{
|
|
|
|
|
int num;
|
|
|
|
|
|
|
|
|
|
if (int.TryParse(number, out num))
|
|
|
|
|
{
|
|
|
|
|
item.IndexNumber = num;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2013-10-25 07:26:22 -07:00
|
|
|
|
case "EpisodeNumberEnd":
|
|
|
|
|
{
|
|
|
|
|
var number = reader.ReadElementContentAsString();
|
|
|
|
|
|
|
|
|
|
if (!string.IsNullOrWhiteSpace(number))
|
|
|
|
|
{
|
|
|
|
|
int num;
|
|
|
|
|
|
|
|
|
|
if (int.TryParse(number, out num))
|
|
|
|
|
{
|
|
|
|
|
item.IndexNumberEnd = num;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-10 13:42:42 -07:00
|
|
|
|
case "absolute_number":
|
|
|
|
|
{
|
|
|
|
|
var val = reader.ReadElementContentAsString();
|
|
|
|
|
|
|
|
|
|
if (!string.IsNullOrWhiteSpace(val))
|
|
|
|
|
{
|
|
|
|
|
int rval;
|
|
|
|
|
|
|
|
|
|
// int.TryParse is local aware, so it can be probamatic, force us culture
|
|
|
|
|
if (int.TryParse(val, NumberStyles.Integer, UsCulture, out rval))
|
|
|
|
|
{
|
|
|
|
|
item.AbsoluteEpisodeNumber = rval;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
}
|
2013-12-08 11:07:45 -07:00
|
|
|
|
case "DVD_episodenumber":
|
|
|
|
|
{
|
|
|
|
|
var number = reader.ReadElementContentAsString();
|
|
|
|
|
|
|
|
|
|
if (!string.IsNullOrWhiteSpace(number))
|
|
|
|
|
{
|
2013-12-08 13:33:24 -07:00
|
|
|
|
float num;
|
2013-12-08 11:07:45 -07:00
|
|
|
|
|
2013-12-08 13:33:24 -07:00
|
|
|
|
if (float.TryParse(number, NumberStyles.Any, UsCulture, out num))
|
2013-12-08 11:07:45 -07:00
|
|
|
|
{
|
2013-12-08 15:16:59 -07:00
|
|
|
|
item.DvdEpisodeNumber = num;
|
2013-12-08 11:07:45 -07:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case "DVD_season":
|
|
|
|
|
{
|
|
|
|
|
var number = reader.ReadElementContentAsString();
|
|
|
|
|
|
|
|
|
|
if (!string.IsNullOrWhiteSpace(number))
|
|
|
|
|
{
|
2013-12-08 13:33:24 -07:00
|
|
|
|
float num;
|
2013-12-08 11:07:45 -07:00
|
|
|
|
|
2013-12-08 13:33:24 -07:00
|
|
|
|
if (float.TryParse(number, NumberStyles.Any, UsCulture, out num))
|
2013-12-08 11:07:45 -07:00
|
|
|
|
{
|
2013-12-08 13:33:24 -07:00
|
|
|
|
item.DvdSeasonNumber = Convert.ToInt32(num);
|
2013-12-08 11:07:45 -07:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2013-11-17 08:27:48 -07:00
|
|
|
|
case "airsbefore_episode":
|
2013-11-15 14:31:33 -07:00
|
|
|
|
{
|
2013-11-17 08:27:48 -07:00
|
|
|
|
var val = reader.ReadElementContentAsString();
|
2013-11-15 14:31:33 -07:00
|
|
|
|
|
2013-11-17 08:27:48 -07:00
|
|
|
|
if (!string.IsNullOrWhiteSpace(val))
|
2013-11-15 14:31:33 -07:00
|
|
|
|
{
|
2013-11-17 08:27:48 -07:00
|
|
|
|
int rval;
|
2013-11-15 14:31:33 -07:00
|
|
|
|
|
2013-11-17 08:27:48 -07:00
|
|
|
|
// int.TryParse is local aware, so it can be probamatic, force us culture
|
|
|
|
|
if (int.TryParse(val, NumberStyles.Integer, UsCulture, out rval))
|
2013-11-15 14:31:33 -07:00
|
|
|
|
{
|
2013-11-17 08:27:48 -07:00
|
|
|
|
item.AirsBeforeEpisodeNumber = rval;
|
2013-11-15 14:31:33 -07:00
|
|
|
|
}
|
|
|
|
|
}
|
2013-11-17 08:27:48 -07:00
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case "airsafter_season":
|
|
|
|
|
{
|
|
|
|
|
var val = reader.ReadElementContentAsString();
|
|
|
|
|
|
|
|
|
|
if (!string.IsNullOrWhiteSpace(val))
|
|
|
|
|
{
|
|
|
|
|
int rval;
|
|
|
|
|
|
|
|
|
|
// int.TryParse is local aware, so it can be probamatic, force us culture
|
|
|
|
|
if (int.TryParse(val, NumberStyles.Integer, UsCulture, out rval))
|
|
|
|
|
{
|
|
|
|
|
item.AirsAfterSeasonNumber = rval;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case "airsbefore_season":
|
|
|
|
|
{
|
|
|
|
|
var val = reader.ReadElementContentAsString();
|
|
|
|
|
|
|
|
|
|
if (!string.IsNullOrWhiteSpace(val))
|
|
|
|
|
{
|
|
|
|
|
int rval;
|
|
|
|
|
|
|
|
|
|
// int.TryParse is local aware, so it can be probamatic, force us culture
|
|
|
|
|
if (int.TryParse(val, NumberStyles.Integer, UsCulture, out rval))
|
|
|
|
|
{
|
|
|
|
|
item.AirsBeforeSeasonNumber = rval;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2013-11-15 14:31:33 -07:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2013-02-20 18:33:05 -07:00
|
|
|
|
case "EpisodeName":
|
|
|
|
|
{
|
|
|
|
|
var name = reader.ReadElementContentAsString();
|
|
|
|
|
|
|
|
|
|
if (!string.IsNullOrWhiteSpace(name))
|
|
|
|
|
{
|
|
|
|
|
item.Name = name;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
default:
|
2015-06-28 18:10:45 -07:00
|
|
|
|
base.FetchDataFromXmlNode(reader, result);
|
2013-02-20 18:33:05 -07:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|