12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040 |
- #pragma warning disable CS1591
- using System;
- using System.Collections.Generic;
- using System.Globalization;
- using System.IO;
- using System.Linq;
- using System.Text;
- using System.Text.RegularExpressions;
- using System.Threading;
- using System.Threading.Tasks;
- using System.Xml;
- using Jellyfin.Data.Enums;
- using Jellyfin.Extensions;
- 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.Model.Configuration;
- using MediaBrowser.Model.Entities;
- using MediaBrowser.Model.IO;
- using MediaBrowser.XbmcMetadata.Configuration;
- using Microsoft.Extensions.Logging;
- namespace MediaBrowser.XbmcMetadata.Savers
- {
- public abstract partial class BaseNfoSaver : IMetadataFileSaver
- {
- public const string DateAddedFormat = "yyyy-MM-dd HH:mm:ss";
- public const string YouTubeWatchUrl = "https://www.youtube.com/watch?v=";
- private static readonly HashSet<string> _commonTags = new HashSet<string>(StringComparer.OrdinalIgnoreCase)
- {
- "plot",
- "customrating",
- "lockdata",
- "dateadded",
- "title",
- "rating",
- "year",
- "sorttitle",
- "mpaa",
- "aspectratio",
- "collectionnumber",
- "tmdbid",
- "rottentomatoesid",
- "language",
- "tvcomid",
- "tagline",
- "studio",
- "genre",
- "tag",
- "runtime",
- "actor",
- "criticrating",
- "fileinfo",
- "director",
- "writer",
- "trailer",
- "premiered",
- "releasedate",
- "outline",
- "id",
- "credits",
- "originaltitle",
- "watched",
- "playcount",
- "lastplayed",
- "art",
- "resume",
- "biography",
- "formed",
- "review",
- "style",
- "imdbid",
- "imdb_id",
- "country",
- "audiodbalbumid",
- "audiodbartistid",
- "enddate",
- "lockedfields",
- "zap2itid",
- "tvrageid",
- "musicbrainzartistid",
- "musicbrainzalbumartistid",
- "musicbrainzalbumid",
- "musicbrainzreleasegroupid",
- "tvdbid",
- "collectionitem",
- "isuserfavorite",
- "userrating",
- "countrycode"
- };
- protected BaseNfoSaver(
- IFileSystem fileSystem,
- IServerConfigurationManager configurationManager,
- ILibraryManager libraryManager,
- IUserManager userManager,
- IUserDataManager userDataManager,
- ILogger<BaseNfoSaver> logger)
- {
- Logger = logger;
- UserDataManager = userDataManager;
- UserManager = userManager;
- LibraryManager = libraryManager;
- ConfigurationManager = configurationManager;
- FileSystem = fileSystem;
- }
- protected IFileSystem FileSystem { get; }
- protected IServerConfigurationManager ConfigurationManager { get; }
- protected ILibraryManager LibraryManager { get; }
- protected IUserManager UserManager { get; }
- protected IUserDataManager UserDataManager { get; }
- protected ILogger<BaseNfoSaver> Logger { get; }
- protected ItemUpdateType MinimumUpdateType
- {
- get
- {
- if (ConfigurationManager.GetNfoConfiguration().SaveImagePathsInNfo)
- {
- return ItemUpdateType.ImageUpdate;
- }
- return ItemUpdateType.MetadataDownload;
- }
- }
- /// <inheritdoc />
- public string Name => SaverName;
- public static string SaverName => "Nfo";
- // filters control characters but allows only properly-formed surrogate sequences
- // http://web.archive.org/web/20181230211547/https://emby.media/community/index.php?/topic/49071-nfo-not-generated-on-actualize-or-rescan-or-identify
- // Web Archive version of link since it's not really explained in the thread.
- [GeneratedRegex(@"(?<![\uD800-\uDBFF])[\uDC00-\uDFFF]|[\uD800-\uDBFF](?![\uDC00-\uDFFF])|[\x00-\x08\x0B\x0C\x0E-\x1F\x7F-\x9F\uFEFF\uFFFE\uFFFF]")]
- private static partial Regex InvalidXMLCharsRegexRegex();
- /// <inheritdoc />
- public string GetSavePath(BaseItem item)
- => GetLocalSavePath(item);
- /// <summary>
- /// Gets the save path.
- /// </summary>
- /// <param name="item">The item.</param>
- /// <returns><see cref="string" />.</returns>
- protected abstract string GetLocalSavePath(BaseItem item);
- /// <summary>
- /// Gets the name of the root element.
- /// </summary>
- /// <param name="item">The item.</param>
- /// <returns><see cref="string" />.</returns>
- protected abstract string GetRootElementName(BaseItem item);
- /// <inheritdoc />
- public abstract bool IsEnabledFor(BaseItem item, ItemUpdateType updateType);
- protected virtual IEnumerable<string> GetTagsUsed(BaseItem item)
- {
- foreach (var providerKey in item.ProviderIds.Keys)
- {
- var providerIdTagName = GetTagForProviderKey(providerKey);
- if (!_commonTags.Contains(providerIdTagName))
- {
- yield return providerIdTagName;
- }
- }
- }
- /// <inheritdoc />
- public async Task SaveAsync(BaseItem item, CancellationToken cancellationToken)
- {
- var path = GetSavePath(item);
- using (var memoryStream = new MemoryStream())
- {
- Save(item, memoryStream, path);
- memoryStream.Position = 0;
- cancellationToken.ThrowIfCancellationRequested();
- await SaveToFileAsync(memoryStream, path).ConfigureAwait(false);
- }
- }
- private async Task SaveToFileAsync(Stream stream, string path)
- {
- var directory = Path.GetDirectoryName(path) ?? throw new ArgumentException($"Provided path ({path}) is not valid.", nameof(path));
- Directory.CreateDirectory(directory);
- // On Windows, saving the file will fail if the file is hidden or readonly
- FileSystem.SetAttributes(path, false, false);
- var fileStreamOptions = new FileStreamOptions()
- {
- Mode = FileMode.Create,
- Access = FileAccess.Write,
- Share = FileShare.None,
- PreallocationSize = stream.Length,
- Options = FileOptions.Asynchronous
- };
- var filestream = new FileStream(path, fileStreamOptions);
- await using (filestream.ConfigureAwait(false))
- {
- await stream.CopyToAsync(filestream).ConfigureAwait(false);
- }
- if (ConfigurationManager.Configuration.SaveMetadataHidden)
- {
- SetHidden(path, true);
- }
- }
- private void SetHidden(string path, bool hidden)
- {
- try
- {
- FileSystem.SetHidden(path, hidden);
- }
- catch (IOException ex)
- {
- Logger.LogError(ex, "Error setting hidden attribute on {Path}", path);
- }
- }
- private void Save(BaseItem item, Stream stream, string xmlPath)
- {
- var settings = new XmlWriterSettings
- {
- Indent = true,
- Encoding = Encoding.UTF8,
- CloseOutput = false
- };
- using (var writer = XmlWriter.Create(stream, settings))
- {
- var root = GetRootElementName(item);
- writer.WriteStartDocument(true);
- writer.WriteStartElement(root);
- var baseItem = item;
- if (baseItem is not null)
- {
- AddCommonNodes(baseItem, writer, LibraryManager, UserManager, UserDataManager, ConfigurationManager);
- }
- WriteCustomElements(item, writer);
- if (baseItem is IHasMediaSources hasMediaSources)
- {
- AddMediaInfo(hasMediaSources, writer);
- }
- var tagsUsed = GetTagsUsed(item).ToList();
- try
- {
- AddCustomTags(xmlPath, tagsUsed, writer, Logger);
- }
- catch (FileNotFoundException)
- {
- }
- catch (IOException)
- {
- }
- catch (XmlException ex)
- {
- Logger.LogError(ex, "Error reading existing nfo");
- }
- writer.WriteEndElement();
- writer.WriteEndDocument();
- }
- }
- protected abstract void WriteCustomElements(BaseItem item, XmlWriter writer);
- public static void AddMediaInfo<T>(T item, XmlWriter writer)
- where T : IHasMediaSources
- {
- writer.WriteStartElement("fileinfo");
- writer.WriteStartElement("streamdetails");
- var mediaStreams = item.GetMediaStreams();
- foreach (var stream in mediaStreams)
- {
- writer.WriteStartElement(stream.Type.ToString().ToLowerInvariant());
- if (!string.IsNullOrEmpty(stream.Codec))
- {
- var codec = stream.Codec;
- if ((stream.CodecTag ?? string.Empty).Contains("xvid", StringComparison.OrdinalIgnoreCase))
- {
- codec = "xvid";
- }
- else if ((stream.CodecTag ?? string.Empty).Contains("divx", StringComparison.OrdinalIgnoreCase))
- {
- codec = "divx";
- }
- writer.WriteElementString("codec", codec);
- writer.WriteElementString("micodec", codec);
- }
- if (stream.BitRate.HasValue)
- {
- writer.WriteElementString("bitrate", stream.BitRate.Value.ToString(CultureInfo.InvariantCulture));
- }
- if (stream.Width.HasValue)
- {
- writer.WriteElementString("width", stream.Width.Value.ToString(CultureInfo.InvariantCulture));
- }
- if (stream.Height.HasValue)
- {
- writer.WriteElementString("height", stream.Height.Value.ToString(CultureInfo.InvariantCulture));
- }
- if (!string.IsNullOrEmpty(stream.AspectRatio))
- {
- writer.WriteElementString("aspect", stream.AspectRatio);
- writer.WriteElementString("aspectratio", stream.AspectRatio);
- }
- var framerate = stream.ReferenceFrameRate;
- if (framerate.HasValue)
- {
- writer.WriteElementString("framerate", framerate.Value.ToString(CultureInfo.InvariantCulture));
- }
- if (!string.IsNullOrEmpty(stream.Language))
- {
- writer.WriteElementString("language", InvalidXMLCharsRegexRegex().Replace(stream.Language, string.Empty));
- }
- var scanType = stream.IsInterlaced ? "interlaced" : "progressive";
- writer.WriteElementString("scantype", scanType);
- if (stream.Channels.HasValue)
- {
- writer.WriteElementString("channels", stream.Channels.Value.ToString(CultureInfo.InvariantCulture));
- }
- if (stream.SampleRate.HasValue)
- {
- writer.WriteElementString("samplingrate", stream.SampleRate.Value.ToString(CultureInfo.InvariantCulture));
- }
- writer.WriteElementString("default", stream.IsDefault.ToString(CultureInfo.InvariantCulture));
- writer.WriteElementString("forced", stream.IsForced.ToString(CultureInfo.InvariantCulture));
- if (stream.Type == MediaStreamType.Video)
- {
- var runtimeTicks = item.RunTimeTicks;
- if (runtimeTicks.HasValue)
- {
- var timespan = TimeSpan.FromTicks(runtimeTicks.Value);
- writer.WriteElementString(
- "duration",
- Math.Floor(timespan.TotalMinutes).ToString(CultureInfo.InvariantCulture));
- writer.WriteElementString(
- "durationinseconds",
- Math.Floor(timespan.TotalSeconds).ToString(CultureInfo.InvariantCulture));
- }
- if (item is Video video)
- {
- // 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();
- }
- /// <summary>
- /// Adds the common nodes.
- /// </summary>
- private void AddCommonNodes(
- BaseItem item,
- XmlWriter writer,
- ILibraryManager libraryManager,
- IUserManager userManager,
- IUserDataManager userDataRepo,
- IServerConfigurationManager config)
- {
- var writtenProviderIds = new HashSet<string>(StringComparer.OrdinalIgnoreCase);
- var overview = (item.Overview ?? string.Empty)
- .StripHtml()
- .Replace(""", "'", StringComparison.Ordinal);
- 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 not Video)
- {
- writer.WriteElementString("outline", overview);
- }
- if (!string.IsNullOrWhiteSpace(item.CustomRating))
- {
- writer.WriteElementString("customrating", item.CustomRating);
- }
- writer.WriteElementString("lockdata", item.IsLocked.ToString(CultureInfo.InvariantCulture).ToLowerInvariant());
- if (item.LockedFields.Length > 0)
- {
- writer.WriteElementString("lockedfields", string.Join('|', item.LockedFields));
- }
- writer.WriteElementString("dateadded", item.DateCreated.ToString(DateAddedFormat, CultureInfo.InvariantCulture));
- 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 => i.IsType(PersonKind.Director))
- .Select(i => i.Name?.Trim())
- .Distinct(StringComparer.OrdinalIgnoreCase)
- .OrderBy(i => i)
- .ToList();
- foreach (var person in directors)
- {
- writer.WriteElementString("director", person);
- }
- var writers = people
- .Where(i => i.IsType(PersonKind.Writer))
- .Select(i => i.Name?.Trim())
- .Distinct(StringComparer.OrdinalIgnoreCase)
- .OrderBy(i => i)
- .ToList();
- foreach (var person in writers)
- {
- writer.WriteElementString("writer", person);
- }
- foreach (var person in writers)
- {
- writer.WriteElementString("credits", person);
- }
- foreach (var trailer in item.RemoteTrailers.OrderBy(t => t.Url?.Trim()))
- {
- writer.WriteElementString("trailer", GetOutputTrailerUrl(trailer.Url));
- }
- if (item.CommunityRating.HasValue)
- {
- writer.WriteElementString("rating", item.CommunityRating.Value.ToString(CultureInfo.InvariantCulture));
- }
- if (item.ProductionYear.HasValue)
- {
- writer.WriteElementString("year", item.ProductionYear.Value.ToString(CultureInfo.InvariantCulture));
- }
- var forcedSortName = item.ForcedSortName;
- if (!string.IsNullOrEmpty(forcedSortName))
- {
- writer.WriteElementString("sorttitle", forcedSortName);
- }
- if (!string.IsNullOrEmpty(item.OfficialRating))
- {
- writer.WriteElementString("mpaa", item.OfficialRating);
- }
- if (item is IHasAspectRatio hasAspectRatio
- && !string.IsNullOrEmpty(hasAspectRatio.AspectRatio))
- {
- writer.WriteElementString("aspectratio", hasAspectRatio.AspectRatio);
- }
- if (item.TryGetProviderId(MetadataProvider.Tmdb, out var tmdbCollection))
- {
- writer.WriteElementString("collectionnumber", tmdbCollection);
- writtenProviderIds.Add(MetadataProvider.TmdbCollection.ToString());
- }
- if (item.TryGetProviderId(MetadataProvider.Imdb, out var imdb))
- {
- if (item is Series)
- {
- writer.WriteElementString("imdb_id", imdb);
- }
- else
- {
- writer.WriteElementString("imdbid", imdb);
- }
- writtenProviderIds.Add(MetadataProvider.Imdb.ToString());
- }
- // Series xml saver already saves this
- if (item is not Series)
- {
- if (item.TryGetProviderId(MetadataProvider.Tvdb, out var tvdb))
- {
- writer.WriteElementString("tvdbid", tvdb);
- writtenProviderIds.Add(MetadataProvider.Tvdb.ToString());
- }
- }
- if (item.TryGetProviderId(MetadataProvider.Tmdb, out var tmdb))
- {
- writer.WriteElementString("tmdbid", tmdb);
- writtenProviderIds.Add(MetadataProvider.Tmdb.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 not Episode)
- {
- var formatString = options.ReleaseDateFormat;
- if (item is MusicArtist)
- {
- writer.WriteElementString(
- "formed",
- item.PremiereDate.Value.ToString(formatString, CultureInfo.InvariantCulture));
- }
- else
- {
- writer.WriteElementString(
- "premiered",
- item.PremiereDate.Value.ToString(formatString, CultureInfo.InvariantCulture));
- writer.WriteElementString(
- "releasedate",
- item.PremiereDate.Value.ToString(formatString, CultureInfo.InvariantCulture));
- }
- }
- if (item.EndDate.HasValue)
- {
- if (item is not Episode)
- {
- var formatString = options.ReleaseDateFormat;
- writer.WriteElementString(
- "enddate",
- item.EndDate.Value.ToString(formatString, CultureInfo.InvariantCulture));
- }
- }
- if (item.CriticRating.HasValue)
- {
- writer.WriteElementString(
- "criticrating",
- item.CriticRating.Value.ToString(CultureInfo.InvariantCulture));
- }
- if (item is IHasDisplayOrder hasDisplayOrder)
- {
- if (!string.IsNullOrEmpty(hasDisplayOrder.DisplayOrder))
- {
- writer.WriteElementString("displayorder", hasDisplayOrder.DisplayOrder);
- }
- }
- // 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.ToInt64(timespan.TotalMinutes).ToString(CultureInfo.InvariantCulture));
- }
- if (!string.IsNullOrWhiteSpace(item.Tagline))
- {
- writer.WriteElementString("tagline", item.Tagline);
- }
- foreach (var country in item.ProductionLocations.Trimmed().OrderBy(country => country))
- {
- writer.WriteElementString("country", country);
- }
- foreach (var genre in item.Genres.Trimmed().OrderBy(genre => genre))
- {
- writer.WriteElementString("genre", genre);
- }
- foreach (var studio in item.Studios.Trimmed().OrderBy(studio => studio))
- {
- writer.WriteElementString("studio", studio);
- }
- foreach (var tag in item.Tags.Trimmed().OrderBy(tag => tag))
- {
- if (item is MusicAlbum || item is MusicArtist)
- {
- writer.WriteElementString("style", tag);
- }
- else
- {
- writer.WriteElementString("tag", tag);
- }
- }
- if (item.TryGetProviderId(MetadataProvider.AudioDbArtist, out var externalId))
- {
- writer.WriteElementString("audiodbartistid", externalId);
- writtenProviderIds.Add(MetadataProvider.AudioDbArtist.ToString());
- }
- if (item.TryGetProviderId(MetadataProvider.AudioDbAlbum, out externalId))
- {
- writer.WriteElementString("audiodbalbumid", externalId);
- writtenProviderIds.Add(MetadataProvider.AudioDbAlbum.ToString());
- }
- if (item.TryGetProviderId(MetadataProvider.Zap2It, out externalId))
- {
- writer.WriteElementString("zap2itid", externalId);
- writtenProviderIds.Add(MetadataProvider.Zap2It.ToString());
- }
- if (item.TryGetProviderId(MetadataProvider.MusicBrainzAlbum, out externalId))
- {
- writer.WriteElementString("musicbrainzalbumid", externalId);
- writtenProviderIds.Add(MetadataProvider.MusicBrainzAlbum.ToString());
- }
- if (item.TryGetProviderId(MetadataProvider.MusicBrainzAlbumArtist, out externalId))
- {
- writer.WriteElementString("musicbrainzalbumartistid", externalId);
- writtenProviderIds.Add(MetadataProvider.MusicBrainzAlbumArtist.ToString());
- }
- if (item.TryGetProviderId(MetadataProvider.MusicBrainzArtist, out externalId))
- {
- writer.WriteElementString("musicbrainzartistid", externalId);
- writtenProviderIds.Add(MetadataProvider.MusicBrainzArtist.ToString());
- }
- if (item.TryGetProviderId(MetadataProvider.MusicBrainzReleaseGroup, out externalId))
- {
- writer.WriteElementString("musicbrainzreleasegroupid", externalId);
- writtenProviderIds.Add(MetadataProvider.MusicBrainzReleaseGroup.ToString());
- }
- if (item.TryGetProviderId(MetadataProvider.TvRage, out externalId))
- {
- writer.WriteElementString("tvrageid", externalId);
- writtenProviderIds.Add(MetadataProvider.TvRage.ToString());
- }
- if (item.ProviderIds is not null)
- {
- foreach (var providerKey in item.ProviderIds.Keys.OrderBy(providerKey => providerKey))
- {
- var providerId = item.ProviderIds[providerKey];
- if (!string.IsNullOrEmpty(providerId) && !writtenProviderIds.Contains(providerKey))
- {
- try
- {
- var tagName = GetTagForProviderKey(providerKey);
- Logger.LogDebug("Verifying custom provider tagname {0}", tagName);
- XmlConvert.VerifyName(tagName);
- Logger.LogDebug("Saving custom provider tagname {0}", tagName);
- writer.WriteElementString(tagName, providerId);
- }
- catch (ArgumentException)
- {
- // catch invalid names without failing the entire operation
- }
- catch (XmlException)
- {
- // catch invalid names without failing the entire operation
- }
- }
- }
- }
- if (options.SaveImagePathsInNfo)
- {
- AddImages(item, writer, libraryManager);
- }
- AddUserData(item, writer, userManager, userDataRepo, options);
- if (item is not MusicAlbum && item is not MusicArtist)
- {
- AddActors(people, writer, libraryManager, options.SaveImagePathsInNfo);
- }
- if (item is BoxSet folder)
- {
- AddCollectionItems(folder, writer);
- }
- }
- private void AddCollectionItems(Folder item, XmlWriter writer)
- {
- var items = item.LinkedChildren
- .Where(i => i.Type == LinkedChildType.Manual)
- .OrderBy(i => i.Path?.Trim())
- .ThenBy(i => i.LibraryItemId?.Trim())
- .ToList();
- foreach (var link in items)
- {
- writer.WriteStartElement("collectionitem");
- if (!string.IsNullOrWhiteSpace(link.Path))
- {
- writer.WriteElementString("path", link.Path);
- }
- if (!string.IsNullOrWhiteSpace(link.LibraryItemId))
- {
- writer.WriteElementString("ItemId", link.LibraryItemId);
- }
- writer.WriteEndElement();
- }
- }
- /// <summary>
- /// Gets the output trailer URL.
- /// </summary>
- /// <param name="url">The URL.</param>
- /// <returns>System.String.</returns>
- private string GetOutputTrailerUrl(string url)
- {
- // This is what xbmc expects
- return url.Replace(YouTubeWatchUrl, "plugin://plugin.video.youtube/play/?video_id=", StringComparison.OrdinalIgnoreCase);
- }
- private void AddImages(BaseItem item, XmlWriter writer, ILibraryManager libraryManager)
- {
- writer.WriteStartElement("art");
- var image = item.GetImageInfo(ImageType.Primary, 0);
- if (image is not null)
- {
- writer.WriteElementString("poster", GetImagePathToSave(image, libraryManager));
- }
- foreach (var backdrop in item.GetImages(ImageType.Backdrop).OrderBy(b => b.Path?.Trim()))
- {
- writer.WriteElementString("fanart", GetImagePathToSave(backdrop, libraryManager));
- }
- writer.WriteEndElement();
- }
- private 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(Guid.Parse(userId));
- if (user is null)
- {
- return;
- }
- if (item.IsFolder)
- {
- return;
- }
- var userdata = userDataRepo.GetUserData(user, item);
- if (userdata is not null)
- {
- writer.WriteElementString(
- "isuserfavorite",
- userdata.IsFavorite.ToString(CultureInfo.InvariantCulture).ToLowerInvariant());
- if (userdata.Rating.HasValue)
- {
- writer.WriteElementString(
- "userrating",
- userdata.Rating.Value.ToString(CultureInfo.InvariantCulture).ToLowerInvariant());
- }
- if (!item.IsFolder)
- {
- writer.WriteElementString(
- "playcount",
- userdata.PlayCount.ToString(CultureInfo.InvariantCulture));
- writer.WriteElementString(
- "watched",
- userdata.Played.ToString(CultureInfo.InvariantCulture).ToLowerInvariant());
- if (userdata.LastPlayedDate.HasValue)
- {
- writer.WriteElementString(
- "lastplayed",
- userdata.LastPlayedDate.Value.ToString("yyyy-MM-dd HH:mm:ss", CultureInfo.InvariantCulture).ToLowerInvariant());
- }
- writer.WriteStartElement("resume");
- var runTimeTicks = item.RunTimeTicks ?? 0;
- writer.WriteElementString(
- "position",
- TimeSpan.FromTicks(userdata.PlaybackPositionTicks).TotalSeconds.ToString(CultureInfo.InvariantCulture));
- writer.WriteElementString(
- "total",
- TimeSpan.FromTicks(runTimeTicks).TotalSeconds.ToString(CultureInfo.InvariantCulture));
- }
- }
- writer.WriteEndElement();
- }
- private void AddActors(IReadOnlyList<PersonInfo> people, XmlWriter writer, ILibraryManager libraryManager, bool saveImagePath)
- {
- foreach (var person in people
- .OrderBy(person => person.SortOrder ?? 0)
- .ThenBy(person => person.Name?.Trim()))
- {
- if (person.IsType(PersonKind.Director) || person.IsType(PersonKind.Writer))
- {
- continue;
- }
- writer.WriteStartElement("actor");
- if (!string.IsNullOrWhiteSpace(person.Name))
- {
- writer.WriteElementString("name", person.Name);
- }
- if (!string.IsNullOrWhiteSpace(person.Role))
- {
- writer.WriteElementString("role", person.Role);
- }
- if (person.Type != PersonKind.Unknown)
- {
- writer.WriteElementString("type", person.Type.ToString());
- }
- if (person.SortOrder.HasValue)
- {
- writer.WriteElementString(
- "sortorder",
- person.SortOrder.Value.ToString(CultureInfo.InvariantCulture));
- }
- if (saveImagePath)
- {
- var personEntity = libraryManager.GetPerson(person.Name);
- var image = personEntity?.GetImageInfo(ImageType.Primary, 0);
- if (image is not null)
- {
- writer.WriteElementString(
- "thumb",
- GetImagePathToSave(image, libraryManager));
- }
- }
- writer.WriteEndElement();
- }
- }
- private string GetImagePathToSave(ItemImageInfo image, ILibraryManager libraryManager)
- {
- if (!image.IsLocalFile)
- {
- return image.Path;
- }
- return libraryManager.GetPathAfterNetworkSubstitution(image.Path);
- }
- private void AddCustomTags(string path, IReadOnlyCollection<string> xmlTagsUsed, XmlWriter writer, ILogger<BaseNfoSaver> logger)
- {
- var settings = new XmlReaderSettings()
- {
- ValidationType = ValidationType.None,
- CheckCharacters = false,
- IgnoreProcessingInstructions = true,
- IgnoreComments = true
- };
- using (var fileStream = File.OpenRead(path))
- using (var streamReader = new StreamReader(fileStream, Encoding.UTF8))
- using (var reader = XmlReader.Create(streamReader, settings))
- {
- try
- {
- reader.MoveToContent();
- }
- catch (Exception ex)
- {
- logger.LogError(ex, "Error reading existing xml tags from {Path}.", 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.Contains(name)
- && !xmlTagsUsed.Contains(name, StringComparison.OrdinalIgnoreCase))
- {
- writer.WriteNode(reader, false);
- }
- else
- {
- reader.Skip();
- }
- }
- else
- {
- reader.Read();
- }
- }
- }
- }
- private string GetTagForProviderKey(string providerKey)
- => providerKey.ToLowerInvariant() + "id";
- protected static string SortNameOrName(BaseItem item)
- {
- if (item is null)
- {
- return string.Empty;
- }
- if (item.SortName is not null)
- {
- string trimmed = item.SortName.Trim();
- if (trimmed.Length > 0)
- {
- return trimmed;
- }
- }
- return (item.Name ?? string.Empty).Trim();
- }
- }
- }
|