ImageService.cs 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Globalization;
  4. using System.IO;
  5. using System.Linq;
  6. using System.Runtime.CompilerServices;
  7. using System.Threading;
  8. using System.Threading.Tasks;
  9. using MediaBrowser.Common.Extensions;
  10. using MediaBrowser.Controller.Configuration;
  11. using MediaBrowser.Controller.Drawing;
  12. using MediaBrowser.Controller.Dto;
  13. using MediaBrowser.Controller.Entities;
  14. using MediaBrowser.Controller.Library;
  15. using MediaBrowser.Controller.Net;
  16. using MediaBrowser.Controller.Providers;
  17. using MediaBrowser.Model.Drawing;
  18. using MediaBrowser.Model.Dto;
  19. using MediaBrowser.Model.Entities;
  20. using MediaBrowser.Model.IO;
  21. using MediaBrowser.Model.Net;
  22. using MediaBrowser.Model.Services;
  23. using Microsoft.Extensions.Logging;
  24. using Microsoft.Net.Http.Headers;
  25. using User = Jellyfin.Data.Entities.User;
  26. namespace MediaBrowser.Api.Images
  27. {
  28. /// <summary>
  29. /// Class GetItemImage.
  30. /// </summary>
  31. [Route("/Items/{Id}/Images", "GET", Summary = "Gets information about an item's images")]
  32. [Authenticated]
  33. public class GetItemImageInfos : IReturn<List<ImageInfo>>
  34. {
  35. /// <summary>
  36. /// Gets or sets the id.
  37. /// </summary>
  38. /// <value>The id.</value>
  39. [ApiMember(Name = "Id", Description = "Item Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  40. public string Id { get; set; }
  41. }
  42. [Route("/Items/{Id}/Images/{Type}", "GET")]
  43. [Route("/Items/{Id}/Images/{Type}/{Index}", "GET")]
  44. [Route("/Items/{Id}/Images/{Type}", "HEAD")]
  45. [Route("/Items/{Id}/Images/{Type}/{Index}", "HEAD")]
  46. [Route("/Items/{Id}/Images/{Type}/{Index}/{Tag}/{Format}/{MaxWidth}/{MaxHeight}/{PercentPlayed}/{UnplayedCount}", "GET")]
  47. [Route("/Items/{Id}/Images/{Type}/{Index}/{Tag}/{Format}/{MaxWidth}/{MaxHeight}/{PercentPlayed}/{UnplayedCount}", "HEAD")]
  48. public class GetItemImage : ImageRequest
  49. {
  50. /// <summary>
  51. /// Gets or sets the id.
  52. /// </summary>
  53. /// <value>The id.</value>
  54. [ApiMember(Name = "Id", Description = "Item Id", IsRequired = true, DataType = "string", ParameterType = "path")]
  55. public Guid Id { get; set; }
  56. }
  57. /// <summary>
  58. /// Class UpdateItemImageIndex
  59. /// </summary>
  60. [Route("/Items/{Id}/Images/{Type}/{Index}/Index", "POST", Summary = "Updates the index for an item image")]
  61. [Authenticated(Roles = "admin")]
  62. public class UpdateItemImageIndex : IReturnVoid
  63. {
  64. /// <summary>
  65. /// Gets or sets the id.
  66. /// </summary>
  67. /// <value>The id.</value>
  68. [ApiMember(Name = "Id", Description = "Item Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "POST")]
  69. public string Id { get; set; }
  70. /// <summary>
  71. /// Gets or sets the type of the image.
  72. /// </summary>
  73. /// <value>The type of the image.</value>
  74. [ApiMember(Name = "Type", Description = "Image Type", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "POST")]
  75. public ImageType Type { get; set; }
  76. /// <summary>
  77. /// Gets or sets the index.
  78. /// </summary>
  79. /// <value>The index.</value>
  80. [ApiMember(Name = "Index", Description = "Image Index", IsRequired = false, DataType = "int", ParameterType = "query", Verb = "POST")]
  81. public int Index { get; set; }
  82. /// <summary>
  83. /// Gets or sets the new index.
  84. /// </summary>
  85. /// <value>The new index.</value>
  86. [ApiMember(Name = "NewIndex", Description = "The new image index", IsRequired = true, DataType = "string", ParameterType = "query", Verb = "POST")]
  87. public int NewIndex { get; set; }
  88. }
  89. /// <summary>
  90. /// Class GetPersonImage
  91. /// </summary>
  92. [Route("/Artists/{Name}/Images/{Type}", "GET")]
  93. [Route("/Artists/{Name}/Images/{Type}/{Index}", "GET")]
  94. [Route("/Genres/{Name}/Images/{Type}", "GET")]
  95. [Route("/Genres/{Name}/Images/{Type}/{Index}", "GET")]
  96. [Route("/MusicGenres/{Name}/Images/{Type}", "GET")]
  97. [Route("/MusicGenres/{Name}/Images/{Type}/{Index}", "GET")]
  98. [Route("/Persons/{Name}/Images/{Type}", "GET")]
  99. [Route("/Persons/{Name}/Images/{Type}/{Index}", "GET")]
  100. [Route("/Studios/{Name}/Images/{Type}", "GET")]
  101. [Route("/Studios/{Name}/Images/{Type}/{Index}", "GET")]
  102. ////[Route("/Years/{Year}/Images/{Type}", "GET")]
  103. ////[Route("/Years/{Year}/Images/{Type}/{Index}", "GET")]
  104. [Route("/Artists/{Name}/Images/{Type}", "HEAD")]
  105. [Route("/Artists/{Name}/Images/{Type}/{Index}", "HEAD")]
  106. [Route("/Genres/{Name}/Images/{Type}", "HEAD")]
  107. [Route("/Genres/{Name}/Images/{Type}/{Index}", "HEAD")]
  108. [Route("/MusicGenres/{Name}/Images/{Type}", "HEAD")]
  109. [Route("/MusicGenres/{Name}/Images/{Type}/{Index}", "HEAD")]
  110. [Route("/Persons/{Name}/Images/{Type}", "HEAD")]
  111. [Route("/Persons/{Name}/Images/{Type}/{Index}", "HEAD")]
  112. [Route("/Studios/{Name}/Images/{Type}", "HEAD")]
  113. [Route("/Studios/{Name}/Images/{Type}/{Index}", "HEAD")]
  114. ////[Route("/Years/{Year}/Images/{Type}", "HEAD")]
  115. ////[Route("/Years/{Year}/Images/{Type}/{Index}", "HEAD")]
  116. public class GetItemByNameImage : ImageRequest
  117. {
  118. /// <summary>
  119. /// Gets or sets the name.
  120. /// </summary>
  121. /// <value>The name.</value>
  122. [ApiMember(Name = "Name", Description = "Item name", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  123. public string Name { get; set; }
  124. }
  125. /// <summary>
  126. /// Class GetUserImage
  127. /// </summary>
  128. [Route("/Users/{Id}/Images/{Type}", "GET")]
  129. [Route("/Users/{Id}/Images/{Type}/{Index}", "GET")]
  130. [Route("/Users/{Id}/Images/{Type}", "HEAD")]
  131. [Route("/Users/{Id}/Images/{Type}/{Index}", "HEAD")]
  132. public class GetUserImage : ImageRequest
  133. {
  134. /// <summary>
  135. /// Gets or sets the id.
  136. /// </summary>
  137. /// <value>The id.</value>
  138. [ApiMember(Name = "Id", Description = "User Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  139. public Guid Id { get; set; }
  140. }
  141. /// <summary>
  142. /// Class DeleteItemImage
  143. /// </summary>
  144. [Route("/Items/{Id}/Images/{Type}", "DELETE")]
  145. [Route("/Items/{Id}/Images/{Type}/{Index}", "DELETE")]
  146. [Authenticated(Roles = "admin")]
  147. public class DeleteItemImage : DeleteImageRequest, IReturnVoid
  148. {
  149. /// <summary>
  150. /// Gets or sets the id.
  151. /// </summary>
  152. /// <value>The id.</value>
  153. [ApiMember(Name = "Id", Description = "Item Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "DELETE")]
  154. public string Id { get; set; }
  155. }
  156. /// <summary>
  157. /// Class DeleteUserImage
  158. /// </summary>
  159. [Route("/Users/{Id}/Images/{Type}", "DELETE")]
  160. [Route("/Users/{Id}/Images/{Type}/{Index}", "DELETE")]
  161. [Authenticated]
  162. public class DeleteUserImage : DeleteImageRequest, IReturnVoid
  163. {
  164. /// <summary>
  165. /// Gets or sets the id.
  166. /// </summary>
  167. /// <value>The id.</value>
  168. [ApiMember(Name = "Id", Description = "User Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "DELETE")]
  169. public Guid Id { get; set; }
  170. }
  171. /// <summary>
  172. /// Class PostUserImage
  173. /// </summary>
  174. [Route("/Users/{Id}/Images/{Type}", "POST")]
  175. [Route("/Users/{Id}/Images/{Type}/{Index}", "POST")]
  176. [Authenticated]
  177. public class PostUserImage : DeleteImageRequest, IRequiresRequestStream, IReturnVoid
  178. {
  179. /// <summary>
  180. /// Gets or sets the id.
  181. /// </summary>
  182. /// <value>The id.</value>
  183. [ApiMember(Name = "Id", Description = "User Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "POST")]
  184. public string Id { get; set; }
  185. /// <summary>
  186. /// The raw Http Request Input Stream
  187. /// </summary>
  188. /// <value>The request stream.</value>
  189. public Stream RequestStream { get; set; }
  190. }
  191. /// <summary>
  192. /// Class PostItemImage
  193. /// </summary>
  194. [Route("/Items/{Id}/Images/{Type}", "POST")]
  195. [Route("/Items/{Id}/Images/{Type}/{Index}", "POST")]
  196. [Authenticated(Roles = "admin")]
  197. public class PostItemImage : DeleteImageRequest, IRequiresRequestStream, IReturnVoid
  198. {
  199. /// <summary>
  200. /// Gets or sets the id.
  201. /// </summary>
  202. /// <value>The id.</value>
  203. [ApiMember(Name = "Id", Description = "Item Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "POST")]
  204. public string Id { get; set; }
  205. /// <summary>
  206. /// The raw Http Request Input Stream
  207. /// </summary>
  208. /// <value>The request stream.</value>
  209. public Stream RequestStream { get; set; }
  210. }
  211. /// <summary>
  212. /// Class ImageService
  213. /// </summary>
  214. public class ImageService : BaseApiService
  215. {
  216. private readonly IUserManager _userManager;
  217. private readonly ILibraryManager _libraryManager;
  218. private readonly IProviderManager _providerManager;
  219. private readonly IImageProcessor _imageProcessor;
  220. private readonly IFileSystem _fileSystem;
  221. private readonly IAuthorizationContext _authContext;
  222. /// <summary>
  223. /// Initializes a new instance of the <see cref="ImageService" /> class.
  224. /// </summary>
  225. public ImageService(
  226. ILogger<ImageService> logger,
  227. IServerConfigurationManager serverConfigurationManager,
  228. IHttpResultFactory httpResultFactory,
  229. IUserManager userManager,
  230. ILibraryManager libraryManager,
  231. IProviderManager providerManager,
  232. IImageProcessor imageProcessor,
  233. IFileSystem fileSystem,
  234. IAuthorizationContext authContext)
  235. : base(logger, serverConfigurationManager, httpResultFactory)
  236. {
  237. _userManager = userManager;
  238. _libraryManager = libraryManager;
  239. _providerManager = providerManager;
  240. _imageProcessor = imageProcessor;
  241. _fileSystem = fileSystem;
  242. _authContext = authContext;
  243. }
  244. /// <summary>
  245. /// Gets the specified request.
  246. /// </summary>
  247. /// <param name="request">The request.</param>
  248. /// <returns>System.Object.</returns>
  249. public object Get(GetItemImageInfos request)
  250. {
  251. var item = _libraryManager.GetItemById(request.Id);
  252. var result = GetItemImageInfos(item);
  253. return ToOptimizedResult(result);
  254. }
  255. /// <summary>
  256. /// Gets the item image infos.
  257. /// </summary>
  258. /// <param name="item">The item.</param>
  259. /// <returns>Task{List{ImageInfo}}.</returns>
  260. public List<ImageInfo> GetItemImageInfos(BaseItem item)
  261. {
  262. var list = new List<ImageInfo>();
  263. var itemImages = item.ImageInfos;
  264. if (itemImages.Length == 0)
  265. {
  266. // short-circuit
  267. return list;
  268. }
  269. _libraryManager.UpdateImages(item); // this makes sure dimensions and hashes are correct
  270. foreach (var image in itemImages)
  271. {
  272. if (!item.AllowsMultipleImages(image.Type))
  273. {
  274. var info = GetImageInfo(item, image, null);
  275. if (info != null)
  276. {
  277. list.Add(info);
  278. }
  279. }
  280. }
  281. foreach (var imageType in itemImages.Select(i => i.Type).Distinct().Where(item.AllowsMultipleImages))
  282. {
  283. var index = 0;
  284. // Prevent implicitly captured closure
  285. var currentImageType = imageType;
  286. foreach (var image in itemImages.Where(i => i.Type == currentImageType))
  287. {
  288. var info = GetImageInfo(item, image, index);
  289. if (info != null)
  290. {
  291. list.Add(info);
  292. }
  293. index++;
  294. }
  295. }
  296. return list;
  297. }
  298. private ImageInfo GetImageInfo(BaseItem item, ItemImageInfo info, int? imageIndex)
  299. {
  300. int? width = null;
  301. int? height = null;
  302. string blurhash = null;
  303. long length = 0;
  304. try
  305. {
  306. if (info.IsLocalFile)
  307. {
  308. var fileInfo = _fileSystem.GetFileInfo(info.Path);
  309. length = fileInfo.Length;
  310. blurhash = info.BlurHash;
  311. width = info.Width;
  312. height = info.Height;
  313. if (width <= 0 || height <= 0)
  314. {
  315. width = null;
  316. height = null;
  317. }
  318. }
  319. }
  320. catch (Exception ex)
  321. {
  322. Logger.LogError(ex, "Error getting image information for {Item}", item.Name);
  323. }
  324. try
  325. {
  326. return new ImageInfo
  327. {
  328. Path = info.Path,
  329. ImageIndex = imageIndex,
  330. ImageType = info.Type,
  331. ImageTag = _imageProcessor.GetImageCacheTag(item, info),
  332. Size = length,
  333. BlurHash = blurhash,
  334. Width = width,
  335. Height = height
  336. };
  337. }
  338. catch (Exception ex)
  339. {
  340. Logger.LogError(ex, "Error getting image information for {Path}", info.Path);
  341. return null;
  342. }
  343. }
  344. /// <summary>
  345. /// Gets the specified request.
  346. /// </summary>
  347. /// <param name="request">The request.</param>
  348. /// <returns>System.Object.</returns>
  349. public object Get(GetItemImage request)
  350. {
  351. return GetImage(request, request.Id, null, false);
  352. }
  353. /// <summary>
  354. /// Gets the specified request.
  355. /// </summary>
  356. /// <param name="request">The request.</param>
  357. /// <returns>System.Object.</returns>
  358. public object Head(GetItemImage request)
  359. {
  360. return GetImage(request, request.Id, null, true);
  361. }
  362. /// <summary>
  363. /// Gets the specified request.
  364. /// </summary>
  365. /// <param name="request">The request.</param>
  366. /// <returns>System.Object.</returns>
  367. public object Get(GetUserImage request)
  368. {
  369. var item = _userManager.GetUserById(request.Id);
  370. return GetImage(request, item, false);
  371. }
  372. public object Head(GetUserImage request)
  373. {
  374. var item = _userManager.GetUserById(request.Id);
  375. return GetImage(request, item, true);
  376. }
  377. public object Get(GetItemByNameImage request)
  378. {
  379. var type = GetPathValue(0).ToString();
  380. var item = GetItemByName(request.Name, type, _libraryManager, new DtoOptions(false));
  381. return GetImage(request, item.Id, item, false);
  382. }
  383. public object Head(GetItemByNameImage request)
  384. {
  385. var type = GetPathValue(0).ToString();
  386. var item = GetItemByName(request.Name, type, _libraryManager, new DtoOptions(false));
  387. return GetImage(request, item.Id, item, true);
  388. }
  389. /// <summary>
  390. /// Posts the specified request.
  391. /// </summary>
  392. /// <param name="request">The request.</param>
  393. public Task Post(PostUserImage request)
  394. {
  395. var id = Guid.Parse(GetPathValue(1));
  396. AssertCanUpdateUser(_authContext, _userManager, id, true);
  397. request.Type = Enum.Parse<ImageType>(GetPathValue(3).ToString(), true);
  398. var user = _userManager.GetUserById(id);
  399. return PostImage(user, request.RequestStream, Request.ContentType);
  400. }
  401. /// <summary>
  402. /// Posts the specified request.
  403. /// </summary>
  404. /// <param name="request">The request.</param>
  405. public Task Post(PostItemImage request)
  406. {
  407. var id = Guid.Parse(GetPathValue(1));
  408. request.Type = Enum.Parse<ImageType>(GetPathValue(3).ToString(), true);
  409. var item = _libraryManager.GetItemById(id);
  410. return PostImage(item, request.RequestStream, request.Type, Request.ContentType);
  411. }
  412. /// <summary>
  413. /// Deletes the specified request.
  414. /// </summary>
  415. /// <param name="request">The request.</param>
  416. public void Delete(DeleteUserImage request)
  417. {
  418. var userId = request.Id;
  419. AssertCanUpdateUser(_authContext, _userManager, userId, true);
  420. var user = _userManager.GetUserById(userId);
  421. try
  422. {
  423. File.Delete(user.ProfileImage.Path);
  424. }
  425. catch (IOException e)
  426. {
  427. Logger.LogError(e, "Error deleting user profile image:");
  428. }
  429. _userManager.ClearProfileImage(user);
  430. _userManager.UpdateUser(user);
  431. }
  432. /// <summary>
  433. /// Deletes the specified request.
  434. /// </summary>
  435. /// <param name="request">The request.</param>
  436. public void Delete(DeleteItemImage request)
  437. {
  438. var item = _libraryManager.GetItemById(request.Id);
  439. item.DeleteImage(request.Type, request.Index ?? 0);
  440. }
  441. /// <summary>
  442. /// Posts the specified request.
  443. /// </summary>
  444. /// <param name="request">The request.</param>
  445. public void Post(UpdateItemImageIndex request)
  446. {
  447. var item = _libraryManager.GetItemById(request.Id);
  448. UpdateItemIndex(item, request.Type, request.Index, request.NewIndex);
  449. }
  450. /// <summary>
  451. /// Updates the index of the item.
  452. /// </summary>
  453. /// <param name="item">The item.</param>
  454. /// <param name="type">The type.</param>
  455. /// <param name="currentIndex">Index of the current.</param>
  456. /// <param name="newIndex">The new index.</param>
  457. /// <returns>Task.</returns>
  458. private void UpdateItemIndex(BaseItem item, ImageType type, int currentIndex, int newIndex)
  459. {
  460. item.SwapImages(type, currentIndex, newIndex);
  461. }
  462. /// <summary>
  463. /// Gets the image.
  464. /// </summary>
  465. /// <param name="request">The request.</param>
  466. /// <param name="item">The item.</param>
  467. /// <param name="isHeadRequest">if set to <c>true</c> [is head request].</param>
  468. /// <returns>System.Object.</returns>
  469. /// <exception cref="ResourceNotFoundException"></exception>
  470. public Task<object> GetImage(ImageRequest request, Guid itemId, BaseItem item, bool isHeadRequest)
  471. {
  472. if (request.PercentPlayed.HasValue)
  473. {
  474. if (request.PercentPlayed.Value <= 0)
  475. {
  476. request.PercentPlayed = null;
  477. }
  478. else if (request.PercentPlayed.Value >= 100)
  479. {
  480. request.PercentPlayed = null;
  481. request.AddPlayedIndicator = true;
  482. }
  483. }
  484. if (request.PercentPlayed.HasValue)
  485. {
  486. request.UnplayedCount = null;
  487. }
  488. if (request.UnplayedCount.HasValue
  489. && request.UnplayedCount.Value <= 0)
  490. {
  491. request.UnplayedCount = null;
  492. }
  493. if (item == null)
  494. {
  495. item = _libraryManager.GetItemById(itemId);
  496. if (item == null)
  497. {
  498. throw new ResourceNotFoundException(string.Format("Item {0} not found.", itemId.ToString("N", CultureInfo.InvariantCulture)));
  499. }
  500. }
  501. var imageInfo = GetImageInfo(request, item);
  502. if (imageInfo == null)
  503. {
  504. throw new ResourceNotFoundException(string.Format("{0} does not have an image of type {1}", item.Name, request.Type));
  505. }
  506. bool cropWhitespace;
  507. if (request.CropWhitespace.HasValue)
  508. {
  509. cropWhitespace = request.CropWhitespace.Value;
  510. }
  511. else
  512. {
  513. cropWhitespace = request.Type == ImageType.Logo || request.Type == ImageType.Art;
  514. }
  515. var outputFormats = GetOutputFormats(request);
  516. TimeSpan? cacheDuration = null;
  517. if (!string.IsNullOrEmpty(request.Tag))
  518. {
  519. cacheDuration = TimeSpan.FromDays(365);
  520. }
  521. var responseHeaders = new Dictionary<string, string>
  522. {
  523. {"transferMode.dlna.org", "Interactive"},
  524. {"realTimeInfo.dlna.org", "DLNA.ORG_TLAG=*"}
  525. };
  526. return GetImageResult(
  527. item,
  528. itemId,
  529. request,
  530. imageInfo,
  531. cropWhitespace,
  532. outputFormats,
  533. cacheDuration,
  534. responseHeaders,
  535. isHeadRequest);
  536. }
  537. public Task<object> GetImage(ImageRequest request, User user, bool isHeadRequest)
  538. {
  539. var imageInfo = GetImageInfo(request, user);
  540. TimeSpan? cacheDuration = null;
  541. if (!string.IsNullOrEmpty(request.Tag))
  542. {
  543. cacheDuration = TimeSpan.FromDays(365);
  544. }
  545. var responseHeaders = new Dictionary<string, string>
  546. {
  547. {"transferMode.dlna.org", "Interactive"},
  548. {"realTimeInfo.dlna.org", "DLNA.ORG_TLAG=*"}
  549. };
  550. var outputFormats = GetOutputFormats(request);
  551. return GetImageResult(user.Id,
  552. request,
  553. imageInfo,
  554. outputFormats,
  555. cacheDuration,
  556. responseHeaders,
  557. isHeadRequest);
  558. }
  559. private async Task<object> GetImageResult(
  560. Guid itemId,
  561. ImageRequest request,
  562. ItemImageInfo info,
  563. IReadOnlyCollection<ImageFormat> supportedFormats,
  564. TimeSpan? cacheDuration,
  565. IDictionary<string, string> headers,
  566. bool isHeadRequest)
  567. {
  568. info.Type = ImageType.Profile;
  569. var options = new ImageProcessingOptions
  570. {
  571. CropWhiteSpace = true,
  572. Height = request.Height,
  573. ImageIndex = request.Index ?? 0,
  574. Image = info,
  575. Item = null, // Hack alert
  576. ItemId = itemId,
  577. MaxHeight = request.MaxHeight,
  578. MaxWidth = request.MaxWidth,
  579. Quality = request.Quality ?? 100,
  580. Width = request.Width,
  581. AddPlayedIndicator = request.AddPlayedIndicator,
  582. PercentPlayed = 0,
  583. UnplayedCount = request.UnplayedCount,
  584. Blur = request.Blur,
  585. BackgroundColor = request.BackgroundColor,
  586. ForegroundLayer = request.ForegroundLayer,
  587. SupportedOutputFormats = supportedFormats
  588. };
  589. var imageResult = await _imageProcessor.ProcessImage(options).ConfigureAwait(false);
  590. headers[HeaderNames.Vary] = HeaderNames.Accept;
  591. return await ResultFactory.GetStaticFileResult(Request, new StaticFileResultOptions
  592. {
  593. CacheDuration = cacheDuration,
  594. ResponseHeaders = headers,
  595. ContentType = imageResult.Item2,
  596. DateLastModified = imageResult.Item3,
  597. IsHeadRequest = isHeadRequest,
  598. Path = imageResult.Item1,
  599. FileShare = FileShare.Read
  600. }).ConfigureAwait(false);
  601. }
  602. private async Task<object> GetImageResult(
  603. BaseItem item,
  604. Guid itemId,
  605. ImageRequest request,
  606. ItemImageInfo image,
  607. bool cropwhitespace,
  608. IReadOnlyCollection<ImageFormat> supportedFormats,
  609. TimeSpan? cacheDuration,
  610. IDictionary<string, string> headers,
  611. bool isHeadRequest)
  612. {
  613. if (!image.IsLocalFile)
  614. {
  615. item ??= _libraryManager.GetItemById(itemId);
  616. image = await _libraryManager.ConvertImageToLocal(item, image, request.Index ?? 0).ConfigureAwait(false);
  617. }
  618. var options = new ImageProcessingOptions
  619. {
  620. CropWhiteSpace = cropwhitespace,
  621. Height = request.Height,
  622. ImageIndex = request.Index ?? 0,
  623. Image = image,
  624. Item = item,
  625. ItemId = itemId,
  626. MaxHeight = request.MaxHeight,
  627. MaxWidth = request.MaxWidth,
  628. Quality = request.Quality ?? 100,
  629. Width = request.Width,
  630. AddPlayedIndicator = request.AddPlayedIndicator,
  631. PercentPlayed = request.PercentPlayed ?? 0,
  632. UnplayedCount = request.UnplayedCount,
  633. Blur = request.Blur,
  634. BackgroundColor = request.BackgroundColor,
  635. ForegroundLayer = request.ForegroundLayer,
  636. SupportedOutputFormats = supportedFormats
  637. };
  638. var imageResult = await _imageProcessor.ProcessImage(options).ConfigureAwait(false);
  639. headers[HeaderNames.Vary] = HeaderNames.Accept;
  640. return await ResultFactory.GetStaticFileResult(Request, new StaticFileResultOptions
  641. {
  642. CacheDuration = cacheDuration,
  643. ResponseHeaders = headers,
  644. ContentType = imageResult.Item2,
  645. DateLastModified = imageResult.Item3,
  646. IsHeadRequest = isHeadRequest,
  647. Path = imageResult.Item1,
  648. FileShare = FileShare.Read
  649. }).ConfigureAwait(false);
  650. }
  651. private ImageFormat[] GetOutputFormats(ImageRequest request)
  652. {
  653. if (!string.IsNullOrWhiteSpace(request.Format)
  654. && Enum.TryParse(request.Format, true, out ImageFormat format))
  655. {
  656. return new[] { format };
  657. }
  658. return GetClientSupportedFormats();
  659. }
  660. private ImageFormat[] GetClientSupportedFormats()
  661. {
  662. var supportedFormats = Request.AcceptTypes ?? Array.Empty<string>();
  663. if (supportedFormats.Length > 0)
  664. {
  665. for (int i = 0; i < supportedFormats.Length; i++)
  666. {
  667. int index = supportedFormats[i].IndexOf(';');
  668. if (index != -1)
  669. {
  670. supportedFormats[i] = supportedFormats[i].Substring(0, index);
  671. }
  672. }
  673. }
  674. var acceptParam = Request.QueryString["accept"];
  675. var supportsWebP = SupportsFormat(supportedFormats, acceptParam, "webp", false);
  676. if (!supportsWebP)
  677. {
  678. var userAgent = Request.UserAgent ?? string.Empty;
  679. if (userAgent.IndexOf("crosswalk", StringComparison.OrdinalIgnoreCase) != -1 &&
  680. userAgent.IndexOf("android", StringComparison.OrdinalIgnoreCase) != -1)
  681. {
  682. supportsWebP = true;
  683. }
  684. }
  685. var formats = new List<ImageFormat>(4);
  686. if (supportsWebP)
  687. {
  688. formats.Add(ImageFormat.Webp);
  689. }
  690. formats.Add(ImageFormat.Jpg);
  691. formats.Add(ImageFormat.Png);
  692. if (SupportsFormat(supportedFormats, acceptParam, "gif", true))
  693. {
  694. formats.Add(ImageFormat.Gif);
  695. }
  696. return formats.ToArray();
  697. }
  698. private bool SupportsFormat(IEnumerable<string> requestAcceptTypes, string acceptParam, string format, bool acceptAll)
  699. {
  700. var mimeType = "image/" + format;
  701. if (requestAcceptTypes.Contains(mimeType))
  702. {
  703. return true;
  704. }
  705. if (acceptAll && requestAcceptTypes.Contains("*/*"))
  706. {
  707. return true;
  708. }
  709. return string.Equals(Request.QueryString["accept"], format, StringComparison.OrdinalIgnoreCase);
  710. }
  711. /// <summary>
  712. /// Gets the image path.
  713. /// </summary>
  714. /// <param name="request">The request.</param>
  715. /// <param name="item">The item.</param>
  716. /// <returns>System.String.</returns>
  717. private static ItemImageInfo GetImageInfo(ImageRequest request, BaseItem item)
  718. {
  719. var index = request.Index ?? 0;
  720. return item.GetImageInfo(request.Type, index);
  721. }
  722. private static ItemImageInfo GetImageInfo(ImageRequest request, User user)
  723. {
  724. var info = new ItemImageInfo
  725. {
  726. Path = user.ProfileImage.Path,
  727. Type = ImageType.Primary,
  728. DateModified = user.ProfileImage.LastModified,
  729. };
  730. if (request.Width.HasValue)
  731. {
  732. info.Width = request.Width.Value;
  733. }
  734. if (request.Height.HasValue)
  735. {
  736. info.Height = request.Height.Value;
  737. }
  738. return info;
  739. }
  740. /// <summary>
  741. /// Posts the image.
  742. /// </summary>
  743. /// <param name="entity">The entity.</param>
  744. /// <param name="inputStream">The input stream.</param>
  745. /// <param name="imageType">Type of the image.</param>
  746. /// <param name="mimeType">Type of the MIME.</param>
  747. /// <returns>Task.</returns>
  748. public async Task PostImage(BaseItem entity, Stream inputStream, ImageType imageType, string mimeType)
  749. {
  750. var memoryStream = await GetMemoryStream(inputStream);
  751. // Handle image/png; charset=utf-8
  752. mimeType = mimeType.Split(';').FirstOrDefault();
  753. await _providerManager.SaveImage(entity, memoryStream, mimeType, imageType, null, CancellationToken.None).ConfigureAwait(false);
  754. entity.UpdateToRepository(ItemUpdateType.ImageUpdate, CancellationToken.None);
  755. }
  756. private static async Task<MemoryStream> GetMemoryStream(Stream inputStream)
  757. {
  758. using var reader = new StreamReader(inputStream);
  759. var text = await reader.ReadToEndAsync().ConfigureAwait(false);
  760. var bytes = Convert.FromBase64String(text);
  761. return new MemoryStream(bytes)
  762. {
  763. Position = 0
  764. };
  765. }
  766. private async Task PostImage(User user, Stream inputStream, string mimeType)
  767. {
  768. var memoryStream = await GetMemoryStream(inputStream);
  769. // Handle image/png; charset=utf-8
  770. mimeType = mimeType.Split(';').FirstOrDefault();
  771. var userDataPath = Path.Combine(ServerConfigurationManager.ApplicationPaths.UserConfigurationDirectoryPath, user.Username);
  772. user.ProfileImage = new Jellyfin.Data.Entities.ImageInfo(Path.Combine(userDataPath, "profile" + MimeTypes.ToExtension(mimeType)));
  773. await _providerManager
  774. .SaveImage(user, memoryStream, mimeType, user.ProfileImage.Path)
  775. .ConfigureAwait(false);
  776. await _userManager.UpdateUserAsync(user);
  777. }
  778. }
  779. }