BaseApiService.cs 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375
  1. using MediaBrowser.Controller.Dto;
  2. using MediaBrowser.Controller.Entities;
  3. using MediaBrowser.Controller.Entities.Audio;
  4. using MediaBrowser.Controller.Library;
  5. using MediaBrowser.Controller.Net;
  6. using MediaBrowser.Controller.Session;
  7. using MediaBrowser.Model.Entities;
  8. using MediaBrowser.Model.Logging;
  9. using ServiceStack.Text.Controller;
  10. using ServiceStack.Web;
  11. using System;
  12. using System.Collections.Generic;
  13. using System.Linq;
  14. using System.Threading.Tasks;
  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. protected void AssertCanUpdateUser(IUserManager userManager, string userId)
  69. {
  70. var auth = AuthorizationContext.GetAuthorizationInfo(Request);
  71. var authenticatedUser = userManager.GetUserById(auth.UserId);
  72. // If they're going to update the record of another user, they must be an administrator
  73. if (!string.Equals(userId, auth.UserId, StringComparison.OrdinalIgnoreCase))
  74. {
  75. if (!authenticatedUser.Policy.IsAdministrator)
  76. {
  77. throw new SecurityException("Unauthorized access.");
  78. }
  79. }
  80. else
  81. {
  82. if (!authenticatedUser.Policy.EnableUserPreferenceAccess)
  83. {
  84. throw new SecurityException("Unauthorized access.");
  85. }
  86. }
  87. }
  88. /// <summary>
  89. /// To the optimized serialized result using cache.
  90. /// </summary>
  91. /// <typeparam name="T"></typeparam>
  92. /// <param name="result">The result.</param>
  93. /// <returns>System.Object.</returns>
  94. protected object ToOptimizedSerializedResultUsingCache<T>(T result)
  95. where T : class
  96. {
  97. return ToOptimizedResult(result);
  98. }
  99. /// <summary>
  100. /// Gets the session.
  101. /// </summary>
  102. /// <returns>SessionInfo.</returns>
  103. protected async Task<SessionInfo> GetSession()
  104. {
  105. var session = await SessionContext.GetSession(Request).ConfigureAwait(false);
  106. if (session == null)
  107. {
  108. throw new ArgumentException("Session not found.");
  109. }
  110. return session;
  111. }
  112. /// <summary>
  113. /// To the static file result.
  114. /// </summary>
  115. /// <param name="path">The path.</param>
  116. /// <returns>System.Object.</returns>
  117. protected object ToStaticFileResult(string path)
  118. {
  119. return ResultFactory.GetStaticFileResult(Request, path);
  120. }
  121. private readonly char[] _dashReplaceChars = { '?', '/', '&' };
  122. private const char SlugChar = '-';
  123. protected DtoOptions GetDtoOptions(object request)
  124. {
  125. var options = new DtoOptions();
  126. options.DeviceId = AuthorizationContext.GetAuthorizationInfo(Request).DeviceId;
  127. var hasFields = request as IHasItemFields;
  128. if (hasFields != null)
  129. {
  130. options.Fields = hasFields.GetItemFields().ToList();
  131. }
  132. var hasDtoOptions = request as IHasDtoOptions;
  133. if (hasDtoOptions != null)
  134. {
  135. options.EnableImages = hasDtoOptions.EnableImages ?? true;
  136. if (hasDtoOptions.ImageTypeLimit.HasValue)
  137. {
  138. options.ImageTypeLimit = hasDtoOptions.ImageTypeLimit.Value;
  139. }
  140. if (!string.IsNullOrWhiteSpace(hasDtoOptions.EnableImageTypes))
  141. {
  142. options.ImageTypes = (hasDtoOptions.EnableImageTypes ?? string.Empty).Split(',').Where(i => !string.IsNullOrWhiteSpace(i)).Select(v => (ImageType)Enum.Parse(typeof(ImageType), v, true)).ToList();
  143. }
  144. }
  145. return options;
  146. }
  147. protected MusicArtist GetArtist(string name, ILibraryManager libraryManager)
  148. {
  149. return libraryManager.GetArtist(DeSlugArtistName(name, libraryManager));
  150. }
  151. protected Studio GetStudio(string name, ILibraryManager libraryManager)
  152. {
  153. return libraryManager.GetStudio(DeSlugStudioName(name, libraryManager));
  154. }
  155. protected Genre GetGenre(string name, ILibraryManager libraryManager)
  156. {
  157. return libraryManager.GetGenre(DeSlugGenreName(name, libraryManager));
  158. }
  159. protected MusicGenre GetMusicGenre(string name, ILibraryManager libraryManager)
  160. {
  161. return libraryManager.GetMusicGenre(DeSlugGenreName(name, libraryManager));
  162. }
  163. protected GameGenre GetGameGenre(string name, ILibraryManager libraryManager)
  164. {
  165. return libraryManager.GetGameGenre(DeSlugGameGenreName(name, libraryManager));
  166. }
  167. protected Person GetPerson(string name, ILibraryManager libraryManager)
  168. {
  169. return libraryManager.GetPerson(DeSlugPersonName(name, libraryManager));
  170. }
  171. /// <summary>
  172. /// Deslugs an artist name by finding the correct entry in the library
  173. /// </summary>
  174. /// <param name="name"></param>
  175. /// <param name="libraryManager"></param>
  176. /// <returns></returns>
  177. protected string DeSlugArtistName(string name, ILibraryManager libraryManager)
  178. {
  179. if (name.IndexOf(SlugChar) == -1)
  180. {
  181. return name;
  182. }
  183. return libraryManager.RootFolder
  184. .GetRecursiveChildren(i => i is IHasArtist)
  185. .Cast<IHasArtist>()
  186. .SelectMany(i => i.AllArtists)
  187. .DistinctNames()
  188. .FirstOrDefault(i =>
  189. {
  190. i = _dashReplaceChars.Aggregate(i, (current, c) => current.Replace(c, SlugChar));
  191. return string.Equals(i, name, StringComparison.OrdinalIgnoreCase);
  192. }) ?? name;
  193. }
  194. /// <summary>
  195. /// Deslugs a genre name by finding the correct entry in the library
  196. /// </summary>
  197. protected string DeSlugGenreName(string name, ILibraryManager libraryManager)
  198. {
  199. if (name.IndexOf(SlugChar) == -1)
  200. {
  201. return name;
  202. }
  203. return libraryManager.RootFolder.GetRecursiveChildren()
  204. .SelectMany(i => i.Genres)
  205. .DistinctNames()
  206. .FirstOrDefault(i =>
  207. {
  208. i = _dashReplaceChars.Aggregate(i, (current, c) => current.Replace(c, SlugChar));
  209. return string.Equals(i, name, StringComparison.OrdinalIgnoreCase);
  210. }) ?? name;
  211. }
  212. protected string DeSlugGameGenreName(string name, ILibraryManager libraryManager)
  213. {
  214. if (name.IndexOf(SlugChar) == -1)
  215. {
  216. return name;
  217. }
  218. return libraryManager.RootFolder
  219. .GetRecursiveChildren(i => i is Game)
  220. .SelectMany(i => i.Genres)
  221. .DistinctNames()
  222. .FirstOrDefault(i =>
  223. {
  224. i = _dashReplaceChars.Aggregate(i, (current, c) => current.Replace(c, SlugChar));
  225. return string.Equals(i, name, StringComparison.OrdinalIgnoreCase);
  226. }) ?? name;
  227. }
  228. /// <summary>
  229. /// Deslugs a studio name by finding the correct entry in the library
  230. /// </summary>
  231. protected string DeSlugStudioName(string name, ILibraryManager libraryManager)
  232. {
  233. if (name.IndexOf(SlugChar) == -1)
  234. {
  235. return name;
  236. }
  237. return libraryManager.RootFolder
  238. .GetRecursiveChildren()
  239. .SelectMany(i => i.Studios)
  240. .DistinctNames()
  241. .FirstOrDefault(i =>
  242. {
  243. i = _dashReplaceChars.Aggregate(i, (current, c) => current.Replace(c, SlugChar));
  244. return string.Equals(i, name, StringComparison.OrdinalIgnoreCase);
  245. }) ?? name;
  246. }
  247. /// <summary>
  248. /// Deslugs a person name by finding the correct entry in the library
  249. /// </summary>
  250. protected string DeSlugPersonName(string name, ILibraryManager libraryManager)
  251. {
  252. if (name.IndexOf(SlugChar) == -1)
  253. {
  254. return name;
  255. }
  256. return libraryManager.GetPeopleNames(new InternalPeopleQuery())
  257. .FirstOrDefault(i =>
  258. {
  259. i = _dashReplaceChars.Aggregate(i, (current, c) => current.Replace(c, SlugChar));
  260. return string.Equals(i, name, StringComparison.OrdinalIgnoreCase);
  261. }) ?? name;
  262. }
  263. protected string GetPathValue(int index)
  264. {
  265. var pathInfo = PathInfo.Parse(Request.PathInfo);
  266. var first = pathInfo.GetArgumentValue<string>(0);
  267. // backwards compatibility
  268. if (string.Equals(first, "mediabrowser", StringComparison.OrdinalIgnoreCase) ||
  269. string.Equals(first, "emby", StringComparison.OrdinalIgnoreCase))
  270. {
  271. index++;
  272. }
  273. return pathInfo.GetArgumentValue<string>(index);
  274. }
  275. /// <summary>
  276. /// Gets the name of the item by.
  277. /// </summary>
  278. /// <param name="name">The name.</param>
  279. /// <param name="type">The type.</param>
  280. /// <param name="libraryManager">The library manager.</param>
  281. /// <returns>Task{BaseItem}.</returns>
  282. protected BaseItem GetItemByName(string name, string type, ILibraryManager libraryManager)
  283. {
  284. BaseItem item;
  285. if (type.IndexOf("Person", StringComparison.OrdinalIgnoreCase) == 0)
  286. {
  287. item = GetPerson(name, libraryManager);
  288. }
  289. else if (type.IndexOf("Artist", StringComparison.OrdinalIgnoreCase) == 0)
  290. {
  291. item = GetArtist(name, libraryManager);
  292. }
  293. else if (type.IndexOf("Genre", StringComparison.OrdinalIgnoreCase) == 0)
  294. {
  295. item = GetGenre(name, libraryManager);
  296. }
  297. else if (type.IndexOf("MusicGenre", StringComparison.OrdinalIgnoreCase) == 0)
  298. {
  299. item = GetMusicGenre(name, libraryManager);
  300. }
  301. else if (type.IndexOf("GameGenre", StringComparison.OrdinalIgnoreCase) == 0)
  302. {
  303. item = GetGameGenre(name, libraryManager);
  304. }
  305. else if (type.IndexOf("Studio", StringComparison.OrdinalIgnoreCase) == 0)
  306. {
  307. item = GetStudio(name, libraryManager);
  308. }
  309. else if (type.IndexOf("Year", StringComparison.OrdinalIgnoreCase) == 0)
  310. {
  311. item = libraryManager.GetYear(int.Parse(name));
  312. }
  313. else
  314. {
  315. throw new ArgumentException();
  316. }
  317. return item;
  318. }
  319. }
  320. }