DidlBuilder.cs 44 KB

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