DidlBuilder.cs 45 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260
  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);
  167. streamInfo = new StreamBuilder(_mediaEncoder, GetStreamBuilderLogger(options)).BuildVideoItem(new VideoOptions
  168. {
  169. ItemId = GetClientId(video),
  170. MediaSources = sources.ToArray(sources.Count),
  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.FirstOrDefault(),
  180. streamInfo.TargetAudioCodec.FirstOrDefault(),
  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(_mediaEncoder, 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.FirstOrDefault(),
  297. streamInfo.TargetVideoCodec.FirstOrDefault(),
  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.Latest)
  329. {
  330. return _localization.GetLocalizedString("Latest");
  331. }
  332. if (itemStubType.HasValue && itemStubType.Value == StubType.Playlists)
  333. {
  334. return _localization.GetLocalizedString("Playlists");
  335. }
  336. if (itemStubType.HasValue && itemStubType.Value == StubType.AlbumArtists)
  337. {
  338. return _localization.GetLocalizedString("HeaderAlbumArtists");
  339. }
  340. if (itemStubType.HasValue && itemStubType.Value == StubType.Albums)
  341. {
  342. return _localization.GetLocalizedString("Albums");
  343. }
  344. if (itemStubType.HasValue && itemStubType.Value == StubType.Artists)
  345. {
  346. return _localization.GetLocalizedString("Artists");
  347. }
  348. if (itemStubType.HasValue && itemStubType.Value == StubType.Songs)
  349. {
  350. return _localization.GetLocalizedString("Songs");
  351. }
  352. if (itemStubType.HasValue && itemStubType.Value == StubType.Genres)
  353. {
  354. return _localization.GetLocalizedString("Genres");
  355. }
  356. if (itemStubType.HasValue && itemStubType.Value == StubType.FavoriteAlbums)
  357. {
  358. return _localization.GetLocalizedString("HeaderFavoriteAlbums");
  359. }
  360. if (itemStubType.HasValue && itemStubType.Value == StubType.FavoriteArtists)
  361. {
  362. return _localization.GetLocalizedString("HeaderFavoriteArtists");
  363. }
  364. if (itemStubType.HasValue && itemStubType.Value == StubType.FavoriteSongs)
  365. {
  366. return _localization.GetLocalizedString("HeaderFavoriteSongs");
  367. }
  368. if (itemStubType.HasValue && itemStubType.Value == StubType.ContinueWatching)
  369. {
  370. return _localization.GetLocalizedString("HeaderContinueWatching");
  371. }
  372. if (itemStubType.HasValue && itemStubType.Value == StubType.Movies)
  373. {
  374. return _localization.GetLocalizedString("Movies");
  375. }
  376. if (itemStubType.HasValue && itemStubType.Value == StubType.Collections)
  377. {
  378. return _localization.GetLocalizedString("Collections");
  379. }
  380. if (itemStubType.HasValue && itemStubType.Value == StubType.Favorites)
  381. {
  382. return _localization.GetLocalizedString("Favorites");
  383. }
  384. if (itemStubType.HasValue && itemStubType.Value == StubType.NextUp)
  385. {
  386. return _localization.GetLocalizedString("HeaderNextUp");
  387. }
  388. if (itemStubType.HasValue && itemStubType.Value == StubType.FavoriteSeries)
  389. {
  390. return _localization.GetLocalizedString("HeaderFavoriteShows");
  391. }
  392. if (itemStubType.HasValue && itemStubType.Value == StubType.FavoriteEpisodes)
  393. {
  394. return _localization.GetLocalizedString("HeaderFavoriteEpisodes");
  395. }
  396. if (itemStubType.HasValue && itemStubType.Value == StubType.Series)
  397. {
  398. return _localization.GetLocalizedString("Shows");
  399. }
  400. var episode = item as Episode;
  401. var season = context as Season;
  402. if (episode != null && season != null)
  403. {
  404. // This is a special embedded within a season
  405. if (item.ParentIndexNumber.HasValue && item.ParentIndexNumber.Value == 0)
  406. {
  407. if (season.IndexNumber.HasValue && season.IndexNumber.Value != 0)
  408. {
  409. return string.Format(_localization.GetLocalizedString("ValueSpecialEpisodeName"), item.Name);
  410. }
  411. }
  412. if (item.IndexNumber.HasValue)
  413. {
  414. var number = item.IndexNumber.Value.ToString("00", CultureInfo.InvariantCulture);
  415. if (episode.IndexNumberEnd.HasValue)
  416. {
  417. number += "-" + episode.IndexNumberEnd.Value.ToString("00", CultureInfo.InvariantCulture);
  418. }
  419. return number + " - " + item.Name;
  420. }
  421. }
  422. return item.Name;
  423. }
  424. private void AddAudioResource(DlnaOptions options, XmlWriter writer, IHasMediaSources audio, string deviceId, Filter filter, StreamInfo streamInfo = null)
  425. {
  426. writer.WriteStartElement(string.Empty, "res", NS_DIDL);
  427. if (streamInfo == null)
  428. {
  429. var sources = _mediaSourceManager.GetStaticMediaSources(audio, true, _user);
  430. streamInfo = new StreamBuilder(_mediaEncoder, GetStreamBuilderLogger(options)).BuildAudioItem(new AudioOptions
  431. {
  432. ItemId = GetClientId(audio),
  433. MediaSources = sources.ToArray(sources.Count),
  434. Profile = _profile,
  435. DeviceId = deviceId
  436. });
  437. }
  438. var url = streamInfo.ToDlnaUrl(_serverAddress, _accessToken);
  439. var mediaSource = streamInfo.MediaSource;
  440. if (mediaSource.RunTimeTicks.HasValue)
  441. {
  442. writer.WriteAttributeString("duration", TimeSpan.FromTicks(mediaSource.RunTimeTicks.Value).ToString("c", _usCulture));
  443. }
  444. if (filter.Contains("res@size"))
  445. {
  446. if (streamInfo.IsDirectStream || streamInfo.EstimateContentLength)
  447. {
  448. var size = streamInfo.TargetSize;
  449. if (size.HasValue)
  450. {
  451. writer.WriteAttributeString("size", size.Value.ToString(_usCulture));
  452. }
  453. }
  454. }
  455. var targetAudioBitrate = streamInfo.TargetAudioBitrate;
  456. var targetSampleRate = streamInfo.TargetAudioSampleRate;
  457. var targetChannels = streamInfo.TargetAudioChannels;
  458. var targetAudioBitDepth = streamInfo.TargetAudioBitDepth;
  459. if (targetChannels.HasValue)
  460. {
  461. writer.WriteAttributeString("nrAudioChannels", targetChannels.Value.ToString(_usCulture));
  462. }
  463. if (targetSampleRate.HasValue)
  464. {
  465. writer.WriteAttributeString("sampleFrequency", targetSampleRate.Value.ToString(_usCulture));
  466. }
  467. if (targetAudioBitrate.HasValue)
  468. {
  469. writer.WriteAttributeString("bitrate", targetAudioBitrate.Value.ToString(_usCulture));
  470. }
  471. var mediaProfile = _profile.GetAudioMediaProfile(streamInfo.Container,
  472. streamInfo.TargetAudioCodec.FirstOrDefault(),
  473. targetChannels,
  474. targetAudioBitrate,
  475. targetSampleRate,
  476. targetAudioBitDepth);
  477. var filename = url.Substring(0, url.IndexOf('?'));
  478. var mimeType = mediaProfile == null || string.IsNullOrEmpty(mediaProfile.MimeType)
  479. ? GetMimeType(filename)
  480. : mediaProfile.MimeType;
  481. var contentFeatures = new ContentFeatureBuilder(_profile).BuildAudioHeader(streamInfo.Container,
  482. streamInfo.TargetAudioCodec.FirstOrDefault(),
  483. targetAudioBitrate,
  484. targetSampleRate,
  485. targetChannels,
  486. targetAudioBitDepth,
  487. streamInfo.IsDirectStream,
  488. streamInfo.RunTimeTicks,
  489. streamInfo.TranscodeSeekInfo);
  490. writer.WriteAttributeString("protocolInfo", String.Format(
  491. "http-get:*:{0}:{1}",
  492. mimeType,
  493. contentFeatures
  494. ));
  495. writer.WriteString(url);
  496. writer.WriteFullEndElement();
  497. }
  498. public static bool IsIdRoot(string id)
  499. {
  500. if (string.IsNullOrWhiteSpace(id) ||
  501. string.Equals(id, "0", StringComparison.OrdinalIgnoreCase)
  502. // Samsung sometimes uses 1 as root
  503. || string.Equals(id, "1", StringComparison.OrdinalIgnoreCase))
  504. {
  505. return true;
  506. }
  507. return false;
  508. }
  509. public void WriteFolderElement(XmlWriter writer, BaseItem folder, StubType? stubType, BaseItem context, int childCount, Filter filter, string requestedId = null)
  510. {
  511. writer.WriteStartElement(string.Empty, "container", NS_DIDL);
  512. writer.WriteAttributeString("restricted", "0");
  513. writer.WriteAttributeString("searchable", "1");
  514. writer.WriteAttributeString("childCount", childCount.ToString(_usCulture));
  515. var clientId = GetClientId(folder, stubType);
  516. if (string.Equals(requestedId, "0"))
  517. {
  518. writer.WriteAttributeString("id", "0");
  519. writer.WriteAttributeString("parentID", "-1");
  520. }
  521. else
  522. {
  523. writer.WriteAttributeString("id", clientId);
  524. if (context != null)
  525. {
  526. writer.WriteAttributeString("parentID", GetClientId(context, null));
  527. }
  528. else
  529. {
  530. var parent = folder.DisplayParentId;
  531. if (!parent.HasValue)
  532. {
  533. writer.WriteAttributeString("parentID", "0");
  534. }
  535. else
  536. {
  537. writer.WriteAttributeString("parentID", GetClientId(parent.Value, null));
  538. }
  539. }
  540. }
  541. AddGeneralProperties(folder, stubType, context, writer, filter);
  542. AddCover(folder, context, stubType, writer);
  543. writer.WriteFullEndElement();
  544. }
  545. private void AddSamsungBookmarkInfo(BaseItem item, User user, XmlWriter writer)
  546. {
  547. if (!item.SupportsPositionTicksResume || item is Folder)
  548. {
  549. return;
  550. }
  551. MediaBrowser.Model.Dlna.XmlAttribute secAttribute = null;
  552. foreach (var attribute in _profile.XmlRootAttributes)
  553. {
  554. if (string.Equals(attribute.Name, "xmlns:sec", StringComparison.OrdinalIgnoreCase))
  555. {
  556. secAttribute = attribute;
  557. break;
  558. }
  559. }
  560. // Not a samsung device
  561. if (secAttribute == null)
  562. {
  563. return;
  564. }
  565. var userdata = _userDataManager.GetUserData(user.Id, item);
  566. if (userdata.PlaybackPositionTicks > 0)
  567. {
  568. var elementValue = string.Format("BM={0}", Convert.ToInt32(TimeSpan.FromTicks(userdata.PlaybackPositionTicks).TotalSeconds).ToString(_usCulture));
  569. AddValue(writer, "sec", "dcmInfo", elementValue, secAttribute.Value);
  570. }
  571. }
  572. /// <summary>
  573. /// Adds fields used by both items and folders
  574. /// </summary>
  575. private void AddCommonFields(BaseItem item, StubType? itemStubType, BaseItem context, XmlWriter writer, Filter filter)
  576. {
  577. // Don't filter on dc:title because not all devices will include it in the filter
  578. // MediaMonkey for example won't display content without a title
  579. //if (filter.Contains("dc:title"))
  580. {
  581. AddValue(writer, "dc", "title", GetDisplayName(item, itemStubType, context), NS_DC);
  582. }
  583. WriteObjectClass(writer, item, itemStubType);
  584. if (filter.Contains("dc:date"))
  585. {
  586. if (item.PremiereDate.HasValue)
  587. {
  588. AddValue(writer, "dc", "date", item.PremiereDate.Value.ToString("o"), NS_DC);
  589. }
  590. }
  591. if (filter.Contains("upnp:genre"))
  592. {
  593. foreach (var genre in item.Genres)
  594. {
  595. AddValue(writer, "upnp", "genre", genre, NS_UPNP);
  596. }
  597. }
  598. foreach (var studio in item.Studios)
  599. {
  600. AddValue(writer, "upnp", "publisher", studio, NS_UPNP);
  601. }
  602. if (filter.Contains("dc:description"))
  603. {
  604. var desc = item.Overview;
  605. if (!string.IsNullOrWhiteSpace(desc))
  606. {
  607. AddValue(writer, "dc", "description", desc, NS_DC);
  608. }
  609. }
  610. if (filter.Contains("upnp:longDescription"))
  611. {
  612. if (!string.IsNullOrWhiteSpace(item.Overview))
  613. {
  614. AddValue(writer, "upnp", "longDescription", item.Overview, NS_UPNP);
  615. }
  616. }
  617. if (!string.IsNullOrEmpty(item.OfficialRating))
  618. {
  619. if (filter.Contains("dc:rating"))
  620. {
  621. AddValue(writer, "dc", "rating", item.OfficialRating, NS_DC);
  622. }
  623. if (filter.Contains("upnp:rating"))
  624. {
  625. AddValue(writer, "upnp", "rating", item.OfficialRating, NS_UPNP);
  626. }
  627. }
  628. AddPeople(item, writer);
  629. }
  630. private void WriteObjectClass(XmlWriter writer, BaseItem item, StubType? stubType)
  631. {
  632. // More types here
  633. // http://oss.linn.co.uk/repos/Public/LibUpnpCil/DidlLite/UpnpAv/Test/TestDidlLite.cs
  634. writer.WriteStartElement("upnp", "class", NS_UPNP);
  635. if (item.IsDisplayedAsFolder || stubType.HasValue)
  636. {
  637. string classType = null;
  638. if (!_profile.RequiresPlainFolders)
  639. {
  640. if (item is MusicAlbum)
  641. {
  642. classType = "object.container.album.musicAlbum";
  643. }
  644. else if (item is MusicArtist)
  645. {
  646. classType = "object.container.person.musicArtist";
  647. }
  648. else if (item is Series || item is Season || item is BoxSet || item is Video)
  649. {
  650. classType = "object.container.album.videoAlbum";
  651. }
  652. else if (item is Playlist)
  653. {
  654. classType = "object.container.playlistContainer";
  655. }
  656. else if (item is PhotoAlbum)
  657. {
  658. classType = "object.container.album.photoAlbum";
  659. }
  660. }
  661. writer.WriteString(classType ?? "object.container.storageFolder");
  662. }
  663. else if (string.Equals(item.MediaType, MediaType.Audio, StringComparison.OrdinalIgnoreCase))
  664. {
  665. writer.WriteString("object.item.audioItem.musicTrack");
  666. }
  667. else if (string.Equals(item.MediaType, MediaType.Photo, StringComparison.OrdinalIgnoreCase))
  668. {
  669. writer.WriteString("object.item.imageItem.photo");
  670. }
  671. else if (string.Equals(item.MediaType, MediaType.Video, StringComparison.OrdinalIgnoreCase))
  672. {
  673. if (!_profile.RequiresPlainVideoItems && item is Movie)
  674. {
  675. writer.WriteString("object.item.videoItem.movie");
  676. }
  677. else if (!_profile.RequiresPlainVideoItems && item is MusicVideo)
  678. {
  679. writer.WriteString("object.item.videoItem.musicVideoClip");
  680. }
  681. else
  682. {
  683. writer.WriteString("object.item.videoItem");
  684. }
  685. }
  686. else if (item is MusicGenre)
  687. {
  688. writer.WriteString(_profile.RequiresPlainFolders ? "object.container.storageFolder" : "object.container.genre.musicGenre");
  689. }
  690. else if (item is Genre || item is GameGenre)
  691. {
  692. writer.WriteString(_profile.RequiresPlainFolders ? "object.container.storageFolder" : "object.container.genre");
  693. }
  694. else
  695. {
  696. writer.WriteString("object.item");
  697. }
  698. writer.WriteFullEndElement();
  699. }
  700. private void AddPeople(BaseItem item, XmlWriter writer)
  701. {
  702. var types = new[]
  703. {
  704. PersonType.Director,
  705. PersonType.Writer,
  706. PersonType.Producer,
  707. PersonType.Composer,
  708. "Creator"
  709. };
  710. var people = _libraryManager.GetPeople(item);
  711. var index = 0;
  712. // Seeing some LG models locking up due content with large lists of people
  713. // The actual issue might just be due to processing a more metadata than it can handle
  714. var limit = 6;
  715. foreach (var actor in people)
  716. {
  717. var type = types.FirstOrDefault(i => string.Equals(i, actor.Type, StringComparison.OrdinalIgnoreCase) || string.Equals(i, actor.Role, StringComparison.OrdinalIgnoreCase))
  718. ?? PersonType.Actor;
  719. AddValue(writer, "upnp", type.ToLower(), actor.Name, NS_UPNP);
  720. index++;
  721. if (index >= limit)
  722. {
  723. break;
  724. }
  725. }
  726. }
  727. private void AddGeneralProperties(BaseItem item, StubType? itemStubType, BaseItem context, XmlWriter writer, Filter filter)
  728. {
  729. AddCommonFields(item, itemStubType, context, writer, filter);
  730. var hasArtists = item as IHasArtist;
  731. var hasAlbumArtists = item as IHasAlbumArtist;
  732. if (hasArtists != null)
  733. {
  734. foreach (var artist in hasArtists.Artists)
  735. {
  736. AddValue(writer, "upnp", "artist", artist, NS_UPNP);
  737. AddValue(writer, "dc", "creator", artist, NS_DC);
  738. // If it doesn't support album artists (musicvideo), then tag as both
  739. if (hasAlbumArtists == null)
  740. {
  741. AddAlbumArtist(writer, artist);
  742. }
  743. }
  744. }
  745. if (hasAlbumArtists != null)
  746. {
  747. foreach (var albumArtist in hasAlbumArtists.AlbumArtists)
  748. {
  749. AddAlbumArtist(writer, albumArtist);
  750. }
  751. }
  752. if (!string.IsNullOrWhiteSpace(item.Album))
  753. {
  754. AddValue(writer, "upnp", "album", item.Album, NS_UPNP);
  755. }
  756. if (item.IndexNumber.HasValue)
  757. {
  758. AddValue(writer, "upnp", "originalTrackNumber", item.IndexNumber.Value.ToString(_usCulture), NS_UPNP);
  759. if (item is Episode)
  760. {
  761. AddValue(writer, "upnp", "episodeNumber", item.IndexNumber.Value.ToString(_usCulture), NS_UPNP);
  762. }
  763. }
  764. }
  765. private void AddAlbumArtist(XmlWriter writer, string name)
  766. {
  767. try
  768. {
  769. writer.WriteStartElement("upnp", "artist", NS_UPNP);
  770. writer.WriteAttributeString("role", "AlbumArtist");
  771. writer.WriteString(name);
  772. writer.WriteFullEndElement();
  773. }
  774. catch (XmlException)
  775. {
  776. //_logger.Error("Error adding xml value: " + value);
  777. }
  778. }
  779. private void AddValue(XmlWriter writer, string prefix, string name, string value, string namespaceUri)
  780. {
  781. try
  782. {
  783. writer.WriteElementString(prefix, name, namespaceUri, value);
  784. }
  785. catch (XmlException)
  786. {
  787. //_logger.Error("Error adding xml value: " + value);
  788. }
  789. }
  790. private void AddCover(BaseItem item, BaseItem context, StubType? stubType, XmlWriter writer)
  791. {
  792. ImageDownloadInfo imageInfo = null;
  793. if (context is UserView)
  794. {
  795. var episode = item as Episode;
  796. if (episode != null)
  797. {
  798. var parent = episode.Series;
  799. if (parent != null)
  800. {
  801. imageInfo = GetImageInfo(parent);
  802. }
  803. }
  804. }
  805. // Finally, just use the image from the item
  806. if (imageInfo == null)
  807. {
  808. imageInfo = GetImageInfo(item);
  809. }
  810. if (imageInfo == null)
  811. {
  812. return;
  813. }
  814. var playbackPercentage = 0;
  815. var unplayedCount = 0;
  816. if (item is Video)
  817. {
  818. var userData = _userDataManager.GetUserDataDto(item, _user);
  819. playbackPercentage = Convert.ToInt32(userData.PlayedPercentage ?? 0);
  820. if (playbackPercentage >= 100 || userData.Played)
  821. {
  822. playbackPercentage = 100;
  823. }
  824. }
  825. else if (item is Series || item is Season || item is BoxSet)
  826. {
  827. var userData = _userDataManager.GetUserDataDto(item, _user);
  828. if (userData.Played)
  829. {
  830. playbackPercentage = 100;
  831. }
  832. else
  833. {
  834. unplayedCount = userData.UnplayedItemCount ?? 0;
  835. }
  836. }
  837. var albumartUrlInfo = GetImageUrl(imageInfo, _profile.MaxAlbumArtWidth, _profile.MaxAlbumArtHeight, playbackPercentage, unplayedCount, "jpg");
  838. writer.WriteStartElement("upnp", "albumArtURI", NS_UPNP);
  839. writer.WriteAttributeString("dlna", "profileID", NS_DLNA, _profile.AlbumArtPn);
  840. writer.WriteString(albumartUrlInfo.Url);
  841. writer.WriteFullEndElement();
  842. // TOOD: Remove these default values
  843. var iconUrlInfo = GetImageUrl(imageInfo, _profile.MaxIconWidth ?? 48, _profile.MaxIconHeight ?? 48, playbackPercentage, unplayedCount, "jpg");
  844. writer.WriteElementString("upnp", "icon", NS_UPNP, iconUrlInfo.Url);
  845. if (!_profile.EnableAlbumArtInDidl)
  846. {
  847. if (string.Equals(item.MediaType, MediaType.Audio, StringComparison.OrdinalIgnoreCase)
  848. || string.Equals(item.MediaType, MediaType.Video, StringComparison.OrdinalIgnoreCase))
  849. {
  850. if (!stubType.HasValue)
  851. {
  852. return;
  853. }
  854. }
  855. }
  856. AddImageResElement(item, writer, 160, 160, playbackPercentage, unplayedCount, "jpg", "JPEG_TN");
  857. if (!_profile.EnableSingleAlbumArtLimit)
  858. {
  859. AddImageResElement(item, writer, 4096, 4096, playbackPercentage, unplayedCount, "jpg", "JPEG_LRG");
  860. AddImageResElement(item, writer, 1024, 768, playbackPercentage, unplayedCount, "jpg", "JPEG_MED");
  861. AddImageResElement(item, writer, 640, 480, playbackPercentage, unplayedCount, "jpg", "JPEG_SM");
  862. AddImageResElement(item, writer, 4096, 4096, playbackPercentage, unplayedCount, "png", "PNG_LRG");
  863. AddImageResElement(item, writer, 160, 160, playbackPercentage, unplayedCount, "png", "PNG_TN");
  864. }
  865. }
  866. private void AddEmbeddedImageAsCover(string name, XmlWriter writer)
  867. {
  868. writer.WriteStartElement("upnp", "albumArtURI", NS_UPNP);
  869. writer.WriteAttributeString("dlna", "profileID", NS_DLNA, _profile.AlbumArtPn);
  870. writer.WriteString(_serverAddress + "/Dlna/icons/people480.jpg");
  871. writer.WriteFullEndElement();
  872. writer.WriteElementString("upnp", "icon", NS_UPNP, _serverAddress + "/Dlna/icons/people48.jpg");
  873. }
  874. private void AddImageResElement(BaseItem item,
  875. XmlWriter writer,
  876. int maxWidth,
  877. int maxHeight,
  878. int playbackPercentage,
  879. int unplayedCount,
  880. string format,
  881. string org_Pn)
  882. {
  883. var imageInfo = GetImageInfo(item);
  884. if (imageInfo == null)
  885. {
  886. return;
  887. }
  888. var albumartUrlInfo = GetImageUrl(imageInfo, maxWidth, maxHeight, playbackPercentage, unplayedCount, format);
  889. writer.WriteStartElement(string.Empty, "res", NS_DIDL);
  890. // Images must have a reported size or many clients (Bubble upnp), will only use the first thumbnail
  891. // rather than using a larger one when available
  892. var width = albumartUrlInfo.Width ?? maxWidth;
  893. var height = albumartUrlInfo.Height ?? maxHeight;
  894. var contentFeatures = new ContentFeatureBuilder(_profile)
  895. .BuildImageHeader(format, width, height, imageInfo.IsDirectStream, org_Pn);
  896. writer.WriteAttributeString("protocolInfo", String.Format(
  897. "http-get:*:{0}:{1}",
  898. GetMimeType("file." + format),
  899. contentFeatures
  900. ));
  901. writer.WriteAttributeString("resolution", string.Format("{0}x{1}", width, height));
  902. writer.WriteString(albumartUrlInfo.Url);
  903. writer.WriteFullEndElement();
  904. }
  905. private ImageDownloadInfo GetImageInfo(BaseItem item)
  906. {
  907. if (item.HasImage(ImageType.Primary))
  908. {
  909. return GetImageInfo(item, ImageType.Primary);
  910. }
  911. if (item.HasImage(ImageType.Thumb))
  912. {
  913. return GetImageInfo(item, ImageType.Thumb);
  914. }
  915. if (item.HasImage(ImageType.Backdrop))
  916. {
  917. if (item is Channel)
  918. {
  919. return GetImageInfo(item, ImageType.Backdrop);
  920. }
  921. }
  922. item = item.GetParents().FirstOrDefault(i => i.HasImage(ImageType.Primary));
  923. if (item != null)
  924. {
  925. if (item.HasImage(ImageType.Primary))
  926. {
  927. return GetImageInfo(item, ImageType.Primary);
  928. }
  929. }
  930. return null;
  931. }
  932. private ImageDownloadInfo GetImageInfo(BaseItem item, ImageType type)
  933. {
  934. var imageInfo = item.GetImageInfo(type, 0);
  935. string tag = null;
  936. try
  937. {
  938. tag = _imageProcessor.GetImageCacheTag(item, type);
  939. }
  940. catch
  941. {
  942. }
  943. int? width = imageInfo.Width;
  944. int? height = imageInfo.Height;
  945. if (width == 0 || height == 0)
  946. {
  947. //_imageProcessor.GetImageSize(item, imageInfo);
  948. width = null;
  949. height = null;
  950. }
  951. else if (width == -1 || height == -1)
  952. {
  953. width = null;
  954. height = null;
  955. }
  956. //try
  957. //{
  958. // var size = _imageProcessor.GetImageSize(imageInfo);
  959. // width = Convert.ToInt32(size.Width);
  960. // height = Convert.ToInt32(size.Height);
  961. //}
  962. //catch
  963. //{
  964. //}
  965. var inputFormat = (Path.GetExtension(imageInfo.Path) ?? string.Empty)
  966. .TrimStart('.')
  967. .Replace("jpeg", "jpg", StringComparison.OrdinalIgnoreCase);
  968. return new ImageDownloadInfo
  969. {
  970. ItemId = item.Id.ToString("N"),
  971. Type = type,
  972. ImageTag = tag,
  973. Width = width,
  974. Height = height,
  975. Format = inputFormat,
  976. ItemImageInfo = imageInfo
  977. };
  978. }
  979. class ImageDownloadInfo
  980. {
  981. internal string ItemId;
  982. internal string ImageTag;
  983. internal ImageType Type;
  984. internal int? Width;
  985. internal int? Height;
  986. internal bool IsDirectStream;
  987. internal string Format;
  988. internal ItemImageInfo ItemImageInfo;
  989. }
  990. class ImageUrlInfo
  991. {
  992. internal string Url;
  993. internal int? Width;
  994. internal int? Height;
  995. }
  996. public static string GetClientId(BaseItem item, StubType? stubType)
  997. {
  998. return GetClientId(item.Id, stubType);
  999. }
  1000. public static string GetClientId(Guid idValue, StubType? stubType)
  1001. {
  1002. var id = idValue.ToString("N");
  1003. if (stubType.HasValue)
  1004. {
  1005. id = stubType.Value.ToString().ToLower() + "_" + id;
  1006. }
  1007. return id;
  1008. }
  1009. public static string GetClientId(IHasMediaSources item)
  1010. {
  1011. var id = item.Id.ToString("N");
  1012. return id;
  1013. }
  1014. private ImageUrlInfo GetImageUrl(ImageDownloadInfo info, int maxWidth, int maxHeight, int playbackPercentage, int unplayedCount, string format)
  1015. {
  1016. var url = string.Format("{0}/Items/{1}/Images/{2}/0/{3}/{4}/{5}/{6}/{7}/{8}",
  1017. _serverAddress,
  1018. info.ItemId,
  1019. info.Type,
  1020. info.ImageTag,
  1021. format,
  1022. maxWidth.ToString(CultureInfo.InvariantCulture),
  1023. maxHeight.ToString(CultureInfo.InvariantCulture),
  1024. playbackPercentage.ToString(CultureInfo.InvariantCulture),
  1025. unplayedCount.ToString(CultureInfo.InvariantCulture)
  1026. );
  1027. var width = info.Width;
  1028. var height = info.Height;
  1029. info.IsDirectStream = false;
  1030. if (width.HasValue && height.HasValue)
  1031. {
  1032. var newSize = DrawingUtils.Resize(new ImageSize
  1033. {
  1034. Height = height.Value,
  1035. Width = width.Value
  1036. }, null, null, maxWidth, maxHeight);
  1037. width = Convert.ToInt32(newSize.Width);
  1038. height = Convert.ToInt32(newSize.Height);
  1039. var normalizedFormat = format
  1040. .Replace("jpeg", "jpg", StringComparison.OrdinalIgnoreCase);
  1041. if (string.Equals(info.Format, normalizedFormat, StringComparison.OrdinalIgnoreCase))
  1042. {
  1043. info.IsDirectStream = maxWidth >= width.Value && maxHeight >= height.Value;
  1044. }
  1045. }
  1046. return new ImageUrlInfo
  1047. {
  1048. Url = url,
  1049. Width = width,
  1050. Height = height
  1051. };
  1052. }
  1053. }
  1054. }