| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150 | using System;using System.Collections.Generic;using System.Collections.Immutable;using System.ComponentModel.DataAnnotations;using System.Diagnostics.CodeAnalysis;using System.Globalization;using System.IO;using System.Linq;using System.Net.Mime;using System.Security.Cryptography;using System.Threading;using System.Threading.Tasks;using Jellyfin.Api.Attributes;using Jellyfin.Api.Extensions;using Jellyfin.Api.Helpers;using Jellyfin.Extensions;using MediaBrowser.Common.Api;using MediaBrowser.Common.Configuration;using MediaBrowser.Controller.Configuration;using MediaBrowser.Controller.Drawing;using MediaBrowser.Controller.Entities;using MediaBrowser.Controller.Library;using MediaBrowser.Controller.Providers;using MediaBrowser.Model.Branding;using MediaBrowser.Model.Drawing;using MediaBrowser.Model.Dto;using MediaBrowser.Model.Entities;using MediaBrowser.Model.IO;using MediaBrowser.Model.Net;using Microsoft.AspNetCore.Authorization;using Microsoft.AspNetCore.Http;using Microsoft.AspNetCore.Mvc;using Microsoft.Extensions.Logging;using Microsoft.Net.Http.Headers;namespace Jellyfin.Api.Controllers;/// <summary>/// Image controller./// </summary>[Route("")]public class ImageController : BaseJellyfinApiController{    private readonly IUserManager _userManager;    private readonly ILibraryManager _libraryManager;    private readonly IProviderManager _providerManager;    private readonly IImageProcessor _imageProcessor;    private readonly IFileSystem _fileSystem;    private readonly ILogger<ImageController> _logger;    private readonly IServerConfigurationManager _serverConfigurationManager;    private readonly IApplicationPaths _appPaths;    /// <summary>    /// Initializes a new instance of the <see cref="ImageController"/> class.    /// </summary>    /// <param name="userManager">Instance of the <see cref="IUserManager"/> interface.</param>    /// <param name="libraryManager">Instance of the <see cref="ILibraryManager"/> interface.</param>    /// <param name="providerManager">Instance of the <see cref="IProviderManager"/> interface.</param>    /// <param name="imageProcessor">Instance of the <see cref="IImageProcessor"/> interface.</param>    /// <param name="fileSystem">Instance of the <see cref="IFileSystem"/> interface.</param>    /// <param name="logger">Instance of the <see cref="ILogger{ImageController}"/> interface.</param>    /// <param name="serverConfigurationManager">Instance of the <see cref="IServerConfigurationManager"/> interface.</param>    /// <param name="appPaths">Instance of the <see cref="IApplicationPaths"/> interface.</param>    public ImageController(        IUserManager userManager,        ILibraryManager libraryManager,        IProviderManager providerManager,        IImageProcessor imageProcessor,        IFileSystem fileSystem,        ILogger<ImageController> logger,        IServerConfigurationManager serverConfigurationManager,        IApplicationPaths appPaths)    {        _userManager = userManager;        _libraryManager = libraryManager;        _providerManager = providerManager;        _imageProcessor = imageProcessor;        _fileSystem = fileSystem;        _logger = logger;        _serverConfigurationManager = serverConfigurationManager;        _appPaths = appPaths;    }    private static CryptoStream GetFromBase64Stream(Stream inputStream)        => new CryptoStream(inputStream, new FromBase64Transform(), CryptoStreamMode.Read);    /// <summary>    /// Sets the user image.    /// </summary>    /// <param name="userId">User Id.</param>    /// <response code="204">Image updated.</response>    /// <response code="403">User does not have permission to delete the image.</response>    /// <response code="404">Item not found.</response>    /// <returns>A <see cref="NoContentResult"/>.</returns>    [HttpPost("UserImage")]    [Authorize]    [AcceptsImageFile]    [ProducesResponseType(StatusCodes.Status204NoContent)]    [ProducesResponseType(StatusCodes.Status400BadRequest)]    [ProducesResponseType(StatusCodes.Status403Forbidden)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    public async Task<ActionResult> PostUserImage(        [FromQuery] Guid? userId)    {        var requestUserId = RequestHelpers.GetUserId(User, userId);        var user = _userManager.GetUserById(requestUserId);        if (user is null)        {            return NotFound();        }        if (!RequestHelpers.AssertCanUpdateUser(HttpContext.User, user, true))        {            return StatusCode(StatusCodes.Status403Forbidden, "User is not allowed to update the image.");        }        if (!TryGetImageExtensionFromContentType(Request.ContentType, out string? extension))        {            return BadRequest("Incorrect ContentType.");        }        var stream = GetFromBase64Stream(Request.Body);        await using (stream.ConfigureAwait(false))        {            // Handle image/png; charset=utf-8            var mimeType = Request.ContentType?.Split(';').FirstOrDefault();            var userDataPath = Path.Combine(_serverConfigurationManager.ApplicationPaths.UserConfigurationDirectoryPath, user.Username);            if (user.ProfileImage is not null)            {                await _userManager.ClearProfileImageAsync(user).ConfigureAwait(false);            }            user.ProfileImage = new Database.Implementations.Entities.ImageInfo(Path.Combine(userDataPath, "profile" + extension));            await _providerManager                .SaveImage(stream, mimeType, user.ProfileImage.Path)                .ConfigureAwait(false);            await _userManager.UpdateUserAsync(user).ConfigureAwait(false);            return NoContent();        }    }    /// <summary>    /// Sets the user image.    /// </summary>    /// <param name="userId">User Id.</param>    /// <param name="imageType">(Unused) Image type.</param>    /// <response code="204">Image updated.</response>    /// <response code="403">User does not have permission to delete the image.</response>    /// <returns>A <see cref="NoContentResult"/>.</returns>    [HttpPost("Users/{userId}/Images/{imageType}")]    [Authorize]    [Obsolete("Kept for backwards compatibility")]    [ApiExplorerSettings(IgnoreApi = true)]    [AcceptsImageFile]    [ProducesResponseType(StatusCodes.Status204NoContent)]    [ProducesResponseType(StatusCodes.Status400BadRequest)]    [ProducesResponseType(StatusCodes.Status403Forbidden)]    [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "imageType", Justification = "Imported from ServiceStack")]    public Task<ActionResult> PostUserImageLegacy(        [FromRoute, Required] Guid userId,        [FromRoute, Required] ImageType imageType)        => PostUserImage(userId);    /// <summary>    /// Sets the user image.    /// </summary>    /// <param name="userId">User Id.</param>    /// <param name="imageType">(Unused) Image type.</param>    /// <param name="index">(Unused) Image index.</param>    /// <response code="204">Image updated.</response>    /// <response code="403">User does not have permission to delete the image.</response>    /// <returns>A <see cref="NoContentResult"/>.</returns>    [HttpPost("Users/{userId}/Images/{imageType}/{index}")]    [Authorize]    [Obsolete("Kept for backwards compatibility")]    [ApiExplorerSettings(IgnoreApi = true)]    [AcceptsImageFile]    [ProducesResponseType(StatusCodes.Status204NoContent)]    [ProducesResponseType(StatusCodes.Status400BadRequest)]    [ProducesResponseType(StatusCodes.Status403Forbidden)]    [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "imageType", Justification = "Imported from ServiceStack")]    [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "index", Justification = "Imported from ServiceStack")]    public Task<ActionResult> PostUserImageByIndexLegacy(        [FromRoute, Required] Guid userId,        [FromRoute, Required] ImageType imageType,        [FromRoute] int index)        => PostUserImage(userId);    /// <summary>    /// Delete the user's image.    /// </summary>    /// <param name="userId">User Id.</param>    /// <response code="204">Image deleted.</response>    /// <response code="403">User does not have permission to delete the image.</response>    /// <returns>A <see cref="NoContentResult"/>.</returns>    [HttpDelete("UserImage")]    [Authorize]    [ProducesResponseType(StatusCodes.Status204NoContent)]    [ProducesResponseType(StatusCodes.Status403Forbidden)]    public async Task<ActionResult> DeleteUserImage(        [FromQuery] Guid? userId)    {        var requestUserId = RequestHelpers.GetUserId(User, userId);        var user = _userManager.GetUserById(requestUserId);        if (user is null)        {            return NotFound();        }        if (!RequestHelpers.AssertCanUpdateUser(HttpContext.User, user, true))        {            return StatusCode(StatusCodes.Status403Forbidden, "User is not allowed to delete the image.");        }        if (user.ProfileImage is null)        {            return NoContent();        }        try        {            System.IO.File.Delete(user.ProfileImage.Path);        }        catch (IOException e)        {            _logger.LogError(e, "Error deleting user profile image:");        }        await _userManager.ClearProfileImageAsync(user).ConfigureAwait(false);        return NoContent();    }    /// <summary>    /// Delete the user's image.    /// </summary>    /// <param name="userId">User Id.</param>    /// <param name="imageType">(Unused) Image type.</param>    /// <param name="index">(Unused) Image index.</param>    /// <response code="204">Image deleted.</response>    /// <response code="403">User does not have permission to delete the image.</response>    /// <returns>A <see cref="NoContentResult"/>.</returns>    [HttpDelete("Users/{userId}/Images/{imageType}")]    [Authorize]    [Obsolete("Kept for backwards compatibility")]    [ApiExplorerSettings(IgnoreApi = true)]    [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "imageType", Justification = "Imported from ServiceStack")]    [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "index", Justification = "Imported from ServiceStack")]    [ProducesResponseType(StatusCodes.Status204NoContent)]    [ProducesResponseType(StatusCodes.Status403Forbidden)]    public Task<ActionResult> DeleteUserImageLegacy(        [FromRoute, Required] Guid userId,        [FromRoute, Required] ImageType imageType,        [FromQuery] int? index = null)        => DeleteUserImage(userId);    /// <summary>    /// Delete the user's image.    /// </summary>    /// <param name="userId">User Id.</param>    /// <param name="imageType">(Unused) Image type.</param>    /// <param name="index">(Unused) Image index.</param>    /// <response code="204">Image deleted.</response>    /// <response code="403">User does not have permission to delete the image.</response>    /// <returns>A <see cref="NoContentResult"/>.</returns>    [HttpDelete("Users/{userId}/Images/{imageType}/{index}")]    [Authorize]    [Obsolete("Kept for backwards compatibility")]    [ApiExplorerSettings(IgnoreApi = true)]    [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "imageType", Justification = "Imported from ServiceStack")]    [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "index", Justification = "Imported from ServiceStack")]    [ProducesResponseType(StatusCodes.Status204NoContent)]    [ProducesResponseType(StatusCodes.Status403Forbidden)]    public Task<ActionResult> DeleteUserImageByIndexLegacy(        [FromRoute, Required] Guid userId,        [FromRoute, Required] ImageType imageType,        [FromRoute] int index)        => DeleteUserImage(userId);    /// <summary>    /// Delete an item's image.    /// </summary>    /// <param name="itemId">Item id.</param>    /// <param name="imageType">Image type.</param>    /// <param name="imageIndex">The image index.</param>    /// <response code="204">Image deleted.</response>    /// <response code="404">Item not found.</response>    /// <returns>A <see cref="NoContentResult"/> on success, or a <see cref="NotFoundResult"/> if item not found.</returns>    [HttpDelete("Items/{itemId}/Images/{imageType}")]    [Authorize(Policy = Policies.RequiresElevation)]    [ProducesResponseType(StatusCodes.Status204NoContent)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    public async Task<ActionResult> DeleteItemImage(        [FromRoute, Required] Guid itemId,        [FromRoute, Required] ImageType imageType,        [FromQuery] int? imageIndex)    {        var item = _libraryManager.GetItemById<BaseItem>(itemId, User.GetUserId());        if (item is null)        {            return NotFound();        }        await item.DeleteImageAsync(imageType, imageIndex ?? 0).ConfigureAwait(false);        return NoContent();    }    /// <summary>    /// Delete an item's image.    /// </summary>    /// <param name="itemId">Item id.</param>    /// <param name="imageType">Image type.</param>    /// <param name="imageIndex">The image index.</param>    /// <response code="204">Image deleted.</response>    /// <response code="404">Item not found.</response>    /// <returns>A <see cref="NoContentResult"/> on success, or a <see cref="NotFoundResult"/> if item not found.</returns>    [HttpDelete("Items/{itemId}/Images/{imageType}/{imageIndex}")]    [Authorize(Policy = Policies.RequiresElevation)]    [ProducesResponseType(StatusCodes.Status204NoContent)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    public async Task<ActionResult> DeleteItemImageByIndex(        [FromRoute, Required] Guid itemId,        [FromRoute, Required] ImageType imageType,        [FromRoute] int imageIndex)    {        var item = _libraryManager.GetItemById<BaseItem>(itemId, User.GetUserId());        if (item is null)        {            return NotFound();        }        await item.DeleteImageAsync(imageType, imageIndex).ConfigureAwait(false);        return NoContent();    }    /// <summary>    /// Set item image.    /// </summary>    /// <param name="itemId">Item id.</param>    /// <param name="imageType">Image type.</param>    /// <response code="204">Image saved.</response>    /// <response code="404">Item not found.</response>    /// <returns>A <see cref="NoContentResult"/> on success, or a <see cref="NotFoundResult"/> if item not found.</returns>    [HttpPost("Items/{itemId}/Images/{imageType}")]    [Authorize(Policy = Policies.RequiresElevation)]    [AcceptsImageFile]    [ProducesResponseType(StatusCodes.Status204NoContent)]    [ProducesResponseType(StatusCodes.Status400BadRequest)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "index", Justification = "Imported from ServiceStack")]    public async Task<ActionResult> SetItemImage(        [FromRoute, Required] Guid itemId,        [FromRoute, Required] ImageType imageType)    {        var item = _libraryManager.GetItemById<BaseItem>(itemId, User.GetUserId());        if (item is null)        {            return NotFound();        }        if (!TryGetImageExtensionFromContentType(Request.ContentType, out _))        {            return BadRequest("Incorrect ContentType.");        }        var stream = GetFromBase64Stream(Request.Body);        await using (stream.ConfigureAwait(false))        {            // Handle image/png; charset=utf-8            var mimeType = Request.ContentType?.Split(';').FirstOrDefault();            await _providerManager.SaveImage(item, stream, mimeType, imageType, null, CancellationToken.None).ConfigureAwait(false);            await item.UpdateToRepositoryAsync(ItemUpdateType.ImageUpdate, CancellationToken.None).ConfigureAwait(false);            return NoContent();        }    }    /// <summary>    /// Set item image.    /// </summary>    /// <param name="itemId">Item id.</param>    /// <param name="imageType">Image type.</param>    /// <param name="imageIndex">(Unused) Image index.</param>    /// <response code="204">Image saved.</response>    /// <response code="404">Item not found.</response>    /// <returns>A <see cref="NoContentResult"/> on success, or a <see cref="NotFoundResult"/> if item not found.</returns>    [HttpPost("Items/{itemId}/Images/{imageType}/{imageIndex}")]    [Authorize(Policy = Policies.RequiresElevation)]    [AcceptsImageFile]    [ProducesResponseType(StatusCodes.Status204NoContent)]    [ProducesResponseType(StatusCodes.Status400BadRequest)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "index", Justification = "Imported from ServiceStack")]    public async Task<ActionResult> SetItemImageByIndex(        [FromRoute, Required] Guid itemId,        [FromRoute, Required] ImageType imageType,        [FromRoute] int imageIndex)    {        var item = _libraryManager.GetItemById<BaseItem>(itemId, User.GetUserId());        if (item is null)        {            return NotFound();        }        if (!TryGetImageExtensionFromContentType(Request.ContentType, out _))        {            return BadRequest("Incorrect ContentType.");        }        var stream = GetFromBase64Stream(Request.Body);        await using (stream.ConfigureAwait(false))        {            // Handle image/png; charset=utf-8            var mimeType = Request.ContentType?.Split(';').FirstOrDefault();            await _providerManager.SaveImage(item, stream, mimeType, imageType, null, CancellationToken.None).ConfigureAwait(false);            await item.UpdateToRepositoryAsync(ItemUpdateType.ImageUpdate, CancellationToken.None).ConfigureAwait(false);            return NoContent();        }    }    /// <summary>    /// Updates the index for an item image.    /// </summary>    /// <param name="itemId">Item id.</param>    /// <param name="imageType">Image type.</param>    /// <param name="imageIndex">Old image index.</param>    /// <param name="newIndex">New image index.</param>    /// <response code="204">Image index updated.</response>    /// <response code="404">Item not found.</response>    /// <returns>A <see cref="NoContentResult"/> on success, or a <see cref="NotFoundResult"/> if item not found.</returns>    [HttpPost("Items/{itemId}/Images/{imageType}/{imageIndex}/Index")]    [Authorize(Policy = Policies.RequiresElevation)]    [ProducesResponseType(StatusCodes.Status204NoContent)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    public async Task<ActionResult> UpdateItemImageIndex(        [FromRoute, Required] Guid itemId,        [FromRoute, Required] ImageType imageType,        [FromRoute, Required] int imageIndex,        [FromQuery, Required] int newIndex)    {        var item = _libraryManager.GetItemById<BaseItem>(itemId, User.GetUserId());        if (item is null)        {            return NotFound();        }        await item.SwapImagesAsync(imageType, imageIndex, newIndex).ConfigureAwait(false);        return NoContent();    }    /// <summary>    /// Get item image infos.    /// </summary>    /// <param name="itemId">Item id.</param>    /// <response code="200">Item images returned.</response>    /// <response code="404">Item not found.</response>    /// <returns>The list of image infos on success, or <see cref="NotFoundResult"/> if item not found.</returns>    [HttpGet("Items/{itemId}/Images")]    [Authorize]    [ProducesResponseType(StatusCodes.Status200OK)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    public async Task<ActionResult<IEnumerable<ImageInfo>>> GetItemImageInfos([FromRoute, Required] Guid itemId)    {        var item = _libraryManager.GetItemById<BaseItem>(itemId, User.GetUserId());        if (item is null)        {            return NotFound();        }        var list = new List<ImageInfo>();        var itemImages = item.ImageInfos;        if (itemImages.Length == 0)        {            // short-circuit            return list;        }        await _libraryManager.UpdateImagesAsync(item).ConfigureAwait(false); // this makes sure dimensions and hashes are correct        foreach (var image in itemImages)        {            if (!item.AllowsMultipleImages(image.Type))            {                var info = GetImageInfo(item, image, null);                if (info is not null)                {                    list.Add(info);                }            }        }        foreach (var imageType in itemImages.Select(i => i.Type).Distinct().Where(item.AllowsMultipleImages))        {            var index = 0;            // Prevent implicitly captured closure            var currentImageType = imageType;            foreach (var image in itemImages.Where(i => i.Type == currentImageType))            {                var info = GetImageInfo(item, image, index);                if (info is not null)                {                    list.Add(info);                }                index++;            }        }        return list;    }    /// <summary>    /// Gets the item's image.    /// </summary>    /// <param name="itemId">Item id.</param>    /// <param name="imageType">Image type.</param>    /// <param name="maxWidth">The maximum image width to return.</param>    /// <param name="maxHeight">The maximum image height to return.</param>    /// <param name="width">The fixed image width to return.</param>    /// <param name="height">The fixed image height to return.</param>    /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>    /// <param name="fillWidth">Width of box to fill.</param>    /// <param name="fillHeight">Height of box to fill.</param>    /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>    /// <param name="format">Optional. The <see cref="ImageFormat"/> of the returned image.</param>    /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>    /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>    /// <param name="blur">Optional. Blur image.</param>    /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>    /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>    /// <param name="imageIndex">Image index.</param>    /// <response code="200">Image stream returned.</response>    /// <response code="404">Item not found.</response>    /// <returns>    /// A <see cref="FileStreamResult"/> containing the file stream on success,    /// or a <see cref="NotFoundResult"/> if item not found.    /// </returns>    [HttpGet("Items/{itemId}/Images/{imageType}")]    [HttpHead("Items/{itemId}/Images/{imageType}", Name = "HeadItemImage")]    [ProducesResponseType(StatusCodes.Status200OK)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    [ProducesImageFile]    public async Task<ActionResult> GetItemImage(        [FromRoute, Required] Guid itemId,        [FromRoute, Required] ImageType imageType,        [FromQuery] int? maxWidth,        [FromQuery] int? maxHeight,        [FromQuery] int? width,        [FromQuery] int? height,        [FromQuery] int? quality,        [FromQuery] int? fillWidth,        [FromQuery] int? fillHeight,        [FromQuery] string? tag,        [FromQuery] ImageFormat? format,        [FromQuery] double? percentPlayed,        [FromQuery] int? unplayedCount,        [FromQuery] int? blur,        [FromQuery] string? backgroundColor,        [FromQuery] string? foregroundLayer,        [FromQuery] int? imageIndex)    {        var item = _libraryManager.GetItemById<BaseItem>(itemId, User.GetUserId());        if (item is null)        {            return NotFound();        }        return await GetImageInternal(                itemId,                imageType,                imageIndex,                tag,                format,                maxWidth,                maxHeight,                percentPlayed,                unplayedCount,                width,                height,                quality,                fillWidth,                fillHeight,                blur,                backgroundColor,                foregroundLayer,                item)            .ConfigureAwait(false);    }    /// <summary>    /// Gets the item's image.    /// </summary>    /// <param name="itemId">Item id.</param>    /// <param name="imageType">Image type.</param>    /// <param name="imageIndex">Image index.</param>    /// <param name="maxWidth">The maximum image width to return.</param>    /// <param name="maxHeight">The maximum image height to return.</param>    /// <param name="width">The fixed image width to return.</param>    /// <param name="height">The fixed image height to return.</param>    /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>    /// <param name="fillWidth">Width of box to fill.</param>    /// <param name="fillHeight">Height of box to fill.</param>    /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>    /// <param name="format">Optional. The <see cref="ImageFormat"/> of the returned image.</param>    /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>    /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>    /// <param name="blur">Optional. Blur image.</param>    /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>    /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>    /// <response code="200">Image stream returned.</response>    /// <response code="404">Item not found.</response>    /// <returns>    /// A <see cref="FileStreamResult"/> containing the file stream on success,    /// or a <see cref="NotFoundResult"/> if item not found.    /// </returns>    [HttpGet("Items/{itemId}/Images/{imageType}/{imageIndex}")]    [HttpHead("Items/{itemId}/Images/{imageType}/{imageIndex}", Name = "HeadItemImageByIndex")]    [ProducesResponseType(StatusCodes.Status200OK)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    [ProducesImageFile]    public async Task<ActionResult> GetItemImageByIndex(        [FromRoute, Required] Guid itemId,        [FromRoute, Required] ImageType imageType,        [FromRoute] int imageIndex,        [FromQuery] int? maxWidth,        [FromQuery] int? maxHeight,        [FromQuery] int? width,        [FromQuery] int? height,        [FromQuery] int? quality,        [FromQuery] int? fillWidth,        [FromQuery] int? fillHeight,        [FromQuery] string? tag,        [FromQuery] ImageFormat? format,        [FromQuery] double? percentPlayed,        [FromQuery] int? unplayedCount,        [FromQuery] int? blur,        [FromQuery] string? backgroundColor,        [FromQuery] string? foregroundLayer)    {        var item = _libraryManager.GetItemById<BaseItem>(itemId, User.GetUserId());        if (item is null)        {            return NotFound();        }        return await GetImageInternal(                itemId,                imageType,                imageIndex,                tag,                format,                maxWidth,                maxHeight,                percentPlayed,                unplayedCount,                width,                height,                quality,                fillWidth,                fillHeight,                blur,                backgroundColor,                foregroundLayer,                item)            .ConfigureAwait(false);    }    /// <summary>    /// Gets the item's image.    /// </summary>    /// <param name="itemId">Item id.</param>    /// <param name="imageType">Image type.</param>    /// <param name="maxWidth">The maximum image width to return.</param>    /// <param name="maxHeight">The maximum image height to return.</param>    /// <param name="width">The fixed image width to return.</param>    /// <param name="height">The fixed image height to return.</param>    /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>    /// <param name="fillWidth">Width of box to fill.</param>    /// <param name="fillHeight">Height of box to fill.</param>    /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>    /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>    /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>    /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>    /// <param name="blur">Optional. Blur image.</param>    /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>    /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>    /// <param name="imageIndex">Image index.</param>    /// <response code="200">Image stream returned.</response>    /// <response code="404">Item not found.</response>    /// <returns>    /// A <see cref="FileStreamResult"/> containing the file stream on success,    /// or a <see cref="NotFoundResult"/> if item not found.    /// </returns>    [HttpGet("Items/{itemId}/Images/{imageType}/{imageIndex}/{tag}/{format}/{maxWidth}/{maxHeight}/{percentPlayed}/{unplayedCount}")]    [HttpHead("Items/{itemId}/Images/{imageType}/{imageIndex}/{tag}/{format}/{maxWidth}/{maxHeight}/{percentPlayed}/{unplayedCount}", Name = "HeadItemImage2")]    [ProducesResponseType(StatusCodes.Status200OK)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    [ProducesImageFile]    public async Task<ActionResult> GetItemImage2(        [FromRoute, Required] Guid itemId,        [FromRoute, Required] ImageType imageType,        [FromRoute, Required] int maxWidth,        [FromRoute, Required] int maxHeight,        [FromQuery] int? width,        [FromQuery] int? height,        [FromQuery] int? quality,        [FromQuery] int? fillWidth,        [FromQuery] int? fillHeight,        [FromRoute, Required] string tag,        [FromRoute, Required] ImageFormat format,        [FromRoute, Required] double percentPlayed,        [FromRoute, Required] int unplayedCount,        [FromQuery] int? blur,        [FromQuery] string? backgroundColor,        [FromQuery] string? foregroundLayer,        [FromRoute, Required] int imageIndex)    {        var item = _libraryManager.GetItemById<BaseItem>(itemId, User.GetUserId());        if (item is null)        {            return NotFound();        }        return await GetImageInternal(                itemId,                imageType,                imageIndex,                tag,                format,                maxWidth,                maxHeight,                percentPlayed,                unplayedCount,                width,                height,                quality,                fillWidth,                fillHeight,                blur,                backgroundColor,                foregroundLayer,                item)            .ConfigureAwait(false);    }    /// <summary>    /// Get artist image by name.    /// </summary>    /// <param name="name">Artist name.</param>    /// <param name="imageType">Image type.</param>    /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>    /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>    /// <param name="maxWidth">The maximum image width to return.</param>    /// <param name="maxHeight">The maximum image height to return.</param>    /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>    /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>    /// <param name="width">The fixed image width to return.</param>    /// <param name="height">The fixed image height to return.</param>    /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>    /// <param name="fillWidth">Width of box to fill.</param>    /// <param name="fillHeight">Height of box to fill.</param>    /// <param name="blur">Optional. Blur image.</param>    /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>    /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>    /// <param name="imageIndex">Image index.</param>    /// <response code="200">Image stream returned.</response>    /// <response code="404">Item not found.</response>    /// <returns>    /// A <see cref="FileStreamResult"/> containing the file stream on success,    /// or a <see cref="NotFoundResult"/> if item not found.    /// </returns>    [HttpGet("Artists/{name}/Images/{imageType}/{imageIndex}")]    [HttpHead("Artists/{name}/Images/{imageType}/{imageIndex}", Name = "HeadArtistImage")]    [ProducesResponseType(StatusCodes.Status200OK)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    [ProducesImageFile]    public async Task<ActionResult> GetArtistImage(        [FromRoute, Required] string name,        [FromRoute, Required] ImageType imageType,        [FromQuery] string? tag,        [FromQuery] ImageFormat? format,        [FromQuery] int? maxWidth,        [FromQuery] int? maxHeight,        [FromQuery] double? percentPlayed,        [FromQuery] int? unplayedCount,        [FromQuery] int? width,        [FromQuery] int? height,        [FromQuery] int? quality,        [FromQuery] int? fillWidth,        [FromQuery] int? fillHeight,        [FromQuery] int? blur,        [FromQuery] string? backgroundColor,        [FromQuery] string? foregroundLayer,        [FromRoute, Required] int imageIndex)    {        var item = _libraryManager.GetArtist(name);        if (item is null)        {            return NotFound();        }        return await GetImageInternal(                item.Id,                imageType,                imageIndex,                tag,                format,                maxWidth,                maxHeight,                percentPlayed,                unplayedCount,                width,                height,                quality,                fillWidth,                fillHeight,                blur,                backgroundColor,                foregroundLayer,                item)            .ConfigureAwait(false);    }    /// <summary>    /// Get genre image by name.    /// </summary>    /// <param name="name">Genre name.</param>    /// <param name="imageType">Image type.</param>    /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>    /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>    /// <param name="maxWidth">The maximum image width to return.</param>    /// <param name="maxHeight">The maximum image height to return.</param>    /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>    /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>    /// <param name="width">The fixed image width to return.</param>    /// <param name="height">The fixed image height to return.</param>    /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>    /// <param name="fillWidth">Width of box to fill.</param>    /// <param name="fillHeight">Height of box to fill.</param>    /// <param name="blur">Optional. Blur image.</param>    /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>    /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>    /// <param name="imageIndex">Image index.</param>    /// <response code="200">Image stream returned.</response>    /// <response code="404">Item not found.</response>    /// <returns>    /// A <see cref="FileStreamResult"/> containing the file stream on success,    /// or a <see cref="NotFoundResult"/> if item not found.    /// </returns>    [HttpGet("Genres/{name}/Images/{imageType}")]    [HttpHead("Genres/{name}/Images/{imageType}", Name = "HeadGenreImage")]    [ProducesResponseType(StatusCodes.Status200OK)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    [ProducesImageFile]    public async Task<ActionResult> GetGenreImage(        [FromRoute, Required] string name,        [FromRoute, Required] ImageType imageType,        [FromQuery] string? tag,        [FromQuery] ImageFormat? format,        [FromQuery] int? maxWidth,        [FromQuery] int? maxHeight,        [FromQuery] double? percentPlayed,        [FromQuery] int? unplayedCount,        [FromQuery] int? width,        [FromQuery] int? height,        [FromQuery] int? quality,        [FromQuery] int? fillWidth,        [FromQuery] int? fillHeight,        [FromQuery] int? blur,        [FromQuery] string? backgroundColor,        [FromQuery] string? foregroundLayer,        [FromQuery] int? imageIndex)    {        var item = _libraryManager.GetGenre(name);        if (item is null)        {            return NotFound();        }        return await GetImageInternal(                item.Id,                imageType,                imageIndex,                tag,                format,                maxWidth,                maxHeight,                percentPlayed,                unplayedCount,                width,                height,                quality,                fillWidth,                fillHeight,                blur,                backgroundColor,                foregroundLayer,                item)            .ConfigureAwait(false);    }    /// <summary>    /// Get genre image by name.    /// </summary>    /// <param name="name">Genre name.</param>    /// <param name="imageType">Image type.</param>    /// <param name="imageIndex">Image index.</param>    /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>    /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>    /// <param name="maxWidth">The maximum image width to return.</param>    /// <param name="maxHeight">The maximum image height to return.</param>    /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>    /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>    /// <param name="width">The fixed image width to return.</param>    /// <param name="height">The fixed image height to return.</param>    /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>    /// <param name="fillWidth">Width of box to fill.</param>    /// <param name="fillHeight">Height of box to fill.</param>    /// <param name="blur">Optional. Blur image.</param>    /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>    /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>    /// <response code="200">Image stream returned.</response>    /// <response code="404">Item not found.</response>    /// <returns>    /// A <see cref="FileStreamResult"/> containing the file stream on success,    /// or a <see cref="NotFoundResult"/> if item not found.    /// </returns>    [HttpGet("Genres/{name}/Images/{imageType}/{imageIndex}")]    [HttpHead("Genres/{name}/Images/{imageType}/{imageIndex}", Name = "HeadGenreImageByIndex")]    [ProducesResponseType(StatusCodes.Status200OK)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    [ProducesImageFile]    public async Task<ActionResult> GetGenreImageByIndex(        [FromRoute, Required] string name,        [FromRoute, Required] ImageType imageType,        [FromRoute, Required] int imageIndex,        [FromQuery] string? tag,        [FromQuery] ImageFormat? format,        [FromQuery] int? maxWidth,        [FromQuery] int? maxHeight,        [FromQuery] double? percentPlayed,        [FromQuery] int? unplayedCount,        [FromQuery] int? width,        [FromQuery] int? height,        [FromQuery] int? quality,        [FromQuery] int? fillWidth,        [FromQuery] int? fillHeight,        [FromQuery] int? blur,        [FromQuery] string? backgroundColor,        [FromQuery] string? foregroundLayer)    {        var item = _libraryManager.GetGenre(name);        if (item is null)        {            return NotFound();        }        return await GetImageInternal(                item.Id,                imageType,                imageIndex,                tag,                format,                maxWidth,                maxHeight,                percentPlayed,                unplayedCount,                width,                height,                quality,                fillWidth,                fillHeight,                blur,                backgroundColor,                foregroundLayer,                item)            .ConfigureAwait(false);    }    /// <summary>    /// Get music genre image by name.    /// </summary>    /// <param name="name">Music genre name.</param>    /// <param name="imageType">Image type.</param>    /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>    /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>    /// <param name="maxWidth">The maximum image width to return.</param>    /// <param name="maxHeight">The maximum image height to return.</param>    /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>    /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>    /// <param name="width">The fixed image width to return.</param>    /// <param name="height">The fixed image height to return.</param>    /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>    /// <param name="fillWidth">Width of box to fill.</param>    /// <param name="fillHeight">Height of box to fill.</param>    /// <param name="blur">Optional. Blur image.</param>    /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>    /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>    /// <param name="imageIndex">Image index.</param>    /// <response code="200">Image stream returned.</response>    /// <response code="404">Item not found.</response>    /// <returns>    /// A <see cref="FileStreamResult"/> containing the file stream on success,    /// or a <see cref="NotFoundResult"/> if item not found.    /// </returns>    [HttpGet("MusicGenres/{name}/Images/{imageType}")]    [HttpHead("MusicGenres/{name}/Images/{imageType}", Name = "HeadMusicGenreImage")]    [ProducesResponseType(StatusCodes.Status200OK)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    [ProducesImageFile]    public async Task<ActionResult> GetMusicGenreImage(        [FromRoute, Required] string name,        [FromRoute, Required] ImageType imageType,        [FromQuery] string? tag,        [FromQuery] ImageFormat? format,        [FromQuery] int? maxWidth,        [FromQuery] int? maxHeight,        [FromQuery] double? percentPlayed,        [FromQuery] int? unplayedCount,        [FromQuery] int? width,        [FromQuery] int? height,        [FromQuery] int? quality,        [FromQuery] int? fillWidth,        [FromQuery] int? fillHeight,        [FromQuery] int? blur,        [FromQuery] string? backgroundColor,        [FromQuery] string? foregroundLayer,        [FromQuery] int? imageIndex)    {        var item = _libraryManager.GetMusicGenre(name);        if (item is null)        {            return NotFound();        }        return await GetImageInternal(                item.Id,                imageType,                imageIndex,                tag,                format,                maxWidth,                maxHeight,                percentPlayed,                unplayedCount,                width,                height,                quality,                fillWidth,                fillHeight,                blur,                backgroundColor,                foregroundLayer,                item)            .ConfigureAwait(false);    }    /// <summary>    /// Get music genre image by name.    /// </summary>    /// <param name="name">Music genre name.</param>    /// <param name="imageType">Image type.</param>    /// <param name="imageIndex">Image index.</param>    /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>    /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>    /// <param name="maxWidth">The maximum image width to return.</param>    /// <param name="maxHeight">The maximum image height to return.</param>    /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>    /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>    /// <param name="width">The fixed image width to return.</param>    /// <param name="height">The fixed image height to return.</param>    /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>    /// <param name="fillWidth">Width of box to fill.</param>    /// <param name="fillHeight">Height of box to fill.</param>    /// <param name="blur">Optional. Blur image.</param>    /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>    /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>    /// <response code="200">Image stream returned.</response>    /// <response code="404">Item not found.</response>    /// <returns>    /// A <see cref="FileStreamResult"/> containing the file stream on success,    /// or a <see cref="NotFoundResult"/> if item not found.    /// </returns>    [HttpGet("MusicGenres/{name}/Images/{imageType}/{imageIndex}")]    [HttpHead("MusicGenres/{name}/Images/{imageType}/{imageIndex}", Name = "HeadMusicGenreImageByIndex")]    [ProducesResponseType(StatusCodes.Status200OK)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    [ProducesImageFile]    public async Task<ActionResult> GetMusicGenreImageByIndex(        [FromRoute, Required] string name,        [FromRoute, Required] ImageType imageType,        [FromRoute, Required] int imageIndex,        [FromQuery] string? tag,        [FromQuery] ImageFormat? format,        [FromQuery] int? maxWidth,        [FromQuery] int? maxHeight,        [FromQuery] double? percentPlayed,        [FromQuery] int? unplayedCount,        [FromQuery] int? width,        [FromQuery] int? height,        [FromQuery] int? quality,        [FromQuery] int? fillWidth,        [FromQuery] int? fillHeight,        [FromQuery] int? blur,        [FromQuery] string? backgroundColor,        [FromQuery] string? foregroundLayer)    {        var item = _libraryManager.GetMusicGenre(name);        if (item is null)        {            return NotFound();        }        return await GetImageInternal(                item.Id,                imageType,                imageIndex,                tag,                format,                maxWidth,                maxHeight,                percentPlayed,                unplayedCount,                width,                height,                quality,                fillWidth,                fillHeight,                blur,                backgroundColor,                foregroundLayer,                item)            .ConfigureAwait(false);    }    /// <summary>    /// Get person image by name.    /// </summary>    /// <param name="name">Person name.</param>    /// <param name="imageType">Image type.</param>    /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>    /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>    /// <param name="maxWidth">The maximum image width to return.</param>    /// <param name="maxHeight">The maximum image height to return.</param>    /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>    /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>    /// <param name="width">The fixed image width to return.</param>    /// <param name="height">The fixed image height to return.</param>    /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>    /// <param name="fillWidth">Width of box to fill.</param>    /// <param name="fillHeight">Height of box to fill.</param>    /// <param name="blur">Optional. Blur image.</param>    /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>    /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>    /// <param name="imageIndex">Image index.</param>    /// <response code="200">Image stream returned.</response>    /// <response code="404">Item not found.</response>    /// <returns>    /// A <see cref="FileStreamResult"/> containing the file stream on success,    /// or a <see cref="NotFoundResult"/> if item not found.    /// </returns>    [HttpGet("Persons/{name}/Images/{imageType}")]    [HttpHead("Persons/{name}/Images/{imageType}", Name = "HeadPersonImage")]    [ProducesResponseType(StatusCodes.Status200OK)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    [ProducesImageFile]    public async Task<ActionResult> GetPersonImage(        [FromRoute, Required] string name,        [FromRoute, Required] ImageType imageType,        [FromQuery] string? tag,        [FromQuery] ImageFormat? format,        [FromQuery] int? maxWidth,        [FromQuery] int? maxHeight,        [FromQuery] double? percentPlayed,        [FromQuery] int? unplayedCount,        [FromQuery] int? width,        [FromQuery] int? height,        [FromQuery] int? quality,        [FromQuery] int? fillWidth,        [FromQuery] int? fillHeight,        [FromQuery] int? blur,        [FromQuery] string? backgroundColor,        [FromQuery] string? foregroundLayer,        [FromQuery] int? imageIndex)    {        var item = _libraryManager.GetPerson(name);        if (item is null)        {            return NotFound();        }        return await GetImageInternal(                item.Id,                imageType,                imageIndex,                tag,                format,                maxWidth,                maxHeight,                percentPlayed,                unplayedCount,                width,                height,                quality,                fillWidth,                fillHeight,                blur,                backgroundColor,                foregroundLayer,                item)            .ConfigureAwait(false);    }    /// <summary>    /// Get person image by name.    /// </summary>    /// <param name="name">Person name.</param>    /// <param name="imageType">Image type.</param>    /// <param name="imageIndex">Image index.</param>    /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>    /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>    /// <param name="maxWidth">The maximum image width to return.</param>    /// <param name="maxHeight">The maximum image height to return.</param>    /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>    /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>    /// <param name="width">The fixed image width to return.</param>    /// <param name="height">The fixed image height to return.</param>    /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>    /// <param name="fillWidth">Width of box to fill.</param>    /// <param name="fillHeight">Height of box to fill.</param>    /// <param name="blur">Optional. Blur image.</param>    /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>    /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>    /// <response code="200">Image stream returned.</response>    /// <response code="404">Item not found.</response>    /// <returns>    /// A <see cref="FileStreamResult"/> containing the file stream on success,    /// or a <see cref="NotFoundResult"/> if item not found.    /// </returns>    [HttpGet("Persons/{name}/Images/{imageType}/{imageIndex}")]    [HttpHead("Persons/{name}/Images/{imageType}/{imageIndex}", Name = "HeadPersonImageByIndex")]    [ProducesResponseType(StatusCodes.Status200OK)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    [ProducesImageFile]    public async Task<ActionResult> GetPersonImageByIndex(        [FromRoute, Required] string name,        [FromRoute, Required] ImageType imageType,        [FromRoute, Required] int imageIndex,        [FromQuery] string? tag,        [FromQuery] ImageFormat? format,        [FromQuery] int? maxWidth,        [FromQuery] int? maxHeight,        [FromQuery] double? percentPlayed,        [FromQuery] int? unplayedCount,        [FromQuery] int? width,        [FromQuery] int? height,        [FromQuery] int? quality,        [FromQuery] int? fillWidth,        [FromQuery] int? fillHeight,        [FromQuery] int? blur,        [FromQuery] string? backgroundColor,        [FromQuery] string? foregroundLayer)    {        var item = _libraryManager.GetPerson(name);        if (item is null)        {            return NotFound();        }        return await GetImageInternal(                item.Id,                imageType,                imageIndex,                tag,                format,                maxWidth,                maxHeight,                percentPlayed,                unplayedCount,                width,                height,                quality,                fillWidth,                fillHeight,                blur,                backgroundColor,                foregroundLayer,                item)            .ConfigureAwait(false);    }    /// <summary>    /// Get studio image by name.    /// </summary>    /// <param name="name">Studio name.</param>    /// <param name="imageType">Image type.</param>    /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>    /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>    /// <param name="maxWidth">The maximum image width to return.</param>    /// <param name="maxHeight">The maximum image height to return.</param>    /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>    /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>    /// <param name="width">The fixed image width to return.</param>    /// <param name="height">The fixed image height to return.</param>    /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>    /// <param name="fillWidth">Width of box to fill.</param>    /// <param name="fillHeight">Height of box to fill.</param>    /// <param name="blur">Optional. Blur image.</param>    /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>    /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>    /// <param name="imageIndex">Image index.</param>    /// <response code="200">Image stream returned.</response>    /// <response code="404">Item not found.</response>    /// <returns>    /// A <see cref="FileStreamResult"/> containing the file stream on success,    /// or a <see cref="NotFoundResult"/> if item not found.    /// </returns>    [HttpGet("Studios/{name}/Images/{imageType}")]    [HttpHead("Studios/{name}/Images/{imageType}", Name = "HeadStudioImage")]    [ProducesResponseType(StatusCodes.Status200OK)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    [ProducesImageFile]    public async Task<ActionResult> GetStudioImage(        [FromRoute, Required] string name,        [FromRoute, Required] ImageType imageType,        [FromQuery] string? tag,        [FromQuery] ImageFormat? format,        [FromQuery] int? maxWidth,        [FromQuery] int? maxHeight,        [FromQuery] double? percentPlayed,        [FromQuery] int? unplayedCount,        [FromQuery] int? width,        [FromQuery] int? height,        [FromQuery] int? quality,        [FromQuery] int? fillWidth,        [FromQuery] int? fillHeight,        [FromQuery] int? blur,        [FromQuery] string? backgroundColor,        [FromQuery] string? foregroundLayer,        [FromQuery] int? imageIndex)    {        var item = _libraryManager.GetStudio(name);        if (item is null)        {            return NotFound();        }        return await GetImageInternal(                item.Id,                imageType,                imageIndex,                tag,                format,                maxWidth,                maxHeight,                percentPlayed,                unplayedCount,                width,                height,                quality,                fillWidth,                fillHeight,                blur,                backgroundColor,                foregroundLayer,                item)            .ConfigureAwait(false);    }    /// <summary>    /// Get studio image by name.    /// </summary>    /// <param name="name">Studio name.</param>    /// <param name="imageType">Image type.</param>    /// <param name="imageIndex">Image index.</param>    /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>    /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>    /// <param name="maxWidth">The maximum image width to return.</param>    /// <param name="maxHeight">The maximum image height to return.</param>    /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>    /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>    /// <param name="width">The fixed image width to return.</param>    /// <param name="height">The fixed image height to return.</param>    /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>    /// <param name="fillWidth">Width of box to fill.</param>    /// <param name="fillHeight">Height of box to fill.</param>    /// <param name="blur">Optional. Blur image.</param>    /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>    /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>    /// <response code="200">Image stream returned.</response>    /// <response code="404">Item not found.</response>    /// <returns>    /// A <see cref="FileStreamResult"/> containing the file stream on success,    /// or a <see cref="NotFoundResult"/> if item not found.    /// </returns>    [HttpGet("Studios/{name}/Images/{imageType}/{imageIndex}")]    [HttpHead("Studios/{name}/Images/{imageType}/{imageIndex}", Name = "HeadStudioImageByIndex")]    [ProducesResponseType(StatusCodes.Status200OK)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    [ProducesImageFile]    public async Task<ActionResult> GetStudioImageByIndex(        [FromRoute, Required] string name,        [FromRoute, Required] ImageType imageType,        [FromRoute, Required] int imageIndex,        [FromQuery] string? tag,        [FromQuery] ImageFormat? format,        [FromQuery] int? maxWidth,        [FromQuery] int? maxHeight,        [FromQuery] double? percentPlayed,        [FromQuery] int? unplayedCount,        [FromQuery] int? width,        [FromQuery] int? height,        [FromQuery] int? quality,        [FromQuery] int? fillWidth,        [FromQuery] int? fillHeight,        [FromQuery] int? blur,        [FromQuery] string? backgroundColor,        [FromQuery] string? foregroundLayer)    {        var item = _libraryManager.GetStudio(name);        if (item is null)        {            return NotFound();        }        return await GetImageInternal(                item.Id,                imageType,                imageIndex,                tag,                format,                maxWidth,                maxHeight,                percentPlayed,                unplayedCount,                width,                height,                quality,                fillWidth,                fillHeight,                blur,                backgroundColor,                foregroundLayer,                item)            .ConfigureAwait(false);    }    /// <summary>    /// Get user profile image.    /// </summary>    /// <param name="userId">User id.</param>    /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>    /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>    /// <param name="maxWidth">The maximum image width to return.</param>    /// <param name="maxHeight">The maximum image height to return.</param>    /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>    /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>    /// <param name="width">The fixed image width to return.</param>    /// <param name="height">The fixed image height to return.</param>    /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>    /// <param name="fillWidth">Width of box to fill.</param>    /// <param name="fillHeight">Height of box to fill.</param>    /// <param name="blur">Optional. Blur image.</param>    /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>    /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>    /// <param name="imageIndex">Image index.</param>    /// <response code="200">Image stream returned.</response>    /// <response code="400">User id not provided.</response>    /// <response code="404">Item not found.</response>    /// <returns>    /// A <see cref="FileStreamResult"/> containing the file stream on success,    /// or a <see cref="NotFoundResult"/> if item not found.    /// </returns>    [HttpGet("UserImage")]    [HttpHead("UserImage", Name = "HeadUserImage")]    [ProducesResponseType(StatusCodes.Status200OK)]    [ProducesResponseType(StatusCodes.Status400BadRequest)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    [ProducesImageFile]    public async Task<ActionResult> GetUserImage(        [FromQuery] Guid? userId,        [FromQuery] string? tag,        [FromQuery] ImageFormat? format,        [FromQuery] int? maxWidth,        [FromQuery] int? maxHeight,        [FromQuery] double? percentPlayed,        [FromQuery] int? unplayedCount,        [FromQuery] int? width,        [FromQuery] int? height,        [FromQuery] int? quality,        [FromQuery] int? fillWidth,        [FromQuery] int? fillHeight,        [FromQuery] int? blur,        [FromQuery] string? backgroundColor,        [FromQuery] string? foregroundLayer,        [FromQuery] int? imageIndex)    {        var requestUserId = userId ?? User.GetUserId();        if (requestUserId.IsEmpty())        {            return BadRequest("UserId is required if unauthenticated");        }        var user = _userManager.GetUserById(requestUserId);        if (user?.ProfileImage is null)        {            return NotFound();        }        var info = new ItemImageInfo        {            Path = user.ProfileImage.Path,            Type = ImageType.Profile,            DateModified = user.ProfileImage.LastModified        };        if (width.HasValue)        {            info.Width = width.Value;        }        if (height.HasValue)        {            info.Height = height.Value;        }        return await GetImageInternal(                user.Id,                ImageType.Profile,                imageIndex,                tag,                format,                maxWidth,                maxHeight,                percentPlayed,                unplayedCount,                width,                height,                quality,                fillWidth,                fillHeight,                blur,                backgroundColor,                foregroundLayer,                null,                info)            .ConfigureAwait(false);    }    /// <summary>    /// Get user profile image.    /// </summary>    /// <param name="userId">User id.</param>    /// <param name="imageType">Image type.</param>    /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>    /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>    /// <param name="maxWidth">The maximum image width to return.</param>    /// <param name="maxHeight">The maximum image height to return.</param>    /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>    /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>    /// <param name="width">The fixed image width to return.</param>    /// <param name="height">The fixed image height to return.</param>    /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>    /// <param name="fillWidth">Width of box to fill.</param>    /// <param name="fillHeight">Height of box to fill.</param>    /// <param name="blur">Optional. Blur image.</param>    /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>    /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>    /// <param name="imageIndex">Image index.</param>    /// <response code="200">Image stream returned.</response>    /// <response code="404">Item not found.</response>    /// <returns>    /// A <see cref="FileStreamResult"/> containing the file stream on success,    /// or a <see cref="NotFoundResult"/> if item not found.    /// </returns>    [HttpGet("Users/{userId}/Images/{imageType}")]    [HttpHead("Users/{userId}/Images/{imageType}", Name = "HeadUserImageLegacy")]    [Obsolete("Kept for backwards compatibility")]    [ApiExplorerSettings(IgnoreApi = true)]    [ProducesResponseType(StatusCodes.Status200OK)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    [ProducesImageFile]    public Task<ActionResult> GetUserImageLegacy(        [FromRoute, Required] Guid userId,        [FromRoute, Required] ImageType imageType,        [FromQuery] string? tag,        [FromQuery] ImageFormat? format,        [FromQuery] int? maxWidth,        [FromQuery] int? maxHeight,        [FromQuery] double? percentPlayed,        [FromQuery] int? unplayedCount,        [FromQuery] int? width,        [FromQuery] int? height,        [FromQuery] int? quality,        [FromQuery] int? fillWidth,        [FromQuery] int? fillHeight,        [FromQuery] int? blur,        [FromQuery] string? backgroundColor,        [FromQuery] string? foregroundLayer,        [FromQuery] int? imageIndex)        => GetUserImage(            userId,            tag,            format,            maxWidth,            maxHeight,            percentPlayed,            unplayedCount,            width,            height,            quality,            fillWidth,            fillHeight,            blur,            backgroundColor,            foregroundLayer,            imageIndex);    /// <summary>    /// Get user profile image.    /// </summary>    /// <param name="userId">User id.</param>    /// <param name="imageType">Image type.</param>    /// <param name="imageIndex">Image index.</param>    /// <param name="tag">Optional. Supply the cache tag from the item object to receive strong caching headers.</param>    /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>    /// <param name="maxWidth">The maximum image width to return.</param>    /// <param name="maxHeight">The maximum image height to return.</param>    /// <param name="percentPlayed">Optional. Percent to render for the percent played overlay.</param>    /// <param name="unplayedCount">Optional. Unplayed count overlay to render.</param>    /// <param name="width">The fixed image width to return.</param>    /// <param name="height">The fixed image height to return.</param>    /// <param name="quality">Optional. Quality setting, from 0-100. Defaults to 90 and should suffice in most cases.</param>    /// <param name="fillWidth">Width of box to fill.</param>    /// <param name="fillHeight">Height of box to fill.</param>    /// <param name="blur">Optional. Blur image.</param>    /// <param name="backgroundColor">Optional. Apply a background color for transparent images.</param>    /// <param name="foregroundLayer">Optional. Apply a foreground layer on top of the image.</param>    /// <response code="200">Image stream returned.</response>    /// <response code="404">Item not found.</response>    /// <returns>    /// A <see cref="FileStreamResult"/> containing the file stream on success,    /// or a <see cref="NotFoundResult"/> if item not found.    /// </returns>    [HttpGet("Users/{userId}/Images/{imageType}/{imageIndex}")]    [HttpHead("Users/{userId}/Images/{imageType}/{imageIndex}", Name = "HeadUserImageByIndexLegacy")]    [Obsolete("Kept for backwards compatibility")]    [ApiExplorerSettings(IgnoreApi = true)]    [ProducesResponseType(StatusCodes.Status200OK)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    [ProducesImageFile]    public Task<ActionResult> GetUserImageByIndexLegacy(        [FromRoute, Required] Guid userId,        [FromRoute, Required] ImageType imageType,        [FromRoute, Required] int imageIndex,        [FromQuery] string? tag,        [FromQuery] ImageFormat? format,        [FromQuery] int? maxWidth,        [FromQuery] int? maxHeight,        [FromQuery] double? percentPlayed,        [FromQuery] int? unplayedCount,        [FromQuery] int? width,        [FromQuery] int? height,        [FromQuery] int? quality,        [FromQuery] int? fillWidth,        [FromQuery] int? fillHeight,        [FromQuery] int? blur,        [FromQuery] string? backgroundColor,        [FromQuery] string? foregroundLayer)        => GetUserImage(            userId,            tag,            format,            maxWidth,            maxHeight,            percentPlayed,            unplayedCount,            width,            height,            quality,            fillWidth,            fillHeight,            blur,            backgroundColor,            foregroundLayer,            imageIndex);    /// <summary>    /// Generates or gets the splashscreen.    /// </summary>    /// <param name="tag">Supply the cache tag from the item object to receive strong caching headers.</param>    /// <param name="format">Determines the output format of the image - original,gif,jpg,png.</param>    /// <param name="maxWidth">The maximum image width to return.</param>    /// <param name="maxHeight">The maximum image height to return.</param>    /// <param name="width">The fixed image width to return.</param>    /// <param name="height">The fixed image height to return.</param>    /// <param name="fillWidth">Width of box to fill.</param>    /// <param name="fillHeight">Height of box to fill.</param>    /// <param name="blur">Blur image.</param>    /// <param name="backgroundColor">Apply a background color for transparent images.</param>    /// <param name="foregroundLayer">Apply a foreground layer on top of the image.</param>    /// <param name="quality">Quality setting, from 0-100.</param>    /// <response code="200">Splashscreen returned successfully.</response>    /// <returns>The splashscreen.</returns>    [HttpGet("Branding/Splashscreen")]    [ProducesResponseType(StatusCodes.Status200OK)]    [ProducesImageFile]    public async Task<ActionResult> GetSplashscreen(        [FromQuery] string? tag,        [FromQuery] ImageFormat? format,        [FromQuery] int? maxWidth,        [FromQuery] int? maxHeight,        [FromQuery] int? width,        [FromQuery] int? height,        [FromQuery] int? fillWidth,        [FromQuery] int? fillHeight,        [FromQuery] int? blur,        [FromQuery] string? backgroundColor,        [FromQuery] string? foregroundLayer,        [FromQuery, Range(0, 100)] int quality = 90)    {        var brandingOptions = _serverConfigurationManager.GetConfiguration<BrandingOptions>("branding");        var isAdmin = User.IsInRole(Constants.UserRoles.Administrator);        if (!brandingOptions.SplashscreenEnabled && !isAdmin)        {            return NotFound();        }        string splashscreenPath;        if (!string.IsNullOrWhiteSpace(brandingOptions.SplashscreenLocation)            && System.IO.File.Exists(brandingOptions.SplashscreenLocation))        {            splashscreenPath = brandingOptions.SplashscreenLocation;        }        else        {            splashscreenPath = Path.Combine(_appPaths.DataPath, "splashscreen.png");            if (!System.IO.File.Exists(splashscreenPath))            {                return NotFound();            }        }        var outputFormats = GetOutputFormats(format);        TimeSpan? cacheDuration = null;        if (!string.IsNullOrEmpty(tag))        {            cacheDuration = TimeSpan.FromDays(365);        }        var options = new ImageProcessingOptions        {            Image = new ItemImageInfo            {                Path = splashscreenPath            },            Height = height,            MaxHeight = maxHeight,            MaxWidth = maxWidth,            FillHeight = fillHeight,            FillWidth = fillWidth,            Quality = quality,            Width = width,            Blur = blur,            BackgroundColor = backgroundColor,            ForegroundLayer = foregroundLayer,            SupportedOutputFormats = outputFormats        };        return await GetImageResult(                options,                cacheDuration,                ImmutableDictionary<string, string>.Empty)            .ConfigureAwait(false);    }    /// <summary>    /// Uploads a custom splashscreen.    /// The body is expected to the image contents base64 encoded.    /// </summary>    /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>    /// <response code="204">Successfully uploaded new splashscreen.</response>    /// <response code="400">Error reading MimeType from uploaded image.</response>    /// <response code="403">User does not have permission to upload splashscreen..</response>    /// <exception cref="ArgumentException">Error reading the image format.</exception>    [HttpPost("Branding/Splashscreen")]    [Authorize(Policy = Policies.RequiresElevation)]    [ProducesResponseType(StatusCodes.Status204NoContent)]    [ProducesResponseType(StatusCodes.Status400BadRequest)]    [ProducesResponseType(StatusCodes.Status403Forbidden)]    [AcceptsImageFile]    public async Task<ActionResult> UploadCustomSplashscreen()    {        if (!TryGetImageExtensionFromContentType(Request.ContentType, out var extension))        {            return BadRequest("Incorrect ContentType.");        }        var stream = GetFromBase64Stream(Request.Body);        await using (stream.ConfigureAwait(false))        {            var filePath = Path.Combine(_appPaths.DataPath, "splashscreen-upload" + extension);            var brandingOptions = _serverConfigurationManager.GetConfiguration<BrandingOptions>("branding");            brandingOptions.SplashscreenLocation = filePath;            _serverConfigurationManager.SaveConfiguration("branding", brandingOptions);            var fs = new FileStream(filePath, FileMode.Create, FileAccess.Write, FileShare.None, IODefaults.FileStreamBufferSize, FileOptions.Asynchronous);            await using (fs.ConfigureAwait(false))            {                await stream.CopyToAsync(fs, CancellationToken.None).ConfigureAwait(false);            }            return NoContent();        }    }    /// <summary>    /// Delete a custom splashscreen.    /// </summary>    /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>    /// <response code="204">Successfully deleted the custom splashscreen.</response>    /// <response code="403">User does not have permission to delete splashscreen..</response>    [HttpDelete("Branding/Splashscreen")]    [Authorize(Policy = Policies.RequiresElevation)]    [ProducesResponseType(StatusCodes.Status204NoContent)]    public ActionResult DeleteCustomSplashscreen()    {        var brandingOptions = _serverConfigurationManager.GetConfiguration<BrandingOptions>("branding");        if (!string.IsNullOrEmpty(brandingOptions.SplashscreenLocation)            && System.IO.File.Exists(brandingOptions.SplashscreenLocation))        {            System.IO.File.Delete(brandingOptions.SplashscreenLocation);            brandingOptions.SplashscreenLocation = null;            _serverConfigurationManager.SaveConfiguration("branding", brandingOptions);        }        return NoContent();    }    private ImageInfo? GetImageInfo(BaseItem item, ItemImageInfo info, int? imageIndex)    {        int? width = null;        int? height = null;        string? blurhash = null;        long length = 0;        try        {            if (info.IsLocalFile)            {                var fileInfo = _fileSystem.GetFileInfo(info.Path);                length = fileInfo.Length;                blurhash = info.BlurHash;                width = info.Width;                height = info.Height;                if (width <= 0 || height <= 0)                {                    width = null;                    height = null;                }            }        }        catch (Exception ex)        {            _logger.LogError(ex, "Error getting image information for {Item}", item.Name);        }        try        {            return new ImageInfo            {                Path = info.Path,                ImageIndex = imageIndex,                ImageType = info.Type,                ImageTag = _imageProcessor.GetImageCacheTag(item, info),                Size = length,                BlurHash = blurhash,                Width = width,                Height = height            };        }        catch (Exception ex)        {            _logger.LogError(ex, "Error getting image information for {Path}", info.Path);            return null;        }    }    private async Task<ActionResult> GetImageInternal(        Guid itemId,        ImageType imageType,        int? imageIndex,        string? tag,        ImageFormat? format,        int? maxWidth,        int? maxHeight,        double? percentPlayed,        int? unplayedCount,        int? width,        int? height,        int? quality,        int? fillWidth,        int? fillHeight,        int? blur,        string? backgroundColor,        string? foregroundLayer,        BaseItem? item,        ItemImageInfo? imageInfo = null)    {        if (percentPlayed.HasValue)        {            if (percentPlayed.Value <= 0)            {                percentPlayed = null;            }            else if (percentPlayed.Value >= 100)            {                percentPlayed = null;            }        }        if (percentPlayed.HasValue)        {            unplayedCount = null;        }        if (unplayedCount.HasValue            && unplayedCount.Value <= 0)        {            unplayedCount = null;        }        if (imageInfo is null)        {            imageInfo = item?.GetImageInfo(imageType, imageIndex ?? 0);            if (imageInfo is null)            {                return NotFound(string.Format(NumberFormatInfo.InvariantInfo, "{0} does not have an image of type {1}", item?.Name, imageType));            }        }        var outputFormats = GetOutputFormats(format);        TimeSpan? cacheDuration = null;        if (!string.IsNullOrEmpty(tag))        {            cacheDuration = TimeSpan.FromDays(365);        }        var responseHeaders = new Dictionary<string, string>        {            { "transferMode.dlna.org", "Interactive" },            { "realTimeInfo.dlna.org", "DLNA.ORG_TLAG=*" }        };        if (!imageInfo.IsLocalFile && item is not null)        {            imageInfo = await _libraryManager.ConvertImageToLocal(item, imageInfo, imageIndex ?? 0).ConfigureAwait(false);        }        var options = new ImageProcessingOptions        {            Height = height,            ImageIndex = imageIndex ?? 0,            Image = imageInfo,            Item = item,            ItemId = itemId,            MaxHeight = maxHeight,            MaxWidth = maxWidth,            FillHeight = fillHeight,            FillWidth = fillWidth,            Quality = quality ?? 100,            Width = width,            PercentPlayed = percentPlayed ?? 0,            UnplayedCount = unplayedCount,            Blur = blur,            BackgroundColor = backgroundColor,            ForegroundLayer = foregroundLayer,            SupportedOutputFormats = outputFormats        };        return await GetImageResult(            options,            cacheDuration,            responseHeaders).ConfigureAwait(false);    }    private ImageFormat[] GetOutputFormats(ImageFormat? format)    {        if (format.HasValue)        {            return [format.Value];        }        return GetClientSupportedFormats();    }    private ImageFormat[] GetClientSupportedFormats()    {        var supportedFormats = Request.Headers.GetCommaSeparatedValues(HeaderNames.Accept);        for (var i = 0; i < supportedFormats.Length; i++)        {            // Remove charsets etc. (anything after semi-colon)            var type = supportedFormats[i];            int index = type.IndexOf(';', StringComparison.Ordinal);            if (index != -1)            {                supportedFormats[i] = type.Substring(0, index);            }        }        var acceptParam = Request.Query[HeaderNames.Accept];        var supportsWebP = SupportsFormat(supportedFormats, acceptParam, ImageFormat.Webp, false);        if (!supportsWebP)        {            var userAgent = Request.Headers[HeaderNames.UserAgent].ToString();            if (userAgent.Contains("crosswalk", StringComparison.OrdinalIgnoreCase)                && userAgent.Contains("android", StringComparison.OrdinalIgnoreCase))            {                supportsWebP = true;            }        }        var formats = new List<ImageFormat>(4);        if (supportsWebP)        {            formats.Add(ImageFormat.Webp);        }        formats.Add(ImageFormat.Jpg);        formats.Add(ImageFormat.Png);        if (SupportsFormat(supportedFormats, acceptParam, ImageFormat.Gif, true))        {            formats.Add(ImageFormat.Gif);        }        return formats.ToArray();    }    private bool SupportsFormat(IReadOnlyCollection<string> requestAcceptTypes, string? acceptParam, ImageFormat format, bool acceptAll)    {        if (requestAcceptTypes.Contains(format.GetMimeType()))        {            return true;        }        if (acceptAll && requestAcceptTypes.Contains("*/*"))        {            return true;        }        // Review if this should be jpeg, jpg or both for ImageFormat.Jpg        var normalized = format.ToString().ToLowerInvariant();        return string.Equals(acceptParam, normalized, StringComparison.OrdinalIgnoreCase);    }    private async Task<ActionResult> GetImageResult(        ImageProcessingOptions imageProcessingOptions,        TimeSpan? cacheDuration,        IDictionary<string, string> headers)    {        var (imagePath, imageContentType, dateImageModified) = await _imageProcessor.ProcessImage(imageProcessingOptions).ConfigureAwait(false);        var disableCaching = Request.Headers[HeaderNames.CacheControl].Contains("no-cache");        var parsingSuccessful = DateTime.TryParse(Request.Headers[HeaderNames.IfModifiedSince], out var ifModifiedSinceHeader);        // if the parsing of the IfModifiedSince header was not successful, disable caching        if (!parsingSuccessful)        {            // disableCaching = true;        }        foreach (var (key, value) in headers)        {            Response.Headers.Append(key, value);        }        Response.ContentType = imageContentType ?? MediaTypeNames.Text.Plain;        Response.Headers.Append(HeaderNames.Age, Convert.ToInt64((DateTime.UtcNow - dateImageModified).TotalSeconds).ToString(CultureInfo.InvariantCulture));        Response.Headers.Append(HeaderNames.Vary, HeaderNames.Accept);        Response.Headers.ContentDisposition = "attachment";        if (disableCaching)        {            Response.Headers.Append(HeaderNames.CacheControl, "no-cache, no-store, must-revalidate");            Response.Headers.Append(HeaderNames.Pragma, "no-cache, no-store, must-revalidate");        }        else        {            if (cacheDuration.HasValue)            {                Response.Headers.Append(HeaderNames.CacheControl, "public, max-age=" + cacheDuration.Value.TotalSeconds);            }            else            {                Response.Headers.Append(HeaderNames.CacheControl, "public");            }            Response.Headers.Append(HeaderNames.LastModified, dateImageModified.ToUniversalTime().ToString("ddd, dd MMM yyyy HH:mm:ss \"GMT\"", CultureInfo.InvariantCulture));            // if the image was not modified since "ifModifiedSinceHeader"-header, return a HTTP status code 304 not modified            if (!(dateImageModified > ifModifiedSinceHeader) && cacheDuration.HasValue)            {                if (ifModifiedSinceHeader.Add(cacheDuration.Value) < DateTime.UtcNow)                {                    Response.StatusCode = StatusCodes.Status304NotModified;                    return new ContentResult();                }            }        }        return PhysicalFile(imagePath, imageContentType ?? MediaTypeNames.Text.Plain);    }    internal static bool TryGetImageExtensionFromContentType(string? contentType, [NotNullWhen(true)] out string? extension)    {        extension = null;        if (string.IsNullOrEmpty(contentType))        {            return false;        }        if (MediaTypeHeaderValue.TryParse(contentType, out var parsedValue)            && parsedValue.MediaType.HasValue            && MimeTypes.IsImage(parsedValue.MediaType.Value))        {            extension = MimeTypes.ToExtension(parsedValue.MediaType.Value);            return extension is not null;        }        return false;    }}
 |