2
0

ImageController.cs 97 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Collections.Immutable;
  4. using System.ComponentModel.DataAnnotations;
  5. using System.Diagnostics.CodeAnalysis;
  6. using System.Globalization;
  7. using System.IO;
  8. using System.Linq;
  9. using System.Net.Mime;
  10. using System.Threading;
  11. using System.Threading.Tasks;
  12. using Jellyfin.Api.Attributes;
  13. using Jellyfin.Api.Constants;
  14. using Jellyfin.Api.Helpers;
  15. using MediaBrowser.Common.Configuration;
  16. using MediaBrowser.Controller.Configuration;
  17. using MediaBrowser.Controller.Drawing;
  18. using MediaBrowser.Controller.Entities;
  19. using MediaBrowser.Controller.Library;
  20. using MediaBrowser.Controller.Providers;
  21. using MediaBrowser.Model.Branding;
  22. using MediaBrowser.Model.Drawing;
  23. using MediaBrowser.Model.Dto;
  24. using MediaBrowser.Model.Entities;
  25. using MediaBrowser.Model.IO;
  26. using MediaBrowser.Model.Net;
  27. using Microsoft.AspNetCore.Authorization;
  28. using Microsoft.AspNetCore.Http;
  29. using Microsoft.AspNetCore.Mvc;
  30. using Microsoft.Extensions.Logging;
  31. using Microsoft.Net.Http.Headers;
  32. namespace Jellyfin.Api.Controllers
  33. {
  34. /// <summary>
  35. /// Image controller.
  36. /// </summary>
  37. [Route("")]
  38. public class ImageController : BaseJellyfinApiController
  39. {
  40. private readonly IUserManager _userManager;
  41. private readonly ILibraryManager _libraryManager;
  42. private readonly IProviderManager _providerManager;
  43. private readonly IImageProcessor _imageProcessor;
  44. private readonly IFileSystem _fileSystem;
  45. private readonly ILogger<ImageController> _logger;
  46. private readonly IServerConfigurationManager _serverConfigurationManager;
  47. private readonly IApplicationPaths _appPaths;
  48. /// <summary>
  49. /// Initializes a new instance of the <see cref="ImageController"/> class.
  50. /// </summary>
  51. /// <param name="userManager">Instance of the <see cref="IUserManager"/> interface.</param>
  52. /// <param name="libraryManager">Instance of the <see cref="ILibraryManager"/> interface.</param>
  53. /// <param name="providerManager">Instance of the <see cref="IProviderManager"/> interface.</param>
  54. /// <param name="imageProcessor">Instance of the <see cref="IImageProcessor"/> interface.</param>
  55. /// <param name="fileSystem">Instance of the <see cref="IFileSystem"/> interface.</param>
  56. /// <param name="logger">Instance of the <see cref="ILogger{ImageController}"/> interface.</param>
  57. /// <param name="serverConfigurationManager">Instance of the <see cref="IServerConfigurationManager"/> interface.</param>
  58. /// <param name="appPaths">Instance of the <see cref="IApplicationPaths"/> interface.</param>
  59. public ImageController(
  60. IUserManager userManager,
  61. ILibraryManager libraryManager,
  62. IProviderManager providerManager,
  63. IImageProcessor imageProcessor,
  64. IFileSystem fileSystem,
  65. ILogger<ImageController> logger,
  66. IServerConfigurationManager serverConfigurationManager,
  67. IApplicationPaths appPaths)
  68. {
  69. _userManager = userManager;
  70. _libraryManager = libraryManager;
  71. _providerManager = providerManager;
  72. _imageProcessor = imageProcessor;
  73. _fileSystem = fileSystem;
  74. _logger = logger;
  75. _serverConfigurationManager = serverConfigurationManager;
  76. _appPaths = appPaths;
  77. }
  78. /// <summary>
  79. /// Sets the user image.
  80. /// </summary>
  81. /// <param name="userId">User Id.</param>
  82. /// <param name="imageType">(Unused) Image type.</param>
  83. /// <param name="index">(Unused) Image index.</param>
  84. /// <response code="204">Image updated.</response>
  85. /// <response code="403">User does not have permission to delete the image.</response>
  86. /// <returns>A <see cref="NoContentResult"/>.</returns>
  87. [HttpPost("Users/{userId}/Images/{imageType}")]
  88. [Authorize(Policy = Policies.DefaultAuthorization)]
  89. [AcceptsImageFile]
  90. [ProducesResponseType(StatusCodes.Status204NoContent)]
  91. [ProducesResponseType(StatusCodes.Status403Forbidden)]
  92. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "imageType", Justification = "Imported from ServiceStack")]
  93. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "index", Justification = "Imported from ServiceStack")]
  94. public async Task<ActionResult> PostUserImage(
  95. [FromRoute, Required] Guid userId,
  96. [FromRoute, Required] ImageType imageType,
  97. [FromQuery] int? index = null)
  98. {
  99. if (!RequestHelpers.AssertCanUpdateUser(_userManager, HttpContext.User, userId, true))
  100. {
  101. return StatusCode(StatusCodes.Status403Forbidden, "User is not allowed to update the image.");
  102. }
  103. var user = _userManager.GetUserById(userId);
  104. await using var memoryStream = await GetMemoryStream(Request.Body).ConfigureAwait(false);
  105. // Handle image/png; charset=utf-8
  106. var mimeType = Request.ContentType?.Split(';').FirstOrDefault();
  107. var userDataPath = Path.Combine(_serverConfigurationManager.ApplicationPaths.UserConfigurationDirectoryPath, user.Username);
  108. if (user.ProfileImage != null)
  109. {
  110. await _userManager.ClearProfileImageAsync(user).ConfigureAwait(false);
  111. }
  112. user.ProfileImage = new Data.Entities.ImageInfo(Path.Combine(userDataPath, "profile" + MimeTypes.ToExtension(mimeType ?? string.Empty)));
  113. await _providerManager
  114. .SaveImage(memoryStream, mimeType, user.ProfileImage.Path)
  115. .ConfigureAwait(false);
  116. await _userManager.UpdateUserAsync(user).ConfigureAwait(false);
  117. return NoContent();
  118. }
  119. /// <summary>
  120. /// Sets the user image.
  121. /// </summary>
  122. /// <param name="userId">User Id.</param>
  123. /// <param name="imageType">(Unused) Image type.</param>
  124. /// <param name="index">(Unused) Image index.</param>
  125. /// <response code="204">Image updated.</response>
  126. /// <response code="403">User does not have permission to delete the image.</response>
  127. /// <returns>A <see cref="NoContentResult"/>.</returns>
  128. [HttpPost("Users/{userId}/Images/{imageType}/{index}")]
  129. [Authorize(Policy = Policies.DefaultAuthorization)]
  130. [AcceptsImageFile]
  131. [ProducesResponseType(StatusCodes.Status204NoContent)]
  132. [ProducesResponseType(StatusCodes.Status403Forbidden)]
  133. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "imageType", Justification = "Imported from ServiceStack")]
  134. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "index", Justification = "Imported from ServiceStack")]
  135. public async Task<ActionResult> PostUserImageByIndex(
  136. [FromRoute, Required] Guid userId,
  137. [FromRoute, Required] ImageType imageType,
  138. [FromRoute] int index)
  139. {
  140. if (!RequestHelpers.AssertCanUpdateUser(_userManager, HttpContext.User, userId, true))
  141. {
  142. return StatusCode(StatusCodes.Status403Forbidden, "User is not allowed to update the image.");
  143. }
  144. var user = _userManager.GetUserById(userId);
  145. await using var memoryStream = await GetMemoryStream(Request.Body).ConfigureAwait(false);
  146. // Handle image/png; charset=utf-8
  147. var mimeType = Request.ContentType?.Split(';').FirstOrDefault();
  148. var userDataPath = Path.Combine(_serverConfigurationManager.ApplicationPaths.UserConfigurationDirectoryPath, user.Username);
  149. if (user.ProfileImage != null)
  150. {
  151. await _userManager.ClearProfileImageAsync(user).ConfigureAwait(false);
  152. }
  153. user.ProfileImage = new Data.Entities.ImageInfo(Path.Combine(userDataPath, "profile" + MimeTypes.ToExtension(mimeType ?? string.Empty)));
  154. await _providerManager
  155. .SaveImage(memoryStream, mimeType, user.ProfileImage.Path)
  156. .ConfigureAwait(false);
  157. await _userManager.UpdateUserAsync(user).ConfigureAwait(false);
  158. return NoContent();
  159. }
  160. /// <summary>
  161. /// Delete the user's image.
  162. /// </summary>
  163. /// <param name="userId">User Id.</param>
  164. /// <param name="imageType">(Unused) Image type.</param>
  165. /// <param name="index">(Unused) Image index.</param>
  166. /// <response code="204">Image deleted.</response>
  167. /// <response code="403">User does not have permission to delete the image.</response>
  168. /// <returns>A <see cref="NoContentResult"/>.</returns>
  169. [HttpDelete("Users/{userId}/Images/{imageType}")]
  170. [Authorize(Policy = Policies.DefaultAuthorization)]
  171. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "imageType", Justification = "Imported from ServiceStack")]
  172. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "index", Justification = "Imported from ServiceStack")]
  173. [ProducesResponseType(StatusCodes.Status204NoContent)]
  174. [ProducesResponseType(StatusCodes.Status403Forbidden)]
  175. public async Task<ActionResult> DeleteUserImage(
  176. [FromRoute, Required] Guid userId,
  177. [FromRoute, Required] ImageType imageType,
  178. [FromQuery] int? index = null)
  179. {
  180. if (!RequestHelpers.AssertCanUpdateUser(_userManager, HttpContext.User, userId, true))
  181. {
  182. return StatusCode(StatusCodes.Status403Forbidden, "User is not allowed to delete the image.");
  183. }
  184. var user = _userManager.GetUserById(userId);
  185. if (user?.ProfileImage == null)
  186. {
  187. return NoContent();
  188. }
  189. try
  190. {
  191. System.IO.File.Delete(user.ProfileImage.Path);
  192. }
  193. catch (IOException e)
  194. {
  195. _logger.LogError(e, "Error deleting user profile image:");
  196. }
  197. await _userManager.ClearProfileImageAsync(user).ConfigureAwait(false);
  198. return NoContent();
  199. }
  200. /// <summary>
  201. /// Delete the user's image.
  202. /// </summary>
  203. /// <param name="userId">User Id.</param>
  204. /// <param name="imageType">(Unused) Image type.</param>
  205. /// <param name="index">(Unused) Image index.</param>
  206. /// <response code="204">Image deleted.</response>
  207. /// <response code="403">User does not have permission to delete the image.</response>
  208. /// <returns>A <see cref="NoContentResult"/>.</returns>
  209. [HttpDelete("Users/{userId}/Images/{imageType}/{index}")]
  210. [Authorize(Policy = Policies.DefaultAuthorization)]
  211. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "imageType", Justification = "Imported from ServiceStack")]
  212. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "index", Justification = "Imported from ServiceStack")]
  213. [ProducesResponseType(StatusCodes.Status204NoContent)]
  214. [ProducesResponseType(StatusCodes.Status403Forbidden)]
  215. public async Task<ActionResult> DeleteUserImageByIndex(
  216. [FromRoute, Required] Guid userId,
  217. [FromRoute, Required] ImageType imageType,
  218. [FromRoute] int index)
  219. {
  220. if (!RequestHelpers.AssertCanUpdateUser(_userManager, HttpContext.User, userId, true))
  221. {
  222. return StatusCode(StatusCodes.Status403Forbidden, "User is not allowed to delete the image.");
  223. }
  224. var user = _userManager.GetUserById(userId);
  225. if (user?.ProfileImage == null)
  226. {
  227. return NoContent();
  228. }
  229. try
  230. {
  231. System.IO.File.Delete(user.ProfileImage.Path);
  232. }
  233. catch (IOException e)
  234. {
  235. _logger.LogError(e, "Error deleting user profile image:");
  236. }
  237. await _userManager.ClearProfileImageAsync(user).ConfigureAwait(false);
  238. return NoContent();
  239. }
  240. /// <summary>
  241. /// Delete an item's image.
  242. /// </summary>
  243. /// <param name="itemId">Item id.</param>
  244. /// <param name="imageType">Image type.</param>
  245. /// <param name="imageIndex">The image index.</param>
  246. /// <response code="204">Image deleted.</response>
  247. /// <response code="404">Item not found.</response>
  248. /// <returns>A <see cref="NoContentResult"/> on success, or a <see cref="NotFoundResult"/> if item not found.</returns>
  249. [HttpDelete("Items/{itemId}/Images/{imageType}")]
  250. [Authorize(Policy = Policies.RequiresElevation)]
  251. [ProducesResponseType(StatusCodes.Status204NoContent)]
  252. [ProducesResponseType(StatusCodes.Status404NotFound)]
  253. public async Task<ActionResult> DeleteItemImage(
  254. [FromRoute, Required] Guid itemId,
  255. [FromRoute, Required] ImageType imageType,
  256. [FromQuery] int? imageIndex)
  257. {
  258. var item = _libraryManager.GetItemById(itemId);
  259. if (item == null)
  260. {
  261. return NotFound();
  262. }
  263. await item.DeleteImageAsync(imageType, imageIndex ?? 0).ConfigureAwait(false);
  264. return NoContent();
  265. }
  266. /// <summary>
  267. /// Delete an item's image.
  268. /// </summary>
  269. /// <param name="itemId">Item id.</param>
  270. /// <param name="imageType">Image type.</param>
  271. /// <param name="imageIndex">The image index.</param>
  272. /// <response code="204">Image deleted.</response>
  273. /// <response code="404">Item not found.</response>
  274. /// <returns>A <see cref="NoContentResult"/> on success, or a <see cref="NotFoundResult"/> if item not found.</returns>
  275. [HttpDelete("Items/{itemId}/Images/{imageType}/{imageIndex}")]
  276. [Authorize(Policy = Policies.RequiresElevation)]
  277. [ProducesResponseType(StatusCodes.Status204NoContent)]
  278. [ProducesResponseType(StatusCodes.Status404NotFound)]
  279. public async Task<ActionResult> DeleteItemImageByIndex(
  280. [FromRoute, Required] Guid itemId,
  281. [FromRoute, Required] ImageType imageType,
  282. [FromRoute] int imageIndex)
  283. {
  284. var item = _libraryManager.GetItemById(itemId);
  285. if (item == null)
  286. {
  287. return NotFound();
  288. }
  289. await item.DeleteImageAsync(imageType, imageIndex).ConfigureAwait(false);
  290. return NoContent();
  291. }
  292. /// <summary>
  293. /// Set item image.
  294. /// </summary>
  295. /// <param name="itemId">Item id.</param>
  296. /// <param name="imageType">Image type.</param>
  297. /// <response code="204">Image saved.</response>
  298. /// <response code="404">Item not found.</response>
  299. /// <returns>A <see cref="NoContentResult"/> on success, or a <see cref="NotFoundResult"/> if item not found.</returns>
  300. [HttpPost("Items/{itemId}/Images/{imageType}")]
  301. [Authorize(Policy = Policies.RequiresElevation)]
  302. [AcceptsImageFile]
  303. [ProducesResponseType(StatusCodes.Status204NoContent)]
  304. [ProducesResponseType(StatusCodes.Status404NotFound)]
  305. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "index", Justification = "Imported from ServiceStack")]
  306. public async Task<ActionResult> SetItemImage(
  307. [FromRoute, Required] Guid itemId,
  308. [FromRoute, Required] ImageType imageType)
  309. {
  310. var item = _libraryManager.GetItemById(itemId);
  311. if (item == null)
  312. {
  313. return NotFound();
  314. }
  315. await using var memoryStream = await GetMemoryStream(Request.Body).ConfigureAwait(false);
  316. // Handle image/png; charset=utf-8
  317. var mimeType = Request.ContentType?.Split(';').FirstOrDefault();
  318. await _providerManager.SaveImage(item, memoryStream, mimeType, imageType, null, CancellationToken.None).ConfigureAwait(false);
  319. await item.UpdateToRepositoryAsync(ItemUpdateType.ImageUpdate, CancellationToken.None).ConfigureAwait(false);
  320. return NoContent();
  321. }
  322. /// <summary>
  323. /// Set item image.
  324. /// </summary>
  325. /// <param name="itemId">Item id.</param>
  326. /// <param name="imageType">Image type.</param>
  327. /// <param name="imageIndex">(Unused) Image index.</param>
  328. /// <response code="204">Image saved.</response>
  329. /// <response code="404">Item not found.</response>
  330. /// <returns>A <see cref="NoContentResult"/> on success, or a <see cref="NotFoundResult"/> if item not found.</returns>
  331. [HttpPost("Items/{itemId}/Images/{imageType}/{imageIndex}")]
  332. [Authorize(Policy = Policies.RequiresElevation)]
  333. [AcceptsImageFile]
  334. [ProducesResponseType(StatusCodes.Status204NoContent)]
  335. [ProducesResponseType(StatusCodes.Status404NotFound)]
  336. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "index", Justification = "Imported from ServiceStack")]
  337. public async Task<ActionResult> SetItemImageByIndex(
  338. [FromRoute, Required] Guid itemId,
  339. [FromRoute, Required] ImageType imageType,
  340. [FromRoute] int imageIndex)
  341. {
  342. var item = _libraryManager.GetItemById(itemId);
  343. if (item == null)
  344. {
  345. return NotFound();
  346. }
  347. await using var memoryStream = await GetMemoryStream(Request.Body).ConfigureAwait(false);
  348. // Handle image/png; charset=utf-8
  349. var mimeType = Request.ContentType?.Split(';').FirstOrDefault();
  350. await _providerManager.SaveImage(item, memoryStream, mimeType, imageType, null, CancellationToken.None).ConfigureAwait(false);
  351. await item.UpdateToRepositoryAsync(ItemUpdateType.ImageUpdate, CancellationToken.None).ConfigureAwait(false);
  352. return NoContent();
  353. }
  354. /// <summary>
  355. /// Updates the index for an item image.
  356. /// </summary>
  357. /// <param name="itemId">Item id.</param>
  358. /// <param name="imageType">Image type.</param>
  359. /// <param name="imageIndex">Old image index.</param>
  360. /// <param name="newIndex">New image index.</param>
  361. /// <response code="204">Image index updated.</response>
  362. /// <response code="404">Item not found.</response>
  363. /// <returns>A <see cref="NoContentResult"/> on success, or a <see cref="NotFoundResult"/> if item not found.</returns>
  364. [HttpPost("Items/{itemId}/Images/{imageType}/{imageIndex}/Index")]
  365. [Authorize(Policy = Policies.RequiresElevation)]
  366. [ProducesResponseType(StatusCodes.Status204NoContent)]
  367. [ProducesResponseType(StatusCodes.Status404NotFound)]
  368. public async Task<ActionResult> UpdateItemImageIndex(
  369. [FromRoute, Required] Guid itemId,
  370. [FromRoute, Required] ImageType imageType,
  371. [FromRoute, Required] int imageIndex,
  372. [FromQuery, Required] int newIndex)
  373. {
  374. var item = _libraryManager.GetItemById(itemId);
  375. if (item == null)
  376. {
  377. return NotFound();
  378. }
  379. await item.SwapImagesAsync(imageType, imageIndex, newIndex).ConfigureAwait(false);
  380. return NoContent();
  381. }
  382. /// <summary>
  383. /// Get item image infos.
  384. /// </summary>
  385. /// <param name="itemId">Item id.</param>
  386. /// <response code="200">Item images returned.</response>
  387. /// <response code="404">Item not found.</response>
  388. /// <returns>The list of image infos on success, or <see cref="NotFoundResult"/> if item not found.</returns>
  389. [HttpGet("Items/{itemId}/Images")]
  390. [Authorize(Policy = Policies.DefaultAuthorization)]
  391. [ProducesResponseType(StatusCodes.Status200OK)]
  392. [ProducesResponseType(StatusCodes.Status404NotFound)]
  393. public async Task<ActionResult<IEnumerable<ImageInfo>>> GetItemImageInfos([FromRoute, Required] Guid itemId)
  394. {
  395. var item = _libraryManager.GetItemById(itemId);
  396. if (item == null)
  397. {
  398. return NotFound();
  399. }
  400. var list = new List<ImageInfo>();
  401. var itemImages = item.ImageInfos;
  402. if (itemImages.Length == 0)
  403. {
  404. // short-circuit
  405. return list;
  406. }
  407. await _libraryManager.UpdateImagesAsync(item).ConfigureAwait(false); // this makes sure dimensions and hashes are correct
  408. foreach (var image in itemImages)
  409. {
  410. if (!item.AllowsMultipleImages(image.Type))
  411. {
  412. var info = GetImageInfo(item, image, null);
  413. if (info != null)
  414. {
  415. list.Add(info);
  416. }
  417. }
  418. }
  419. foreach (var imageType in itemImages.Select(i => i.Type).Distinct().Where(item.AllowsMultipleImages))
  420. {
  421. var index = 0;
  422. // Prevent implicitly captured closure
  423. var currentImageType = imageType;
  424. foreach (var image in itemImages.Where(i => i.Type == currentImageType))
  425. {
  426. var info = GetImageInfo(item, image, index);
  427. if (info != null)
  428. {
  429. list.Add(info);
  430. }
  431. index++;
  432. }
  433. }
  434. return list;
  435. }
  436. /// <summary>
  437. /// Gets the item's image.
  438. /// </summary>
  439. /// <param name="itemId">Item id.</param>
  440. /// <param name="imageType">Image type.</param>
  441. /// <param name="maxWidth">The maximum image width to return.</param>
  442. /// <param name="maxHeight">The maximum image height to return.</param>
  443. /// <param name="width">The fixed image width to return.</param>
  444. /// <param name="height">The fixed image height to return.</param>
  445. /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>
  446. /// <param name="fillWidth">Width of box to fill.</param>
  447. /// <param name="fillHeight">Height of box to fill.</param>
  448. /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>
  449. /// <param name="cropWhitespace">Optional. Specify if whitespace should be cropped out of the image. True/False. If unspecified, whitespace will be cropped from logos and clear art.</param>
  450. /// <param name="format">Optional. The <see cref="ImageFormat"/> of the returned image.</param>
  451. /// <param name="addPlayedIndicator">Optional. Add a played indicator.</param>
  452. /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>
  453. /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>
  454. /// <param name="blur">Optional. Blur image.</param>
  455. /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>
  456. /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>
  457. /// <param name="imageIndex">Image index.</param>
  458. /// <response code="200">Image stream returned.</response>
  459. /// <response code="404">Item not found.</response>
  460. /// <returns>
  461. /// A <see cref="FileStreamResult"/> containing the file stream on success,
  462. /// or a <see cref="NotFoundResult"/> if item not found.
  463. /// </returns>
  464. [HttpGet("Items/{itemId}/Images/{imageType}")]
  465. [HttpHead("Items/{itemId}/Images/{imageType}", Name = "HeadItemImage")]
  466. [ProducesResponseType(StatusCodes.Status200OK)]
  467. [ProducesResponseType(StatusCodes.Status404NotFound)]
  468. [ProducesImageFile]
  469. public async Task<ActionResult> GetItemImage(
  470. [FromRoute, Required] Guid itemId,
  471. [FromRoute, Required] ImageType imageType,
  472. [FromQuery] int? maxWidth,
  473. [FromQuery] int? maxHeight,
  474. [FromQuery] int? width,
  475. [FromQuery] int? height,
  476. [FromQuery] int? quality,
  477. [FromQuery] int? fillWidth,
  478. [FromQuery] int? fillHeight,
  479. [FromQuery] string? tag,
  480. [FromQuery, ParameterObsolete] bool? cropWhitespace,
  481. [FromQuery] ImageFormat? format,
  482. [FromQuery] bool? addPlayedIndicator,
  483. [FromQuery] double? percentPlayed,
  484. [FromQuery] int? unplayedCount,
  485. [FromQuery] int? blur,
  486. [FromQuery] string? backgroundColor,
  487. [FromQuery] string? foregroundLayer,
  488. [FromQuery] int? imageIndex)
  489. {
  490. var item = _libraryManager.GetItemById(itemId);
  491. if (item == null)
  492. {
  493. return NotFound();
  494. }
  495. return await GetImageInternal(
  496. itemId,
  497. imageType,
  498. imageIndex,
  499. tag,
  500. format,
  501. maxWidth,
  502. maxHeight,
  503. percentPlayed,
  504. unplayedCount,
  505. width,
  506. height,
  507. quality,
  508. fillWidth,
  509. fillHeight,
  510. addPlayedIndicator,
  511. blur,
  512. backgroundColor,
  513. foregroundLayer,
  514. item)
  515. .ConfigureAwait(false);
  516. }
  517. /// <summary>
  518. /// Gets the item's image.
  519. /// </summary>
  520. /// <param name="itemId">Item id.</param>
  521. /// <param name="imageType">Image type.</param>
  522. /// <param name="imageIndex">Image index.</param>
  523. /// <param name="maxWidth">The maximum image width to return.</param>
  524. /// <param name="maxHeight">The maximum image height to return.</param>
  525. /// <param name="width">The fixed image width to return.</param>
  526. /// <param name="height">The fixed image height to return.</param>
  527. /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>
  528. /// <param name="fillWidth">Width of box to fill.</param>
  529. /// <param name="fillHeight">Height of box to fill.</param>
  530. /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>
  531. /// <param name="cropWhitespace">Optional. Specify if whitespace should be cropped out of the image. True/False. If unspecified, whitespace will be cropped from logos and clear art.</param>
  532. /// <param name="format">Optional. The <see cref="ImageFormat"/> of the returned image.</param>
  533. /// <param name="addPlayedIndicator">Optional. Add a played indicator.</param>
  534. /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>
  535. /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>
  536. /// <param name="blur">Optional. Blur image.</param>
  537. /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>
  538. /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>
  539. /// <response code="200">Image stream returned.</response>
  540. /// <response code="404">Item not found.</response>
  541. /// <returns>
  542. /// A <see cref="FileStreamResult"/> containing the file stream on success,
  543. /// or a <see cref="NotFoundResult"/> if item not found.
  544. /// </returns>
  545. [HttpGet("Items/{itemId}/Images/{imageType}/{imageIndex}")]
  546. [HttpHead("Items/{itemId}/Images/{imageType}/{imageIndex}", Name = "HeadItemImageByIndex")]
  547. [ProducesResponseType(StatusCodes.Status200OK)]
  548. [ProducesResponseType(StatusCodes.Status404NotFound)]
  549. [ProducesImageFile]
  550. public async Task<ActionResult> GetItemImageByIndex(
  551. [FromRoute, Required] Guid itemId,
  552. [FromRoute, Required] ImageType imageType,
  553. [FromRoute] int imageIndex,
  554. [FromQuery] int? maxWidth,
  555. [FromQuery] int? maxHeight,
  556. [FromQuery] int? width,
  557. [FromQuery] int? height,
  558. [FromQuery] int? quality,
  559. [FromQuery] int? fillWidth,
  560. [FromQuery] int? fillHeight,
  561. [FromQuery] string? tag,
  562. [FromQuery, ParameterObsolete] bool? cropWhitespace,
  563. [FromQuery] ImageFormat? format,
  564. [FromQuery] bool? addPlayedIndicator,
  565. [FromQuery] double? percentPlayed,
  566. [FromQuery] int? unplayedCount,
  567. [FromQuery] int? blur,
  568. [FromQuery] string? backgroundColor,
  569. [FromQuery] string? foregroundLayer)
  570. {
  571. var item = _libraryManager.GetItemById(itemId);
  572. if (item == null)
  573. {
  574. return NotFound();
  575. }
  576. return await GetImageInternal(
  577. itemId,
  578. imageType,
  579. imageIndex,
  580. tag,
  581. format,
  582. maxWidth,
  583. maxHeight,
  584. percentPlayed,
  585. unplayedCount,
  586. width,
  587. height,
  588. quality,
  589. fillWidth,
  590. fillHeight,
  591. addPlayedIndicator,
  592. blur,
  593. backgroundColor,
  594. foregroundLayer,
  595. item)
  596. .ConfigureAwait(false);
  597. }
  598. /// <summary>
  599. /// Gets the item's image.
  600. /// </summary>
  601. /// <param name="itemId">Item id.</param>
  602. /// <param name="imageType">Image type.</param>
  603. /// <param name="maxWidth">The maximum image width to return.</param>
  604. /// <param name="maxHeight">The maximum image height to return.</param>
  605. /// <param name="width">The fixed image width to return.</param>
  606. /// <param name="height">The fixed image height to return.</param>
  607. /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>
  608. /// <param name="fillWidth">Width of box to fill.</param>
  609. /// <param name="fillHeight">Height of box to fill.</param>
  610. /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>
  611. /// <param name="cropWhitespace">Optional. Specify if whitespace should be cropped out of the image. True/False. If unspecified, whitespace will be cropped from logos and clear art.</param>
  612. /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>
  613. /// <param name="addPlayedIndicator">Optional. Add a played indicator.</param>
  614. /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>
  615. /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>
  616. /// <param name="blur">Optional. Blur image.</param>
  617. /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>
  618. /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>
  619. /// <param name="imageIndex">Image index.</param>
  620. /// <response code="200">Image stream returned.</response>
  621. /// <response code="404">Item not found.</response>
  622. /// <returns>
  623. /// A <see cref="FileStreamResult"/> containing the file stream on success,
  624. /// or a <see cref="NotFoundResult"/> if item not found.
  625. /// </returns>
  626. [HttpGet("Items/{itemId}/Images/{imageType}/{imageIndex}/{tag}/{format}/{maxWidth}/{maxHeight}/{percentPlayed}/{unplayedCount}")]
  627. [HttpHead("Items/{itemId}/Images/{imageType}/{imageIndex}/{tag}/{format}/{maxWidth}/{maxHeight}/{percentPlayed}/{unplayedCount}", Name = "HeadItemImage2")]
  628. [ProducesResponseType(StatusCodes.Status200OK)]
  629. [ProducesResponseType(StatusCodes.Status404NotFound)]
  630. [ProducesImageFile]
  631. public async Task<ActionResult> GetItemImage2(
  632. [FromRoute, Required] Guid itemId,
  633. [FromRoute, Required] ImageType imageType,
  634. [FromRoute, Required] int maxWidth,
  635. [FromRoute, Required] int maxHeight,
  636. [FromQuery] int? width,
  637. [FromQuery] int? height,
  638. [FromQuery] int? quality,
  639. [FromQuery] int? fillWidth,
  640. [FromQuery] int? fillHeight,
  641. [FromRoute, Required] string tag,
  642. [FromQuery, ParameterObsolete] bool? cropWhitespace,
  643. [FromRoute, Required] ImageFormat format,
  644. [FromQuery] bool? addPlayedIndicator,
  645. [FromRoute, Required] double percentPlayed,
  646. [FromRoute, Required] int unplayedCount,
  647. [FromQuery] int? blur,
  648. [FromQuery] string? backgroundColor,
  649. [FromQuery] string? foregroundLayer,
  650. [FromRoute, Required] int imageIndex)
  651. {
  652. var item = _libraryManager.GetItemById(itemId);
  653. if (item == null)
  654. {
  655. return NotFound();
  656. }
  657. return await GetImageInternal(
  658. itemId,
  659. imageType,
  660. imageIndex,
  661. tag,
  662. format,
  663. maxWidth,
  664. maxHeight,
  665. percentPlayed,
  666. unplayedCount,
  667. width,
  668. height,
  669. quality,
  670. fillWidth,
  671. fillHeight,
  672. addPlayedIndicator,
  673. blur,
  674. backgroundColor,
  675. foregroundLayer,
  676. item)
  677. .ConfigureAwait(false);
  678. }
  679. /// <summary>
  680. /// Get artist image by name.
  681. /// </summary>
  682. /// <param name="name">Artist name.</param>
  683. /// <param name="imageType">Image type.</param>
  684. /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>
  685. /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>
  686. /// <param name="maxWidth">The maximum image width to return.</param>
  687. /// <param name="maxHeight">The maximum image height to return.</param>
  688. /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>
  689. /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>
  690. /// <param name="width">The fixed image width to return.</param>
  691. /// <param name="height">The fixed image height to return.</param>
  692. /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>
  693. /// <param name="fillWidth">Width of box to fill.</param>
  694. /// <param name="fillHeight">Height of box to fill.</param>
  695. /// <param name="cropWhitespace">Optional. Specify if whitespace should be cropped out of the image. True/False. If unspecified, whitespace will be cropped from logos and clear art.</param>
  696. /// <param name="addPlayedIndicator">Optional. Add a played indicator.</param>
  697. /// <param name="blur">Optional. Blur image.</param>
  698. /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>
  699. /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>
  700. /// <param name="imageIndex">Image index.</param>
  701. /// <response code="200">Image stream returned.</response>
  702. /// <response code="404">Item not found.</response>
  703. /// <returns>
  704. /// A <see cref="FileStreamResult"/> containing the file stream on success,
  705. /// or a <see cref="NotFoundResult"/> if item not found.
  706. /// </returns>
  707. [HttpGet("Artists/{name}/Images/{imageType}/{imageIndex}")]
  708. [HttpHead("Artists/{name}/Images/{imageType}/{imageIndex}", Name = "HeadArtistImage")]
  709. [ProducesResponseType(StatusCodes.Status200OK)]
  710. [ProducesResponseType(StatusCodes.Status404NotFound)]
  711. [ProducesImageFile]
  712. public async Task<ActionResult> GetArtistImage(
  713. [FromRoute, Required] string name,
  714. [FromRoute, Required] ImageType imageType,
  715. [FromQuery] string? tag,
  716. [FromQuery] ImageFormat? format,
  717. [FromQuery] int? maxWidth,
  718. [FromQuery] int? maxHeight,
  719. [FromQuery] double? percentPlayed,
  720. [FromQuery] int? unplayedCount,
  721. [FromQuery] int? width,
  722. [FromQuery] int? height,
  723. [FromQuery] int? quality,
  724. [FromQuery] int? fillWidth,
  725. [FromQuery] int? fillHeight,
  726. [FromQuery, ParameterObsolete] bool? cropWhitespace,
  727. [FromQuery] bool? addPlayedIndicator,
  728. [FromQuery] int? blur,
  729. [FromQuery] string? backgroundColor,
  730. [FromQuery] string? foregroundLayer,
  731. [FromRoute, Required] int imageIndex)
  732. {
  733. var item = _libraryManager.GetArtist(name);
  734. if (item == null)
  735. {
  736. return NotFound();
  737. }
  738. return await GetImageInternal(
  739. item.Id,
  740. imageType,
  741. imageIndex,
  742. tag,
  743. format,
  744. maxWidth,
  745. maxHeight,
  746. percentPlayed,
  747. unplayedCount,
  748. width,
  749. height,
  750. quality,
  751. fillWidth,
  752. fillHeight,
  753. addPlayedIndicator,
  754. blur,
  755. backgroundColor,
  756. foregroundLayer,
  757. item)
  758. .ConfigureAwait(false);
  759. }
  760. /// <summary>
  761. /// Get genre image by name.
  762. /// </summary>
  763. /// <param name="name">Genre name.</param>
  764. /// <param name="imageType">Image type.</param>
  765. /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>
  766. /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>
  767. /// <param name="maxWidth">The maximum image width to return.</param>
  768. /// <param name="maxHeight">The maximum image height to return.</param>
  769. /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>
  770. /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>
  771. /// <param name="width">The fixed image width to return.</param>
  772. /// <param name="height">The fixed image height to return.</param>
  773. /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>
  774. /// <param name="fillWidth">Width of box to fill.</param>
  775. /// <param name="fillHeight">Height of box to fill.</param>
  776. /// <param name="cropWhitespace">Optional. Specify if whitespace should be cropped out of the image. True/False. If unspecified, whitespace will be cropped from logos and clear art.</param>
  777. /// <param name="addPlayedIndicator">Optional. Add a played indicator.</param>
  778. /// <param name="blur">Optional. Blur image.</param>
  779. /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>
  780. /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>
  781. /// <param name="imageIndex">Image index.</param>
  782. /// <response code="200">Image stream returned.</response>
  783. /// <response code="404">Item not found.</response>
  784. /// <returns>
  785. /// A <see cref="FileStreamResult"/> containing the file stream on success,
  786. /// or a <see cref="NotFoundResult"/> if item not found.
  787. /// </returns>
  788. [HttpGet("Genres/{name}/Images/{imageType}")]
  789. [HttpHead("Genres/{name}/Images/{imageType}", Name = "HeadGenreImage")]
  790. [ProducesResponseType(StatusCodes.Status200OK)]
  791. [ProducesResponseType(StatusCodes.Status404NotFound)]
  792. [ProducesImageFile]
  793. public async Task<ActionResult> GetGenreImage(
  794. [FromRoute, Required] string name,
  795. [FromRoute, Required] ImageType imageType,
  796. [FromQuery] string? tag,
  797. [FromQuery] ImageFormat? format,
  798. [FromQuery] int? maxWidth,
  799. [FromQuery] int? maxHeight,
  800. [FromQuery] double? percentPlayed,
  801. [FromQuery] int? unplayedCount,
  802. [FromQuery] int? width,
  803. [FromQuery] int? height,
  804. [FromQuery] int? quality,
  805. [FromQuery] int? fillWidth,
  806. [FromQuery] int? fillHeight,
  807. [FromQuery, ParameterObsolete] bool? cropWhitespace,
  808. [FromQuery] bool? addPlayedIndicator,
  809. [FromQuery] int? blur,
  810. [FromQuery] string? backgroundColor,
  811. [FromQuery] string? foregroundLayer,
  812. [FromQuery] int? imageIndex)
  813. {
  814. var item = _libraryManager.GetGenre(name);
  815. if (item == null)
  816. {
  817. return NotFound();
  818. }
  819. return await GetImageInternal(
  820. item.Id,
  821. imageType,
  822. imageIndex,
  823. tag,
  824. format,
  825. maxWidth,
  826. maxHeight,
  827. percentPlayed,
  828. unplayedCount,
  829. width,
  830. height,
  831. quality,
  832. fillWidth,
  833. fillHeight,
  834. addPlayedIndicator,
  835. blur,
  836. backgroundColor,
  837. foregroundLayer,
  838. item)
  839. .ConfigureAwait(false);
  840. }
  841. /// <summary>
  842. /// Get genre image by name.
  843. /// </summary>
  844. /// <param name="name">Genre name.</param>
  845. /// <param name="imageType">Image type.</param>
  846. /// <param name="imageIndex">Image index.</param>
  847. /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>
  848. /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>
  849. /// <param name="maxWidth">The maximum image width to return.</param>
  850. /// <param name="maxHeight">The maximum image height to return.</param>
  851. /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>
  852. /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>
  853. /// <param name="width">The fixed image width to return.</param>
  854. /// <param name="height">The fixed image height to return.</param>
  855. /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>
  856. /// <param name="fillWidth">Width of box to fill.</param>
  857. /// <param name="fillHeight">Height of box to fill.</param>
  858. /// <param name="cropWhitespace">Optional. Specify if whitespace should be cropped out of the image. True/False. If unspecified, whitespace will be cropped from logos and clear art.</param>
  859. /// <param name="addPlayedIndicator">Optional. Add a played indicator.</param>
  860. /// <param name="blur">Optional. Blur image.</param>
  861. /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>
  862. /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>
  863. /// <response code="200">Image stream returned.</response>
  864. /// <response code="404">Item not found.</response>
  865. /// <returns>
  866. /// A <see cref="FileStreamResult"/> containing the file stream on success,
  867. /// or a <see cref="NotFoundResult"/> if item not found.
  868. /// </returns>
  869. [HttpGet("Genres/{name}/Images/{imageType}/{imageIndex}")]
  870. [HttpHead("Genres/{name}/Images/{imageType}/{imageIndex}", Name = "HeadGenreImageByIndex")]
  871. [ProducesResponseType(StatusCodes.Status200OK)]
  872. [ProducesResponseType(StatusCodes.Status404NotFound)]
  873. [ProducesImageFile]
  874. public async Task<ActionResult> GetGenreImageByIndex(
  875. [FromRoute, Required] string name,
  876. [FromRoute, Required] ImageType imageType,
  877. [FromRoute, Required] int imageIndex,
  878. [FromQuery] string? tag,
  879. [FromQuery] ImageFormat? format,
  880. [FromQuery] int? maxWidth,
  881. [FromQuery] int? maxHeight,
  882. [FromQuery] double? percentPlayed,
  883. [FromQuery] int? unplayedCount,
  884. [FromQuery] int? width,
  885. [FromQuery] int? height,
  886. [FromQuery] int? quality,
  887. [FromQuery] int? fillWidth,
  888. [FromQuery] int? fillHeight,
  889. [FromQuery, ParameterObsolete] bool? cropWhitespace,
  890. [FromQuery] bool? addPlayedIndicator,
  891. [FromQuery] int? blur,
  892. [FromQuery] string? backgroundColor,
  893. [FromQuery] string? foregroundLayer)
  894. {
  895. var item = _libraryManager.GetGenre(name);
  896. if (item == null)
  897. {
  898. return NotFound();
  899. }
  900. return await GetImageInternal(
  901. item.Id,
  902. imageType,
  903. imageIndex,
  904. tag,
  905. format,
  906. maxWidth,
  907. maxHeight,
  908. percentPlayed,
  909. unplayedCount,
  910. width,
  911. height,
  912. quality,
  913. fillWidth,
  914. fillHeight,
  915. addPlayedIndicator,
  916. blur,
  917. backgroundColor,
  918. foregroundLayer,
  919. item)
  920. .ConfigureAwait(false);
  921. }
  922. /// <summary>
  923. /// Get music genre image by name.
  924. /// </summary>
  925. /// <param name="name">Music genre name.</param>
  926. /// <param name="imageType">Image type.</param>
  927. /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>
  928. /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>
  929. /// <param name="maxWidth">The maximum image width to return.</param>
  930. /// <param name="maxHeight">The maximum image height to return.</param>
  931. /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>
  932. /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>
  933. /// <param name="width">The fixed image width to return.</param>
  934. /// <param name="height">The fixed image height to return.</param>
  935. /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>
  936. /// <param name="fillWidth">Width of box to fill.</param>
  937. /// <param name="fillHeight">Height of box to fill.</param>
  938. /// <param name="cropWhitespace">Optional. Specify if whitespace should be cropped out of the image. True/False. If unspecified, whitespace will be cropped from logos and clear art.</param>
  939. /// <param name="addPlayedIndicator">Optional. Add a played indicator.</param>
  940. /// <param name="blur">Optional. Blur image.</param>
  941. /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>
  942. /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>
  943. /// <param name="imageIndex">Image index.</param>
  944. /// <response code="200">Image stream returned.</response>
  945. /// <response code="404">Item not found.</response>
  946. /// <returns>
  947. /// A <see cref="FileStreamResult"/> containing the file stream on success,
  948. /// or a <see cref="NotFoundResult"/> if item not found.
  949. /// </returns>
  950. [HttpGet("MusicGenres/{name}/Images/{imageType}")]
  951. [HttpHead("MusicGenres/{name}/Images/{imageType}", Name = "HeadMusicGenreImage")]
  952. [ProducesResponseType(StatusCodes.Status200OK)]
  953. [ProducesResponseType(StatusCodes.Status404NotFound)]
  954. [ProducesImageFile]
  955. public async Task<ActionResult> GetMusicGenreImage(
  956. [FromRoute, Required] string name,
  957. [FromRoute, Required] ImageType imageType,
  958. [FromQuery] string? tag,
  959. [FromQuery] ImageFormat? format,
  960. [FromQuery] int? maxWidth,
  961. [FromQuery] int? maxHeight,
  962. [FromQuery] double? percentPlayed,
  963. [FromQuery] int? unplayedCount,
  964. [FromQuery] int? width,
  965. [FromQuery] int? height,
  966. [FromQuery] int? quality,
  967. [FromQuery] int? fillWidth,
  968. [FromQuery] int? fillHeight,
  969. [FromQuery, ParameterObsolete] bool? cropWhitespace,
  970. [FromQuery] bool? addPlayedIndicator,
  971. [FromQuery] int? blur,
  972. [FromQuery] string? backgroundColor,
  973. [FromQuery] string? foregroundLayer,
  974. [FromQuery] int? imageIndex)
  975. {
  976. var item = _libraryManager.GetMusicGenre(name);
  977. if (item == null)
  978. {
  979. return NotFound();
  980. }
  981. return await GetImageInternal(
  982. item.Id,
  983. imageType,
  984. imageIndex,
  985. tag,
  986. format,
  987. maxWidth,
  988. maxHeight,
  989. percentPlayed,
  990. unplayedCount,
  991. width,
  992. height,
  993. quality,
  994. fillWidth,
  995. fillHeight,
  996. addPlayedIndicator,
  997. blur,
  998. backgroundColor,
  999. foregroundLayer,
  1000. item)
  1001. .ConfigureAwait(false);
  1002. }
  1003. /// <summary>
  1004. /// Get music genre image by name.
  1005. /// </summary>
  1006. /// <param name="name">Music genre name.</param>
  1007. /// <param name="imageType">Image type.</param>
  1008. /// <param name="imageIndex">Image index.</param>
  1009. /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>
  1010. /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>
  1011. /// <param name="maxWidth">The maximum image width to return.</param>
  1012. /// <param name="maxHeight">The maximum image height to return.</param>
  1013. /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>
  1014. /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>
  1015. /// <param name="width">The fixed image width to return.</param>
  1016. /// <param name="height">The fixed image height to return.</param>
  1017. /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>
  1018. /// <param name="fillWidth">Width of box to fill.</param>
  1019. /// <param name="fillHeight">Height of box to fill.</param>
  1020. /// <param name="cropWhitespace">Optional. Specify if whitespace should be cropped out of the image. True/False. If unspecified, whitespace will be cropped from logos and clear art.</param>
  1021. /// <param name="addPlayedIndicator">Optional. Add a played indicator.</param>
  1022. /// <param name="blur">Optional. Blur image.</param>
  1023. /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>
  1024. /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>
  1025. /// <response code="200">Image stream returned.</response>
  1026. /// <response code="404">Item not found.</response>
  1027. /// <returns>
  1028. /// A <see cref="FileStreamResult"/> containing the file stream on success,
  1029. /// or a <see cref="NotFoundResult"/> if item not found.
  1030. /// </returns>
  1031. [HttpGet("MusicGenres/{name}/Images/{imageType}/{imageIndex}")]
  1032. [HttpHead("MusicGenres/{name}/Images/{imageType}/{imageIndex}", Name = "HeadMusicGenreImageByIndex")]
  1033. [ProducesResponseType(StatusCodes.Status200OK)]
  1034. [ProducesResponseType(StatusCodes.Status404NotFound)]
  1035. [ProducesImageFile]
  1036. public async Task<ActionResult> GetMusicGenreImageByIndex(
  1037. [FromRoute, Required] string name,
  1038. [FromRoute, Required] ImageType imageType,
  1039. [FromRoute, Required] int imageIndex,
  1040. [FromQuery] string? tag,
  1041. [FromQuery] ImageFormat? format,
  1042. [FromQuery] int? maxWidth,
  1043. [FromQuery] int? maxHeight,
  1044. [FromQuery] double? percentPlayed,
  1045. [FromQuery] int? unplayedCount,
  1046. [FromQuery] int? width,
  1047. [FromQuery] int? height,
  1048. [FromQuery] int? quality,
  1049. [FromQuery] int? fillWidth,
  1050. [FromQuery] int? fillHeight,
  1051. [FromQuery, ParameterObsolete] bool? cropWhitespace,
  1052. [FromQuery] bool? addPlayedIndicator,
  1053. [FromQuery] int? blur,
  1054. [FromQuery] string? backgroundColor,
  1055. [FromQuery] string? foregroundLayer)
  1056. {
  1057. var item = _libraryManager.GetMusicGenre(name);
  1058. if (item == null)
  1059. {
  1060. return NotFound();
  1061. }
  1062. return await GetImageInternal(
  1063. item.Id,
  1064. imageType,
  1065. imageIndex,
  1066. tag,
  1067. format,
  1068. maxWidth,
  1069. maxHeight,
  1070. percentPlayed,
  1071. unplayedCount,
  1072. width,
  1073. height,
  1074. quality,
  1075. fillWidth,
  1076. fillHeight,
  1077. addPlayedIndicator,
  1078. blur,
  1079. backgroundColor,
  1080. foregroundLayer,
  1081. item)
  1082. .ConfigureAwait(false);
  1083. }
  1084. /// <summary>
  1085. /// Get person image by name.
  1086. /// </summary>
  1087. /// <param name="name">Person name.</param>
  1088. /// <param name="imageType">Image type.</param>
  1089. /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>
  1090. /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>
  1091. /// <param name="maxWidth">The maximum image width to return.</param>
  1092. /// <param name="maxHeight">The maximum image height to return.</param>
  1093. /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>
  1094. /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>
  1095. /// <param name="width">The fixed image width to return.</param>
  1096. /// <param name="height">The fixed image height to return.</param>
  1097. /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>
  1098. /// <param name="fillWidth">Width of box to fill.</param>
  1099. /// <param name="fillHeight">Height of box to fill.</param>
  1100. /// <param name="cropWhitespace">Optional. Specify if whitespace should be cropped out of the image. True/False. If unspecified, whitespace will be cropped from logos and clear art.</param>
  1101. /// <param name="addPlayedIndicator">Optional. Add a played indicator.</param>
  1102. /// <param name="blur">Optional. Blur image.</param>
  1103. /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>
  1104. /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>
  1105. /// <param name="imageIndex">Image index.</param>
  1106. /// <response code="200">Image stream returned.</response>
  1107. /// <response code="404">Item not found.</response>
  1108. /// <returns>
  1109. /// A <see cref="FileStreamResult"/> containing the file stream on success,
  1110. /// or a <see cref="NotFoundResult"/> if item not found.
  1111. /// </returns>
  1112. [HttpGet("Persons/{name}/Images/{imageType}")]
  1113. [HttpHead("Persons/{name}/Images/{imageType}", Name = "HeadPersonImage")]
  1114. [ProducesResponseType(StatusCodes.Status200OK)]
  1115. [ProducesResponseType(StatusCodes.Status404NotFound)]
  1116. [ProducesImageFile]
  1117. public async Task<ActionResult> GetPersonImage(
  1118. [FromRoute, Required] string name,
  1119. [FromRoute, Required] ImageType imageType,
  1120. [FromQuery] string? tag,
  1121. [FromQuery] ImageFormat? format,
  1122. [FromQuery] int? maxWidth,
  1123. [FromQuery] int? maxHeight,
  1124. [FromQuery] double? percentPlayed,
  1125. [FromQuery] int? unplayedCount,
  1126. [FromQuery] int? width,
  1127. [FromQuery] int? height,
  1128. [FromQuery] int? quality,
  1129. [FromQuery] int? fillWidth,
  1130. [FromQuery] int? fillHeight,
  1131. [FromQuery, ParameterObsolete] bool? cropWhitespace,
  1132. [FromQuery] bool? addPlayedIndicator,
  1133. [FromQuery] int? blur,
  1134. [FromQuery] string? backgroundColor,
  1135. [FromQuery] string? foregroundLayer,
  1136. [FromQuery] int? imageIndex)
  1137. {
  1138. var item = _libraryManager.GetPerson(name);
  1139. if (item == null)
  1140. {
  1141. return NotFound();
  1142. }
  1143. return await GetImageInternal(
  1144. item.Id,
  1145. imageType,
  1146. imageIndex,
  1147. tag,
  1148. format,
  1149. maxWidth,
  1150. maxHeight,
  1151. percentPlayed,
  1152. unplayedCount,
  1153. width,
  1154. height,
  1155. quality,
  1156. fillWidth,
  1157. fillHeight,
  1158. addPlayedIndicator,
  1159. blur,
  1160. backgroundColor,
  1161. foregroundLayer,
  1162. item)
  1163. .ConfigureAwait(false);
  1164. }
  1165. /// <summary>
  1166. /// Get person image by name.
  1167. /// </summary>
  1168. /// <param name="name">Person name.</param>
  1169. /// <param name="imageType">Image type.</param>
  1170. /// <param name="imageIndex">Image index.</param>
  1171. /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>
  1172. /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>
  1173. /// <param name="maxWidth">The maximum image width to return.</param>
  1174. /// <param name="maxHeight">The maximum image height to return.</param>
  1175. /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>
  1176. /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>
  1177. /// <param name="width">The fixed image width to return.</param>
  1178. /// <param name="height">The fixed image height to return.</param>
  1179. /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>
  1180. /// <param name="fillWidth">Width of box to fill.</param>
  1181. /// <param name="fillHeight">Height of box to fill.</param>
  1182. /// <param name="cropWhitespace">Optional. Specify if whitespace should be cropped out of the image. True/False. If unspecified, whitespace will be cropped from logos and clear art.</param>
  1183. /// <param name="addPlayedIndicator">Optional. Add a played indicator.</param>
  1184. /// <param name="blur">Optional. Blur image.</param>
  1185. /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>
  1186. /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>
  1187. /// <response code="200">Image stream returned.</response>
  1188. /// <response code="404">Item not found.</response>
  1189. /// <returns>
  1190. /// A <see cref="FileStreamResult"/> containing the file stream on success,
  1191. /// or a <see cref="NotFoundResult"/> if item not found.
  1192. /// </returns>
  1193. [HttpGet("Persons/{name}/Images/{imageType}/{imageIndex}")]
  1194. [HttpHead("Persons/{name}/Images/{imageType}/{imageIndex}", Name = "HeadPersonImageByIndex")]
  1195. [ProducesResponseType(StatusCodes.Status200OK)]
  1196. [ProducesResponseType(StatusCodes.Status404NotFound)]
  1197. [ProducesImageFile]
  1198. public async Task<ActionResult> GetPersonImageByIndex(
  1199. [FromRoute, Required] string name,
  1200. [FromRoute, Required] ImageType imageType,
  1201. [FromRoute, Required] int imageIndex,
  1202. [FromQuery] string? tag,
  1203. [FromQuery] ImageFormat? format,
  1204. [FromQuery] int? maxWidth,
  1205. [FromQuery] int? maxHeight,
  1206. [FromQuery] double? percentPlayed,
  1207. [FromQuery] int? unplayedCount,
  1208. [FromQuery] int? width,
  1209. [FromQuery] int? height,
  1210. [FromQuery] int? quality,
  1211. [FromQuery] int? fillWidth,
  1212. [FromQuery] int? fillHeight,
  1213. [FromQuery, ParameterObsolete] bool? cropWhitespace,
  1214. [FromQuery] bool? addPlayedIndicator,
  1215. [FromQuery] int? blur,
  1216. [FromQuery] string? backgroundColor,
  1217. [FromQuery] string? foregroundLayer)
  1218. {
  1219. var item = _libraryManager.GetPerson(name);
  1220. if (item == null)
  1221. {
  1222. return NotFound();
  1223. }
  1224. return await GetImageInternal(
  1225. item.Id,
  1226. imageType,
  1227. imageIndex,
  1228. tag,
  1229. format,
  1230. maxWidth,
  1231. maxHeight,
  1232. percentPlayed,
  1233. unplayedCount,
  1234. width,
  1235. height,
  1236. quality,
  1237. fillWidth,
  1238. fillHeight,
  1239. addPlayedIndicator,
  1240. blur,
  1241. backgroundColor,
  1242. foregroundLayer,
  1243. item)
  1244. .ConfigureAwait(false);
  1245. }
  1246. /// <summary>
  1247. /// Get studio image by name.
  1248. /// </summary>
  1249. /// <param name="name">Studio name.</param>
  1250. /// <param name="imageType">Image type.</param>
  1251. /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>
  1252. /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>
  1253. /// <param name="maxWidth">The maximum image width to return.</param>
  1254. /// <param name="maxHeight">The maximum image height to return.</param>
  1255. /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>
  1256. /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>
  1257. /// <param name="width">The fixed image width to return.</param>
  1258. /// <param name="height">The fixed image height to return.</param>
  1259. /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>
  1260. /// <param name="fillWidth">Width of box to fill.</param>
  1261. /// <param name="fillHeight">Height of box to fill.</param>
  1262. /// <param name="cropWhitespace">Optional. Specify if whitespace should be cropped out of the image. True/False. If unspecified, whitespace will be cropped from logos and clear art.</param>
  1263. /// <param name="addPlayedIndicator">Optional. Add a played indicator.</param>
  1264. /// <param name="blur">Optional. Blur image.</param>
  1265. /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>
  1266. /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>
  1267. /// <param name="imageIndex">Image index.</param>
  1268. /// <response code="200">Image stream returned.</response>
  1269. /// <response code="404">Item not found.</response>
  1270. /// <returns>
  1271. /// A <see cref="FileStreamResult"/> containing the file stream on success,
  1272. /// or a <see cref="NotFoundResult"/> if item not found.
  1273. /// </returns>
  1274. [HttpGet("Studios/{name}/Images/{imageType}")]
  1275. [HttpHead("Studios/{name}/Images/{imageType}", Name = "HeadStudioImage")]
  1276. [ProducesResponseType(StatusCodes.Status200OK)]
  1277. [ProducesResponseType(StatusCodes.Status404NotFound)]
  1278. [ProducesImageFile]
  1279. public async Task<ActionResult> GetStudioImage(
  1280. [FromRoute, Required] string name,
  1281. [FromRoute, Required] ImageType imageType,
  1282. [FromQuery] string? tag,
  1283. [FromQuery] ImageFormat? format,
  1284. [FromQuery] int? maxWidth,
  1285. [FromQuery] int? maxHeight,
  1286. [FromQuery] double? percentPlayed,
  1287. [FromQuery] int? unplayedCount,
  1288. [FromQuery] int? width,
  1289. [FromQuery] int? height,
  1290. [FromQuery] int? quality,
  1291. [FromQuery] int? fillWidth,
  1292. [FromQuery] int? fillHeight,
  1293. [FromQuery, ParameterObsolete] bool? cropWhitespace,
  1294. [FromQuery] bool? addPlayedIndicator,
  1295. [FromQuery] int? blur,
  1296. [FromQuery] string? backgroundColor,
  1297. [FromQuery] string? foregroundLayer,
  1298. [FromQuery] int? imageIndex)
  1299. {
  1300. var item = _libraryManager.GetStudio(name);
  1301. if (item == null)
  1302. {
  1303. return NotFound();
  1304. }
  1305. return await GetImageInternal(
  1306. item.Id,
  1307. imageType,
  1308. imageIndex,
  1309. tag,
  1310. format,
  1311. maxWidth,
  1312. maxHeight,
  1313. percentPlayed,
  1314. unplayedCount,
  1315. width,
  1316. height,
  1317. quality,
  1318. fillWidth,
  1319. fillHeight,
  1320. addPlayedIndicator,
  1321. blur,
  1322. backgroundColor,
  1323. foregroundLayer,
  1324. item)
  1325. .ConfigureAwait(false);
  1326. }
  1327. /// <summary>
  1328. /// Get studio image by name.
  1329. /// </summary>
  1330. /// <param name="name">Studio name.</param>
  1331. /// <param name="imageType">Image type.</param>
  1332. /// <param name="imageIndex">Image index.</param>
  1333. /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>
  1334. /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>
  1335. /// <param name="maxWidth">The maximum image width to return.</param>
  1336. /// <param name="maxHeight">The maximum image height to return.</param>
  1337. /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>
  1338. /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>
  1339. /// <param name="width">The fixed image width to return.</param>
  1340. /// <param name="height">The fixed image height to return.</param>
  1341. /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>
  1342. /// <param name="fillWidth">Width of box to fill.</param>
  1343. /// <param name="fillHeight">Height of box to fill.</param>
  1344. /// <param name="cropWhitespace">Optional. Specify if whitespace should be cropped out of the image. True/False. If unspecified, whitespace will be cropped from logos and clear art.</param>
  1345. /// <param name="addPlayedIndicator">Optional. Add a played indicator.</param>
  1346. /// <param name="blur">Optional. Blur image.</param>
  1347. /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>
  1348. /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>
  1349. /// <response code="200">Image stream returned.</response>
  1350. /// <response code="404">Item not found.</response>
  1351. /// <returns>
  1352. /// A <see cref="FileStreamResult"/> containing the file stream on success,
  1353. /// or a <see cref="NotFoundResult"/> if item not found.
  1354. /// </returns>
  1355. [HttpGet("Studios/{name}/Images/{imageType}/{imageIndex}")]
  1356. [HttpHead("Studios/{name}/Images/{imageType}/{imageIndex}", Name = "HeadStudioImageByIndex")]
  1357. [ProducesResponseType(StatusCodes.Status200OK)]
  1358. [ProducesResponseType(StatusCodes.Status404NotFound)]
  1359. [ProducesImageFile]
  1360. public async Task<ActionResult> GetStudioImageByIndex(
  1361. [FromRoute, Required] string name,
  1362. [FromRoute, Required] ImageType imageType,
  1363. [FromRoute, Required] int imageIndex,
  1364. [FromQuery] string? tag,
  1365. [FromQuery] ImageFormat? format,
  1366. [FromQuery] int? maxWidth,
  1367. [FromQuery] int? maxHeight,
  1368. [FromQuery] double? percentPlayed,
  1369. [FromQuery] int? unplayedCount,
  1370. [FromQuery] int? width,
  1371. [FromQuery] int? height,
  1372. [FromQuery] int? quality,
  1373. [FromQuery] int? fillWidth,
  1374. [FromQuery] int? fillHeight,
  1375. [FromQuery, ParameterObsolete] bool? cropWhitespace,
  1376. [FromQuery] bool? addPlayedIndicator,
  1377. [FromQuery] int? blur,
  1378. [FromQuery] string? backgroundColor,
  1379. [FromQuery] string? foregroundLayer)
  1380. {
  1381. var item = _libraryManager.GetStudio(name);
  1382. if (item == null)
  1383. {
  1384. return NotFound();
  1385. }
  1386. return await GetImageInternal(
  1387. item.Id,
  1388. imageType,
  1389. imageIndex,
  1390. tag,
  1391. format,
  1392. maxWidth,
  1393. maxHeight,
  1394. percentPlayed,
  1395. unplayedCount,
  1396. width,
  1397. height,
  1398. quality,
  1399. fillWidth,
  1400. fillHeight,
  1401. addPlayedIndicator,
  1402. blur,
  1403. backgroundColor,
  1404. foregroundLayer,
  1405. item)
  1406. .ConfigureAwait(false);
  1407. }
  1408. /// <summary>
  1409. /// Get user profile image.
  1410. /// </summary>
  1411. /// <param name="userId">User id.</param>
  1412. /// <param name="imageType">Image type.</param>
  1413. /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>
  1414. /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>
  1415. /// <param name="maxWidth">The maximum image width to return.</param>
  1416. /// <param name="maxHeight">The maximum image height to return.</param>
  1417. /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>
  1418. /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>
  1419. /// <param name="width">The fixed image width to return.</param>
  1420. /// <param name="height">The fixed image height to return.</param>
  1421. /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>
  1422. /// <param name="fillWidth">Width of box to fill.</param>
  1423. /// <param name="fillHeight">Height of box to fill.</param>
  1424. /// <param name="cropWhitespace">Optional. Specify if whitespace should be cropped out of the image. True/False. If unspecified, whitespace will be cropped from logos and clear art.</param>
  1425. /// <param name="addPlayedIndicator">Optional. Add a played indicator.</param>
  1426. /// <param name="blur">Optional. Blur image.</param>
  1427. /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>
  1428. /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>
  1429. /// <param name="imageIndex">Image index.</param>
  1430. /// <response code="200">Image stream returned.</response>
  1431. /// <response code="404">Item not found.</response>
  1432. /// <returns>
  1433. /// A <see cref="FileStreamResult"/> containing the file stream on success,
  1434. /// or a <see cref="NotFoundResult"/> if item not found.
  1435. /// </returns>
  1436. [HttpGet("Users/{userId}/Images/{imageType}")]
  1437. [HttpHead("Users/{userId}/Images/{imageType}", Name = "HeadUserImage")]
  1438. [ProducesResponseType(StatusCodes.Status200OK)]
  1439. [ProducesResponseType(StatusCodes.Status404NotFound)]
  1440. [ProducesImageFile]
  1441. public async Task<ActionResult> GetUserImage(
  1442. [FromRoute, Required] Guid userId,
  1443. [FromRoute, Required] ImageType imageType,
  1444. [FromQuery] string? tag,
  1445. [FromQuery] ImageFormat? format,
  1446. [FromQuery] int? maxWidth,
  1447. [FromQuery] int? maxHeight,
  1448. [FromQuery] double? percentPlayed,
  1449. [FromQuery] int? unplayedCount,
  1450. [FromQuery] int? width,
  1451. [FromQuery] int? height,
  1452. [FromQuery] int? quality,
  1453. [FromQuery] int? fillWidth,
  1454. [FromQuery] int? fillHeight,
  1455. [FromQuery, ParameterObsolete] bool? cropWhitespace,
  1456. [FromQuery] bool? addPlayedIndicator,
  1457. [FromQuery] int? blur,
  1458. [FromQuery] string? backgroundColor,
  1459. [FromQuery] string? foregroundLayer,
  1460. [FromQuery] int? imageIndex)
  1461. {
  1462. var user = _userManager.GetUserById(userId);
  1463. if (user?.ProfileImage == null)
  1464. {
  1465. return NotFound();
  1466. }
  1467. var info = new ItemImageInfo
  1468. {
  1469. Path = user.ProfileImage.Path,
  1470. Type = ImageType.Profile,
  1471. DateModified = user.ProfileImage.LastModified
  1472. };
  1473. if (width.HasValue)
  1474. {
  1475. info.Width = width.Value;
  1476. }
  1477. if (height.HasValue)
  1478. {
  1479. info.Height = height.Value;
  1480. }
  1481. return await GetImageInternal(
  1482. user.Id,
  1483. imageType,
  1484. imageIndex,
  1485. tag,
  1486. format,
  1487. maxWidth,
  1488. maxHeight,
  1489. percentPlayed,
  1490. unplayedCount,
  1491. width,
  1492. height,
  1493. quality,
  1494. fillWidth,
  1495. fillHeight,
  1496. addPlayedIndicator,
  1497. blur,
  1498. backgroundColor,
  1499. foregroundLayer,
  1500. null,
  1501. info)
  1502. .ConfigureAwait(false);
  1503. }
  1504. /// <summary>
  1505. /// Get user profile image.
  1506. /// </summary>
  1507. /// <param name="userId">User id.</param>
  1508. /// <param name="imageType">Image type.</param>
  1509. /// <param name="imageIndex">Image index.</param>
  1510. /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>
  1511. /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>
  1512. /// <param name="maxWidth">The maximum image width to return.</param>
  1513. /// <param name="maxHeight">The maximum image height to return.</param>
  1514. /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>
  1515. /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>
  1516. /// <param name="width">The fixed image width to return.</param>
  1517. /// <param name="height">The fixed image height to return.</param>
  1518. /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>
  1519. /// <param name="fillWidth">Width of box to fill.</param>
  1520. /// <param name="fillHeight">Height of box to fill.</param>
  1521. /// <param name="cropWhitespace">Optional. Specify if whitespace should be cropped out of the image. True/False. If unspecified, whitespace will be cropped from logos and clear art.</param>
  1522. /// <param name="addPlayedIndicator">Optional. Add a played indicator.</param>
  1523. /// <param name="blur">Optional. Blur image.</param>
  1524. /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>
  1525. /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>
  1526. /// <response code="200">Image stream returned.</response>
  1527. /// <response code="404">Item not found.</response>
  1528. /// <returns>
  1529. /// A <see cref="FileStreamResult"/> containing the file stream on success,
  1530. /// or a <see cref="NotFoundResult"/> if item not found.
  1531. /// </returns>
  1532. [HttpGet("Users/{userId}/Images/{imageType}/{imageIndex}")]
  1533. [HttpHead("Users/{userId}/Images/{imageType}/{imageIndex}", Name = "HeadUserImageByIndex")]
  1534. [ProducesResponseType(StatusCodes.Status200OK)]
  1535. [ProducesResponseType(StatusCodes.Status404NotFound)]
  1536. [ProducesImageFile]
  1537. public async Task<ActionResult> GetUserImageByIndex(
  1538. [FromRoute, Required] Guid userId,
  1539. [FromRoute, Required] ImageType imageType,
  1540. [FromRoute, Required] int imageIndex,
  1541. [FromQuery] string? tag,
  1542. [FromQuery] ImageFormat? format,
  1543. [FromQuery] int? maxWidth,
  1544. [FromQuery] int? maxHeight,
  1545. [FromQuery] double? percentPlayed,
  1546. [FromQuery] int? unplayedCount,
  1547. [FromQuery] int? width,
  1548. [FromQuery] int? height,
  1549. [FromQuery] int? quality,
  1550. [FromQuery] int? fillWidth,
  1551. [FromQuery] int? fillHeight,
  1552. [FromQuery, ParameterObsolete] bool? cropWhitespace,
  1553. [FromQuery] bool? addPlayedIndicator,
  1554. [FromQuery] int? blur,
  1555. [FromQuery] string? backgroundColor,
  1556. [FromQuery] string? foregroundLayer)
  1557. {
  1558. var user = _userManager.GetUserById(userId);
  1559. if (user?.ProfileImage == null)
  1560. {
  1561. return NotFound();
  1562. }
  1563. var info = new ItemImageInfo
  1564. {
  1565. Path = user.ProfileImage.Path,
  1566. Type = ImageType.Profile,
  1567. DateModified = user.ProfileImage.LastModified
  1568. };
  1569. if (width.HasValue)
  1570. {
  1571. info.Width = width.Value;
  1572. }
  1573. if (height.HasValue)
  1574. {
  1575. info.Height = height.Value;
  1576. }
  1577. return await GetImageInternal(
  1578. user.Id,
  1579. imageType,
  1580. imageIndex,
  1581. tag,
  1582. format,
  1583. maxWidth,
  1584. maxHeight,
  1585. percentPlayed,
  1586. unplayedCount,
  1587. width,
  1588. height,
  1589. quality,
  1590. fillWidth,
  1591. fillHeight,
  1592. addPlayedIndicator,
  1593. blur,
  1594. backgroundColor,
  1595. foregroundLayer,
  1596. null,
  1597. info)
  1598. .ConfigureAwait(false);
  1599. }
  1600. /// <summary>
  1601. /// Generates or gets the splashscreen.
  1602. /// </summary>
  1603. /// <param name="tag">Supply the cache tag from the item object to receive strong caching headers.</param>
  1604. /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>
  1605. /// <param name="maxWidth">The maximum image width to return.</param>
  1606. /// <param name="maxHeight">The maximum image height to return.</param>
  1607. /// <param name="width">The fixed image width to return.</param>
  1608. /// <param name="height">The fixed image height to return.</param>
  1609. /// <param name="fillWidth">Width of box to fill.</param>
  1610. /// <param name="fillHeight">Height of box to fill.</param>
  1611. /// <param name="blur">Blur image.</param>
  1612. /// <param name="backgroundColor">Apply a background color for transparent images.</param>
  1613. /// <param name="foregroundLayer">Apply a foreground layer on top of the image.</param>
  1614. /// <param name="quality">Quality setting, from 0-100.</param>
  1615. /// <response code="200">Splashscreen returned successfully.</response>
  1616. /// <returns>The splashscreen.</returns>
  1617. [HttpGet("Branding/Splashscreen")]
  1618. [ProducesResponseType(StatusCodes.Status200OK)]
  1619. [ProducesImageFile]
  1620. public async Task<ActionResult> GetSplashscreen(
  1621. [FromQuery] string? tag,
  1622. [FromQuery] ImageFormat? format,
  1623. [FromQuery] int? maxWidth,
  1624. [FromQuery] int? maxHeight,
  1625. [FromQuery] int? width,
  1626. [FromQuery] int? height,
  1627. [FromQuery] int? fillWidth,
  1628. [FromQuery] int? fillHeight,
  1629. [FromQuery] int? blur,
  1630. [FromQuery] string? backgroundColor,
  1631. [FromQuery] string? foregroundLayer,
  1632. [FromQuery, Range(0, 100)] int quality = 90)
  1633. {
  1634. var brandingOptions = _serverConfigurationManager.GetConfiguration<BrandingOptions>("branding");
  1635. if (!brandingOptions.SplashscreenEnabled)
  1636. {
  1637. return NotFound();
  1638. }
  1639. string splashscreenPath;
  1640. if (!string.IsNullOrWhiteSpace(brandingOptions.SplashscreenLocation)
  1641. && System.IO.File.Exists(brandingOptions.SplashscreenLocation))
  1642. {
  1643. splashscreenPath = brandingOptions.SplashscreenLocation;
  1644. }
  1645. else
  1646. {
  1647. splashscreenPath = Path.Combine(_appPaths.DataPath, "splashscreen.png");
  1648. if (!System.IO.File.Exists(splashscreenPath))
  1649. {
  1650. return NotFound();
  1651. }
  1652. }
  1653. var outputFormats = GetOutputFormats(format);
  1654. TimeSpan? cacheDuration = null;
  1655. if (!string.IsNullOrEmpty(tag))
  1656. {
  1657. cacheDuration = TimeSpan.FromDays(365);
  1658. }
  1659. var options = new ImageProcessingOptions
  1660. {
  1661. Image = new ItemImageInfo
  1662. {
  1663. Path = splashscreenPath
  1664. },
  1665. Height = height,
  1666. MaxHeight = maxHeight,
  1667. MaxWidth = maxWidth,
  1668. FillHeight = fillHeight,
  1669. FillWidth = fillWidth,
  1670. Quality = quality,
  1671. Width = width,
  1672. Blur = blur,
  1673. BackgroundColor = backgroundColor,
  1674. ForegroundLayer = foregroundLayer,
  1675. SupportedOutputFormats = outputFormats
  1676. };
  1677. return await GetImageResult(
  1678. options,
  1679. cacheDuration,
  1680. ImmutableDictionary<string, string>.Empty)
  1681. .ConfigureAwait(false);
  1682. }
  1683. /// <summary>
  1684. /// Uploads a custom splashscreen.
  1685. /// The body is expected to the image contents base64 encoded.
  1686. /// </summary>
  1687. /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
  1688. /// <response code="204">Successfully uploaded new splashscreen.</response>
  1689. /// <response code="400">Error reading MimeType from uploaded image.</response>
  1690. /// <response code="403">User does not have permission to upload splashscreen..</response>
  1691. /// <exception cref="ArgumentException">Error reading the image format.</exception>
  1692. [HttpPost("Branding/Splashscreen")]
  1693. [Authorize(Policy = Policies.RequiresElevation)]
  1694. [ProducesResponseType(StatusCodes.Status204NoContent)]
  1695. [ProducesResponseType(StatusCodes.Status400BadRequest)]
  1696. [ProducesResponseType(StatusCodes.Status403Forbidden)]
  1697. [AcceptsImageFile]
  1698. public async Task<ActionResult> UploadCustomSplashscreen()
  1699. {
  1700. await using var memoryStream = await GetMemoryStream(Request.Body).ConfigureAwait(false);
  1701. var mimeType = MediaTypeHeaderValue.Parse(Request.ContentType).MediaType;
  1702. if (!mimeType.HasValue)
  1703. {
  1704. return BadRequest("Error reading mimetype from uploaded image");
  1705. }
  1706. var extension = MimeTypes.ToExtension(mimeType.Value);
  1707. if (string.IsNullOrEmpty(extension))
  1708. {
  1709. return BadRequest("Error converting mimetype to an image extension");
  1710. }
  1711. var filePath = Path.Combine(_appPaths.DataPath, "splashscreen-upload" + extension);
  1712. var brandingOptions = _serverConfigurationManager.GetConfiguration<BrandingOptions>("branding");
  1713. brandingOptions.SplashscreenLocation = filePath;
  1714. _serverConfigurationManager.SaveConfiguration("branding", brandingOptions);
  1715. await using (var fs = new FileStream(filePath, FileMode.Create, FileAccess.Write, FileShare.None, IODefaults.FileStreamBufferSize, FileOptions.Asynchronous))
  1716. {
  1717. await memoryStream.CopyToAsync(fs, CancellationToken.None).ConfigureAwait(false);
  1718. }
  1719. return NoContent();
  1720. }
  1721. /// <summary>
  1722. /// Delete a custom splashscreen.
  1723. /// </summary>
  1724. /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
  1725. /// <response code="204">Successfully deleted the custom splashscreen.</response>
  1726. /// <response code="403">User does not have permission to delete splashscreen..</response>
  1727. [HttpDelete("Branding/Splashscreen")]
  1728. [Authorize(Policy = Policies.RequiresElevation)]
  1729. [ProducesResponseType(StatusCodes.Status204NoContent)]
  1730. public ActionResult DeleteCustomSplashscreen()
  1731. {
  1732. var brandingOptions = _serverConfigurationManager.GetConfiguration<BrandingOptions>("branding");
  1733. if (!string.IsNullOrEmpty(brandingOptions.SplashscreenLocation)
  1734. && System.IO.File.Exists(brandingOptions.SplashscreenLocation))
  1735. {
  1736. System.IO.File.Delete(brandingOptions.SplashscreenLocation);
  1737. brandingOptions.SplashscreenLocation = null;
  1738. _serverConfigurationManager.SaveConfiguration("branding", brandingOptions);
  1739. }
  1740. return NoContent();
  1741. }
  1742. private static async Task<MemoryStream> GetMemoryStream(Stream inputStream)
  1743. {
  1744. using var reader = new StreamReader(inputStream);
  1745. var text = await reader.ReadToEndAsync().ConfigureAwait(false);
  1746. var bytes = Convert.FromBase64String(text);
  1747. return new MemoryStream(bytes, 0, bytes.Length, false, true);
  1748. }
  1749. private ImageInfo? GetImageInfo(BaseItem item, ItemImageInfo info, int? imageIndex)
  1750. {
  1751. int? width = null;
  1752. int? height = null;
  1753. string? blurhash = null;
  1754. long length = 0;
  1755. try
  1756. {
  1757. if (info.IsLocalFile)
  1758. {
  1759. var fileInfo = _fileSystem.GetFileInfo(info.Path);
  1760. length = fileInfo.Length;
  1761. blurhash = info.BlurHash;
  1762. width = info.Width;
  1763. height = info.Height;
  1764. if (width <= 0 || height <= 0)
  1765. {
  1766. width = null;
  1767. height = null;
  1768. }
  1769. }
  1770. }
  1771. catch (Exception ex)
  1772. {
  1773. _logger.LogError(ex, "Error getting image information for {Item}", item.Name);
  1774. }
  1775. try
  1776. {
  1777. return new ImageInfo
  1778. {
  1779. Path = info.Path,
  1780. ImageIndex = imageIndex,
  1781. ImageType = info.Type,
  1782. ImageTag = _imageProcessor.GetImageCacheTag(item, info),
  1783. Size = length,
  1784. BlurHash = blurhash,
  1785. Width = width,
  1786. Height = height
  1787. };
  1788. }
  1789. catch (Exception ex)
  1790. {
  1791. _logger.LogError(ex, "Error getting image information for {Path}", info.Path);
  1792. return null;
  1793. }
  1794. }
  1795. private async Task<ActionResult> GetImageInternal(
  1796. Guid itemId,
  1797. ImageType imageType,
  1798. int? imageIndex,
  1799. string? tag,
  1800. ImageFormat? format,
  1801. int? maxWidth,
  1802. int? maxHeight,
  1803. double? percentPlayed,
  1804. int? unplayedCount,
  1805. int? width,
  1806. int? height,
  1807. int? quality,
  1808. int? fillWidth,
  1809. int? fillHeight,
  1810. bool? addPlayedIndicator,
  1811. int? blur,
  1812. string? backgroundColor,
  1813. string? foregroundLayer,
  1814. BaseItem? item,
  1815. ItemImageInfo? imageInfo = null)
  1816. {
  1817. if (percentPlayed.HasValue)
  1818. {
  1819. if (percentPlayed.Value <= 0)
  1820. {
  1821. percentPlayed = null;
  1822. }
  1823. else if (percentPlayed.Value >= 100)
  1824. {
  1825. percentPlayed = null;
  1826. addPlayedIndicator = true;
  1827. }
  1828. }
  1829. if (percentPlayed.HasValue)
  1830. {
  1831. unplayedCount = null;
  1832. }
  1833. if (unplayedCount.HasValue
  1834. && unplayedCount.Value <= 0)
  1835. {
  1836. unplayedCount = null;
  1837. }
  1838. if (imageInfo == null)
  1839. {
  1840. imageInfo = item?.GetImageInfo(imageType, imageIndex ?? 0);
  1841. if (imageInfo == null)
  1842. {
  1843. return NotFound(string.Format(NumberFormatInfo.InvariantInfo, "{0} does not have an image of type {1}", item?.Name, imageType));
  1844. }
  1845. }
  1846. var outputFormats = GetOutputFormats(format);
  1847. TimeSpan? cacheDuration = null;
  1848. if (!string.IsNullOrEmpty(tag))
  1849. {
  1850. cacheDuration = TimeSpan.FromDays(365);
  1851. }
  1852. var responseHeaders = new Dictionary<string, string>
  1853. {
  1854. { "transferMode.dlna.org", "Interactive" },
  1855. { "realTimeInfo.dlna.org", "DLNA.ORG_TLAG=*" }
  1856. };
  1857. if (!imageInfo.IsLocalFile && item != null)
  1858. {
  1859. imageInfo = await _libraryManager.ConvertImageToLocal(item, imageInfo, imageIndex ?? 0).ConfigureAwait(false);
  1860. }
  1861. var options = new ImageProcessingOptions
  1862. {
  1863. Height = height,
  1864. ImageIndex = imageIndex ?? 0,
  1865. Image = imageInfo,
  1866. Item = item,
  1867. ItemId = itemId,
  1868. MaxHeight = maxHeight,
  1869. MaxWidth = maxWidth,
  1870. FillHeight = fillHeight,
  1871. FillWidth = fillWidth,
  1872. Quality = quality ?? 100,
  1873. Width = width,
  1874. AddPlayedIndicator = addPlayedIndicator ?? false,
  1875. PercentPlayed = percentPlayed ?? 0,
  1876. UnplayedCount = unplayedCount,
  1877. Blur = blur,
  1878. BackgroundColor = backgroundColor,
  1879. ForegroundLayer = foregroundLayer,
  1880. SupportedOutputFormats = outputFormats
  1881. };
  1882. return await GetImageResult(
  1883. options,
  1884. cacheDuration,
  1885. responseHeaders).ConfigureAwait(false);
  1886. }
  1887. private ImageFormat[] GetOutputFormats(ImageFormat? format)
  1888. {
  1889. if (format.HasValue)
  1890. {
  1891. return new[] { format.Value };
  1892. }
  1893. return GetClientSupportedFormats();
  1894. }
  1895. private ImageFormat[] GetClientSupportedFormats()
  1896. {
  1897. var supportedFormats = Request.Headers.GetCommaSeparatedValues(HeaderNames.Accept);
  1898. for (var i = 0; i < supportedFormats.Length; i++)
  1899. {
  1900. // Remove charsets etc. (anything after semi-colon)
  1901. var type = supportedFormats[i];
  1902. int index = type.IndexOf(';', StringComparison.Ordinal);
  1903. if (index != -1)
  1904. {
  1905. supportedFormats[i] = type.Substring(0, index);
  1906. }
  1907. }
  1908. var acceptParam = Request.Query[HeaderNames.Accept];
  1909. var supportsWebP = SupportsFormat(supportedFormats, acceptParam, ImageFormat.Webp, false);
  1910. if (!supportsWebP)
  1911. {
  1912. var userAgent = Request.Headers[HeaderNames.UserAgent].ToString();
  1913. if (userAgent.Contains("crosswalk", StringComparison.OrdinalIgnoreCase)
  1914. && userAgent.Contains("android", StringComparison.OrdinalIgnoreCase))
  1915. {
  1916. supportsWebP = true;
  1917. }
  1918. }
  1919. var formats = new List<ImageFormat>(4);
  1920. if (supportsWebP)
  1921. {
  1922. formats.Add(ImageFormat.Webp);
  1923. }
  1924. formats.Add(ImageFormat.Jpg);
  1925. formats.Add(ImageFormat.Png);
  1926. if (SupportsFormat(supportedFormats, acceptParam, ImageFormat.Gif, true))
  1927. {
  1928. formats.Add(ImageFormat.Gif);
  1929. }
  1930. return formats.ToArray();
  1931. }
  1932. private bool SupportsFormat(IReadOnlyCollection<string> requestAcceptTypes, string acceptParam, ImageFormat format, bool acceptAll)
  1933. {
  1934. if (requestAcceptTypes.Contains(format.GetMimeType()))
  1935. {
  1936. return true;
  1937. }
  1938. if (acceptAll && requestAcceptTypes.Contains("*/*"))
  1939. {
  1940. return true;
  1941. }
  1942. // Review if this should be jpeg, jpg or both for ImageFormat.Jpg
  1943. var normalized = format.ToString().ToLowerInvariant();
  1944. return string.Equals(acceptParam, normalized, StringComparison.OrdinalIgnoreCase);
  1945. }
  1946. private async Task<ActionResult> GetImageResult(
  1947. ImageProcessingOptions imageProcessingOptions,
  1948. TimeSpan? cacheDuration,
  1949. IDictionary<string, string> headers)
  1950. {
  1951. var (imagePath, imageContentType, dateImageModified) = await _imageProcessor.ProcessImage(imageProcessingOptions).ConfigureAwait(false);
  1952. var disableCaching = Request.Headers[HeaderNames.CacheControl].Contains("no-cache");
  1953. var parsingSuccessful = DateTime.TryParse(Request.Headers[HeaderNames.IfModifiedSince], out var ifModifiedSinceHeader);
  1954. // if the parsing of the IfModifiedSince header was not successful, disable caching
  1955. if (!parsingSuccessful)
  1956. {
  1957. // disableCaching = true;
  1958. }
  1959. foreach (var (key, value) in headers)
  1960. {
  1961. Response.Headers.Add(key, value);
  1962. }
  1963. Response.ContentType = imageContentType ?? MediaTypeNames.Text.Plain;
  1964. Response.Headers.Add(HeaderNames.Age, Convert.ToInt64((DateTime.UtcNow - dateImageModified).TotalSeconds).ToString(CultureInfo.InvariantCulture));
  1965. Response.Headers.Add(HeaderNames.Vary, HeaderNames.Accept);
  1966. if (disableCaching)
  1967. {
  1968. Response.Headers.Add(HeaderNames.CacheControl, "no-cache, no-store, must-revalidate");
  1969. Response.Headers.Add(HeaderNames.Pragma, "no-cache, no-store, must-revalidate");
  1970. }
  1971. else
  1972. {
  1973. if (cacheDuration.HasValue)
  1974. {
  1975. Response.Headers.Add(HeaderNames.CacheControl, "public, max-age=" + cacheDuration.Value.TotalSeconds);
  1976. }
  1977. else
  1978. {
  1979. Response.Headers.Add(HeaderNames.CacheControl, "public");
  1980. }
  1981. Response.Headers.Add(HeaderNames.LastModified, dateImageModified.ToUniversalTime().ToString("ddd, dd MMM yyyy HH:mm:ss \"GMT\"", CultureInfo.InvariantCulture));
  1982. // if the image was not modified since "ifModifiedSinceHeader"-header, return a HTTP status code 304 not modified
  1983. if (!(dateImageModified > ifModifiedSinceHeader) && cacheDuration.HasValue)
  1984. {
  1985. if (ifModifiedSinceHeader.Add(cacheDuration.Value) < DateTime.UtcNow)
  1986. {
  1987. Response.StatusCode = StatusCodes.Status304NotModified;
  1988. return new ContentResult();
  1989. }
  1990. }
  1991. }
  1992. return PhysicalFile(imagePath, imageContentType ?? MediaTypeNames.Text.Plain);
  1993. }
  1994. }
  1995. }