| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097 | using MediaBrowser.Common.Extensions;using MediaBrowser.Controller.Configuration;using MediaBrowser.Controller.Entities;using MediaBrowser.Controller.Entities.Audio;using MediaBrowser.Controller.Entities.Movies;using MediaBrowser.Controller.Entities.TV;using MediaBrowser.Controller.Library;using MediaBrowser.Controller.Persistence;using MediaBrowser.Model.Configuration;using MediaBrowser.Model.Entities;using MediaBrowser.Model.Logging;using MediaBrowser.XbmcMetadata.Configuration;using System;using System.Collections.Generic;using System.Globalization;using System.IO;using System.Linq;using System.Text;using System.Threading;using System.Xml;using MediaBrowser.Common.IO;using MediaBrowser.Controller.IO;using MediaBrowser.Model.Extensions;using MediaBrowser.Model.IO;using MediaBrowser.Model.Xml;namespace MediaBrowser.XbmcMetadata.Savers{    public abstract class BaseNfoSaver : IMetadataFileSaver    {        private static readonly CultureInfo UsCulture = new CultureInfo("en-US");        private static readonly Dictionary<string, string> CommonTags = new[] {                    "plot",                    "customrating",                    "lockdata",                    "type",                    "dateadded",                    "title",                    "rating",                    "year",                    "sorttitle",                    "mpaa",                    "mpaadescription",                    "aspectratio",                    "website",                    "collectionnumber",                    "tmdbid",                    "rottentomatoesid",                    "language",                    "tvcomid",                    "budget",                    "revenue",                    "tagline",                    "studio",                    "genre",                    "tag",                    "runtime",                    "actor",                    "criticratingsummary",                    "criticrating",                    "fileinfo",                    "director",                    "writer",                    "trailer",                    "premiered",                    "releasedate",                    "outline",                    "id",                    "votes",                    "credits",                    "originaltitle",                    "watched",                    "playcount",                    "lastplayed",                    "art",                    "resume",                    "biography",                    "formed",                    "review",                    "style",                    "imdbid",                    "imdb_id",                    "plotkeyword",                    "country",                    "audiodbalbumid",                    "audiodbartistid",                    "awardsummary",                    "enddate",                    "lockedfields",                    "metascore",                    "zap2itid",                    "tvrageid",                    "gamesdbid",                    "musicbrainzartistid",                    "musicbrainzalbumartistid",                    "musicbrainzalbumid",                    "musicbrainzreleasegroupid",                    "tvdbid",                    "collectionitem",                    "isuserfavorite",                    "userrating",                    "countrycode"        }.ToDictionary(i => i, StringComparer.OrdinalIgnoreCase);        protected BaseNfoSaver(IFileSystem fileSystem, IServerConfigurationManager configurationManager, ILibraryManager libraryManager, IUserManager userManager, IUserDataManager userDataManager, ILogger logger, IXmlReaderSettingsFactory xmlReaderSettingsFactory)        {            Logger = logger;            XmlReaderSettingsFactory = xmlReaderSettingsFactory;            UserDataManager = userDataManager;            UserManager = userManager;            LibraryManager = libraryManager;            ConfigurationManager = configurationManager;            FileSystem = fileSystem;        }        protected IFileSystem FileSystem { get; private set; }        protected IServerConfigurationManager ConfigurationManager { get; private set; }        protected ILibraryManager LibraryManager { get; private set; }        protected IUserManager UserManager { get; private set; }        protected IUserDataManager UserDataManager { get; private set; }        protected ILogger Logger { get; private set; }        protected IXmlReaderSettingsFactory XmlReaderSettingsFactory { get; private set; }        protected ItemUpdateType MinimumUpdateType        {            get            {                if (ConfigurationManager.GetNfoConfiguration().SaveImagePathsInNfo)                {                    return ItemUpdateType.ImageUpdate;                }                return ItemUpdateType.MetadataDownload;            }        }        public string Name        {            get            {                return SaverName;            }        }        public static string SaverName        {            get            {                return "Nfo";            }        }        public string GetSavePath(IHasMetadata item)        {            return GetLocalSavePath(item);        }        /// <summary>        /// Gets the save path.        /// </summary>        /// <param name="item">The item.</param>        /// <returns>System.String.</returns>        protected abstract string GetLocalSavePath(IHasMetadata item);        /// <summary>        /// Gets the name of the root element.        /// </summary>        /// <param name="item">The item.</param>        /// <returns>System.String.</returns>        protected abstract string GetRootElementName(IHasMetadata item);        /// <summary>        /// Determines whether [is enabled for] [the specified item].        /// </summary>        /// <param name="item">The item.</param>        /// <param name="updateType">Type of the update.</param>        /// <returns><c>true</c> if [is enabled for] [the specified item]; otherwise, <c>false</c>.</returns>        public abstract bool IsEnabledFor(IHasMetadata item, ItemUpdateType updateType);        protected virtual List<string> GetTagsUsed()        {            return new List<string>();        }        public void Save(IHasMetadata item, CancellationToken cancellationToken)        {            var path = GetSavePath(item);            using (var memoryStream = new MemoryStream())            {                Save(item, memoryStream, path);                memoryStream.Position = 0;                cancellationToken.ThrowIfCancellationRequested();                SaveToFile(memoryStream, path);            }        }        private void SaveToFile(Stream stream, string path)        {            FileSystem.CreateDirectory(Path.GetDirectoryName(path));            var file = FileSystem.GetFileInfo(path);            var wasHidden = false;            // This will fail if the file is hidden            if (file.Exists)            {                if (file.IsHidden)                {                    FileSystem.SetHidden(path, false);                    wasHidden = true;                }                if (file.IsReadOnly)                {                    FileSystem.SetReadOnly(path, false);                }            }            using (var filestream = FileSystem.GetFileStream(path, FileOpenMode.Create, FileAccessMode.Write, FileShareMode.Read))            {                stream.CopyTo(filestream);            }            if (wasHidden || ConfigurationManager.Configuration.SaveMetadataHidden)            {                FileSystem.SetHidden(path, true);            }        }        private void Save(IHasMetadata item, Stream stream, string xmlPath)        {            var settings = new XmlWriterSettings            {                Indent = true,                Encoding = Encoding.UTF8,                CloseOutput = false            };            using (XmlWriter writer = XmlWriter.Create(stream, settings))            {                var root = GetRootElementName(item);                writer.WriteStartDocument(true);                writer.WriteStartElement(root);                var baseItem = item as BaseItem;                if (baseItem != null)                {                    AddCommonNodes(baseItem, writer, LibraryManager, UserManager, UserDataManager, FileSystem, ConfigurationManager);                }                WriteCustomElements(item, writer);                var hasMediaSources = baseItem as IHasMediaSources;                if (hasMediaSources != null)                {                    AddMediaInfo(hasMediaSources, writer);                }                var tagsUsed = GetTagsUsed();                try                {                    AddCustomTags(xmlPath, tagsUsed, writer, Logger, FileSystem);                }                catch (FileNotFoundException)                {                }                catch (IOException)                {                }                catch (XmlException ex)                {                    Logger.ErrorException("Error reading existng nfo", ex);                }                writer.WriteEndElement();                writer.WriteEndDocument();            }        }        protected abstract void WriteCustomElements(IHasMetadata item, XmlWriter writer);        public static void AddMediaInfo<T>(T item, XmlWriter writer)         where T : IHasMediaSources        {            writer.WriteStartElement("fileinfo");            writer.WriteStartElement("streamdetails");            var mediaSource = item.GetMediaSources(false).First();            foreach (var stream in mediaSource.MediaStreams)            {                writer.WriteStartElement(stream.Type.ToString().ToLower());                if (!string.IsNullOrEmpty(stream.Codec))                {                    var codec = stream.Codec;                    if ((stream.CodecTag ?? string.Empty).IndexOf("xvid", StringComparison.OrdinalIgnoreCase) != -1)                    {                        codec = "xvid;";                    }                    else if ((stream.CodecTag ?? string.Empty).IndexOf("divx", StringComparison.OrdinalIgnoreCase) != -1)                    {                        codec = "divx;";                    }                    writer.WriteElementString("codec", codec);                    writer.WriteElementString("micodec", codec);                }                if (stream.BitRate.HasValue)                {                    writer.WriteElementString("bitrate", stream.BitRate.Value.ToString(UsCulture));                }                if (stream.Width.HasValue)                {                    writer.WriteElementString("width", stream.Width.Value.ToString(UsCulture));                }                if (stream.Height.HasValue)                {                    writer.WriteElementString("height", stream.Height.Value.ToString(UsCulture));                }                if (!string.IsNullOrEmpty(stream.AspectRatio))                {                    writer.WriteElementString("aspect", stream.AspectRatio);                    writer.WriteElementString("aspectratio", stream.AspectRatio);                }                var framerate = stream.AverageFrameRate ?? stream.RealFrameRate;                if (framerate.HasValue)                {                    writer.WriteElementString("framerate", framerate.Value.ToString(UsCulture));                }                if (!string.IsNullOrEmpty(stream.Language))                {                    writer.WriteElementString("language", stream.Language);                }                var scanType = stream.IsInterlaced ? "interlaced" : "progressive";                if (!string.IsNullOrEmpty(scanType))                {                    writer.WriteElementString("scantype", scanType);                }                if (stream.Channels.HasValue)                {                    writer.WriteElementString("channels", stream.Channels.Value.ToString(UsCulture));                }                if (stream.SampleRate.HasValue)                {                    writer.WriteElementString("samplingrate", stream.SampleRate.Value.ToString(UsCulture));                }                writer.WriteElementString("default", stream.IsDefault.ToString());                writer.WriteElementString("forced", stream.IsForced.ToString());                if (stream.Type == MediaStreamType.Video)                {                    if (mediaSource.RunTimeTicks.HasValue)                    {                        var timespan = TimeSpan.FromTicks(mediaSource.RunTimeTicks.Value);                        writer.WriteElementString("duration", Convert.ToInt32(timespan.TotalMinutes).ToString(UsCulture));                        writer.WriteElementString("durationinseconds", Convert.ToInt32(timespan.TotalSeconds).ToString(UsCulture));                    }                    var video = item as Video;                    if (video != null)                    {                        //AddChapters(video, builder, itemRepository);                        if (video.Video3DFormat.HasValue)                        {                            switch (video.Video3DFormat.Value)                            {                                case Video3DFormat.FullSideBySide:                                    writer.WriteElementString("format3d", "FSBS");                                    break;                                case Video3DFormat.FullTopAndBottom:                                    writer.WriteElementString("format3d", "FTAB");                                    break;                                case Video3DFormat.HalfSideBySide:                                    writer.WriteElementString("format3d", "HSBS");                                    break;                                case Video3DFormat.HalfTopAndBottom:                                    writer.WriteElementString("format3d", "HTAB");                                    break;                                case Video3DFormat.MVC:                                    writer.WriteElementString("format3d", "MVC");                                    break;                            }                        }                    }                }                writer.WriteEndElement();            }            writer.WriteEndElement();            writer.WriteEndElement();        }        public const string DateAddedFormat = "yyyy-MM-dd HH:mm:ss";        /// <summary>        /// Adds the common nodes.        /// </summary>        /// <returns>Task.</returns>        public static void AddCommonNodes(BaseItem item, XmlWriter writer, ILibraryManager libraryManager, IUserManager userManager, IUserDataManager userDataRepo, IFileSystem fileSystem, IServerConfigurationManager config)        {            var writtenProviderIds = new HashSet<string>(StringComparer.OrdinalIgnoreCase);            var overview = (item.Overview ?? string.Empty)                .StripHtml()                .Replace(""", "'");            var options = config.GetNfoConfiguration();            if (item is MusicArtist)            {                writer.WriteElementString("biography", overview);            }            else if (item is MusicAlbum)            {                writer.WriteElementString("review", overview);            }            else            {                writer.WriteElementString("plot", overview);            }            if (item is Video)            {                var outline = (item.ShortOverview ?? string.Empty)                    .StripHtml()                    .Replace(""", "'");                writer.WriteElementString("outline", outline);            }            else            {                writer.WriteElementString("outline", overview);            }            if (!string.IsNullOrWhiteSpace(item.CustomRating))            {                writer.WriteElementString("customrating", item.CustomRating);            }            writer.WriteElementString("lockdata", item.IsLocked.ToString().ToLower());            if (item.LockedFields.Count > 0)            {                writer.WriteElementString("lockedfields", string.Join("|", item.LockedFields.Select(i => i.ToString()).ToArray()));            }            if (!string.IsNullOrEmpty(item.DisplayMediaType))            {                writer.WriteElementString("type", item.DisplayMediaType);            }            writer.WriteElementString("dateadded", item.DateCreated.ToLocalTime().ToString(DateAddedFormat));            writer.WriteElementString("title", item.Name ?? string.Empty);            if (!string.IsNullOrWhiteSpace(item.OriginalTitle))            {                writer.WriteElementString("originaltitle", item.OriginalTitle);            }            var people = libraryManager.GetPeople(item);            var directors = people                .Where(i => IsPersonType(i, PersonType.Director))                .Select(i => i.Name)                .ToList();            foreach (var person in directors)            {                writer.WriteElementString("director", person);            }            var writers = people                .Where(i => IsPersonType(i, PersonType.Writer))                .Select(i => i.Name)                .Distinct(StringComparer.OrdinalIgnoreCase)                .ToList();            foreach (var person in writers)            {                writer.WriteElementString("writer", person);            }            foreach (var person in writers)            {                writer.WriteElementString("credits", person);            }            var hasTrailer = item as IHasTrailers;            if (hasTrailer != null)            {                foreach (var trailer in hasTrailer.RemoteTrailers)                {                    writer.WriteElementString("trailer", GetOutputTrailerUrl(trailer.Url));                }            }            if (item.CommunityRating.HasValue)            {                writer.WriteElementString("rating", item.CommunityRating.Value.ToString(UsCulture));            }            if (item.ProductionYear.HasValue)            {                writer.WriteElementString("year", item.ProductionYear.Value.ToString(UsCulture));            }            if (!string.IsNullOrEmpty(item.ForcedSortName))            {                writer.WriteElementString("sorttitle", item.ForcedSortName);            }            if (!string.IsNullOrEmpty(item.OfficialRating))            {                writer.WriteElementString("mpaa", item.OfficialRating);            }            if (!string.IsNullOrEmpty(item.OfficialRatingDescription))            {                writer.WriteElementString("mpaadescription", item.OfficialRatingDescription);            }            var hasAspectRatio = item as IHasAspectRatio;            if (hasAspectRatio != null)            {                if (!string.IsNullOrEmpty(hasAspectRatio.AspectRatio))                {                    writer.WriteElementString("aspectratio", hasAspectRatio.AspectRatio);                }            }            if (!string.IsNullOrEmpty(item.HomePageUrl))            {                writer.WriteElementString("website", item.HomePageUrl);            }            var rt = item.GetProviderId(MetadataProviders.RottenTomatoes);            if (!string.IsNullOrEmpty(rt))            {                writer.WriteElementString("rottentomatoesid", rt);                writtenProviderIds.Add(MetadataProviders.RottenTomatoes.ToString());            }            var tmdbCollection = item.GetProviderId(MetadataProviders.TmdbCollection);            if (!string.IsNullOrEmpty(tmdbCollection))            {                writer.WriteElementString("collectionnumber", tmdbCollection);                writtenProviderIds.Add(MetadataProviders.TmdbCollection.ToString());            }            var imdb = item.GetProviderId(MetadataProviders.Imdb);            if (!string.IsNullOrEmpty(imdb))            {                if (item is Series)                {                    writer.WriteElementString("imdb_id", imdb);                }                else                {                    writer.WriteElementString("imdbid", imdb);                }                writtenProviderIds.Add(MetadataProviders.Imdb.ToString());            }            // Series xml saver already saves this            if (!(item is Series))            {                var tvdb = item.GetProviderId(MetadataProviders.Tvdb);                if (!string.IsNullOrEmpty(tvdb))                {                    writer.WriteElementString("tvdbid", tvdb);                    writtenProviderIds.Add(MetadataProviders.Tvdb.ToString());                }            }            var tmdb = item.GetProviderId(MetadataProviders.Tmdb);            if (!string.IsNullOrEmpty(tmdb))            {                writer.WriteElementString("tmdbid", tmdb);                writtenProviderIds.Add(MetadataProviders.Tmdb.ToString());            }            var tvcom = item.GetProviderId(MetadataProviders.Tvcom);            if (!string.IsNullOrEmpty(tvcom))            {                writer.WriteElementString("tvcomid", tvcom);                writtenProviderIds.Add(MetadataProviders.Tvcom.ToString());            }            if (!string.IsNullOrEmpty(item.PreferredMetadataLanguage))            {                writer.WriteElementString("language", item.PreferredMetadataLanguage);            }            if (!string.IsNullOrEmpty(item.PreferredMetadataCountryCode))            {                writer.WriteElementString("countrycode", item.PreferredMetadataCountryCode);            }            if (item.PremiereDate.HasValue && !(item is Episode))            {                var formatString = options.ReleaseDateFormat;                if (item is MusicArtist)                {                    writer.WriteElementString("formed", item.PremiereDate.Value.ToLocalTime().ToString(formatString));                }                else                {                    writer.WriteElementString("premiered", item.PremiereDate.Value.ToLocalTime().ToString(formatString));                    writer.WriteElementString("releasedate", item.PremiereDate.Value.ToLocalTime().ToString(formatString));                }            }            if (item.EndDate.HasValue)            {                if (!(item is Episode))                {                    var formatString = options.ReleaseDateFormat;                    writer.WriteElementString("enddate", item.EndDate.Value.ToLocalTime().ToString(formatString));                }            }            if (item.CriticRating.HasValue)            {                writer.WriteElementString("criticrating", item.CriticRating.Value.ToString(UsCulture));            }            if (!string.IsNullOrEmpty(item.CriticRatingSummary))            {                writer.WriteElementString("criticratingsummary", item.CriticRatingSummary);            }            var hasDisplayOrder = item as IHasDisplayOrder;            if (hasDisplayOrder != null)            {                if (!string.IsNullOrEmpty(hasDisplayOrder.DisplayOrder))                {                    writer.WriteElementString("displayorder", hasDisplayOrder.DisplayOrder);                }            }            if (item.VoteCount.HasValue)            {                writer.WriteElementString("votes", item.VoteCount.Value.ToString(UsCulture));            }            var hasBudget = item as IHasBudget;            if (hasBudget != null)            {                if (hasBudget.Budget.HasValue)                {                    writer.WriteElementString("budget", hasBudget.Budget.Value.ToString(UsCulture));                }                if (hasBudget.Revenue.HasValue)                {                    writer.WriteElementString("revenue", hasBudget.Revenue.Value.ToString(UsCulture));                }            }            var hasMetascore = item as IHasMetascore;            if (hasMetascore != null && hasMetascore.Metascore.HasValue)            {                writer.WriteElementString("metascore", hasMetascore.Metascore.Value.ToString(UsCulture));            }            // Use original runtime here, actual file runtime later in MediaInfo            var runTimeTicks = item.RunTimeTicks;            if (runTimeTicks.HasValue)            {                var timespan = TimeSpan.FromTicks(runTimeTicks.Value);                writer.WriteElementString("runtime", Convert.ToInt32(timespan.TotalMinutes).ToString(UsCulture));            }            if (!string.IsNullOrWhiteSpace(item.Tagline))            {                writer.WriteElementString("tagline", item.Tagline);            }            foreach (var country in item.ProductionLocations)            {                writer.WriteElementString("country", country);            }            foreach (var genre in item.Genres)            {                writer.WriteElementString("genre", genre);            }            foreach (var studio in item.Studios)            {                writer.WriteElementString("studio", studio);            }            foreach (var tag in item.Tags)            {                if (item is MusicAlbum || item is MusicArtist)                {                    writer.WriteElementString("style", tag);                }                else                {                    writer.WriteElementString("tag", tag);                }            }            foreach (var tag in item.Keywords)            {                writer.WriteElementString("plotkeyword", tag);            }            var hasAwards = item as IHasAwards;            if (hasAwards != null && !string.IsNullOrEmpty(hasAwards.AwardSummary))            {                writer.WriteElementString("awardsummary", hasAwards.AwardSummary);            }            var externalId = item.GetProviderId(MetadataProviders.AudioDbArtist);            if (!string.IsNullOrEmpty(externalId))            {                writer.WriteElementString("audiodbartistid", externalId);                writtenProviderIds.Add(MetadataProviders.AudioDbArtist.ToString());            }            externalId = item.GetProviderId(MetadataProviders.AudioDbAlbum);            if (!string.IsNullOrEmpty(externalId))            {                writer.WriteElementString("audiodbalbumid", externalId);                writtenProviderIds.Add(MetadataProviders.AudioDbAlbum.ToString());            }            externalId = item.GetProviderId(MetadataProviders.Zap2It);            if (!string.IsNullOrEmpty(externalId))            {                writer.WriteElementString("zap2itid", externalId);                writtenProviderIds.Add(MetadataProviders.Zap2It.ToString());            }            externalId = item.GetProviderId(MetadataProviders.MusicBrainzAlbum);            if (!string.IsNullOrEmpty(externalId))            {                writer.WriteElementString("musicbrainzalbumid", externalId);                writtenProviderIds.Add(MetadataProviders.MusicBrainzAlbum.ToString());            }            externalId = item.GetProviderId(MetadataProviders.MusicBrainzAlbumArtist);            if (!string.IsNullOrEmpty(externalId))            {                writer.WriteElementString("musicbrainzalbumartistid", externalId);                writtenProviderIds.Add(MetadataProviders.MusicBrainzAlbumArtist.ToString());            }            externalId = item.GetProviderId(MetadataProviders.MusicBrainzArtist);            if (!string.IsNullOrEmpty(externalId))            {                writer.WriteElementString("musicbrainzartistid", externalId);                writtenProviderIds.Add(MetadataProviders.MusicBrainzArtist.ToString());            }            externalId = item.GetProviderId(MetadataProviders.MusicBrainzReleaseGroup);            if (!string.IsNullOrEmpty(externalId))            {                writer.WriteElementString("musicbrainzreleasegroupid", externalId);                writtenProviderIds.Add(MetadataProviders.MusicBrainzReleaseGroup.ToString());            }            externalId = item.GetProviderId(MetadataProviders.Gamesdb);            if (!string.IsNullOrEmpty(externalId))            {                writer.WriteElementString("gamesdbid", externalId);                writtenProviderIds.Add(MetadataProviders.Gamesdb.ToString());            }            externalId = item.GetProviderId(MetadataProviders.TvRage);            if (!string.IsNullOrEmpty(externalId))            {                writer.WriteElementString("tvrageid", externalId);                writtenProviderIds.Add(MetadataProviders.TvRage.ToString());            }            if (item.ProviderIds != null)            {                foreach (var providerKey in item.ProviderIds.Keys)                {                    var providerId = item.ProviderIds[providerKey];                    if (!string.IsNullOrEmpty(providerId) && !writtenProviderIds.Contains(providerKey))                    {                        writer.WriteElementString(providerKey.ToLower() + "id", providerId);                    }                }            }            if (options.SaveImagePathsInNfo)            {                AddImages(item, writer, libraryManager, config);            }            AddUserData(item, writer, userManager, userDataRepo, options);            AddActors(people, writer, libraryManager, fileSystem, config, options.SaveImagePathsInNfo);            var folder = item as BoxSet;            if (folder != null)            {                AddCollectionItems(folder, writer);            }        }        public static void AddChapters(Video item, XmlWriter writer, IItemRepository repository)        {            var chapters = repository.GetChapters(item.Id);            foreach (var chapter in chapters)            {                writer.WriteStartElement("chapter");                writer.WriteElementString("name", chapter.Name);                var time = TimeSpan.FromTicks(chapter.StartPositionTicks);                var ms = Convert.ToInt64(time.TotalMilliseconds);                writer.WriteElementString("startpositionms", ms.ToString(UsCulture));                writer.WriteEndElement();            }        }        private static void AddCollectionItems(Folder item, XmlWriter writer)        {            var items = item.LinkedChildren                .Where(i => i.Type == LinkedChildType.Manual)                .ToList();            foreach (var link in items)            {                writer.WriteStartElement("collectionitem");                if (!string.IsNullOrWhiteSpace(link.Path))                {                    writer.WriteElementString("path", link.Path);                }                writer.WriteEndElement();            }        }        /// <summary>        /// Gets the output trailer URL.        /// </summary>        /// <param name="url">The URL.</param>        /// <returns>System.String.</returns>        private static string GetOutputTrailerUrl(string url)        {            // This is what xbmc expects            return url.Replace("https://www.youtube.com/watch?v=",                "plugin://plugin.video.youtube/?action=play_video&videoid=",                StringComparison.OrdinalIgnoreCase);        }        private static void AddImages(BaseItem item, XmlWriter writer, ILibraryManager libraryManager, IServerConfigurationManager config)        {            writer.WriteStartElement("art");            var image = item.GetImageInfo(ImageType.Primary, 0);            if (image != null)            {                writer.WriteElementString("poster", GetImagePathToSave(image, libraryManager, config));            }            foreach (var backdrop in item.GetImages(ImageType.Backdrop))            {                writer.WriteElementString("fanart", GetImagePathToSave(backdrop, libraryManager, config));            }            writer.WriteEndElement();        }        private static void AddUserData(BaseItem item, XmlWriter writer, IUserManager userManager, IUserDataManager userDataRepo, XbmcMetadataOptions options)        {            var userId = options.UserId;            if (string.IsNullOrWhiteSpace(userId))            {                return;            }            var user = userManager.GetUserById(userId);            if (user == null)            {                return;            }            if (item.IsFolder)            {                return;            }            var userdata = userDataRepo.GetUserData(user, item);            writer.WriteElementString("isuserfavorite", userdata.IsFavorite.ToString().ToLower());            if (userdata.Rating.HasValue)            {                writer.WriteElementString("userrating", userdata.Rating.Value.ToString(CultureInfo.InvariantCulture).ToLower());            }            if (!item.IsFolder)            {                writer.WriteElementString("playcount", userdata.PlayCount.ToString(UsCulture));                writer.WriteElementString("watched", userdata.Played.ToString().ToLower());                if (userdata.LastPlayedDate.HasValue)                {                    writer.WriteElementString("lastplayed", userdata.LastPlayedDate.Value.ToLocalTime().ToString("yyyy-MM-dd HH:mm:ss").ToLower());                }                writer.WriteStartElement("resume");                var runTimeTicks = item.RunTimeTicks ?? 0;                writer.WriteElementString("position", TimeSpan.FromTicks(userdata.PlaybackPositionTicks).TotalSeconds.ToString(UsCulture));                writer.WriteElementString("total", TimeSpan.FromTicks(runTimeTicks).TotalSeconds.ToString(UsCulture));            }            writer.WriteEndElement();        }        private static void AddActors(List<PersonInfo> people, XmlWriter writer, ILibraryManager libraryManager, IFileSystem fileSystem, IServerConfigurationManager config, bool saveImagePath)        {            var actors = people                .Where(i => !IsPersonType(i, PersonType.Director) && !IsPersonType(i, PersonType.Writer))                .ToList();            foreach (var person in actors)            {                writer.WriteStartElement("actor");                if (!string.IsNullOrWhiteSpace(person.Name))                {                    writer.WriteElementString("name", person.Name);                }                if (!string.IsNullOrWhiteSpace(person.Role))                {                    writer.WriteElementString("role", person.Role);                }                if (!string.IsNullOrWhiteSpace(person.Type))                {                    writer.WriteElementString("type", person.Type);                }                if (person.SortOrder.HasValue)                {                    writer.WriteElementString("sortorder", person.SortOrder.Value.ToString(UsCulture));                }                if (saveImagePath)                {                    try                    {                        var personEntity = libraryManager.GetPerson(person.Name);                        var image = personEntity.GetImageInfo(ImageType.Primary, 0);                        if (image != null)                        {                            writer.WriteElementString("thumb", GetImagePathToSave(image, libraryManager, config));                        }                    }                    catch (Exception)                    {                        // Already logged in core                    }                }                writer.WriteEndElement();            }        }        private static string GetImagePathToSave(ItemImageInfo image, ILibraryManager libraryManager, IServerConfigurationManager config)        {            if (!image.IsLocalFile)            {                return image.Path;            }            return libraryManager.GetPathAfterNetworkSubstitution(image.Path);        }        private static bool IsPersonType(PersonInfo person, string type)        {            return string.Equals(person.Type, type, StringComparison.OrdinalIgnoreCase) || string.Equals(person.Role, type, StringComparison.OrdinalIgnoreCase);        }        private void AddCustomTags(string path, List<string> xmlTagsUsed, XmlWriter writer, ILogger logger, IFileSystem fileSystem)        {            var settings = XmlReaderSettingsFactory.Create(false);            settings.CheckCharacters = false;            settings.IgnoreProcessingInstructions = true;            settings.IgnoreComments = true;            using (var fileStream = fileSystem.OpenRead(path))            {                using (var streamReader = new StreamReader(fileStream, Encoding.UTF8))                {                    // Use XmlReader for best performance                    using (var reader = XmlReader.Create(streamReader, settings))                    {                        try                        {                            reader.MoveToContent();                        }                        catch (Exception ex)                        {                            logger.ErrorException("Error reading existing xml tags from {0}.", ex, path);                            return;                        }                        reader.Read();                        // Loop through each element                        while (!reader.EOF && reader.ReadState == ReadState.Interactive)                        {                            if (reader.NodeType == XmlNodeType.Element)                            {                                var name = reader.Name;                                if (!CommonTags.ContainsKey(name) && !xmlTagsUsed.Contains(name, StringComparer.OrdinalIgnoreCase))                                {                                    writer.WriteNode(reader, false);                                }                                else                                {                                    reader.Skip();                                }                            }                            else                            {                                reader.Read();                            }                        }                    }                }            }        }    }}
 |