DidlBuilder.cs 42 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203
  1. using MediaBrowser.Model.Extensions;
  2. using MediaBrowser.Controller.Channels;
  3. using MediaBrowser.Controller.Drawing;
  4. using MediaBrowser.Controller.Entities;
  5. using MediaBrowser.Controller.Entities.Audio;
  6. using MediaBrowser.Controller.Entities.Movies;
  7. using MediaBrowser.Controller.Entities.TV;
  8. using MediaBrowser.Controller.Library;
  9. using MediaBrowser.Controller.Playlists;
  10. using Emby.Dlna.ContentDirectory;
  11. using MediaBrowser.Model.Dlna;
  12. using MediaBrowser.Model.Drawing;
  13. using MediaBrowser.Model.Entities;
  14. using MediaBrowser.Model.Logging;
  15. using MediaBrowser.Model.Net;
  16. using System;
  17. using System.Globalization;
  18. using System.IO;
  19. using System.Linq;
  20. using System.Text;
  21. using System.Xml;
  22. using MediaBrowser.Controller.MediaEncoding;
  23. using MediaBrowser.Model.Configuration;
  24. using MediaBrowser.Model.Globalization;
  25. namespace Emby.Dlna.Didl
  26. {
  27. public class DidlBuilder
  28. {
  29. private readonly CultureInfo _usCulture = new CultureInfo("en-US");
  30. private const string NS_DIDL = "urn:schemas-upnp-org:metadata-1-0/DIDL-Lite/";
  31. private const string NS_DC = "http://purl.org/dc/elements/1.1/";
  32. private const string NS_UPNP = "urn:schemas-upnp-org:metadata-1-0/upnp/";
  33. private const string NS_DLNA = "urn:schemas-dlna-org:metadata-1-0/";
  34. private readonly DeviceProfile _profile;
  35. private readonly IImageProcessor _imageProcessor;
  36. private readonly string _serverAddress;
  37. private readonly string _accessToken;
  38. private readonly User _user;
  39. private readonly IUserDataManager _userDataManager;
  40. private readonly ILocalizationManager _localization;
  41. private readonly IMediaSourceManager _mediaSourceManager;
  42. private readonly ILogger _logger;
  43. private readonly ILibraryManager _libraryManager;
  44. private readonly IMediaEncoder _mediaEncoder;
  45. public DidlBuilder(DeviceProfile profile, User user, IImageProcessor imageProcessor, string serverAddress, string accessToken, IUserDataManager userDataManager, ILocalizationManager localization, IMediaSourceManager mediaSourceManager, ILogger logger, ILibraryManager libraryManager, IMediaEncoder mediaEncoder)
  46. {
  47. _profile = profile;
  48. _imageProcessor = imageProcessor;
  49. _serverAddress = serverAddress;
  50. _userDataManager = userDataManager;
  51. _localization = localization;
  52. _mediaSourceManager = mediaSourceManager;
  53. _logger = logger;
  54. _libraryManager = libraryManager;
  55. _mediaEncoder = mediaEncoder;
  56. _accessToken = accessToken;
  57. _user = user;
  58. }
  59. public string GetItemDidl(DlnaOptions options, BaseItem item, User user, BaseItem context, string deviceId, Filter filter, StreamInfo streamInfo)
  60. {
  61. var settings = new XmlWriterSettings
  62. {
  63. Encoding = Encoding.UTF8,
  64. CloseOutput = false,
  65. OmitXmlDeclaration = true,
  66. ConformanceLevel = ConformanceLevel.Fragment
  67. };
  68. StringWriter builder = new StringWriterWithEncoding(Encoding.UTF8);
  69. using (XmlWriter writer = XmlWriter.Create(builder, settings))
  70. {
  71. //writer.WriteStartDocument();
  72. writer.WriteStartElement(string.Empty, "DIDL-Lite", NS_DIDL);
  73. writer.WriteAttributeString("xmlns", "dc", null, NS_DC);
  74. writer.WriteAttributeString("xmlns", "dlna", null, NS_DLNA);
  75. writer.WriteAttributeString("xmlns", "upnp", null, NS_UPNP);
  76. //didl.SetAttribute("xmlns:sec", NS_SEC);
  77. WriteXmlRootAttributes(_profile, writer);
  78. WriteItemElement(options, writer, item, user, context, null, deviceId, filter, streamInfo);
  79. writer.WriteFullEndElement();
  80. //writer.WriteEndDocument();
  81. }
  82. return builder.ToString();
  83. }
  84. public static void WriteXmlRootAttributes(DeviceProfile profile, XmlWriter writer)
  85. {
  86. foreach (var att in profile.XmlRootAttributes)
  87. {
  88. var parts = att.Name.Split(new[] { ':' }, StringSplitOptions.RemoveEmptyEntries);
  89. if (parts.Length == 2)
  90. {
  91. writer.WriteAttributeString(parts[0], parts[1], null, att.Value);
  92. }
  93. else
  94. {
  95. writer.WriteAttributeString(att.Name, att.Value);
  96. }
  97. }
  98. }
  99. public void WriteItemElement(DlnaOptions options,
  100. XmlWriter writer,
  101. BaseItem item,
  102. User user,
  103. BaseItem context,
  104. StubType? contextStubType,
  105. string deviceId,
  106. Filter filter,
  107. StreamInfo streamInfo = null)
  108. {
  109. var clientId = GetClientId(item, null);
  110. writer.WriteStartElement(string.Empty, "item", NS_DIDL);
  111. writer.WriteAttributeString("restricted", "1");
  112. writer.WriteAttributeString("id", clientId);
  113. if (context != null)
  114. {
  115. writer.WriteAttributeString("parentID", GetClientId(context, contextStubType));
  116. }
  117. else
  118. {
  119. var parent = item.DisplayParentId;
  120. if (parent.HasValue)
  121. {
  122. writer.WriteAttributeString("parentID", GetClientId(parent.Value, null));
  123. }
  124. }
  125. AddGeneralProperties(item, null, context, writer, filter);
  126. AddSamsungBookmarkInfo(item, user, writer);
  127. // refID?
  128. // storeAttribute(itemNode, object, ClassProperties.REF_ID, false);
  129. var hasMediaSources = item as IHasMediaSources;
  130. if (hasMediaSources != null)
  131. {
  132. if (string.Equals(item.MediaType, MediaType.Audio, StringComparison.OrdinalIgnoreCase))
  133. {
  134. AddAudioResource(options, writer, hasMediaSources, deviceId, filter, streamInfo);
  135. }
  136. else if (string.Equals(item.MediaType, MediaType.Video, StringComparison.OrdinalIgnoreCase))
  137. {
  138. AddVideoResource(options, writer, hasMediaSources, deviceId, filter, streamInfo);
  139. }
  140. }
  141. AddCover(item, context, null, writer);
  142. writer.WriteFullEndElement();
  143. }
  144. private ILogger GetStreamBuilderLogger(DlnaOptions options)
  145. {
  146. if (options.EnableDebugLog)
  147. {
  148. return _logger;
  149. }
  150. return new NullLogger();
  151. }
  152. private string GetMimeType(string input)
  153. {
  154. var mime = MimeTypes.GetMimeType(input);
  155. if (string.Equals(mime, "video/mp2t", StringComparison.OrdinalIgnoreCase))
  156. {
  157. mime = "video/mpeg";
  158. }
  159. return mime;
  160. }
  161. private void AddVideoResource(DlnaOptions options, XmlWriter writer, IHasMediaSources video, string deviceId, Filter filter, StreamInfo streamInfo = null)
  162. {
  163. if (streamInfo == null)
  164. {
  165. var sources = _mediaSourceManager.GetStaticMediaSources(video, true, _user).ToList();
  166. streamInfo = new StreamBuilder(_mediaEncoder, GetStreamBuilderLogger(options)).BuildVideoItem(new VideoOptions
  167. {
  168. ItemId = GetClientId(video),
  169. MediaSources = sources,
  170. Profile = _profile,
  171. DeviceId = deviceId,
  172. MaxBitrate = _profile.MaxStreamingBitrate
  173. });
  174. }
  175. var targetWidth = streamInfo.TargetWidth;
  176. var targetHeight = streamInfo.TargetHeight;
  177. var contentFeatureList = new ContentFeatureBuilder(_profile).BuildVideoHeader(streamInfo.Container,
  178. streamInfo.TargetVideoCodec,
  179. streamInfo.TargetAudioCodec,
  180. targetWidth,
  181. targetHeight,
  182. streamInfo.TargetVideoBitDepth,
  183. streamInfo.TargetVideoBitrate,
  184. streamInfo.TargetTimestamp,
  185. streamInfo.IsDirectStream,
  186. streamInfo.RunTimeTicks,
  187. streamInfo.TargetVideoProfile,
  188. streamInfo.TargetVideoLevel,
  189. streamInfo.TargetFramerate,
  190. streamInfo.TargetPacketLength,
  191. streamInfo.TranscodeSeekInfo,
  192. streamInfo.IsTargetAnamorphic,
  193. streamInfo.TargetRefFrames,
  194. streamInfo.TargetVideoStreamCount,
  195. streamInfo.TargetAudioStreamCount,
  196. streamInfo.TargetVideoCodecTag,
  197. streamInfo.IsTargetAVC);
  198. foreach (var contentFeature in contentFeatureList)
  199. {
  200. AddVideoResource(writer, video, deviceId, filter, contentFeature, streamInfo);
  201. }
  202. var subtitleProfiles = streamInfo.GetSubtitleProfiles(false, _serverAddress, _accessToken)
  203. .Where(subtitle => subtitle.DeliveryMethod == SubtitleDeliveryMethod.External)
  204. .ToList();
  205. foreach (var subtitle in subtitleProfiles)
  206. {
  207. var subtitleAdded = AddSubtitleElement(writer, subtitle);
  208. if (subtitleAdded && _profile.EnableSingleSubtitleLimit)
  209. {
  210. break;
  211. }
  212. }
  213. }
  214. private bool AddSubtitleElement(XmlWriter writer, SubtitleStreamInfo info)
  215. {
  216. var subtitleProfile = _profile.SubtitleProfiles
  217. .FirstOrDefault(i => string.Equals(info.Format, i.Format, StringComparison.OrdinalIgnoreCase) && i.Method == SubtitleDeliveryMethod.External);
  218. if (subtitleProfile == null)
  219. {
  220. return false;
  221. }
  222. var subtitleMode = subtitleProfile.DidlMode;
  223. if (string.Equals(subtitleMode, "CaptionInfoEx", StringComparison.OrdinalIgnoreCase))
  224. {
  225. // <sec:CaptionInfoEx sec:type="srt">http://192.168.1.3:9999/video.srt</sec:CaptionInfoEx>
  226. // <sec:CaptionInfo sec:type="srt">http://192.168.1.3:9999/video.srt</sec:CaptionInfo>
  227. writer.WriteStartElement("sec", "CaptionInfoEx", null);
  228. writer.WriteAttributeString("sec", "type", null, info.Format.ToLower());
  229. writer.WriteString(info.Url);
  230. writer.WriteFullEndElement();
  231. }
  232. else if (string.Equals(subtitleMode, "smi", StringComparison.OrdinalIgnoreCase))
  233. {
  234. writer.WriteStartElement(string.Empty, "res", NS_DIDL);
  235. writer.WriteAttributeString("protocolInfo", "http-get:*:smi/caption:*");
  236. writer.WriteString(info.Url);
  237. writer.WriteFullEndElement();
  238. }
  239. else
  240. {
  241. writer.WriteStartElement(string.Empty, "res", NS_DIDL);
  242. var protocolInfo = string.Format("http-get:*:text/{0}:*", info.Format.ToLower());
  243. writer.WriteAttributeString("protocolInfo", protocolInfo);
  244. writer.WriteString(info.Url);
  245. writer.WriteFullEndElement();
  246. }
  247. return true;
  248. }
  249. private void AddVideoResource(XmlWriter writer, IHasMediaSources video, string deviceId, Filter filter, string contentFeatures, StreamInfo streamInfo)
  250. {
  251. writer.WriteStartElement(string.Empty, "res", NS_DIDL);
  252. var url = streamInfo.ToDlnaUrl(_serverAddress, _accessToken);
  253. var mediaSource = streamInfo.MediaSource;
  254. if (mediaSource.RunTimeTicks.HasValue)
  255. {
  256. writer.WriteAttributeString("duration", TimeSpan.FromTicks(mediaSource.RunTimeTicks.Value).ToString("c", _usCulture));
  257. }
  258. if (filter.Contains("res@size"))
  259. {
  260. if (streamInfo.IsDirectStream || streamInfo.EstimateContentLength)
  261. {
  262. var size = streamInfo.TargetSize;
  263. if (size.HasValue)
  264. {
  265. writer.WriteAttributeString("size", size.Value.ToString(_usCulture));
  266. }
  267. }
  268. }
  269. var totalBitrate = streamInfo.TargetTotalBitrate;
  270. var targetSampleRate = streamInfo.TargetAudioSampleRate;
  271. var targetChannels = streamInfo.TargetAudioChannels;
  272. var targetWidth = streamInfo.TargetWidth;
  273. var targetHeight = streamInfo.TargetHeight;
  274. if (targetChannels.HasValue)
  275. {
  276. writer.WriteAttributeString("nrAudioChannels", targetChannels.Value.ToString(_usCulture));
  277. }
  278. if (filter.Contains("res@resolution"))
  279. {
  280. if (targetWidth.HasValue && targetHeight.HasValue)
  281. {
  282. writer.WriteAttributeString("resolution", string.Format("{0}x{1}", targetWidth.Value, targetHeight.Value));
  283. }
  284. }
  285. if (targetSampleRate.HasValue)
  286. {
  287. writer.WriteAttributeString("sampleFrequency", targetSampleRate.Value.ToString(_usCulture));
  288. }
  289. if (totalBitrate.HasValue)
  290. {
  291. writer.WriteAttributeString("bitrate", totalBitrate.Value.ToString(_usCulture));
  292. }
  293. var mediaProfile = _profile.GetVideoMediaProfile(streamInfo.Container,
  294. streamInfo.TargetAudioCodec,
  295. streamInfo.TargetVideoCodec,
  296. streamInfo.TargetAudioBitrate,
  297. targetWidth,
  298. targetHeight,
  299. streamInfo.TargetVideoBitDepth,
  300. streamInfo.TargetVideoProfile,
  301. streamInfo.TargetVideoLevel,
  302. streamInfo.TargetFramerate,
  303. streamInfo.TargetPacketLength,
  304. streamInfo.TargetTimestamp,
  305. streamInfo.IsTargetAnamorphic,
  306. streamInfo.TargetRefFrames,
  307. streamInfo.TargetVideoStreamCount,
  308. streamInfo.TargetAudioStreamCount,
  309. streamInfo.TargetVideoCodecTag,
  310. streamInfo.IsTargetAVC);
  311. var filename = url.Substring(0, url.IndexOf('?'));
  312. var mimeType = mediaProfile == null || string.IsNullOrEmpty(mediaProfile.MimeType)
  313. ? GetMimeType(filename)
  314. : mediaProfile.MimeType;
  315. writer.WriteAttributeString("protocolInfo", String.Format(
  316. "http-get:*:{0}:{1}",
  317. mimeType,
  318. contentFeatures
  319. ));
  320. writer.WriteString(url);
  321. writer.WriteFullEndElement();
  322. }
  323. private string GetDisplayName(BaseItem item, StubType? itemStubType, BaseItem context)
  324. {
  325. if (itemStubType.HasValue && itemStubType.Value == StubType.People)
  326. {
  327. if (item is Video)
  328. {
  329. return _localization.GetLocalizedString("HeaderCastCrew");
  330. }
  331. return _localization.GetLocalizedString("HeaderPeople");
  332. }
  333. var episode = item as Episode;
  334. var season = context as Season;
  335. if (episode != null && season != null)
  336. {
  337. // This is a special embedded within a season
  338. if (item.ParentIndexNumber.HasValue && item.ParentIndexNumber.Value == 0)
  339. {
  340. if (season.IndexNumber.HasValue && season.IndexNumber.Value != 0)
  341. {
  342. return string.Format(_localization.GetLocalizedString("ValueSpecialEpisodeName"), item.Name);
  343. }
  344. }
  345. if (item.IndexNumber.HasValue)
  346. {
  347. var number = item.IndexNumber.Value.ToString("00", CultureInfo.InvariantCulture);
  348. if (episode.IndexNumberEnd.HasValue)
  349. {
  350. number += "-" + episode.IndexNumberEnd.Value.ToString("00", CultureInfo.InvariantCulture);
  351. }
  352. return number + " - " + item.Name;
  353. }
  354. }
  355. return item.Name;
  356. }
  357. private void AddAudioResource(DlnaOptions options, XmlWriter writer, IHasMediaSources audio, string deviceId, Filter filter, StreamInfo streamInfo = null)
  358. {
  359. writer.WriteStartElement(string.Empty, "res", NS_DIDL);
  360. if (streamInfo == null)
  361. {
  362. var sources = _mediaSourceManager.GetStaticMediaSources(audio, true, _user).ToList();
  363. streamInfo = new StreamBuilder(_mediaEncoder, GetStreamBuilderLogger(options)).BuildAudioItem(new AudioOptions
  364. {
  365. ItemId = GetClientId(audio),
  366. MediaSources = sources,
  367. Profile = _profile,
  368. DeviceId = deviceId
  369. });
  370. }
  371. var url = streamInfo.ToDlnaUrl(_serverAddress, _accessToken);
  372. var mediaSource = streamInfo.MediaSource;
  373. if (mediaSource.RunTimeTicks.HasValue)
  374. {
  375. writer.WriteAttributeString("duration", TimeSpan.FromTicks(mediaSource.RunTimeTicks.Value).ToString("c", _usCulture));
  376. }
  377. if (filter.Contains("res@size"))
  378. {
  379. if (streamInfo.IsDirectStream || streamInfo.EstimateContentLength)
  380. {
  381. var size = streamInfo.TargetSize;
  382. if (size.HasValue)
  383. {
  384. writer.WriteAttributeString("size", size.Value.ToString(_usCulture));
  385. }
  386. }
  387. }
  388. var targetAudioBitrate = streamInfo.TargetAudioBitrate;
  389. var targetSampleRate = streamInfo.TargetAudioSampleRate;
  390. var targetChannels = streamInfo.TargetAudioChannels;
  391. if (targetChannels.HasValue)
  392. {
  393. writer.WriteAttributeString("nrAudioChannels", targetChannels.Value.ToString(_usCulture));
  394. }
  395. if (targetSampleRate.HasValue)
  396. {
  397. writer.WriteAttributeString("sampleFrequency", targetSampleRate.Value.ToString(_usCulture));
  398. }
  399. if (targetAudioBitrate.HasValue)
  400. {
  401. writer.WriteAttributeString("bitrate", targetAudioBitrate.Value.ToString(_usCulture));
  402. }
  403. var mediaProfile = _profile.GetAudioMediaProfile(streamInfo.Container,
  404. streamInfo.TargetAudioCodec,
  405. targetChannels,
  406. targetAudioBitrate);
  407. var filename = url.Substring(0, url.IndexOf('?'));
  408. var mimeType = mediaProfile == null || string.IsNullOrEmpty(mediaProfile.MimeType)
  409. ? GetMimeType(filename)
  410. : mediaProfile.MimeType;
  411. var contentFeatures = new ContentFeatureBuilder(_profile).BuildAudioHeader(streamInfo.Container,
  412. streamInfo.TargetAudioCodec,
  413. targetAudioBitrate,
  414. targetSampleRate,
  415. targetChannels,
  416. streamInfo.IsDirectStream,
  417. streamInfo.RunTimeTicks,
  418. streamInfo.TranscodeSeekInfo);
  419. writer.WriteAttributeString("protocolInfo", String.Format(
  420. "http-get:*:{0}:{1}",
  421. mimeType,
  422. contentFeatures
  423. ));
  424. writer.WriteString(url);
  425. writer.WriteFullEndElement();
  426. }
  427. public static bool IsIdRoot(string id)
  428. {
  429. if (string.IsNullOrWhiteSpace(id) ||
  430. string.Equals(id, "0", StringComparison.OrdinalIgnoreCase)
  431. // Samsung sometimes uses 1 as root
  432. || string.Equals(id, "1", StringComparison.OrdinalIgnoreCase))
  433. {
  434. return true;
  435. }
  436. return false;
  437. }
  438. public void WriteFolderElement(XmlWriter writer, BaseItem folder, StubType? stubType, BaseItem context, int childCount, Filter filter, string requestedId = null)
  439. {
  440. writer.WriteStartElement(string.Empty, "container", NS_DIDL);
  441. writer.WriteAttributeString("restricted", "0");
  442. writer.WriteAttributeString("searchable", "1");
  443. writer.WriteAttributeString("childCount", childCount.ToString(_usCulture));
  444. var clientId = GetClientId(folder, stubType);
  445. if (string.Equals(requestedId, "0"))
  446. {
  447. writer.WriteAttributeString("id", "0");
  448. writer.WriteAttributeString("parentID", "-1");
  449. }
  450. else
  451. {
  452. writer.WriteAttributeString("id", clientId);
  453. if (context != null)
  454. {
  455. writer.WriteAttributeString("parentID", GetClientId(context, null));
  456. }
  457. else
  458. {
  459. var parent = folder.DisplayParentId;
  460. if (!parent.HasValue)
  461. {
  462. writer.WriteAttributeString("parentID", "0");
  463. }
  464. else
  465. {
  466. writer.WriteAttributeString("parentID", GetClientId(parent.Value, null));
  467. }
  468. }
  469. }
  470. AddGeneralProperties(folder, stubType, context, writer, filter);
  471. AddCover(folder, context, stubType, writer);
  472. writer.WriteFullEndElement();
  473. }
  474. private void AddSamsungBookmarkInfo(BaseItem item, User user, XmlWriter writer)
  475. {
  476. if (!item.SupportsPositionTicksResume || item is Folder)
  477. {
  478. return;
  479. }
  480. XmlAttribute secAttribute = null;
  481. foreach (var attribute in _profile.XmlRootAttributes)
  482. {
  483. if (string.Equals(attribute.Name, "xmlns:sec", StringComparison.OrdinalIgnoreCase))
  484. {
  485. secAttribute = attribute;
  486. break;
  487. }
  488. }
  489. // Not a samsung device
  490. if (secAttribute == null)
  491. {
  492. return;
  493. }
  494. var userdata = _userDataManager.GetUserData(user.Id, item);
  495. if (userdata.PlaybackPositionTicks > 0)
  496. {
  497. var elementValue = string.Format("BM={0}", Convert.ToInt32(TimeSpan.FromTicks(userdata.PlaybackPositionTicks).TotalSeconds).ToString(_usCulture));
  498. AddValue(writer, "sec", "dcmInfo", elementValue, secAttribute.Value);
  499. }
  500. }
  501. /// <summary>
  502. /// Adds fields used by both items and folders
  503. /// </summary>
  504. private void AddCommonFields(BaseItem item, StubType? itemStubType, BaseItem context, XmlWriter writer, Filter filter)
  505. {
  506. // Don't filter on dc:title because not all devices will include it in the filter
  507. // MediaMonkey for example won't display content without a title
  508. //if (filter.Contains("dc:title"))
  509. {
  510. AddValue(writer, "dc", "title", GetDisplayName(item, itemStubType, context), NS_DC);
  511. }
  512. WriteObjectClass(writer, item, itemStubType);
  513. if (filter.Contains("dc:date"))
  514. {
  515. if (item.PremiereDate.HasValue)
  516. {
  517. AddValue(writer, "dc", "date", item.PremiereDate.Value.ToString("o"), NS_DC);
  518. }
  519. }
  520. if (filter.Contains("upnp:genre"))
  521. {
  522. foreach (var genre in item.Genres)
  523. {
  524. AddValue(writer, "upnp", "genre", genre, NS_UPNP);
  525. }
  526. }
  527. foreach (var studio in item.Studios)
  528. {
  529. AddValue(writer, "upnp", "publisher", studio, NS_UPNP);
  530. }
  531. if (filter.Contains("dc:description"))
  532. {
  533. var desc = item.Overview;
  534. if (!string.IsNullOrWhiteSpace(desc))
  535. {
  536. AddValue(writer, "dc", "description", desc, NS_DC);
  537. }
  538. }
  539. if (filter.Contains("upnp:longDescription"))
  540. {
  541. if (!string.IsNullOrWhiteSpace(item.Overview))
  542. {
  543. AddValue(writer, "upnp", "longDescription", item.Overview, NS_UPNP);
  544. }
  545. }
  546. if (!string.IsNullOrEmpty(item.OfficialRating))
  547. {
  548. if (filter.Contains("dc:rating"))
  549. {
  550. AddValue(writer, "dc", "rating", item.OfficialRating, NS_DC);
  551. }
  552. if (filter.Contains("upnp:rating"))
  553. {
  554. AddValue(writer, "upnp", "rating", item.OfficialRating, NS_UPNP);
  555. }
  556. }
  557. AddPeople(item, writer);
  558. }
  559. private void WriteObjectClass(XmlWriter writer, BaseItem item, StubType? stubType)
  560. {
  561. // More types here
  562. // http://oss.linn.co.uk/repos/Public/LibUpnpCil/DidlLite/UpnpAv/Test/TestDidlLite.cs
  563. writer.WriteStartElement("upnp", "class", NS_UPNP);
  564. if (item.IsDisplayedAsFolder || stubType.HasValue)
  565. {
  566. string classType = null;
  567. if (!_profile.RequiresPlainFolders)
  568. {
  569. if (item is MusicAlbum)
  570. {
  571. classType = "object.container.album.musicAlbum";
  572. }
  573. else if (item is MusicArtist)
  574. {
  575. classType = "object.container.person.musicArtist";
  576. }
  577. else if (item is Series || item is Season || item is BoxSet || item is Video)
  578. {
  579. classType = "object.container.album.videoAlbum";
  580. }
  581. else if (item is Playlist)
  582. {
  583. classType = "object.container.playlistContainer";
  584. }
  585. else if (item is PhotoAlbum)
  586. {
  587. classType = "object.container.album.photoAlbum";
  588. }
  589. }
  590. writer.WriteString(classType ?? "object.container.storageFolder");
  591. }
  592. else if (string.Equals(item.MediaType, MediaType.Audio, StringComparison.OrdinalIgnoreCase))
  593. {
  594. writer.WriteString("object.item.audioItem.musicTrack");
  595. }
  596. else if (string.Equals(item.MediaType, MediaType.Photo, StringComparison.OrdinalIgnoreCase))
  597. {
  598. writer.WriteString("object.item.imageItem.photo");
  599. }
  600. else if (string.Equals(item.MediaType, MediaType.Video, StringComparison.OrdinalIgnoreCase))
  601. {
  602. if (!_profile.RequiresPlainVideoItems && item is Movie)
  603. {
  604. writer.WriteString("object.item.videoItem.movie");
  605. }
  606. else if (!_profile.RequiresPlainVideoItems && item is MusicVideo)
  607. {
  608. writer.WriteString("object.item.videoItem.musicVideoClip");
  609. }
  610. else
  611. {
  612. writer.WriteString("object.item.videoItem");
  613. }
  614. }
  615. else if (item is MusicGenre)
  616. {
  617. writer.WriteString(_profile.RequiresPlainFolders ? "object.container.storageFolder" : "object.container.genre.musicGenre");
  618. }
  619. else if (item is Genre || item is GameGenre)
  620. {
  621. writer.WriteString(_profile.RequiresPlainFolders ? "object.container.storageFolder" : "object.container.genre");
  622. }
  623. else
  624. {
  625. writer.WriteString("object.item");
  626. }
  627. writer.WriteFullEndElement();
  628. }
  629. private void AddPeople(BaseItem item, XmlWriter writer)
  630. {
  631. var types = new[]
  632. {
  633. PersonType.Director,
  634. PersonType.Writer,
  635. PersonType.Producer,
  636. PersonType.Composer,
  637. "Creator"
  638. };
  639. var people = _libraryManager.GetPeople(item);
  640. var index = 0;
  641. // Seeing some LG models locking up due content with large lists of people
  642. // The actual issue might just be due to processing a more metadata than it can handle
  643. var limit = 6;
  644. foreach (var actor in people)
  645. {
  646. var type = types.FirstOrDefault(i => string.Equals(i, actor.Type, StringComparison.OrdinalIgnoreCase) || string.Equals(i, actor.Role, StringComparison.OrdinalIgnoreCase))
  647. ?? PersonType.Actor;
  648. AddValue(writer, "upnp", type.ToLower(), actor.Name, NS_UPNP);
  649. index++;
  650. if (index >= limit)
  651. {
  652. break;
  653. }
  654. }
  655. }
  656. private void AddGeneralProperties(BaseItem item, StubType? itemStubType, BaseItem context, XmlWriter writer, Filter filter)
  657. {
  658. AddCommonFields(item, itemStubType, context, writer, filter);
  659. var audio = item as Audio;
  660. if (audio != null)
  661. {
  662. foreach (var artist in audio.Artists)
  663. {
  664. AddValue(writer, "upnp", "artist", artist, NS_UPNP);
  665. }
  666. if (!string.IsNullOrEmpty(audio.Album))
  667. {
  668. AddValue(writer, "upnp", "album", audio.Album, NS_UPNP);
  669. }
  670. foreach (var artist in audio.AlbumArtists)
  671. {
  672. AddAlbumArtist(writer, artist);
  673. }
  674. }
  675. var album = item as MusicAlbum;
  676. if (album != null)
  677. {
  678. foreach (var artist in album.AlbumArtists)
  679. {
  680. AddAlbumArtist(writer, artist);
  681. AddValue(writer, "upnp", "artist", artist, NS_UPNP);
  682. }
  683. foreach (var artist in album.Artists)
  684. {
  685. AddValue(writer, "upnp", "artist", artist, NS_UPNP);
  686. }
  687. }
  688. var musicVideo = item as MusicVideo;
  689. if (musicVideo != null)
  690. {
  691. foreach (var artist in musicVideo.Artists)
  692. {
  693. AddValue(writer, "upnp", "artist", artist, NS_UPNP);
  694. AddAlbumArtist(writer, artist);
  695. }
  696. if (!string.IsNullOrEmpty(musicVideo.Album))
  697. {
  698. AddValue(writer, "upnp", "album", musicVideo.Album, NS_UPNP);
  699. }
  700. }
  701. if (item.IndexNumber.HasValue)
  702. {
  703. AddValue(writer, "upnp", "originalTrackNumber", item.IndexNumber.Value.ToString(_usCulture), NS_UPNP);
  704. if (item is Episode)
  705. {
  706. AddValue(writer, "upnp", "episodeNumber", item.IndexNumber.Value.ToString(_usCulture), NS_UPNP);
  707. }
  708. }
  709. }
  710. private void AddAlbumArtist(XmlWriter writer, string name)
  711. {
  712. try
  713. {
  714. writer.WriteStartElement("upnp", "artist", NS_UPNP);
  715. writer.WriteAttributeString("role", "AlbumArtist");
  716. writer.WriteString(name);
  717. writer.WriteFullEndElement();
  718. }
  719. catch (XmlException)
  720. {
  721. //_logger.Error("Error adding xml value: " + value);
  722. }
  723. }
  724. private void AddValue(XmlWriter writer, string prefix, string name, string value, string namespaceUri)
  725. {
  726. try
  727. {
  728. writer.WriteElementString(prefix, name, namespaceUri, value);
  729. }
  730. catch (XmlException)
  731. {
  732. //_logger.Error("Error adding xml value: " + value);
  733. }
  734. }
  735. private void AddCover(BaseItem item, BaseItem context, StubType? stubType, XmlWriter writer)
  736. {
  737. if (stubType.HasValue && stubType.Value == StubType.People)
  738. {
  739. AddEmbeddedImageAsCover("people", writer);
  740. return;
  741. }
  742. ImageDownloadInfo imageInfo = null;
  743. if (context is UserView)
  744. {
  745. var episode = item as Episode;
  746. if (episode != null)
  747. {
  748. var parent = episode.Series;
  749. if (parent != null)
  750. {
  751. imageInfo = GetImageInfo(parent);
  752. }
  753. }
  754. }
  755. // Finally, just use the image from the item
  756. if (imageInfo == null)
  757. {
  758. imageInfo = GetImageInfo(item);
  759. }
  760. if (imageInfo == null)
  761. {
  762. return;
  763. }
  764. var playbackPercentage = 0;
  765. var unplayedCount = 0;
  766. if (item is Video)
  767. {
  768. var userData = _userDataManager.GetUserDataDto(item, _user).Result;
  769. playbackPercentage = Convert.ToInt32(userData.PlayedPercentage ?? 0);
  770. if (playbackPercentage >= 100 || userData.Played)
  771. {
  772. playbackPercentage = 100;
  773. }
  774. }
  775. else if (item is Series || item is Season || item is BoxSet)
  776. {
  777. var userData = _userDataManager.GetUserDataDto(item, _user).Result;
  778. if (userData.Played)
  779. {
  780. playbackPercentage = 100;
  781. }
  782. else
  783. {
  784. unplayedCount = userData.UnplayedItemCount ?? 0;
  785. }
  786. }
  787. var albumartUrlInfo = GetImageUrl(imageInfo, _profile.MaxAlbumArtWidth, _profile.MaxAlbumArtHeight, playbackPercentage, unplayedCount, "jpg");
  788. writer.WriteStartElement("upnp", "albumArtURI", NS_UPNP);
  789. writer.WriteAttributeString("dlna", "profileID", NS_DLNA, _profile.AlbumArtPn);
  790. writer.WriteString(albumartUrlInfo.Url);
  791. writer.WriteFullEndElement();
  792. // TOOD: Remove these default values
  793. var iconUrlInfo = GetImageUrl(imageInfo, _profile.MaxIconWidth ?? 48, _profile.MaxIconHeight ?? 48, playbackPercentage, unplayedCount, "jpg");
  794. writer.WriteElementString("upnp", "icon", NS_UPNP, iconUrlInfo.Url);
  795. if (!_profile.EnableAlbumArtInDidl)
  796. {
  797. if (string.Equals(item.MediaType, MediaType.Audio, StringComparison.OrdinalIgnoreCase)
  798. || string.Equals(item.MediaType, MediaType.Video, StringComparison.OrdinalIgnoreCase))
  799. {
  800. if (!stubType.HasValue)
  801. {
  802. return;
  803. }
  804. }
  805. }
  806. AddImageResElement(item, writer, 160, 160, playbackPercentage, unplayedCount, "jpg", "JPEG_TN");
  807. if (!_profile.EnableSingleAlbumArtLimit)
  808. {
  809. AddImageResElement(item, writer, 4096, 4096, playbackPercentage, unplayedCount, "jpg", "JPEG_LRG");
  810. AddImageResElement(item, writer, 1024, 768, playbackPercentage, unplayedCount, "jpg", "JPEG_MED");
  811. AddImageResElement(item, writer, 640, 480, playbackPercentage, unplayedCount, "jpg", "JPEG_SM");
  812. AddImageResElement(item, writer, 4096, 4096, playbackPercentage, unplayedCount, "png", "PNG_LRG");
  813. AddImageResElement(item, writer, 160, 160, playbackPercentage, unplayedCount, "png", "PNG_TN");
  814. }
  815. }
  816. private void AddEmbeddedImageAsCover(string name, XmlWriter writer)
  817. {
  818. writer.WriteStartElement("upnp", "albumArtURI", NS_UPNP);
  819. writer.WriteAttributeString("dlna", "profileID", NS_DLNA, _profile.AlbumArtPn);
  820. writer.WriteString(_serverAddress + "/Dlna/icons/people480.jpg");
  821. writer.WriteFullEndElement();
  822. writer.WriteElementString("upnp", "icon", NS_UPNP, _serverAddress + "/Dlna/icons/people48.jpg");
  823. }
  824. private void AddImageResElement(BaseItem item,
  825. XmlWriter writer,
  826. int maxWidth,
  827. int maxHeight,
  828. int playbackPercentage,
  829. int unplayedCount,
  830. string format,
  831. string org_Pn)
  832. {
  833. var imageInfo = GetImageInfo(item);
  834. if (imageInfo == null)
  835. {
  836. return;
  837. }
  838. var albumartUrlInfo = GetImageUrl(imageInfo, maxWidth, maxHeight, playbackPercentage, unplayedCount, format);
  839. writer.WriteStartElement(string.Empty, "res", NS_DIDL);
  840. var width = albumartUrlInfo.Width;
  841. var height = albumartUrlInfo.Height;
  842. var contentFeatures = new ContentFeatureBuilder(_profile)
  843. .BuildImageHeader(format, width, height, imageInfo.IsDirectStream, org_Pn);
  844. writer.WriteAttributeString("protocolInfo", String.Format(
  845. "http-get:*:{0}:{1}",
  846. GetMimeType("file." + format),
  847. contentFeatures
  848. ));
  849. if (width.HasValue && height.HasValue)
  850. {
  851. writer.WriteAttributeString("resolution", string.Format("{0}x{1}", width.Value, height.Value));
  852. }
  853. writer.WriteString(albumartUrlInfo.Url);
  854. writer.WriteFullEndElement();
  855. }
  856. private ImageDownloadInfo GetImageInfo(BaseItem item)
  857. {
  858. if (item.HasImage(ImageType.Primary))
  859. {
  860. return GetImageInfo(item, ImageType.Primary);
  861. }
  862. if (item.HasImage(ImageType.Thumb))
  863. {
  864. return GetImageInfo(item, ImageType.Thumb);
  865. }
  866. if (item.HasImage(ImageType.Backdrop))
  867. {
  868. if (item is Channel)
  869. {
  870. return GetImageInfo(item, ImageType.Backdrop);
  871. }
  872. }
  873. item = item.GetParents().FirstOrDefault(i => i.HasImage(ImageType.Primary));
  874. if (item != null)
  875. {
  876. if (item.HasImage(ImageType.Primary))
  877. {
  878. return GetImageInfo(item, ImageType.Primary);
  879. }
  880. }
  881. return null;
  882. }
  883. private ImageDownloadInfo GetImageInfo(BaseItem item, ImageType type)
  884. {
  885. var imageInfo = item.GetImageInfo(type, 0);
  886. string tag = null;
  887. try
  888. {
  889. tag = _imageProcessor.GetImageCacheTag(item, type);
  890. }
  891. catch
  892. {
  893. }
  894. int? width = null;
  895. int? height = null;
  896. try
  897. {
  898. var size = _imageProcessor.GetImageSize(imageInfo);
  899. width = Convert.ToInt32(size.Width);
  900. height = Convert.ToInt32(size.Height);
  901. }
  902. catch
  903. {
  904. }
  905. var inputFormat = (Path.GetExtension(imageInfo.Path) ?? string.Empty)
  906. .TrimStart('.')
  907. .Replace("jpeg", "jpg", StringComparison.OrdinalIgnoreCase);
  908. return new ImageDownloadInfo
  909. {
  910. ItemId = item.Id.ToString("N"),
  911. Type = type,
  912. ImageTag = tag,
  913. Width = width,
  914. Height = height,
  915. Format = inputFormat,
  916. ItemImageInfo = imageInfo
  917. };
  918. }
  919. class ImageDownloadInfo
  920. {
  921. internal string ItemId;
  922. internal string ImageTag;
  923. internal ImageType Type;
  924. internal int? Width;
  925. internal int? Height;
  926. internal bool IsDirectStream;
  927. internal string Format;
  928. internal ItemImageInfo ItemImageInfo;
  929. }
  930. class ImageUrlInfo
  931. {
  932. internal string Url;
  933. internal int? Width;
  934. internal int? Height;
  935. }
  936. public static string GetClientId(BaseItem item, StubType? stubType)
  937. {
  938. return GetClientId(item.Id, stubType);
  939. }
  940. public static string GetClientId(Guid idValue, StubType? stubType)
  941. {
  942. var id = idValue.ToString("N");
  943. if (stubType.HasValue)
  944. {
  945. id = stubType.Value.ToString().ToLower() + "_" + id;
  946. }
  947. return id;
  948. }
  949. public static string GetClientId(IHasMediaSources item)
  950. {
  951. var id = item.Id.ToString("N");
  952. return id;
  953. }
  954. private ImageUrlInfo GetImageUrl(ImageDownloadInfo info, int maxWidth, int maxHeight, int playbackPercentage, int unplayedCount, string format)
  955. {
  956. var url = string.Format("{0}/Items/{1}/Images/{2}/0/{3}/{4}/{5}/{6}/{7}/{8}",
  957. _serverAddress,
  958. info.ItemId,
  959. info.Type,
  960. info.ImageTag,
  961. format,
  962. maxWidth.ToString(CultureInfo.InvariantCulture),
  963. maxHeight.ToString(CultureInfo.InvariantCulture),
  964. playbackPercentage.ToString(CultureInfo.InvariantCulture),
  965. unplayedCount.ToString(CultureInfo.InvariantCulture)
  966. );
  967. var width = info.Width;
  968. var height = info.Height;
  969. info.IsDirectStream = false;
  970. if (width.HasValue && height.HasValue)
  971. {
  972. var newSize = DrawingUtils.Resize(new ImageSize
  973. {
  974. Height = height.Value,
  975. Width = width.Value
  976. }, null, null, maxWidth, maxHeight);
  977. width = Convert.ToInt32(newSize.Width);
  978. height = Convert.ToInt32(newSize.Height);
  979. var normalizedFormat = format
  980. .Replace("jpeg", "jpg", StringComparison.OrdinalIgnoreCase);
  981. if (string.Equals(info.Format, normalizedFormat, StringComparison.OrdinalIgnoreCase))
  982. {
  983. info.IsDirectStream = maxWidth >= width.Value && maxHeight >= height.Value;
  984. }
  985. }
  986. return new ImageUrlInfo
  987. {
  988. Url = url,
  989. Width = width,
  990. Height = height
  991. };
  992. }
  993. }
  994. }