BaseNfoSaver.cs 38 KB

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