ControlHandler.cs 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628
  1. using MediaBrowser.Common.Extensions;
  2. using MediaBrowser.Controller.Channels;
  3. using MediaBrowser.Controller.Configuration;
  4. using MediaBrowser.Controller.Drawing;
  5. using MediaBrowser.Controller.Entities;
  6. using MediaBrowser.Controller.Entities.Movies;
  7. using MediaBrowser.Controller.Entities.TV;
  8. using MediaBrowser.Controller.Library;
  9. using Emby.Dlna.Didl;
  10. using Emby.Dlna.Server;
  11. using Emby.Dlna.Service;
  12. using MediaBrowser.Model.Configuration;
  13. using MediaBrowser.Model.Dlna;
  14. using MediaBrowser.Model.Entities;
  15. using MediaBrowser.Model.Logging;
  16. using MediaBrowser.Model.Querying;
  17. using System;
  18. using System.Collections.Generic;
  19. using System.Globalization;
  20. using System.IO;
  21. using System.Linq;
  22. using System.Text;
  23. using System.Threading;
  24. using System.Threading.Tasks;
  25. using System.Xml;
  26. using MediaBrowser.Controller.MediaEncoding;
  27. using MediaBrowser.Model.Globalization;
  28. using MediaBrowser.Model.Xml;
  29. namespace Emby.Dlna.ContentDirectory
  30. {
  31. public class ControlHandler : BaseControlHandler
  32. {
  33. private readonly ILibraryManager _libraryManager;
  34. private readonly IChannelManager _channelManager;
  35. private readonly IUserDataManager _userDataManager;
  36. private readonly IServerConfigurationManager _config;
  37. private readonly User _user;
  38. private readonly IUserViewManager _userViewManager;
  39. private const string NS_DC = "http://purl.org/dc/elements/1.1/";
  40. private const string NS_DIDL = "urn:schemas-upnp-org:metadata-1-0/DIDL-Lite/";
  41. private const string NS_DLNA = "urn:schemas-dlna-org:metadata-1-0/";
  42. private const string NS_UPNP = "urn:schemas-upnp-org:metadata-1-0/upnp/";
  43. private readonly int _systemUpdateId;
  44. private readonly CultureInfo _usCulture = new CultureInfo("en-US");
  45. private readonly DidlBuilder _didlBuilder;
  46. private readonly DeviceProfile _profile;
  47. public ControlHandler(ILogger logger, ILibraryManager libraryManager, DeviceProfile profile, string serverAddress, string accessToken, IImageProcessor imageProcessor, IUserDataManager userDataManager, User user, int systemUpdateId, IServerConfigurationManager config, ILocalizationManager localization, IChannelManager channelManager, IMediaSourceManager mediaSourceManager, IUserViewManager userViewManager, IMediaEncoder mediaEncoder, IXmlReaderSettingsFactory xmlReaderSettingsFactory)
  48. : base(config, logger, xmlReaderSettingsFactory)
  49. {
  50. _libraryManager = libraryManager;
  51. _userDataManager = userDataManager;
  52. _user = user;
  53. _systemUpdateId = systemUpdateId;
  54. _channelManager = channelManager;
  55. _userViewManager = userViewManager;
  56. _profile = profile;
  57. _config = config;
  58. _didlBuilder = new DidlBuilder(profile, user, imageProcessor, serverAddress, accessToken, userDataManager, localization, mediaSourceManager, Logger, libraryManager, mediaEncoder);
  59. }
  60. protected override IEnumerable<KeyValuePair<string, string>> GetResult(string methodName, IDictionary<string, string> methodParams)
  61. {
  62. var deviceId = "test";
  63. var user = _user;
  64. if (string.Equals(methodName, "GetSearchCapabilities", StringComparison.OrdinalIgnoreCase))
  65. return HandleGetSearchCapabilities();
  66. if (string.Equals(methodName, "GetSortCapabilities", StringComparison.OrdinalIgnoreCase))
  67. return HandleGetSortCapabilities();
  68. if (string.Equals(methodName, "GetSortExtensionCapabilities", StringComparison.OrdinalIgnoreCase))
  69. return HandleGetSortExtensionCapabilities();
  70. if (string.Equals(methodName, "GetSystemUpdateID", StringComparison.OrdinalIgnoreCase))
  71. return HandleGetSystemUpdateID();
  72. if (string.Equals(methodName, "Browse", StringComparison.OrdinalIgnoreCase))
  73. return HandleBrowse(methodParams, user, deviceId).Result;
  74. if (string.Equals(methodName, "X_GetFeatureList", StringComparison.OrdinalIgnoreCase))
  75. return HandleXGetFeatureList();
  76. if (string.Equals(methodName, "GetFeatureList", StringComparison.OrdinalIgnoreCase))
  77. return HandleGetFeatureList();
  78. if (string.Equals(methodName, "X_SetBookmark", StringComparison.OrdinalIgnoreCase))
  79. return HandleXSetBookmark(methodParams, user);
  80. if (string.Equals(methodName, "Search", StringComparison.OrdinalIgnoreCase))
  81. return HandleSearch(methodParams, user, deviceId).Result;
  82. if (string.Equals(methodName, "X_BrowseByLetter", StringComparison.OrdinalIgnoreCase))
  83. return HandleX_BrowseByLetter(methodParams, user, deviceId).Result;
  84. throw new ResourceNotFoundException("Unexpected control request name: " + methodName);
  85. }
  86. private IEnumerable<KeyValuePair<string, string>> HandleXSetBookmark(IDictionary<string, string> sparams, User user)
  87. {
  88. var id = sparams["ObjectID"];
  89. var serverItem = GetItemFromObjectId(id, user);
  90. var item = serverItem.Item;
  91. var newbookmark = int.Parse(sparams["PosSecond"], _usCulture);
  92. var userdata = _userDataManager.GetUserData(user, item);
  93. userdata.PlaybackPositionTicks = TimeSpan.FromSeconds(newbookmark).Ticks;
  94. _userDataManager.SaveUserData(user.Id, item, userdata, UserDataSaveReason.TogglePlayed,
  95. CancellationToken.None);
  96. return new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase);
  97. }
  98. private IEnumerable<KeyValuePair<string, string>> HandleGetSearchCapabilities()
  99. {
  100. return new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase)
  101. {
  102. { "SearchCaps", "res@resolution,res@size,res@duration,dc:title,dc:creator,upnp:actor,upnp:artist,upnp:genre,upnp:album,dc:date,upnp:class,@id,@refID,@protocolInfo,upnp:author,dc:description,pv:avKeywords" }
  103. };
  104. }
  105. private IEnumerable<KeyValuePair<string, string>> HandleGetSortCapabilities()
  106. {
  107. return new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase)
  108. {
  109. { "SortCaps", "res@duration,res@size,res@bitrate,dc:date,dc:title,dc:size,upnp:album,upnp:artist,upnp:albumArtist,upnp:episodeNumber,upnp:genre,upnp:originalTrackNumber,upnp:rating" }
  110. };
  111. }
  112. private IEnumerable<KeyValuePair<string, string>> HandleGetSortExtensionCapabilities()
  113. {
  114. return new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase)
  115. {
  116. { "SortExtensionCaps", "res@duration,res@size,res@bitrate,dc:date,dc:title,dc:size,upnp:album,upnp:artist,upnp:albumArtist,upnp:episodeNumber,upnp:genre,upnp:originalTrackNumber,upnp:rating" }
  117. };
  118. }
  119. private IEnumerable<KeyValuePair<string, string>> HandleGetSystemUpdateID()
  120. {
  121. var headers = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase);
  122. headers.Add("Id", _systemUpdateId.ToString(_usCulture));
  123. return headers;
  124. }
  125. private IEnumerable<KeyValuePair<string, string>> HandleGetFeatureList()
  126. {
  127. return new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase)
  128. {
  129. { "FeatureList", GetFeatureListXml() }
  130. };
  131. }
  132. private IEnumerable<KeyValuePair<string, string>> HandleXGetFeatureList()
  133. {
  134. return new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase)
  135. {
  136. { "FeatureList", GetFeatureListXml() }
  137. };
  138. }
  139. private string GetFeatureListXml()
  140. {
  141. var builder = new StringBuilder();
  142. builder.Append("<?xml version=\"1.0\" encoding=\"UTF-8\"?>");
  143. builder.Append("<Features xmlns=\"urn:schemas-upnp-org:av:avs\" xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xsi:schemaLocation=\"urn:schemas-upnp-org:av:avs http://www.upnp.org/schemas/av/avs.xsd\">");
  144. builder.Append("<Feature name=\"samsung.com_BASICVIEW\" version=\"1\">");
  145. builder.Append("<container id=\"I\" type=\"object.item.imageItem\"/>");
  146. builder.Append("<container id=\"A\" type=\"object.item.audioItem\"/>");
  147. builder.Append("<container id=\"V\" type=\"object.item.videoItem\"/>");
  148. builder.Append("</Feature>");
  149. builder.Append("</Features>");
  150. return builder.ToString();
  151. }
  152. public string GetValueOrDefault(IDictionary<string, string> sparams, string key, string defaultValue)
  153. {
  154. string val;
  155. if (sparams.TryGetValue(key, out val))
  156. {
  157. return val;
  158. }
  159. return defaultValue;
  160. }
  161. private async Task<IEnumerable<KeyValuePair<string, string>>> HandleBrowse(IDictionary<string, string> sparams, User user, string deviceId)
  162. {
  163. var id = sparams["ObjectID"];
  164. var flag = sparams["BrowseFlag"];
  165. var filter = new Filter(GetValueOrDefault(sparams, "Filter", "*"));
  166. var sortCriteria = new SortCriteria(GetValueOrDefault(sparams, "SortCriteria", ""));
  167. var provided = 0;
  168. // Default to null instead of 0
  169. // Upnp inspector sends 0 as requestedCount when it wants everything
  170. int? requestedCount = null;
  171. int? start = 0;
  172. int requestedVal;
  173. if (sparams.ContainsKey("RequestedCount") && int.TryParse(sparams["RequestedCount"], out requestedVal) && requestedVal > 0)
  174. {
  175. requestedCount = requestedVal;
  176. }
  177. int startVal;
  178. if (sparams.ContainsKey("StartingIndex") && int.TryParse(sparams["StartingIndex"], out startVal) && startVal > 0)
  179. {
  180. start = startVal;
  181. }
  182. var settings = new XmlWriterSettings
  183. {
  184. Encoding = Encoding.UTF8,
  185. CloseOutput = false,
  186. OmitXmlDeclaration = true,
  187. ConformanceLevel = ConformanceLevel.Fragment
  188. };
  189. StringWriter builder = new StringWriterWithEncoding(Encoding.UTF8);
  190. int totalCount;
  191. using (XmlWriter writer = XmlWriter.Create(builder, settings))
  192. {
  193. //writer.WriteStartDocument();
  194. writer.WriteStartElement(string.Empty, "DIDL-Lite", NS_DIDL);
  195. writer.WriteAttributeString("xmlns", "dc", null, NS_DC);
  196. writer.WriteAttributeString("xmlns", "dlna", null, NS_DLNA);
  197. writer.WriteAttributeString("xmlns", "upnp", null, NS_UPNP);
  198. //didl.SetAttribute("xmlns:sec", NS_SEC);
  199. DidlBuilder.WriteXmlRootAttributes(_profile, writer);
  200. var serverItem = GetItemFromObjectId(id, user);
  201. var item = serverItem.Item;
  202. if (string.Equals(flag, "BrowseMetadata"))
  203. {
  204. totalCount = 1;
  205. if (item.IsFolder || serverItem.StubType.HasValue)
  206. {
  207. var childrenResult = (await GetUserItems(item, serverItem.StubType, user, sortCriteria, start, requestedCount).ConfigureAwait(false));
  208. _didlBuilder.WriteFolderElement(writer, item, serverItem.StubType, null, childrenResult.TotalRecordCount, filter, id);
  209. }
  210. else
  211. {
  212. _didlBuilder.WriteItemElement(_config.GetDlnaConfiguration(), writer, item, null, null, deviceId, filter);
  213. }
  214. provided++;
  215. }
  216. else
  217. {
  218. var childrenResult = (await GetUserItems(item, serverItem.StubType, user, sortCriteria, start, requestedCount).ConfigureAwait(false));
  219. totalCount = childrenResult.TotalRecordCount;
  220. provided = childrenResult.Items.Length;
  221. foreach (var i in childrenResult.Items)
  222. {
  223. var childItem = i.Item;
  224. var displayStubType = i.StubType;
  225. if (childItem.IsFolder || displayStubType.HasValue)
  226. {
  227. var childCount = (await GetUserItems(childItem, displayStubType, user, sortCriteria, null, 0).ConfigureAwait(false))
  228. .TotalRecordCount;
  229. _didlBuilder.WriteFolderElement(writer, childItem, displayStubType, item, childCount, filter);
  230. }
  231. else
  232. {
  233. _didlBuilder.WriteItemElement(_config.GetDlnaConfiguration(), writer, childItem, item, serverItem.StubType, deviceId, filter);
  234. }
  235. }
  236. }
  237. writer.WriteFullEndElement();
  238. //writer.WriteEndDocument();
  239. }
  240. var resXML = builder.ToString();
  241. return new List<KeyValuePair<string, string>>
  242. {
  243. new KeyValuePair<string,string>("Result", resXML),
  244. new KeyValuePair<string,string>("NumberReturned", provided.ToString(_usCulture)),
  245. new KeyValuePair<string,string>("TotalMatches", totalCount.ToString(_usCulture)),
  246. new KeyValuePair<string,string>("UpdateID", _systemUpdateId.ToString(_usCulture))
  247. };
  248. }
  249. private Task<IEnumerable<KeyValuePair<string, string>>> HandleX_BrowseByLetter(IDictionary<string, string> sparams, User user, string deviceId)
  250. {
  251. // TODO: Implement this method
  252. return HandleSearch(sparams, user, deviceId);
  253. }
  254. private async Task<IEnumerable<KeyValuePair<string, string>>> HandleSearch(IDictionary<string, string> sparams, User user, string deviceId)
  255. {
  256. var searchCriteria = new SearchCriteria(GetValueOrDefault(sparams, "SearchCriteria", ""));
  257. var sortCriteria = new SortCriteria(GetValueOrDefault(sparams, "SortCriteria", ""));
  258. var filter = new Filter(GetValueOrDefault(sparams, "Filter", "*"));
  259. // sort example: dc:title, dc:date
  260. // Default to null instead of 0
  261. // Upnp inspector sends 0 as requestedCount when it wants everything
  262. int? requestedCount = null;
  263. int? start = 0;
  264. int requestedVal;
  265. if (sparams.ContainsKey("RequestedCount") && int.TryParse(sparams["RequestedCount"], out requestedVal) && requestedVal > 0)
  266. {
  267. requestedCount = requestedVal;
  268. }
  269. int startVal;
  270. if (sparams.ContainsKey("StartingIndex") && int.TryParse(sparams["StartingIndex"], out startVal) && startVal > 0)
  271. {
  272. start = startVal;
  273. }
  274. var settings = new XmlWriterSettings
  275. {
  276. Encoding = Encoding.UTF8,
  277. CloseOutput = false,
  278. OmitXmlDeclaration = true,
  279. ConformanceLevel = ConformanceLevel.Fragment
  280. };
  281. StringWriter builder = new StringWriterWithEncoding(Encoding.UTF8);
  282. int totalCount = 0;
  283. int provided = 0;
  284. using (XmlWriter writer = XmlWriter.Create(builder, settings))
  285. {
  286. //writer.WriteStartDocument();
  287. writer.WriteStartElement(string.Empty, "DIDL-Lite", NS_DIDL);
  288. writer.WriteAttributeString("xmlns", "dc", null, NS_DC);
  289. writer.WriteAttributeString("xmlns", "dlna", null, NS_DLNA);
  290. writer.WriteAttributeString("xmlns", "upnp", null, NS_UPNP);
  291. //didl.SetAttribute("xmlns:sec", NS_SEC);
  292. DidlBuilder.WriteXmlRootAttributes(_profile, writer);
  293. var serverItem = GetItemFromObjectId(sparams["ContainerID"], user);
  294. var item = serverItem.Item;
  295. var childrenResult = (await GetChildrenSorted(item, user, searchCriteria, sortCriteria, start, requestedCount).ConfigureAwait(false));
  296. totalCount = childrenResult.TotalRecordCount;
  297. provided = childrenResult.Items.Length;
  298. foreach (var i in childrenResult.Items)
  299. {
  300. if (i.IsFolder)
  301. {
  302. var childCount = (await GetChildrenSorted(i, user, searchCriteria, sortCriteria, null, 0).ConfigureAwait(false))
  303. .TotalRecordCount;
  304. _didlBuilder.WriteFolderElement(writer, i, null, item, childCount, filter);
  305. }
  306. else
  307. {
  308. _didlBuilder.WriteItemElement(_config.GetDlnaConfiguration(), writer, i, item, serverItem.StubType, deviceId, filter);
  309. }
  310. }
  311. writer.WriteFullEndElement();
  312. //writer.WriteEndDocument();
  313. }
  314. var resXML = builder.ToString();
  315. return new List<KeyValuePair<string, string>>
  316. {
  317. new KeyValuePair<string,string>("Result", resXML),
  318. new KeyValuePair<string,string>("NumberReturned", provided.ToString(_usCulture)),
  319. new KeyValuePair<string,string>("TotalMatches", totalCount.ToString(_usCulture)),
  320. new KeyValuePair<string,string>("UpdateID", _systemUpdateId.ToString(_usCulture))
  321. };
  322. }
  323. private Task<QueryResult<BaseItem>> GetChildrenSorted(BaseItem item, User user, SearchCriteria search, SortCriteria sort, int? startIndex, int? limit)
  324. {
  325. var folder = (Folder)item;
  326. var sortOrders = new List<string>();
  327. if (!folder.IsPreSorted)
  328. {
  329. sortOrders.Add(ItemSortBy.SortName);
  330. }
  331. var mediaTypes = new List<string>();
  332. bool? isFolder = null;
  333. if (search.SearchType == SearchType.Audio)
  334. {
  335. mediaTypes.Add(MediaType.Audio);
  336. isFolder = false;
  337. }
  338. else if (search.SearchType == SearchType.Video)
  339. {
  340. mediaTypes.Add(MediaType.Video);
  341. isFolder = false;
  342. }
  343. else if (search.SearchType == SearchType.Image)
  344. {
  345. mediaTypes.Add(MediaType.Photo);
  346. isFolder = false;
  347. }
  348. else if (search.SearchType == SearchType.Playlist)
  349. {
  350. //items = items.OfType<Playlist>();
  351. isFolder = true;
  352. }
  353. else if (search.SearchType == SearchType.MusicAlbum)
  354. {
  355. //items = items.OfType<MusicAlbum>();
  356. isFolder = true;
  357. }
  358. return folder.GetItems(new InternalItemsQuery
  359. {
  360. Limit = limit,
  361. StartIndex = startIndex,
  362. SortBy = sortOrders.ToArray(),
  363. SortOrder = sort.SortOrder,
  364. User = user,
  365. Recursive = true,
  366. IsMissing = false,
  367. ExcludeItemTypes = new[] { typeof(Game).Name, typeof(Book).Name },
  368. IsFolder = isFolder,
  369. MediaTypes = mediaTypes.ToArray()
  370. });
  371. }
  372. private async Task<QueryResult<ServerItem>> GetUserItems(BaseItem item, StubType? stubType, User user, SortCriteria sort, int? startIndex, int? limit)
  373. {
  374. if (stubType.HasValue)
  375. {
  376. if (stubType.Value == StubType.People)
  377. {
  378. var items = _libraryManager.GetPeopleItems(new InternalPeopleQuery
  379. {
  380. ItemId = item.Id
  381. }).ToArray();
  382. var result = new QueryResult<ServerItem>
  383. {
  384. Items = items.Select(i => new ServerItem { Item = i, StubType = StubType.Folder }).ToArray(),
  385. TotalRecordCount = items.Length
  386. };
  387. return ApplyPaging(result, startIndex, limit);
  388. }
  389. var person = item as Person;
  390. if (person != null)
  391. {
  392. return GetItemsFromPerson(person, user, startIndex, limit);
  393. }
  394. return ApplyPaging(new QueryResult<ServerItem>(), startIndex, limit);
  395. }
  396. var folder = (Folder)item;
  397. var sortOrders = new List<string>();
  398. if (!folder.IsPreSorted)
  399. {
  400. sortOrders.Add(ItemSortBy.SortName);
  401. }
  402. var queryResult = await folder.GetItems(new InternalItemsQuery
  403. {
  404. Limit = limit,
  405. StartIndex = startIndex,
  406. SortBy = sortOrders.ToArray(),
  407. SortOrder = sort.SortOrder,
  408. User = user,
  409. IsMissing = false,
  410. PresetViews = new[] { CollectionType.Movies, CollectionType.TvShows, CollectionType.Music },
  411. ExcludeItemTypes = new[] { typeof(Game).Name, typeof(Book).Name },
  412. IsPlaceHolder = false
  413. }).ConfigureAwait(false);
  414. var serverItems = queryResult
  415. .Items
  416. .Select(i => new ServerItem
  417. {
  418. Item = i
  419. })
  420. .ToArray();
  421. return new QueryResult<ServerItem>
  422. {
  423. TotalRecordCount = queryResult.TotalRecordCount,
  424. Items = serverItems
  425. };
  426. }
  427. private QueryResult<ServerItem> GetItemsFromPerson(Person person, User user, int? startIndex, int? limit)
  428. {
  429. var itemsResult = _libraryManager.GetItemsResult(new InternalItemsQuery(user)
  430. {
  431. PersonIds = new[] { person.Id.ToString("N") },
  432. IncludeItemTypes = new[] { typeof(Movie).Name, typeof(Series).Name, typeof(Trailer).Name },
  433. SortBy = new[] { ItemSortBy.SortName },
  434. Limit = limit,
  435. StartIndex = startIndex
  436. });
  437. var serverItems = itemsResult.Items.Select(i => new ServerItem
  438. {
  439. Item = i,
  440. StubType = null
  441. })
  442. .ToArray();
  443. return new QueryResult<ServerItem>
  444. {
  445. TotalRecordCount = itemsResult.TotalRecordCount,
  446. Items = serverItems
  447. };
  448. }
  449. private QueryResult<ServerItem> ApplyPaging(QueryResult<ServerItem> result, int? startIndex, int? limit)
  450. {
  451. result.Items = result.Items.Skip(startIndex ?? 0).Take(limit ?? int.MaxValue).ToArray();
  452. return result;
  453. }
  454. private ServerItem GetItemFromObjectId(string id, User user)
  455. {
  456. return DidlBuilder.IsIdRoot(id)
  457. ? new ServerItem { Item = user.RootFolder }
  458. : ParseItemId(id, user);
  459. }
  460. private ServerItem ParseItemId(string id, User user)
  461. {
  462. Guid itemId;
  463. StubType? stubType = null;
  464. // After using PlayTo, MediaMonkey sends a request to the server trying to get item info
  465. const string paramsSrch = "Params=";
  466. var paramsIndex = id.IndexOf(paramsSrch, StringComparison.OrdinalIgnoreCase);
  467. if (paramsIndex != -1)
  468. {
  469. id = id.Substring(paramsIndex + paramsSrch.Length);
  470. var parts = id.Split(';');
  471. id = parts[23];
  472. }
  473. if (id.StartsWith("folder_", StringComparison.OrdinalIgnoreCase))
  474. {
  475. stubType = StubType.Folder;
  476. id = id.Split(new[] { '_' }, 2)[1];
  477. }
  478. else if (id.StartsWith("people_", StringComparison.OrdinalIgnoreCase))
  479. {
  480. stubType = StubType.People;
  481. id = id.Split(new[] { '_' }, 2)[1];
  482. }
  483. if (Guid.TryParse(id, out itemId))
  484. {
  485. var item = _libraryManager.GetItemById(itemId);
  486. return new ServerItem
  487. {
  488. Item = item,
  489. StubType = stubType
  490. };
  491. }
  492. Logger.Error("Error parsing item Id: {0}. Returning user root folder.", id);
  493. return new ServerItem { Item = user.RootFolder };
  494. }
  495. }
  496. internal class ServerItem
  497. {
  498. public BaseItem Item { get; set; }
  499. public StubType? StubType { get; set; }
  500. }
  501. public enum StubType
  502. {
  503. Folder = 0,
  504. People = 1
  505. }
  506. }