BaseNfoSaver.cs 38 KB

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