BaseApiService.cs 12 KB

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