BaseNfoSaver.cs 37 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040
  1. using MediaBrowser.Common.Extensions;
  2. using MediaBrowser.Controller.Configuration;
  3. using MediaBrowser.Controller.Entities;
  4. using MediaBrowser.Controller.Entities.Audio;
  5. using MediaBrowser.Controller.Entities.Movies;
  6. using MediaBrowser.Controller.Entities.TV;
  7. using MediaBrowser.Controller.Library;
  8. using MediaBrowser.Controller.Persistence;
  9. using MediaBrowser.Model.Configuration;
  10. using MediaBrowser.Model.Entities;
  11. using MediaBrowser.Model.Logging;
  12. using MediaBrowser.XbmcMetadata.Configuration;
  13. using System;
  14. using System.Collections.Generic;
  15. using System.Globalization;
  16. using System.IO;
  17. using System.Linq;
  18. using System.Text;
  19. using System.Text.RegularExpressions;
  20. using System.Threading;
  21. using System.Xml;
  22. using MediaBrowser.Controller.Extensions;
  23. using MediaBrowser.Model.Extensions;
  24. using MediaBrowser.Model.IO;
  25. using MediaBrowser.Model.Xml;
  26. namespace MediaBrowser.XbmcMetadata.Savers
  27. {
  28. public abstract class BaseNfoSaver : IMetadataFileSaver
  29. {
  30. public static readonly string YouTubeWatchUrl = "https://www.youtube.com/watch?v=";
  31. private static readonly CultureInfo UsCulture = new CultureInfo("en-US");
  32. private static readonly Dictionary<string, string> CommonTags = new[] {
  33. "plot",
  34. "customrating",
  35. "lockdata",
  36. "dateadded",
  37. "title",
  38. "rating",
  39. "year",
  40. "sorttitle",
  41. "mpaa",
  42. "aspectratio",
  43. "collectionnumber",
  44. "tmdbid",
  45. "rottentomatoesid",
  46. "language",
  47. "tvcomid",
  48. "tagline",
  49. "studio",
  50. "genre",
  51. "tag",
  52. "runtime",
  53. "actor",
  54. "criticrating",
  55. "fileinfo",
  56. "director",
  57. "writer",
  58. "trailer",
  59. "premiered",
  60. "releasedate",
  61. "outline",
  62. "id",
  63. "credits",
  64. "originaltitle",
  65. "watched",
  66. "playcount",
  67. "lastplayed",
  68. "art",
  69. "resume",
  70. "biography",
  71. "formed",
  72. "review",
  73. "style",
  74. "imdbid",
  75. "imdb_id",
  76. "country",
  77. "audiodbalbumid",
  78. "audiodbartistid",
  79. "enddate",
  80. "lockedfields",
  81. "zap2itid",
  82. "tvrageid",
  83. "gamesdbid",
  84. "musicbrainzartistid",
  85. "musicbrainzalbumartistid",
  86. "musicbrainzalbumid",
  87. "musicbrainzreleasegroupid",
  88. "tvdbid",
  89. "collectionitem",
  90. "isuserfavorite",
  91. "userrating",
  92. "countrycode"
  93. }.ToDictionary(i => i, StringComparer.OrdinalIgnoreCase);
  94. protected BaseNfoSaver(IFileSystem fileSystem, IServerConfigurationManager configurationManager, ILibraryManager libraryManager, IUserManager userManager, IUserDataManager userDataManager, ILogger logger, IXmlReaderSettingsFactory xmlReaderSettingsFactory)
  95. {
  96. Logger = logger;
  97. XmlReaderSettingsFactory = xmlReaderSettingsFactory;
  98. UserDataManager = userDataManager;
  99. UserManager = userManager;
  100. LibraryManager = libraryManager;
  101. ConfigurationManager = configurationManager;
  102. FileSystem = fileSystem;
  103. }
  104. protected IFileSystem FileSystem { get; private set; }
  105. protected IServerConfigurationManager ConfigurationManager { get; private set; }
  106. protected ILibraryManager LibraryManager { get; private set; }
  107. protected IUserManager UserManager { get; private set; }
  108. protected IUserDataManager UserDataManager { get; private set; }
  109. protected ILogger Logger { get; private set; }
  110. protected IXmlReaderSettingsFactory XmlReaderSettingsFactory { get; private set; }
  111. protected ItemUpdateType MinimumUpdateType
  112. {
  113. get
  114. {
  115. if (ConfigurationManager.GetNfoConfiguration().SaveImagePathsInNfo)
  116. {
  117. return ItemUpdateType.ImageUpdate;
  118. }
  119. return ItemUpdateType.MetadataDownload;
  120. }
  121. }
  122. public string Name
  123. {
  124. get
  125. {
  126. return SaverName;
  127. }
  128. }
  129. public static string SaverName
  130. {
  131. get
  132. {
  133. return "Nfo";
  134. }
  135. }
  136. public string GetSavePath(BaseItem item)
  137. {
  138. return GetLocalSavePath(item);
  139. }
  140. /// <summary>
  141. /// Gets the save path.
  142. /// </summary>
  143. /// <param name="item">The item.</param>
  144. /// <returns>System.String.</returns>
  145. protected abstract string GetLocalSavePath(BaseItem item);
  146. /// <summary>
  147. /// Gets the name of the root element.
  148. /// </summary>
  149. /// <param name="item">The item.</param>
  150. /// <returns>System.String.</returns>
  151. protected abstract string GetRootElementName(BaseItem item);
  152. /// <summary>
  153. /// Determines whether [is enabled for] [the specified item].
  154. /// </summary>
  155. /// <param name="item">The item.</param>
  156. /// <param name="updateType">Type of the update.</param>
  157. /// <returns><c>true</c> if [is enabled for] [the specified item]; otherwise, <c>false</c>.</returns>
  158. public abstract bool IsEnabledFor(BaseItem item, ItemUpdateType updateType);
  159. protected virtual List<string> GetTagsUsed(BaseItem item)
  160. {
  161. var list = new List<string>();
  162. foreach (var providerKey in item.ProviderIds.Keys)
  163. {
  164. var providerIdTagName = GetTagForProviderKey(providerKey);
  165. if (!CommonTags.ContainsKey(providerIdTagName))
  166. {
  167. list.Add(providerIdTagName);
  168. }
  169. }
  170. return list;
  171. }
  172. public void Save(BaseItem item, CancellationToken cancellationToken)
  173. {
  174. var path = GetSavePath(item);
  175. using (var memoryStream = new MemoryStream())
  176. {
  177. Save(item, memoryStream, path);
  178. memoryStream.Position = 0;
  179. cancellationToken.ThrowIfCancellationRequested();
  180. SaveToFile(memoryStream, path);
  181. }
  182. }
  183. private void SaveToFile(Stream stream, string path)
  184. {
  185. FileSystem.CreateDirectory(FileSystem.GetDirectoryName(path));
  186. // On Windows, savint the file will fail if the file is hidden or readonly
  187. FileSystem.SetAttributes(path, false, false);
  188. using (var filestream = FileSystem.GetFileStream(path, FileOpenMode.Create, FileAccessMode.Write, FileShareMode.Read))
  189. {
  190. stream.CopyTo(filestream);
  191. }
  192. if (ConfigurationManager.Configuration.SaveMetadataHidden)
  193. {
  194. SetHidden(path, true);
  195. }
  196. }
  197. private void SetHidden(string path, bool hidden)
  198. {
  199. try
  200. {
  201. FileSystem.SetHidden(path, hidden);
  202. }
  203. catch (Exception ex)
  204. {
  205. Logger.Error("Error setting hidden attribute on {0} - {1}", path, ex.Message);
  206. }
  207. }
  208. private void Save(BaseItem item, Stream stream, string xmlPath)
  209. {
  210. var settings = new XmlWriterSettings
  211. {
  212. Indent = true,
  213. Encoding = Encoding.UTF8,
  214. CloseOutput = false
  215. };
  216. using (XmlWriter writer = XmlWriter.Create(stream, settings))
  217. {
  218. var root = GetRootElementName(item);
  219. writer.WriteStartDocument(true);
  220. writer.WriteStartElement(root);
  221. var baseItem = item;
  222. if (baseItem != null)
  223. {
  224. AddCommonNodes(baseItem, writer, LibraryManager, UserManager, UserDataManager, FileSystem, ConfigurationManager);
  225. }
  226. WriteCustomElements(item, writer);
  227. var hasMediaSources = baseItem as IHasMediaSources;
  228. if (hasMediaSources != null)
  229. {
  230. AddMediaInfo(hasMediaSources, writer);
  231. }
  232. var tagsUsed = GetTagsUsed(item);
  233. try
  234. {
  235. AddCustomTags(xmlPath, tagsUsed, writer, Logger, FileSystem);
  236. }
  237. catch (FileNotFoundException)
  238. {
  239. }
  240. catch (IOException)
  241. {
  242. }
  243. catch (XmlException ex)
  244. {
  245. Logger.ErrorException("Error reading existng nfo", ex);
  246. }
  247. writer.WriteEndElement();
  248. writer.WriteEndDocument();
  249. }
  250. }
  251. protected abstract void WriteCustomElements(BaseItem item, XmlWriter writer);
  252. public static void AddMediaInfo<T>(T item, XmlWriter writer)
  253. where T : IHasMediaSources
  254. {
  255. writer.WriteStartElement("fileinfo");
  256. writer.WriteStartElement("streamdetails");
  257. var mediaStreams = item.GetMediaStreams();
  258. foreach (var stream in mediaStreams)
  259. {
  260. writer.WriteStartElement(stream.Type.ToString().ToLower());
  261. if (!string.IsNullOrEmpty(stream.Codec))
  262. {
  263. var codec = stream.Codec;
  264. if ((stream.CodecTag ?? string.Empty).IndexOf("xvid", StringComparison.OrdinalIgnoreCase) != -1)
  265. {
  266. codec = "xvid";
  267. }
  268. else if ((stream.CodecTag ?? string.Empty).IndexOf("divx", StringComparison.OrdinalIgnoreCase) != -1)
  269. {
  270. codec = "divx";
  271. }
  272. writer.WriteElementString("codec", codec);
  273. writer.WriteElementString("micodec", codec);
  274. }
  275. if (stream.BitRate.HasValue)
  276. {
  277. writer.WriteElementString("bitrate", stream.BitRate.Value.ToString(UsCulture));
  278. }
  279. if (stream.Width.HasValue)
  280. {
  281. writer.WriteElementString("width", stream.Width.Value.ToString(UsCulture));
  282. }
  283. if (stream.Height.HasValue)
  284. {
  285. writer.WriteElementString("height", stream.Height.Value.ToString(UsCulture));
  286. }
  287. if (!string.IsNullOrEmpty(stream.AspectRatio))
  288. {
  289. writer.WriteElementString("aspect", stream.AspectRatio);
  290. writer.WriteElementString("aspectratio", stream.AspectRatio);
  291. }
  292. var framerate = stream.AverageFrameRate ?? stream.RealFrameRate;
  293. if (framerate.HasValue)
  294. {
  295. writer.WriteElementString("framerate", framerate.Value.ToString(UsCulture));
  296. }
  297. if (!string.IsNullOrEmpty(stream.Language))
  298. {
  299. // https://emby.media/community/index.php?/topic/49071-nfo-not-generated-on-actualize-or-rescan-or-identify
  300. writer.WriteElementString("language", RemoveInvalidXMLChars(stream.Language));
  301. }
  302. var scanType = stream.IsInterlaced ? "interlaced" : "progressive";
  303. if (!string.IsNullOrEmpty(scanType))
  304. {
  305. writer.WriteElementString("scantype", scanType);
  306. }
  307. if (stream.Channels.HasValue)
  308. {
  309. writer.WriteElementString("channels", stream.Channels.Value.ToString(UsCulture));
  310. }
  311. if (stream.SampleRate.HasValue)
  312. {
  313. writer.WriteElementString("samplingrate", stream.SampleRate.Value.ToString(UsCulture));
  314. }
  315. writer.WriteElementString("default", stream.IsDefault.ToString());
  316. writer.WriteElementString("forced", stream.IsForced.ToString());
  317. if (stream.Type == MediaStreamType.Video)
  318. {
  319. var runtimeTicks = item.RunTimeTicks;
  320. if (runtimeTicks.HasValue)
  321. {
  322. var timespan = TimeSpan.FromTicks(runtimeTicks.Value);
  323. writer.WriteElementString("duration", Math.Floor(timespan.TotalMinutes).ToString(UsCulture));
  324. writer.WriteElementString("durationinseconds", Math.Floor(timespan.TotalSeconds).ToString(UsCulture));
  325. }
  326. var video = item as Video;
  327. if (video != null)
  328. {
  329. //AddChapters(video, builder, itemRepository);
  330. if (video.Video3DFormat.HasValue)
  331. {
  332. switch (video.Video3DFormat.Value)
  333. {
  334. case Video3DFormat.FullSideBySide:
  335. writer.WriteElementString("format3d", "FSBS");
  336. break;
  337. case Video3DFormat.FullTopAndBottom:
  338. writer.WriteElementString("format3d", "FTAB");
  339. break;
  340. case Video3DFormat.HalfSideBySide:
  341. writer.WriteElementString("format3d", "HSBS");
  342. break;
  343. case Video3DFormat.HalfTopAndBottom:
  344. writer.WriteElementString("format3d", "HTAB");
  345. break;
  346. case Video3DFormat.MVC:
  347. writer.WriteElementString("format3d", "MVC");
  348. break;
  349. }
  350. }
  351. }
  352. }
  353. writer.WriteEndElement();
  354. }
  355. writer.WriteEndElement();
  356. writer.WriteEndElement();
  357. }
  358. // filters control characters but allows only properly-formed surrogate sequences
  359. private static Regex _invalidXMLChars = new Regex(
  360. @"(?<![\uD800-\uDBFF])[\uDC00-\uDFFF]|[\uD800-\uDBFF](?![\uDC00-\uDFFF])|[\x00-\x08\x0B\x0C\x0E-\x1F\x7F-\x9F\uFEFF\uFFFE\uFFFF]");
  361. /// <summary>
  362. /// removes any unusual unicode characters that can't be encoded into XML
  363. /// </summary>
  364. public static string RemoveInvalidXMLChars(string text)
  365. {
  366. if (string.IsNullOrEmpty(text)) return string.Empty;
  367. return _invalidXMLChars.Replace(text, string.Empty);
  368. }
  369. public const string DateAddedFormat = "yyyy-MM-dd HH:mm:ss";
  370. /// <summary>
  371. /// Adds the common nodes.
  372. /// </summary>
  373. /// <returns>Task.</returns>
  374. private void AddCommonNodes(BaseItem item, XmlWriter writer, ILibraryManager libraryManager, IUserManager userManager, IUserDataManager userDataRepo, IFileSystem fileSystem, IServerConfigurationManager config)
  375. {
  376. var writtenProviderIds = new HashSet<string>(StringComparer.OrdinalIgnoreCase);
  377. var overview = (item.Overview ?? string.Empty)
  378. .StripHtml()
  379. .Replace("&quot;", "'");
  380. var options = config.GetNfoConfiguration();
  381. if (item is MusicArtist)
  382. {
  383. writer.WriteElementString("biography", overview);
  384. }
  385. else if (item is MusicAlbum)
  386. {
  387. writer.WriteElementString("review", overview);
  388. }
  389. else
  390. {
  391. writer.WriteElementString("plot", overview);
  392. }
  393. if (item is Video)
  394. {
  395. var outline = (item.Tagline ?? string.Empty)
  396. .StripHtml()
  397. .Replace("&quot;", "'");
  398. writer.WriteElementString("outline", outline);
  399. }
  400. else
  401. {
  402. writer.WriteElementString("outline", overview);
  403. }
  404. if (!string.IsNullOrWhiteSpace(item.CustomRating))
  405. {
  406. writer.WriteElementString("customrating", item.CustomRating);
  407. }
  408. writer.WriteElementString("lockdata", item.IsLocked.ToString().ToLower());
  409. if (item.LockedFields.Length > 0)
  410. {
  411. writer.WriteElementString("lockedfields", string.Join("|", item.LockedFields));
  412. }
  413. writer.WriteElementString("dateadded", item.DateCreated.ToLocalTime().ToString(DateAddedFormat));
  414. writer.WriteElementString("title", item.Name ?? string.Empty);
  415. if (!string.IsNullOrWhiteSpace(item.OriginalTitle))
  416. {
  417. writer.WriteElementString("originaltitle", item.OriginalTitle);
  418. }
  419. var people = libraryManager.GetPeople(item);
  420. var directors = people
  421. .Where(i => IsPersonType(i, PersonType.Director))
  422. .Select(i => i.Name)
  423. .ToList();
  424. foreach (var person in directors)
  425. {
  426. writer.WriteElementString("director", person);
  427. }
  428. var writers = people
  429. .Where(i => IsPersonType(i, PersonType.Writer))
  430. .Select(i => i.Name)
  431. .Distinct(StringComparer.OrdinalIgnoreCase)
  432. .ToList();
  433. foreach (var person in writers)
  434. {
  435. writer.WriteElementString("writer", person);
  436. }
  437. foreach (var person in writers)
  438. {
  439. writer.WriteElementString("credits", person);
  440. }
  441. foreach (var trailer in item.RemoteTrailers)
  442. {
  443. writer.WriteElementString("trailer", GetOutputTrailerUrl(trailer.Url));
  444. }
  445. if (item.CommunityRating.HasValue)
  446. {
  447. writer.WriteElementString("rating", item.CommunityRating.Value.ToString(UsCulture));
  448. }
  449. if (item.ProductionYear.HasValue)
  450. {
  451. writer.WriteElementString("year", item.ProductionYear.Value.ToString(UsCulture));
  452. }
  453. var forcedSortName = item.ForcedSortName;
  454. if (!string.IsNullOrEmpty(forcedSortName))
  455. {
  456. writer.WriteElementString("sorttitle", forcedSortName);
  457. }
  458. if (!string.IsNullOrEmpty(item.OfficialRating))
  459. {
  460. writer.WriteElementString("mpaa", item.OfficialRating);
  461. }
  462. var hasAspectRatio = item as IHasAspectRatio;
  463. if (hasAspectRatio != null)
  464. {
  465. if (!string.IsNullOrEmpty(hasAspectRatio.AspectRatio))
  466. {
  467. writer.WriteElementString("aspectratio", hasAspectRatio.AspectRatio);
  468. }
  469. }
  470. var tmdbCollection = item.GetProviderId(MetadataProviders.TmdbCollection);
  471. if (!string.IsNullOrEmpty(tmdbCollection))
  472. {
  473. writer.WriteElementString("collectionnumber", tmdbCollection);
  474. writtenProviderIds.Add(MetadataProviders.TmdbCollection.ToString());
  475. }
  476. var imdb = item.GetProviderId(MetadataProviders.Imdb);
  477. if (!string.IsNullOrEmpty(imdb))
  478. {
  479. if (item is Series)
  480. {
  481. writer.WriteElementString("imdb_id", imdb);
  482. }
  483. else
  484. {
  485. writer.WriteElementString("imdbid", imdb);
  486. }
  487. writtenProviderIds.Add(MetadataProviders.Imdb.ToString());
  488. }
  489. // Series xml saver already saves this
  490. if (!(item is Series))
  491. {
  492. var tvdb = item.GetProviderId(MetadataProviders.Tvdb);
  493. if (!string.IsNullOrEmpty(tvdb))
  494. {
  495. writer.WriteElementString("tvdbid", tvdb);
  496. writtenProviderIds.Add(MetadataProviders.Tvdb.ToString());
  497. }
  498. }
  499. var tmdb = item.GetProviderId(MetadataProviders.Tmdb);
  500. if (!string.IsNullOrEmpty(tmdb))
  501. {
  502. writer.WriteElementString("tmdbid", tmdb);
  503. writtenProviderIds.Add(MetadataProviders.Tmdb.ToString());
  504. }
  505. if (!string.IsNullOrEmpty(item.PreferredMetadataLanguage))
  506. {
  507. writer.WriteElementString("language", item.PreferredMetadataLanguage);
  508. }
  509. if (!string.IsNullOrEmpty(item.PreferredMetadataCountryCode))
  510. {
  511. writer.WriteElementString("countrycode", item.PreferredMetadataCountryCode);
  512. }
  513. if (item.PremiereDate.HasValue && !(item is Episode))
  514. {
  515. var formatString = options.ReleaseDateFormat;
  516. if (item is MusicArtist)
  517. {
  518. writer.WriteElementString("formed", item.PremiereDate.Value.ToLocalTime().ToString(formatString));
  519. }
  520. else
  521. {
  522. writer.WriteElementString("premiered", item.PremiereDate.Value.ToLocalTime().ToString(formatString));
  523. writer.WriteElementString("releasedate", item.PremiereDate.Value.ToLocalTime().ToString(formatString));
  524. }
  525. }
  526. if (item.EndDate.HasValue)
  527. {
  528. if (!(item is Episode))
  529. {
  530. var formatString = options.ReleaseDateFormat;
  531. writer.WriteElementString("enddate", item.EndDate.Value.ToLocalTime().ToString(formatString));
  532. }
  533. }
  534. if (item.CriticRating.HasValue)
  535. {
  536. writer.WriteElementString("criticrating", item.CriticRating.Value.ToString(UsCulture));
  537. }
  538. var hasDisplayOrder = item as IHasDisplayOrder;
  539. if (hasDisplayOrder != null)
  540. {
  541. if (!string.IsNullOrEmpty(hasDisplayOrder.DisplayOrder))
  542. {
  543. writer.WriteElementString("displayorder", hasDisplayOrder.DisplayOrder);
  544. }
  545. }
  546. // Use original runtime here, actual file runtime later in MediaInfo
  547. var runTimeTicks = item.RunTimeTicks;
  548. if (runTimeTicks.HasValue)
  549. {
  550. var timespan = TimeSpan.FromTicks(runTimeTicks.Value);
  551. writer.WriteElementString("runtime", Convert.ToInt64(timespan.TotalMinutes).ToString(UsCulture));
  552. }
  553. if (!string.IsNullOrWhiteSpace(item.Tagline))
  554. {
  555. writer.WriteElementString("tagline", item.Tagline);
  556. }
  557. foreach (var country in item.ProductionLocations)
  558. {
  559. writer.WriteElementString("country", country);
  560. }
  561. foreach (var genre in item.Genres)
  562. {
  563. writer.WriteElementString("genre", genre);
  564. }
  565. foreach (var studio in item.Studios)
  566. {
  567. writer.WriteElementString("studio", studio);
  568. }
  569. foreach (var tag in item.Tags)
  570. {
  571. if (item is MusicAlbum || item is MusicArtist)
  572. {
  573. writer.WriteElementString("style", tag);
  574. }
  575. else
  576. {
  577. writer.WriteElementString("tag", tag);
  578. }
  579. }
  580. var externalId = item.GetProviderId(MetadataProviders.AudioDbArtist);
  581. if (!string.IsNullOrEmpty(externalId))
  582. {
  583. writer.WriteElementString("audiodbartistid", externalId);
  584. writtenProviderIds.Add(MetadataProviders.AudioDbArtist.ToString());
  585. }
  586. externalId = item.GetProviderId(MetadataProviders.AudioDbAlbum);
  587. if (!string.IsNullOrEmpty(externalId))
  588. {
  589. writer.WriteElementString("audiodbalbumid", externalId);
  590. writtenProviderIds.Add(MetadataProviders.AudioDbAlbum.ToString());
  591. }
  592. externalId = item.GetProviderId(MetadataProviders.Zap2It);
  593. if (!string.IsNullOrEmpty(externalId))
  594. {
  595. writer.WriteElementString("zap2itid", externalId);
  596. writtenProviderIds.Add(MetadataProviders.Zap2It.ToString());
  597. }
  598. externalId = item.GetProviderId(MetadataProviders.MusicBrainzAlbum);
  599. if (!string.IsNullOrEmpty(externalId))
  600. {
  601. writer.WriteElementString("musicbrainzalbumid", externalId);
  602. writtenProviderIds.Add(MetadataProviders.MusicBrainzAlbum.ToString());
  603. }
  604. externalId = item.GetProviderId(MetadataProviders.MusicBrainzAlbumArtist);
  605. if (!string.IsNullOrEmpty(externalId))
  606. {
  607. writer.WriteElementString("musicbrainzalbumartistid", externalId);
  608. writtenProviderIds.Add(MetadataProviders.MusicBrainzAlbumArtist.ToString());
  609. }
  610. externalId = item.GetProviderId(MetadataProviders.MusicBrainzArtist);
  611. if (!string.IsNullOrEmpty(externalId))
  612. {
  613. writer.WriteElementString("musicbrainzartistid", externalId);
  614. writtenProviderIds.Add(MetadataProviders.MusicBrainzArtist.ToString());
  615. }
  616. externalId = item.GetProviderId(MetadataProviders.MusicBrainzReleaseGroup);
  617. if (!string.IsNullOrEmpty(externalId))
  618. {
  619. writer.WriteElementString("musicbrainzreleasegroupid", externalId);
  620. writtenProviderIds.Add(MetadataProviders.MusicBrainzReleaseGroup.ToString());
  621. }
  622. externalId = item.GetProviderId(MetadataProviders.Gamesdb);
  623. if (!string.IsNullOrEmpty(externalId))
  624. {
  625. writer.WriteElementString("gamesdbid", externalId);
  626. writtenProviderIds.Add(MetadataProviders.Gamesdb.ToString());
  627. }
  628. externalId = item.GetProviderId(MetadataProviders.TvRage);
  629. if (!string.IsNullOrEmpty(externalId))
  630. {
  631. writer.WriteElementString("tvrageid", externalId);
  632. writtenProviderIds.Add(MetadataProviders.TvRage.ToString());
  633. }
  634. if (item.ProviderIds != null)
  635. {
  636. foreach (var providerKey in item.ProviderIds.Keys)
  637. {
  638. var providerId = item.ProviderIds[providerKey];
  639. if (!string.IsNullOrEmpty(providerId) && !writtenProviderIds.Contains(providerKey))
  640. {
  641. try
  642. {
  643. var tagName = GetTagForProviderKey(providerKey);
  644. //Logger.Debug("Verifying custom provider tagname {0}", tagName);
  645. XmlConvert.VerifyName(tagName);
  646. //Logger.Debug("Saving custom provider tagname {0}", tagName);
  647. writer.WriteElementString(GetTagForProviderKey(providerKey), providerId);
  648. }
  649. catch (ArgumentException)
  650. {
  651. // catch invalid names without failing the entire operation
  652. }
  653. catch (XmlException)
  654. {
  655. // catch invalid names without failing the entire operation
  656. }
  657. }
  658. }
  659. }
  660. if (options.SaveImagePathsInNfo)
  661. {
  662. AddImages(item, writer, libraryManager, config);
  663. }
  664. AddUserData(item, writer, userManager, userDataRepo, options);
  665. AddActors(people, writer, libraryManager, fileSystem, config, options.SaveImagePathsInNfo);
  666. var folder = item as BoxSet;
  667. if (folder != null)
  668. {
  669. AddCollectionItems(folder, writer);
  670. }
  671. }
  672. private void AddCollectionItems(Folder item, XmlWriter writer)
  673. {
  674. var items = item.LinkedChildren
  675. .Where(i => i.Type == LinkedChildType.Manual)
  676. .ToList();
  677. foreach (var link in items)
  678. {
  679. writer.WriteStartElement("collectionitem");
  680. if (!string.IsNullOrWhiteSpace(link.Path))
  681. {
  682. writer.WriteElementString("path", link.Path);
  683. }
  684. if (!string.IsNullOrWhiteSpace(link.LibraryItemId))
  685. {
  686. writer.WriteElementString("ItemId", link.LibraryItemId);
  687. }
  688. writer.WriteEndElement();
  689. }
  690. }
  691. /// <summary>
  692. /// Gets the output trailer URL.
  693. /// </summary>
  694. /// <param name="url">The URL.</param>
  695. /// <returns>System.String.</returns>
  696. private string GetOutputTrailerUrl(string url)
  697. {
  698. // This is what xbmc expects
  699. return url.Replace(YouTubeWatchUrl, "plugin://plugin.video.youtube/?action=play_video&videoid=", StringComparison.OrdinalIgnoreCase);
  700. }
  701. private void AddImages(BaseItem item, XmlWriter writer, ILibraryManager libraryManager, IServerConfigurationManager config)
  702. {
  703. writer.WriteStartElement("art");
  704. var image = item.GetImageInfo(ImageType.Primary, 0);
  705. if (image != null)
  706. {
  707. writer.WriteElementString("poster", GetImagePathToSave(image, libraryManager, config));
  708. }
  709. foreach (var backdrop in item.GetImages(ImageType.Backdrop))
  710. {
  711. writer.WriteElementString("fanart", GetImagePathToSave(backdrop, libraryManager, config));
  712. }
  713. writer.WriteEndElement();
  714. }
  715. private void AddUserData(BaseItem item, XmlWriter writer, IUserManager userManager, IUserDataManager userDataRepo, XbmcMetadataOptions options)
  716. {
  717. var userId = options.UserId;
  718. if (string.IsNullOrWhiteSpace(userId))
  719. {
  720. return;
  721. }
  722. var user = userManager.GetUserById(userId);
  723. if (user == null)
  724. {
  725. return;
  726. }
  727. if (item.IsFolder)
  728. {
  729. return;
  730. }
  731. var userdata = userDataRepo.GetUserData(user, item);
  732. writer.WriteElementString("isuserfavorite", userdata.IsFavorite.ToString().ToLower());
  733. if (userdata.Rating.HasValue)
  734. {
  735. writer.WriteElementString("userrating", userdata.Rating.Value.ToString(CultureInfo.InvariantCulture).ToLower());
  736. }
  737. if (!item.IsFolder)
  738. {
  739. writer.WriteElementString("playcount", userdata.PlayCount.ToString(UsCulture));
  740. writer.WriteElementString("watched", userdata.Played.ToString().ToLower());
  741. if (userdata.LastPlayedDate.HasValue)
  742. {
  743. writer.WriteElementString("lastplayed", userdata.LastPlayedDate.Value.ToLocalTime().ToString("yyyy-MM-dd HH:mm:ss").ToLower());
  744. }
  745. writer.WriteStartElement("resume");
  746. var runTimeTicks = item.RunTimeTicks ?? 0;
  747. writer.WriteElementString("position", TimeSpan.FromTicks(userdata.PlaybackPositionTicks).TotalSeconds.ToString(UsCulture));
  748. writer.WriteElementString("total", TimeSpan.FromTicks(runTimeTicks).TotalSeconds.ToString(UsCulture));
  749. }
  750. writer.WriteEndElement();
  751. }
  752. private void AddActors(List<PersonInfo> people, XmlWriter writer, ILibraryManager libraryManager, IFileSystem fileSystem, IServerConfigurationManager config, bool saveImagePath)
  753. {
  754. var actors = people
  755. .Where(i => !IsPersonType(i, PersonType.Director) && !IsPersonType(i, PersonType.Writer))
  756. .ToList();
  757. foreach (var person in actors)
  758. {
  759. writer.WriteStartElement("actor");
  760. if (!string.IsNullOrWhiteSpace(person.Name))
  761. {
  762. writer.WriteElementString("name", person.Name);
  763. }
  764. if (!string.IsNullOrWhiteSpace(person.Role))
  765. {
  766. writer.WriteElementString("role", person.Role);
  767. }
  768. if (!string.IsNullOrWhiteSpace(person.Type))
  769. {
  770. writer.WriteElementString("type", person.Type);
  771. }
  772. if (person.SortOrder.HasValue)
  773. {
  774. writer.WriteElementString("sortorder", person.SortOrder.Value.ToString(UsCulture));
  775. }
  776. if (saveImagePath)
  777. {
  778. try
  779. {
  780. var personEntity = libraryManager.GetPerson(person.Name);
  781. var image = personEntity.GetImageInfo(ImageType.Primary, 0);
  782. if (image != null)
  783. {
  784. writer.WriteElementString("thumb", GetImagePathToSave(image, libraryManager, config));
  785. }
  786. }
  787. catch (Exception)
  788. {
  789. // Already logged in core
  790. }
  791. }
  792. writer.WriteEndElement();
  793. }
  794. }
  795. private string GetImagePathToSave(ItemImageInfo image, ILibraryManager libraryManager, IServerConfigurationManager config)
  796. {
  797. if (!image.IsLocalFile)
  798. {
  799. return image.Path;
  800. }
  801. return libraryManager.GetPathAfterNetworkSubstitution(image.Path);
  802. }
  803. private bool IsPersonType(PersonInfo person, string type)
  804. {
  805. return string.Equals(person.Type, type, StringComparison.OrdinalIgnoreCase) || string.Equals(person.Role, type, StringComparison.OrdinalIgnoreCase);
  806. }
  807. private void AddCustomTags(string path, List<string> xmlTagsUsed, XmlWriter writer, ILogger logger, IFileSystem fileSystem)
  808. {
  809. var settings = XmlReaderSettingsFactory.Create(false);
  810. settings.CheckCharacters = false;
  811. settings.IgnoreProcessingInstructions = true;
  812. settings.IgnoreComments = true;
  813. using (var fileStream = fileSystem.OpenRead(path))
  814. {
  815. using (var streamReader = new StreamReader(fileStream, Encoding.UTF8))
  816. {
  817. // Use XmlReader for best performance
  818. using (var reader = XmlReader.Create(streamReader, settings))
  819. {
  820. try
  821. {
  822. reader.MoveToContent();
  823. }
  824. catch (Exception ex)
  825. {
  826. logger.ErrorException("Error reading existing xml tags from {0}.", ex, path);
  827. return;
  828. }
  829. reader.Read();
  830. // Loop through each element
  831. while (!reader.EOF && reader.ReadState == ReadState.Interactive)
  832. {
  833. if (reader.NodeType == XmlNodeType.Element)
  834. {
  835. var name = reader.Name;
  836. if (!CommonTags.ContainsKey(name) && !xmlTagsUsed.Contains(name, StringComparer.OrdinalIgnoreCase))
  837. {
  838. writer.WriteNode(reader, false);
  839. }
  840. else
  841. {
  842. reader.Skip();
  843. }
  844. }
  845. else
  846. {
  847. reader.Read();
  848. }
  849. }
  850. }
  851. }
  852. }
  853. }
  854. private string GetTagForProviderKey(string providerKey)
  855. {
  856. return providerKey.ToLower() + "id";
  857. }
  858. }
  859. }