UserLibraryService.cs 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547
  1. using MediaBrowser.Common.Net;
  2. using MediaBrowser.Controller;
  3. using MediaBrowser.Controller.Entities;
  4. using MediaBrowser.Controller.Entities.Movies;
  5. using MediaBrowser.Controller.Library;
  6. using MediaBrowser.Model.Dto;
  7. using MediaBrowser.Model.Entities;
  8. using MediaBrowser.Model.Serialization;
  9. using ServiceStack.ServiceHost;
  10. using System;
  11. using System.Collections.Generic;
  12. using System.IO;
  13. using System.Linq;
  14. using System.Threading.Tasks;
  15. using ServiceStack.Text.Controller;
  16. namespace MediaBrowser.Api.UserLibrary
  17. {
  18. /// <summary>
  19. /// Class GetItem
  20. /// </summary>
  21. [Route("/Users/{UserId}/Items/{Id}", "GET")]
  22. [Route("/Users/{UserId}/Items/Root", "GET")]
  23. public class GetItem : IReturn<BaseItemDto>
  24. {
  25. /// <summary>
  26. /// Gets or sets the user id.
  27. /// </summary>
  28. /// <value>The user id.</value>
  29. public Guid UserId { get; set; }
  30. /// <summary>
  31. /// Gets or sets the id.
  32. /// </summary>
  33. /// <value>The id.</value>
  34. public string Id { get; set; }
  35. }
  36. /// <summary>
  37. /// Class GetIntros
  38. /// </summary>
  39. [Route("/Users/{UserId}/Items/{Id}/Intros", "GET")]
  40. [ServiceStack.ServiceHost.Api(("Gets intros to play before the main media item plays"))]
  41. public class GetIntros : IReturn<List<string>>
  42. {
  43. /// <summary>
  44. /// Gets or sets the user id.
  45. /// </summary>
  46. /// <value>The user id.</value>
  47. public Guid UserId { get; set; }
  48. /// <summary>
  49. /// Gets or sets the item id.
  50. /// </summary>
  51. /// <value>The item id.</value>
  52. public string Id { get; set; }
  53. }
  54. /// <summary>
  55. /// Class UpdateDisplayPreferences
  56. /// </summary>
  57. [Route("/Users/{UserId}/Items/{Id}/DisplayPreferences", "GET")]
  58. [ServiceStack.ServiceHost.Api(("Updates a user's display preferences for an item"))]
  59. public class UpdateDisplayPreferences : IReturnVoid, IRequiresRequestStream
  60. {
  61. /// <summary>
  62. /// Gets or sets the user id.
  63. /// </summary>
  64. /// <value>The user id.</value>
  65. public Guid UserId { get; set; }
  66. /// <summary>
  67. /// Gets or sets the id.
  68. /// </summary>
  69. /// <value>The id.</value>
  70. public string Id { get; set; }
  71. /// <summary>
  72. /// The raw Http Request Input Stream
  73. /// </summary>
  74. /// <value>The request stream.</value>
  75. public Stream RequestStream { get; set; }
  76. }
  77. /// <summary>
  78. /// Class GetVirtualFolders
  79. /// </summary>
  80. [Route("/Users/{UserId}/VirtualFolders", "GET")]
  81. public class GetVirtualFolders : IReturn<List<VirtualFolderInfo>>
  82. {
  83. /// <summary>
  84. /// Gets or sets the user id.
  85. /// </summary>
  86. /// <value>The user id.</value>
  87. public Guid UserId { get; set; }
  88. }
  89. /// <summary>
  90. /// Class MarkFavoriteItem
  91. /// </summary>
  92. [Route("/Users/{UserId}/FavoriteItems/{Id}", "POST")]
  93. public class MarkFavoriteItem : IReturnVoid
  94. {
  95. /// <summary>
  96. /// Gets or sets the user id.
  97. /// </summary>
  98. /// <value>The user id.</value>
  99. public Guid UserId { get; set; }
  100. /// <summary>
  101. /// Gets or sets the id.
  102. /// </summary>
  103. /// <value>The id.</value>
  104. public string Id { get; set; }
  105. }
  106. /// <summary>
  107. /// Class UnmarkFavoriteItem
  108. /// </summary>
  109. [Route("/Users/{UserId}/FavoriteItems/{Id}", "DELETE")]
  110. public class UnmarkFavoriteItem : IReturnVoid
  111. {
  112. /// <summary>
  113. /// Gets or sets the user id.
  114. /// </summary>
  115. /// <value>The user id.</value>
  116. public Guid UserId { get; set; }
  117. /// <summary>
  118. /// Gets or sets the id.
  119. /// </summary>
  120. /// <value>The id.</value>
  121. public string Id { get; set; }
  122. }
  123. /// <summary>
  124. /// Class ClearUserItemRating
  125. /// </summary>
  126. [Route("/Users/{UserId}/Items/{Id}/Rating", "DELETE")]
  127. public class DeleteUserItemRating : IReturnVoid
  128. {
  129. /// <summary>
  130. /// Gets or sets the user id.
  131. /// </summary>
  132. /// <value>The user id.</value>
  133. public Guid UserId { get; set; }
  134. /// <summary>
  135. /// Gets or sets the id.
  136. /// </summary>
  137. /// <value>The id.</value>
  138. public string Id { get; set; }
  139. }
  140. /// <summary>
  141. /// Class UpdateUserItemRating
  142. /// </summary>
  143. [Route("/Users/{UserId}/Items/{Id}/Rating", "POST")]
  144. public class UpdateUserItemRating : IReturnVoid
  145. {
  146. /// <summary>
  147. /// Gets or sets the user id.
  148. /// </summary>
  149. /// <value>The user id.</value>
  150. public Guid UserId { get; set; }
  151. /// <summary>
  152. /// Gets or sets the id.
  153. /// </summary>
  154. /// <value>The id.</value>
  155. public string Id { get; set; }
  156. /// <summary>
  157. /// Gets or sets a value indicating whether this <see cref="UpdateUserItemRating" /> is likes.
  158. /// </summary>
  159. /// <value><c>true</c> if likes; otherwise, <c>false</c>.</value>
  160. public bool Likes { get; set; }
  161. }
  162. /// <summary>
  163. /// Class MarkPlayedItem
  164. /// </summary>
  165. [Route("/Users/{UserId}/PlayedItems/{Id}", "POST")]
  166. public class MarkPlayedItem : IReturnVoid
  167. {
  168. /// <summary>
  169. /// Gets or sets the user id.
  170. /// </summary>
  171. /// <value>The user id.</value>
  172. public Guid UserId { get; set; }
  173. /// <summary>
  174. /// Gets or sets the id.
  175. /// </summary>
  176. /// <value>The id.</value>
  177. public string Id { get; set; }
  178. }
  179. /// <summary>
  180. /// Class MarkUnplayedItem
  181. /// </summary>
  182. [Route("/Users/{UserId}/PlayedItems/{Id}", "DELETE")]
  183. public class MarkUnplayedItem : IReturnVoid
  184. {
  185. /// <summary>
  186. /// Gets or sets the user id.
  187. /// </summary>
  188. /// <value>The user id.</value>
  189. public Guid UserId { get; set; }
  190. /// <summary>
  191. /// Gets or sets the id.
  192. /// </summary>
  193. /// <value>The id.</value>
  194. public string Id { get; set; }
  195. }
  196. /// <summary>
  197. /// Class GetLocalTrailers
  198. /// </summary>
  199. [Route("/Users/{UserId}/Items/{Id}/LocalTrailers", "GET")]
  200. public class GetLocalTrailers : IReturn<List<BaseItemDto>>
  201. {
  202. /// <summary>
  203. /// Gets or sets the user id.
  204. /// </summary>
  205. /// <value>The user id.</value>
  206. public Guid UserId { get; set; }
  207. /// <summary>
  208. /// Gets or sets the id.
  209. /// </summary>
  210. /// <value>The id.</value>
  211. public string Id { get; set; }
  212. }
  213. /// <summary>
  214. /// Class GetSpecialFeatures
  215. /// </summary>
  216. [Route("/Users/{UserId}/Items/{Id}/SpecialFeatures", "GET")]
  217. public class GetSpecialFeatures : IReturn<List<BaseItemDto>>
  218. {
  219. /// <summary>
  220. /// Gets or sets the user id.
  221. /// </summary>
  222. /// <value>The user id.</value>
  223. public Guid UserId { get; set; }
  224. /// <summary>
  225. /// Gets or sets the id.
  226. /// </summary>
  227. /// <value>The id.</value>
  228. public string Id { get; set; }
  229. }
  230. /// <summary>
  231. /// Class UserLibraryService
  232. /// </summary>
  233. public class UserLibraryService : BaseRestService
  234. {
  235. /// <summary>
  236. /// The _json serializer
  237. /// </summary>
  238. private readonly IJsonSerializer _jsonSerializer;
  239. /// <summary>
  240. /// Initializes a new instance of the <see cref="UserLibraryService" /> class.
  241. /// </summary>
  242. /// <param name="jsonSerializer">The json serializer.</param>
  243. /// <exception cref="System.ArgumentNullException">jsonSerializer</exception>
  244. public UserLibraryService(IJsonSerializer jsonSerializer)
  245. : base()
  246. {
  247. if (jsonSerializer == null)
  248. {
  249. throw new ArgumentNullException("jsonSerializer");
  250. }
  251. _jsonSerializer = jsonSerializer;
  252. }
  253. /// <summary>
  254. /// Gets the specified request.
  255. /// </summary>
  256. /// <param name="request">The request.</param>
  257. /// <returns>System.Object.</returns>
  258. public object Get(GetSpecialFeatures request)
  259. {
  260. var kernel = (Kernel)Kernel;
  261. var user = kernel.GetUserById(request.UserId);
  262. var item = DtoBuilder.GetItemByClientId(request.Id, user.Id);
  263. // Get everything
  264. var fields = Enum.GetNames(typeof(ItemFields)).Select(i => (ItemFields)Enum.Parse(typeof(ItemFields), i, true)).ToList();
  265. var movie = (Movie)item;
  266. var dtoBuilder = new DtoBuilder(Logger);
  267. var items = movie.SpecialFeatures.Select(i => dtoBuilder.GetDtoBaseItem(item, user, fields)).AsParallel().Select(t => t.Result).ToList();
  268. return ToOptimizedResult(items);
  269. }
  270. /// <summary>
  271. /// Gets the specified request.
  272. /// </summary>
  273. /// <param name="request">The request.</param>
  274. /// <returns>System.Object.</returns>
  275. public object Get(GetLocalTrailers request)
  276. {
  277. var kernel = (Kernel)Kernel;
  278. var user = kernel.GetUserById(request.UserId);
  279. var item = DtoBuilder.GetItemByClientId(request.Id, user.Id);
  280. // Get everything
  281. var fields = Enum.GetNames(typeof(ItemFields)).Select(i => (ItemFields)Enum.Parse(typeof(ItemFields), i, true)).ToList();
  282. var dtoBuilder = new DtoBuilder(Logger);
  283. var items = item.LocalTrailers.Select(i => dtoBuilder.GetDtoBaseItem(item, user, fields)).AsParallel().Select(t => t.Result).ToList();
  284. return ToOptimizedResult(items);
  285. }
  286. /// <summary>
  287. /// Gets the specified request.
  288. /// </summary>
  289. /// <param name="request">The request.</param>
  290. /// <returns>System.Object.</returns>
  291. public object Get(GetItem request)
  292. {
  293. var kernel = (Kernel)Kernel;
  294. var user = kernel.GetUserById(request.UserId);
  295. var item = string.IsNullOrEmpty(request.Id) ? user.RootFolder : DtoBuilder.GetItemByClientId(request.Id, user.Id);
  296. // Get everything
  297. var fields = Enum.GetNames(typeof(ItemFields)).Select(i => (ItemFields)Enum.Parse(typeof(ItemFields), i, true)).ToList();
  298. var dtoBuilder = new DtoBuilder(Logger);
  299. var result = dtoBuilder.GetDtoBaseItem(item, user, fields).Result;
  300. return ToOptimizedResult(result);
  301. }
  302. /// <summary>
  303. /// Gets the specified request.
  304. /// </summary>
  305. /// <param name="request">The request.</param>
  306. /// <returns>System.Object.</returns>
  307. public object Get(GetVirtualFolders request)
  308. {
  309. var kernel = (Kernel)Kernel;
  310. var user = kernel.GetUserById(request.UserId);
  311. var result = kernel.LibraryManager.GetVirtualFolders(user).ToList();
  312. return ToOptimizedResult(result);
  313. }
  314. /// <summary>
  315. /// Gets the specified request.
  316. /// </summary>
  317. /// <param name="request">The request.</param>
  318. /// <returns>System.Object.</returns>
  319. public object Get(GetIntros request)
  320. {
  321. var kernel = (Kernel)Kernel;
  322. var user = kernel.GetUserById(request.UserId);
  323. var item = DtoBuilder.GetItemByClientId(request.Id, user.Id);
  324. var result = kernel.IntroProviders.SelectMany(i => i.GetIntros(item, user));
  325. return ToOptimizedResult(result);
  326. }
  327. /// <summary>
  328. /// Posts the specified request.
  329. /// </summary>
  330. /// <param name="request">The request.</param>
  331. public void Post(UpdateDisplayPreferences request)
  332. {
  333. // We need to parse this manually because we told service stack not to with IRequiresRequestStream
  334. // https://code.google.com/p/servicestack/source/browse/trunk/Common/ServiceStack.Text/ServiceStack.Text/Controller/PathInfo.cs
  335. var pathInfo = PathInfo.Parse(Request.PathInfo);
  336. var userId = new Guid(pathInfo.GetArgumentValue<string>(1));
  337. var itemId = pathInfo.GetArgumentValue<string>(3);
  338. var kernel = (Kernel)Kernel;
  339. var user = kernel.GetUserById(userId);
  340. var item = (Folder)DtoBuilder.GetItemByClientId(itemId, user.Id);
  341. var displayPreferences = _jsonSerializer.DeserializeFromStream<DisplayPreferences>(request.RequestStream);
  342. var task = kernel.LibraryManager.SaveDisplayPreferencesForFolder(user, item, displayPreferences);
  343. Task.WaitAll(task);
  344. }
  345. /// <summary>
  346. /// Posts the specified request.
  347. /// </summary>
  348. /// <param name="request">The request.</param>
  349. public void Post(MarkFavoriteItem request)
  350. {
  351. var kernel = (Kernel)Kernel;
  352. var user = kernel.GetUserById(request.UserId);
  353. var item = (Folder)DtoBuilder.GetItemByClientId(request.Id, user.Id);
  354. // Get the user data for this item
  355. var data = item.GetUserData(user, true);
  356. // Set favorite status
  357. data.IsFavorite = true;
  358. var task = kernel.UserDataManager.SaveUserDataForItem(user, item, data);
  359. Task.WaitAll(task);
  360. }
  361. /// <summary>
  362. /// Deletes the specified request.
  363. /// </summary>
  364. /// <param name="request">The request.</param>
  365. public void Delete(UnmarkFavoriteItem request)
  366. {
  367. var kernel = (Kernel)Kernel;
  368. var user = kernel.GetUserById(request.UserId);
  369. var item = (Folder)DtoBuilder.GetItemByClientId(request.Id, user.Id);
  370. // Get the user data for this item
  371. var data = item.GetUserData(user, true);
  372. // Set favorite status
  373. data.IsFavorite = false;
  374. var task = kernel.UserDataManager.SaveUserDataForItem(user, item, data);
  375. Task.WaitAll(task);
  376. }
  377. /// <summary>
  378. /// Deletes the specified request.
  379. /// </summary>
  380. /// <param name="request">The request.</param>
  381. public void Delete(DeleteUserItemRating request)
  382. {
  383. var kernel = (Kernel)Kernel;
  384. var user = kernel.GetUserById(request.UserId);
  385. var item = (Folder)DtoBuilder.GetItemByClientId(request.Id, user.Id);
  386. // Get the user data for this item
  387. var data = item.GetUserData(user, true);
  388. data.Rating = null;
  389. var task = kernel.UserDataManager.SaveUserDataForItem(user, item, data);
  390. Task.WaitAll(task);
  391. }
  392. /// <summary>
  393. /// Posts the specified request.
  394. /// </summary>
  395. /// <param name="request">The request.</param>
  396. public void Post(UpdateUserItemRating request)
  397. {
  398. var kernel = (Kernel)Kernel;
  399. var user = kernel.GetUserById(request.UserId);
  400. var item = (Folder)DtoBuilder.GetItemByClientId(request.Id, user.Id);
  401. // Get the user data for this item
  402. var data = item.GetUserData(user, true);
  403. data.Likes = request.Likes;
  404. var task = kernel.UserDataManager.SaveUserDataForItem(user, item, data);
  405. Task.WaitAll(task);
  406. }
  407. /// <summary>
  408. /// Posts the specified request.
  409. /// </summary>
  410. /// <param name="request">The request.</param>
  411. public void Post(MarkPlayedItem request)
  412. {
  413. var kernel = (Kernel)Kernel;
  414. var user = kernel.GetUserById(request.UserId);
  415. var task = UpdatePlayedStatus(user, request.Id, true);
  416. Task.WaitAll(task);
  417. }
  418. /// <summary>
  419. /// Deletes the specified request.
  420. /// </summary>
  421. /// <param name="request">The request.</param>
  422. public void Delete(MarkUnplayedItem request)
  423. {
  424. var kernel = (Kernel)Kernel;
  425. var user = kernel.GetUserById(request.UserId);
  426. var task = UpdatePlayedStatus(user, request.Id, false);
  427. Task.WaitAll(task);
  428. }
  429. /// <summary>
  430. /// Updates the played status.
  431. /// </summary>
  432. /// <param name="user">The user.</param>
  433. /// <param name="itemId">The item id.</param>
  434. /// <param name="wasPlayed">if set to <c>true</c> [was played].</param>
  435. /// <returns>Task.</returns>
  436. private Task UpdatePlayedStatus(User user, string itemId, bool wasPlayed)
  437. {
  438. var item = DtoBuilder.GetItemByClientId(itemId, user.Id);
  439. return item.SetPlayedStatus(user, wasPlayed);
  440. }
  441. }
  442. }