ImageByNameService.cs 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277
  1. using System;
  2. using System.Collections.Generic;
  3. using System.IO;
  4. using System.Linq;
  5. using System.Threading.Tasks;
  6. using MediaBrowser.Common.Extensions;
  7. using MediaBrowser.Controller;
  8. using MediaBrowser.Controller.Configuration;
  9. using MediaBrowser.Controller.Entities;
  10. using MediaBrowser.Controller.Net;
  11. using MediaBrowser.Model.Dto;
  12. using MediaBrowser.Model.IO;
  13. using MediaBrowser.Model.Services;
  14. using Microsoft.Extensions.Logging;
  15. namespace MediaBrowser.Api.Images
  16. {
  17. /// <summary>
  18. /// Class GetGeneralImage.
  19. /// </summary>
  20. [Route("/Images/General/{Name}/{Type}", "GET", Summary = "Gets a general image by name")]
  21. public class GetGeneralImage
  22. {
  23. /// <summary>
  24. /// Gets or sets the name.
  25. /// </summary>
  26. /// <value>The name.</value>
  27. [ApiMember(Name = "Name", Description = "The name of the image", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  28. public string Name { get; set; }
  29. [ApiMember(Name = "Type", Description = "Image Type (primary, backdrop, logo, etc).", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  30. public string Type { get; set; }
  31. }
  32. /// <summary>
  33. /// Class GetRatingImage.
  34. /// </summary>
  35. [Route("/Images/Ratings/{Theme}/{Name}", "GET", Summary = "Gets a rating image by name")]
  36. public class GetRatingImage
  37. {
  38. /// <summary>
  39. /// Gets or sets the name.
  40. /// </summary>
  41. /// <value>The name.</value>
  42. [ApiMember(Name = "Name", Description = "The name of the image", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  43. public string Name { get; set; }
  44. /// <summary>
  45. /// Gets or sets the theme.
  46. /// </summary>
  47. /// <value>The theme.</value>
  48. [ApiMember(Name = "Theme", Description = "The theme to get the image from", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  49. public string Theme { get; set; }
  50. }
  51. /// <summary>
  52. /// Class GetMediaInfoImage.
  53. /// </summary>
  54. [Route("/Images/MediaInfo/{Theme}/{Name}", "GET", Summary = "Gets a media info image by name")]
  55. public class GetMediaInfoImage
  56. {
  57. /// <summary>
  58. /// Gets or sets the name.
  59. /// </summary>
  60. /// <value>The name.</value>
  61. [ApiMember(Name = "Name", Description = "The name of the image", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  62. public string Name { get; set; }
  63. /// <summary>
  64. /// Gets or sets the theme.
  65. /// </summary>
  66. /// <value>The theme.</value>
  67. [ApiMember(Name = "Theme", Description = "The theme to get the image from", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  68. public string Theme { get; set; }
  69. }
  70. [Route("/Images/MediaInfo", "GET", Summary = "Gets all media info image by name")]
  71. [Authenticated]
  72. public class GetMediaInfoImages : IReturn<List<ImageByNameInfo>>
  73. {
  74. }
  75. [Route("/Images/Ratings", "GET", Summary = "Gets all rating images by name")]
  76. [Authenticated]
  77. public class GetRatingImages : IReturn<List<ImageByNameInfo>>
  78. {
  79. }
  80. [Route("/Images/General", "GET", Summary = "Gets all general images by name")]
  81. [Authenticated]
  82. public class GetGeneralImages : IReturn<List<ImageByNameInfo>>
  83. {
  84. }
  85. /// <summary>
  86. /// Class ImageByNameService.
  87. /// </summary>
  88. public class ImageByNameService : BaseApiService
  89. {
  90. /// <summary>
  91. /// The _app paths.
  92. /// </summary>
  93. private readonly IServerApplicationPaths _appPaths;
  94. private readonly IFileSystem _fileSystem;
  95. /// <summary>
  96. /// Initializes a new instance of the <see cref="ImageByNameService" /> class.
  97. /// </summary>
  98. public ImageByNameService(
  99. ILogger<ImageByNameService> logger,
  100. IServerConfigurationManager serverConfigurationManager,
  101. IHttpResultFactory resultFactory,
  102. IFileSystem fileSystem)
  103. : base(logger, serverConfigurationManager, resultFactory)
  104. {
  105. _appPaths = serverConfigurationManager.ApplicationPaths;
  106. _fileSystem = fileSystem;
  107. }
  108. public object Get(GetMediaInfoImages request)
  109. {
  110. return ToOptimizedResult(GetImageList(_appPaths.MediaInfoImagesPath, true));
  111. }
  112. public object Get(GetRatingImages request)
  113. {
  114. return ToOptimizedResult(GetImageList(_appPaths.RatingsPath, true));
  115. }
  116. public object Get(GetGeneralImages request)
  117. {
  118. return ToOptimizedResult(GetImageList(_appPaths.GeneralPath, false));
  119. }
  120. private List<ImageByNameInfo> GetImageList(string path, bool supportsThemes)
  121. {
  122. try
  123. {
  124. return _fileSystem.GetFiles(path, BaseItem.SupportedImageExtensions, false, true)
  125. .Select(i => new ImageByNameInfo
  126. {
  127. Name = _fileSystem.GetFileNameWithoutExtension(i),
  128. FileLength = i.Length,
  129. // For themeable images, use the Theme property
  130. // For general images, the same object structure is fine,
  131. // but it's not owned by a theme, so call it Context
  132. Theme = supportsThemes ? GetThemeName(i.FullName, path) : null,
  133. Context = supportsThemes ? null : GetThemeName(i.FullName, path),
  134. Format = i.Extension.ToLowerInvariant().TrimStart('.')
  135. })
  136. .OrderBy(i => i.Name)
  137. .ToList();
  138. }
  139. catch (IOException)
  140. {
  141. return new List<ImageByNameInfo>();
  142. }
  143. }
  144. private string GetThemeName(string path, string rootImagePath)
  145. {
  146. var parentName = Path.GetDirectoryName(path);
  147. if (string.Equals(parentName, rootImagePath, StringComparison.OrdinalIgnoreCase))
  148. {
  149. return null;
  150. }
  151. parentName = Path.GetFileName(parentName);
  152. return string.Equals(parentName, "all", StringComparison.OrdinalIgnoreCase) ?
  153. null :
  154. parentName;
  155. }
  156. /// <summary>
  157. /// Gets the specified request.
  158. /// </summary>
  159. /// <param name="request">The request.</param>
  160. /// <returns>System.Object.</returns>
  161. public Task<object> Get(GetGeneralImage request)
  162. {
  163. var filename = string.Equals(request.Type, "primary", StringComparison.OrdinalIgnoreCase)
  164. ? "folder"
  165. : request.Type;
  166. var paths = BaseItem.SupportedImageExtensions.Select(i => Path.Combine(_appPaths.GeneralPath, request.Name, filename + i)).ToList();
  167. var path = paths.FirstOrDefault(File.Exists) ?? paths.FirstOrDefault();
  168. return ResultFactory.GetStaticFileResult(Request, path);
  169. }
  170. /// <summary>
  171. /// Gets the specified request.
  172. /// </summary>
  173. /// <param name="request">The request.</param>
  174. /// <returns>System.Object.</returns>
  175. public object Get(GetRatingImage request)
  176. {
  177. var themeFolder = Path.Combine(_appPaths.RatingsPath, request.Theme);
  178. if (Directory.Exists(themeFolder))
  179. {
  180. var path = BaseItem.SupportedImageExtensions
  181. .Select(i => Path.Combine(themeFolder, request.Name + i))
  182. .FirstOrDefault(File.Exists);
  183. if (!string.IsNullOrEmpty(path))
  184. {
  185. return ResultFactory.GetStaticFileResult(Request, path);
  186. }
  187. }
  188. var allFolder = Path.Combine(_appPaths.RatingsPath, "all");
  189. if (Directory.Exists(allFolder))
  190. {
  191. // Avoid implicitly captured closure
  192. var currentRequest = request;
  193. var path = BaseItem.SupportedImageExtensions
  194. .Select(i => Path.Combine(allFolder, currentRequest.Name + i))
  195. .FirstOrDefault(File.Exists);
  196. if (!string.IsNullOrEmpty(path))
  197. {
  198. return ResultFactory.GetStaticFileResult(Request, path);
  199. }
  200. }
  201. throw new ResourceNotFoundException("MediaInfo image not found: " + request.Name);
  202. }
  203. /// <summary>
  204. /// Gets the specified request.
  205. /// </summary>
  206. /// <param name="request">The request.</param>
  207. /// <returns>System.Object.</returns>
  208. public Task<object> Get(GetMediaInfoImage request)
  209. {
  210. var themeFolder = Path.Combine(_appPaths.MediaInfoImagesPath, request.Theme);
  211. if (Directory.Exists(themeFolder))
  212. {
  213. var path = BaseItem.SupportedImageExtensions.Select(i => Path.Combine(themeFolder, request.Name + i))
  214. .FirstOrDefault(File.Exists);
  215. if (!string.IsNullOrEmpty(path))
  216. {
  217. return ResultFactory.GetStaticFileResult(Request, path);
  218. }
  219. }
  220. var allFolder = Path.Combine(_appPaths.MediaInfoImagesPath, "all");
  221. if (Directory.Exists(allFolder))
  222. {
  223. // Avoid implicitly captured closure
  224. var currentRequest = request;
  225. var path = BaseItem.SupportedImageExtensions.Select(i => Path.Combine(allFolder, currentRequest.Name + i))
  226. .FirstOrDefault(File.Exists);
  227. if (!string.IsNullOrEmpty(path))
  228. {
  229. return ResultFactory.GetStaticFileResult(Request, path);
  230. }
  231. }
  232. throw new ResourceNotFoundException("MediaInfo image not found: " + request.Name);
  233. }
  234. }
  235. }