BaseApiService.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347
  1. using MediaBrowser.Controller.Entities;
  2. using MediaBrowser.Controller.Entities.Audio;
  3. using MediaBrowser.Controller.Library;
  4. using MediaBrowser.Controller.Net;
  5. using MediaBrowser.Controller.Session;
  6. using MediaBrowser.Model.Logging;
  7. using ServiceStack.Web;
  8. using System;
  9. using System.Collections.Generic;
  10. using System.Linq;
  11. namespace MediaBrowser.Api
  12. {
  13. /// <summary>
  14. /// Class BaseApiService
  15. /// </summary>
  16. public class BaseApiService : IHasResultFactory, IRestfulService, IHasSession
  17. {
  18. /// <summary>
  19. /// Gets or sets the logger.
  20. /// </summary>
  21. /// <value>The logger.</value>
  22. public ILogger Logger { get; set; }
  23. /// <summary>
  24. /// Gets or sets the HTTP result factory.
  25. /// </summary>
  26. /// <value>The HTTP result factory.</value>
  27. public IHttpResultFactory ResultFactory { get; set; }
  28. /// <summary>
  29. /// Gets or sets the request context.
  30. /// </summary>
  31. /// <value>The request context.</value>
  32. public IRequest Request { get; set; }
  33. public ISessionContext SessionContext { get; set; }
  34. public string GetHeader(string name)
  35. {
  36. return Request.Headers[name];
  37. }
  38. /// <summary>
  39. /// To the optimized result.
  40. /// </summary>
  41. /// <typeparam name="T"></typeparam>
  42. /// <param name="result">The result.</param>
  43. /// <returns>System.Object.</returns>
  44. protected object ToOptimizedResult<T>(T result)
  45. where T : class
  46. {
  47. return ResultFactory.GetOptimizedResult(Request, result);
  48. }
  49. /// <summary>
  50. /// To the optimized result using cache.
  51. /// </summary>
  52. /// <typeparam name="T"></typeparam>
  53. /// <param name="cacheKey">The cache key.</param>
  54. /// <param name="lastDateModified">The last date modified.</param>
  55. /// <param name="cacheDuration">Duration of the cache.</param>
  56. /// <param name="factoryFn">The factory function.</param>
  57. /// <returns>System.Object.</returns>
  58. protected object ToOptimizedResultUsingCache<T>(Guid cacheKey, DateTime? lastDateModified, TimeSpan? cacheDuration, Func<T> factoryFn)
  59. where T : class
  60. {
  61. return ResultFactory.GetOptimizedResultUsingCache(Request, cacheKey, lastDateModified, cacheDuration, factoryFn);
  62. }
  63. /// <summary>
  64. /// To the optimized serialized result using cache.
  65. /// </summary>
  66. /// <typeparam name="T"></typeparam>
  67. /// <param name="result">The result.</param>
  68. /// <returns>System.Object.</returns>
  69. protected object ToOptimizedSerializedResultUsingCache<T>(T result)
  70. where T : class
  71. {
  72. return ToOptimizedResult(result);
  73. }
  74. /// <summary>
  75. /// Gets the session.
  76. /// </summary>
  77. /// <returns>SessionInfo.</returns>
  78. /// <exception cref="System.ArgumentException">Session not found.</exception>
  79. protected SessionInfo GetSession()
  80. {
  81. var session = SessionContext.GetSession(Request);
  82. if (session == null)
  83. {
  84. throw new ArgumentException("Session not found.");
  85. }
  86. return session;
  87. }
  88. /// <summary>
  89. /// To the cached result.
  90. /// </summary>
  91. /// <typeparam name="T"></typeparam>
  92. /// <param name="cacheKey">The cache key.</param>
  93. /// <param name="lastDateModified">The last date modified.</param>
  94. /// <param name="cacheDuration">Duration of the cache.</param>
  95. /// <param name="factoryFn">The factory fn.</param>
  96. /// <param name="contentType">Type of the content.</param>
  97. /// <param name="responseHeaders">The response headers.</param>
  98. /// <returns>System.Object.</returns>
  99. /// <exception cref="System.ArgumentNullException">cacheKey</exception>
  100. protected object ToCachedResult<T>(Guid cacheKey, DateTime? lastDateModified, TimeSpan? cacheDuration, Func<T> factoryFn, string contentType, IDictionary<string,string> responseHeaders = null)
  101. where T : class
  102. {
  103. return ResultFactory.GetCachedResult(Request, cacheKey, lastDateModified, cacheDuration, factoryFn, contentType, responseHeaders);
  104. }
  105. /// <summary>
  106. /// To the static file result.
  107. /// </summary>
  108. /// <param name="path">The path.</param>
  109. /// <returns>System.Object.</returns>
  110. protected object ToStaticFileResult(string path)
  111. {
  112. return ResultFactory.GetStaticFileResult(Request, path);
  113. }
  114. private readonly char[] _dashReplaceChars = new[] { '?', '/' };
  115. private const char SlugChar = '-';
  116. protected MusicArtist GetArtist(string name, ILibraryManager libraryManager)
  117. {
  118. return libraryManager.GetArtist(DeSlugArtistName(name, libraryManager));
  119. }
  120. protected Studio GetStudio(string name, ILibraryManager libraryManager)
  121. {
  122. return libraryManager.GetStudio(DeSlugStudioName(name, libraryManager));
  123. }
  124. protected Genre GetGenre(string name, ILibraryManager libraryManager)
  125. {
  126. return libraryManager.GetGenre(DeSlugGenreName(name, libraryManager));
  127. }
  128. protected MusicGenre GetMusicGenre(string name, ILibraryManager libraryManager)
  129. {
  130. return libraryManager.GetMusicGenre(DeSlugGenreName(name, libraryManager));
  131. }
  132. protected GameGenre GetGameGenre(string name, ILibraryManager libraryManager)
  133. {
  134. return libraryManager.GetGameGenre(DeSlugGameGenreName(name, libraryManager));
  135. }
  136. protected Person GetPerson(string name, ILibraryManager libraryManager)
  137. {
  138. return libraryManager.GetPerson(DeSlugPersonName(name, libraryManager));
  139. }
  140. protected IEnumerable<BaseItem> GetAllLibraryItems(Guid? userId, IUserManager userManager, ILibraryManager libraryManager, string parentId = null)
  141. {
  142. if (!string.IsNullOrEmpty(parentId))
  143. {
  144. var folder = (Folder) libraryManager.GetItemById(new Guid(parentId));
  145. if (userId.HasValue)
  146. {
  147. var user = userManager.GetUserById(userId.Value);
  148. return folder.GetRecursiveChildren(user);
  149. }
  150. return folder.GetRecursiveChildren();
  151. }
  152. if (userId.HasValue)
  153. {
  154. var user = userManager.GetUserById(userId.Value);
  155. return userManager.GetUserById(userId.Value).RootFolder.GetRecursiveChildren(user);
  156. }
  157. return libraryManager.RootFolder.GetRecursiveChildren();
  158. }
  159. /// <summary>
  160. /// Deslugs an artist name by finding the correct entry in the library
  161. /// </summary>
  162. /// <param name="name"></param>
  163. /// <param name="libraryManager"></param>
  164. /// <returns></returns>
  165. protected string DeSlugArtistName(string name, ILibraryManager libraryManager)
  166. {
  167. if (name.IndexOf(SlugChar) == -1)
  168. {
  169. return name;
  170. }
  171. return libraryManager.RootFolder.RecursiveChildren
  172. .OfType<Audio>()
  173. .SelectMany(i => i.AllArtists)
  174. .Distinct(StringComparer.OrdinalIgnoreCase)
  175. .FirstOrDefault(i =>
  176. {
  177. i = _dashReplaceChars.Aggregate(i, (current, c) => current.Replace(c, SlugChar));
  178. return string.Equals(i, name, StringComparison.OrdinalIgnoreCase);
  179. }) ?? name;
  180. }
  181. /// <summary>
  182. /// Deslugs a genre name by finding the correct entry in the library
  183. /// </summary>
  184. protected string DeSlugGenreName(string name, ILibraryManager libraryManager)
  185. {
  186. if (name.IndexOf(SlugChar) == -1)
  187. {
  188. return name;
  189. }
  190. return libraryManager.RootFolder.GetRecursiveChildren()
  191. .SelectMany(i => i.Genres)
  192. .Distinct(StringComparer.OrdinalIgnoreCase)
  193. .FirstOrDefault(i =>
  194. {
  195. i = _dashReplaceChars.Aggregate(i, (current, c) => current.Replace(c, SlugChar));
  196. return string.Equals(i, name, StringComparison.OrdinalIgnoreCase);
  197. }) ?? name;
  198. }
  199. protected string DeSlugGameGenreName(string name, ILibraryManager libraryManager)
  200. {
  201. if (name.IndexOf(SlugChar) == -1)
  202. {
  203. return name;
  204. }
  205. return libraryManager.RootFolder.GetRecursiveChildren()
  206. .OfType<Game>()
  207. .SelectMany(i => i.Genres)
  208. .Distinct(StringComparer.OrdinalIgnoreCase)
  209. .FirstOrDefault(i =>
  210. {
  211. i = _dashReplaceChars.Aggregate(i, (current, c) => current.Replace(c, SlugChar));
  212. return string.Equals(i, name, StringComparison.OrdinalIgnoreCase);
  213. }) ?? name;
  214. }
  215. /// <summary>
  216. /// Deslugs a studio name by finding the correct entry in the library
  217. /// </summary>
  218. protected string DeSlugStudioName(string name, ILibraryManager libraryManager)
  219. {
  220. if (name.IndexOf(SlugChar) == -1)
  221. {
  222. return name;
  223. }
  224. return libraryManager.RootFolder.GetRecursiveChildren()
  225. .SelectMany(i => i.Studios)
  226. .Distinct(StringComparer.OrdinalIgnoreCase)
  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.RootFolder.GetRecursiveChildren()
  243. .SelectMany(i => i.People)
  244. .Select(i => i.Name)
  245. .Distinct(StringComparer.OrdinalIgnoreCase)
  246. .FirstOrDefault(i =>
  247. {
  248. i = _dashReplaceChars.Aggregate(i, (current, c) => current.Replace(c, SlugChar));
  249. return string.Equals(i, name, StringComparison.OrdinalIgnoreCase);
  250. }) ?? name;
  251. }
  252. /// <summary>
  253. /// Gets the name of the item by.
  254. /// </summary>
  255. /// <param name="name">The name.</param>
  256. /// <param name="type">The type.</param>
  257. /// <param name="libraryManager">The library manager.</param>
  258. /// <returns>Task{BaseItem}.</returns>
  259. /// <exception cref="System.ArgumentException"></exception>
  260. protected BaseItem GetItemByName(string name, string type, ILibraryManager libraryManager)
  261. {
  262. BaseItem item;
  263. if (type.IndexOf("Person", StringComparison.OrdinalIgnoreCase) == 0)
  264. {
  265. item = GetPerson(name, libraryManager);
  266. }
  267. else if (type.IndexOf("Artist", StringComparison.OrdinalIgnoreCase) == 0)
  268. {
  269. item = GetArtist(name, libraryManager);
  270. }
  271. else if (type.IndexOf("Genre", StringComparison.OrdinalIgnoreCase) == 0)
  272. {
  273. item = GetGenre(name, libraryManager);
  274. }
  275. else if (type.IndexOf("MusicGenre", StringComparison.OrdinalIgnoreCase) == 0)
  276. {
  277. item = GetMusicGenre(name, libraryManager);
  278. }
  279. else if (type.IndexOf("GameGenre", StringComparison.OrdinalIgnoreCase) == 0)
  280. {
  281. item = GetGameGenre(name, libraryManager);
  282. }
  283. else if (type.IndexOf("Studio", StringComparison.OrdinalIgnoreCase) == 0)
  284. {
  285. item = GetStudio(name, libraryManager);
  286. }
  287. else if (type.IndexOf("Year", StringComparison.OrdinalIgnoreCase) == 0)
  288. {
  289. item = libraryManager.GetYear(int.Parse(name));
  290. }
  291. else
  292. {
  293. throw new ArgumentException();
  294. }
  295. return item;
  296. }
  297. }
  298. }