BaseApiService.cs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433
  1. using System.Threading.Tasks;
  2. using MediaBrowser.Controller.Dto;
  3. using MediaBrowser.Controller.Entities;
  4. using MediaBrowser.Controller.Entities.Audio;
  5. using MediaBrowser.Controller.Library;
  6. using MediaBrowser.Controller.Net;
  7. using MediaBrowser.Controller.Session;
  8. using MediaBrowser.Model.Entities;
  9. using MediaBrowser.Model.Logging;
  10. using ServiceStack.Text.Controller;
  11. using ServiceStack.Web;
  12. using System;
  13. using System.Collections.Generic;
  14. using System.Linq;
  15. namespace MediaBrowser.Api
  16. {
  17. /// <summary>
  18. /// Class BaseApiService
  19. /// </summary>
  20. public class BaseApiService : IHasResultFactory, IRestfulService, IHasSession
  21. {
  22. /// <summary>
  23. /// Gets or sets the logger.
  24. /// </summary>
  25. /// <value>The logger.</value>
  26. public ILogger Logger { get; set; }
  27. /// <summary>
  28. /// Gets or sets the HTTP result factory.
  29. /// </summary>
  30. /// <value>The HTTP result factory.</value>
  31. public IHttpResultFactory ResultFactory { get; set; }
  32. /// <summary>
  33. /// Gets or sets the request context.
  34. /// </summary>
  35. /// <value>The request context.</value>
  36. public IRequest Request { get; set; }
  37. public ISessionContext SessionContext { get; set; }
  38. public IAuthorizationContext AuthorizationContext { get; set; }
  39. public string GetHeader(string name)
  40. {
  41. return Request.Headers[name];
  42. }
  43. /// <summary>
  44. /// To the optimized result.
  45. /// </summary>
  46. /// <typeparam name="T"></typeparam>
  47. /// <param name="result">The result.</param>
  48. /// <returns>System.Object.</returns>
  49. protected object ToOptimizedResult<T>(T result)
  50. where T : class
  51. {
  52. return ResultFactory.GetOptimizedResult(Request, result);
  53. }
  54. /// <summary>
  55. /// To the optimized result using cache.
  56. /// </summary>
  57. /// <typeparam name="T"></typeparam>
  58. /// <param name="cacheKey">The cache key.</param>
  59. /// <param name="lastDateModified">The last date modified.</param>
  60. /// <param name="cacheDuration">Duration of the cache.</param>
  61. /// <param name="factoryFn">The factory function.</param>
  62. /// <returns>System.Object.</returns>
  63. protected object ToOptimizedResultUsingCache<T>(Guid cacheKey, DateTime? lastDateModified, TimeSpan? cacheDuration, Func<T> factoryFn)
  64. where T : class
  65. {
  66. return ResultFactory.GetOptimizedResultUsingCache(Request, cacheKey, lastDateModified, cacheDuration, factoryFn);
  67. }
  68. /// <summary>
  69. /// Infers the server address from the url
  70. /// </summary>
  71. /// <returns></returns>
  72. protected string GetServerAddress()
  73. {
  74. var index = Request.AbsoluteUri.IndexOf(Request.PathInfo, StringComparison.OrdinalIgnoreCase);
  75. return Request.AbsoluteUri.Substring(0, index);
  76. }
  77. protected void AssertCanUpdateUser(IUserManager userManager, string userId)
  78. {
  79. var auth = AuthorizationContext.GetAuthorizationInfo(Request);
  80. var authenticatedUser = userManager.GetUserById(auth.UserId);
  81. // If they're going to update the record of another user, they must be an administrator
  82. if (!string.Equals(userId, auth.UserId, StringComparison.OrdinalIgnoreCase))
  83. {
  84. if (!authenticatedUser.Policy.IsAdministrator)
  85. {
  86. throw new SecurityException("Unauthorized access.");
  87. }
  88. }
  89. else
  90. {
  91. if (!authenticatedUser.Policy.EnableUserPreferenceAccess)
  92. {
  93. throw new SecurityException("Unauthorized access.");
  94. }
  95. }
  96. }
  97. /// <summary>
  98. /// To the optimized serialized result using cache.
  99. /// </summary>
  100. /// <typeparam name="T"></typeparam>
  101. /// <param name="result">The result.</param>
  102. /// <returns>System.Object.</returns>
  103. protected object ToOptimizedSerializedResultUsingCache<T>(T result)
  104. where T : class
  105. {
  106. return ToOptimizedResult(result);
  107. }
  108. /// <summary>
  109. /// Gets the session.
  110. /// </summary>
  111. /// <returns>SessionInfo.</returns>
  112. protected async Task<SessionInfo> GetSession()
  113. {
  114. var session = await SessionContext.GetSession(Request).ConfigureAwait(false);
  115. if (session == null)
  116. {
  117. throw new ArgumentException("Session not found.");
  118. }
  119. return session;
  120. }
  121. /// <summary>
  122. /// To the static file result.
  123. /// </summary>
  124. /// <param name="path">The path.</param>
  125. /// <returns>System.Object.</returns>
  126. protected object ToStaticFileResult(string path)
  127. {
  128. return ResultFactory.GetStaticFileResult(Request, path);
  129. }
  130. private readonly char[] _dashReplaceChars = { '?', '/', '&' };
  131. private const char SlugChar = '-';
  132. protected DtoOptions GetDtoOptions(object request)
  133. {
  134. var options = new DtoOptions();
  135. options.DeviceId = AuthorizationContext.GetAuthorizationInfo(Request).DeviceId;
  136. var hasFields = request as IHasItemFields;
  137. if (hasFields != null)
  138. {
  139. options.Fields = hasFields.GetItemFields().ToList();
  140. }
  141. var hasDtoOptions = request as IHasDtoOptions;
  142. if (hasDtoOptions != null)
  143. {
  144. options.EnableImages = hasDtoOptions.EnableImages ?? true;
  145. if (hasDtoOptions.ImageTypeLimit.HasValue)
  146. {
  147. options.ImageTypeLimit = hasDtoOptions.ImageTypeLimit.Value;
  148. }
  149. if (!string.IsNullOrWhiteSpace(hasDtoOptions.EnableImageTypes))
  150. {
  151. options.ImageTypes = (hasDtoOptions.EnableImageTypes ?? string.Empty).Split(',').Where(i => !string.IsNullOrWhiteSpace(i)).Select(v => (ImageType)Enum.Parse(typeof(ImageType), v, true)).ToList();
  152. }
  153. }
  154. return options;
  155. }
  156. protected MusicArtist GetArtist(string name, ILibraryManager libraryManager)
  157. {
  158. return libraryManager.GetArtist(DeSlugArtistName(name, libraryManager));
  159. }
  160. protected Studio GetStudio(string name, ILibraryManager libraryManager)
  161. {
  162. return libraryManager.GetStudio(DeSlugStudioName(name, libraryManager));
  163. }
  164. protected Genre GetGenre(string name, ILibraryManager libraryManager)
  165. {
  166. return libraryManager.GetGenre(DeSlugGenreName(name, libraryManager));
  167. }
  168. protected MusicGenre GetMusicGenre(string name, ILibraryManager libraryManager)
  169. {
  170. return libraryManager.GetMusicGenre(DeSlugGenreName(name, libraryManager));
  171. }
  172. protected GameGenre GetGameGenre(string name, ILibraryManager libraryManager)
  173. {
  174. return libraryManager.GetGameGenre(DeSlugGameGenreName(name, libraryManager));
  175. }
  176. protected Person GetPerson(string name, ILibraryManager libraryManager)
  177. {
  178. return libraryManager.GetPerson(DeSlugPersonName(name, libraryManager));
  179. }
  180. protected IList<BaseItem> GetAllLibraryItems(Guid? userId, IUserManager userManager, ILibraryManager libraryManager, string parentId, Func<BaseItem,bool> filter)
  181. {
  182. if (!string.IsNullOrEmpty(parentId))
  183. {
  184. var folder = (Folder)libraryManager.GetItemById(new Guid(parentId));
  185. if (userId.HasValue)
  186. {
  187. var user = userManager.GetUserById(userId.Value);
  188. if (user == null)
  189. {
  190. throw new ArgumentException("User not found");
  191. }
  192. return folder
  193. .GetRecursiveChildren(user, filter)
  194. .ToList();
  195. }
  196. return folder
  197. .GetRecursiveChildren(filter);
  198. }
  199. if (userId.HasValue)
  200. {
  201. var user = userManager.GetUserById(userId.Value);
  202. if (user == null)
  203. {
  204. throw new ArgumentException("User not found");
  205. }
  206. return userManager
  207. .GetUserById(userId.Value)
  208. .RootFolder
  209. .GetRecursiveChildren(user, filter)
  210. .ToList();
  211. }
  212. return libraryManager
  213. .RootFolder
  214. .GetRecursiveChildren(filter);
  215. }
  216. /// <summary>
  217. /// Deslugs an artist name by finding the correct entry in the library
  218. /// </summary>
  219. /// <param name="name"></param>
  220. /// <param name="libraryManager"></param>
  221. /// <returns></returns>
  222. protected string DeSlugArtistName(string name, ILibraryManager libraryManager)
  223. {
  224. if (name.IndexOf(SlugChar) == -1)
  225. {
  226. return name;
  227. }
  228. return libraryManager.RootFolder
  229. .GetRecursiveChildren(i => i is IHasArtist)
  230. .Cast<IHasArtist>()
  231. .SelectMany(i => i.AllArtists)
  232. .Distinct(StringComparer.OrdinalIgnoreCase)
  233. .FirstOrDefault(i =>
  234. {
  235. i = _dashReplaceChars.Aggregate(i, (current, c) => current.Replace(c, SlugChar));
  236. return string.Equals(i, name, StringComparison.OrdinalIgnoreCase);
  237. }) ?? name;
  238. }
  239. /// <summary>
  240. /// Deslugs a genre name by finding the correct entry in the library
  241. /// </summary>
  242. protected string DeSlugGenreName(string name, ILibraryManager libraryManager)
  243. {
  244. if (name.IndexOf(SlugChar) == -1)
  245. {
  246. return name;
  247. }
  248. return libraryManager.RootFolder.GetRecursiveChildren()
  249. .SelectMany(i => i.Genres)
  250. .Distinct(StringComparer.OrdinalIgnoreCase)
  251. .FirstOrDefault(i =>
  252. {
  253. i = _dashReplaceChars.Aggregate(i, (current, c) => current.Replace(c, SlugChar));
  254. return string.Equals(i, name, StringComparison.OrdinalIgnoreCase);
  255. }) ?? name;
  256. }
  257. protected string DeSlugGameGenreName(string name, ILibraryManager libraryManager)
  258. {
  259. if (name.IndexOf(SlugChar) == -1)
  260. {
  261. return name;
  262. }
  263. return libraryManager.RootFolder
  264. .GetRecursiveChildren(i => i is Game)
  265. .SelectMany(i => i.Genres)
  266. .Distinct(StringComparer.OrdinalIgnoreCase)
  267. .FirstOrDefault(i =>
  268. {
  269. i = _dashReplaceChars.Aggregate(i, (current, c) => current.Replace(c, SlugChar));
  270. return string.Equals(i, name, StringComparison.OrdinalIgnoreCase);
  271. }) ?? name;
  272. }
  273. /// <summary>
  274. /// Deslugs a studio name by finding the correct entry in the library
  275. /// </summary>
  276. protected string DeSlugStudioName(string name, ILibraryManager libraryManager)
  277. {
  278. if (name.IndexOf(SlugChar) == -1)
  279. {
  280. return name;
  281. }
  282. return libraryManager.RootFolder
  283. .GetRecursiveChildren()
  284. .SelectMany(i => i.Studios)
  285. .Distinct(StringComparer.OrdinalIgnoreCase)
  286. .FirstOrDefault(i =>
  287. {
  288. i = _dashReplaceChars.Aggregate(i, (current, c) => current.Replace(c, SlugChar));
  289. return string.Equals(i, name, StringComparison.OrdinalIgnoreCase);
  290. }) ?? name;
  291. }
  292. /// <summary>
  293. /// Deslugs a person name by finding the correct entry in the library
  294. /// </summary>
  295. protected string DeSlugPersonName(string name, ILibraryManager libraryManager)
  296. {
  297. if (name.IndexOf(SlugChar) == -1)
  298. {
  299. return name;
  300. }
  301. return libraryManager.RootFolder
  302. .GetRecursiveChildren()
  303. .SelectMany(i => i.People)
  304. .Select(i => i.Name)
  305. .Distinct(StringComparer.OrdinalIgnoreCase)
  306. .FirstOrDefault(i =>
  307. {
  308. i = _dashReplaceChars.Aggregate(i, (current, c) => current.Replace(c, SlugChar));
  309. return string.Equals(i, name, StringComparison.OrdinalIgnoreCase);
  310. }) ?? name;
  311. }
  312. protected string GetPathValue(int index)
  313. {
  314. var pathInfo = PathInfo.Parse(Request.PathInfo);
  315. var first = pathInfo.GetArgumentValue<string>(0);
  316. // backwards compatibility
  317. if (string.Equals(first, "mediabrowser", StringComparison.OrdinalIgnoreCase))
  318. {
  319. index++;
  320. }
  321. return pathInfo.GetArgumentValue<string>(index);
  322. }
  323. /// <summary>
  324. /// Gets the name of the item by.
  325. /// </summary>
  326. /// <param name="name">The name.</param>
  327. /// <param name="type">The type.</param>
  328. /// <param name="libraryManager">The library manager.</param>
  329. /// <returns>Task{BaseItem}.</returns>
  330. protected BaseItem GetItemByName(string name, string type, ILibraryManager libraryManager)
  331. {
  332. BaseItem item;
  333. if (type.IndexOf("Person", StringComparison.OrdinalIgnoreCase) == 0)
  334. {
  335. item = GetPerson(name, libraryManager);
  336. }
  337. else if (type.IndexOf("Artist", StringComparison.OrdinalIgnoreCase) == 0)
  338. {
  339. item = GetArtist(name, libraryManager);
  340. }
  341. else if (type.IndexOf("Genre", StringComparison.OrdinalIgnoreCase) == 0)
  342. {
  343. item = GetGenre(name, libraryManager);
  344. }
  345. else if (type.IndexOf("MusicGenre", StringComparison.OrdinalIgnoreCase) == 0)
  346. {
  347. item = GetMusicGenre(name, libraryManager);
  348. }
  349. else if (type.IndexOf("GameGenre", StringComparison.OrdinalIgnoreCase) == 0)
  350. {
  351. item = GetGameGenre(name, libraryManager);
  352. }
  353. else if (type.IndexOf("Studio", StringComparison.OrdinalIgnoreCase) == 0)
  354. {
  355. item = GetStudio(name, libraryManager);
  356. }
  357. else if (type.IndexOf("Year", StringComparison.OrdinalIgnoreCase) == 0)
  358. {
  359. item = libraryManager.GetYear(int.Parse(name));
  360. }
  361. else
  362. {
  363. throw new ArgumentException();
  364. }
  365. return item;
  366. }
  367. }
  368. }