2020-11-21 13:26:03 +00:00
|
|
|
using System;
|
2020-06-27 10:26:43 +00:00
|
|
|
using System.Buffers;
|
2020-08-06 14:17:45 +00:00
|
|
|
using System.ComponentModel.DataAnnotations;
|
2020-06-27 10:26:43 +00:00
|
|
|
using System.Linq;
|
|
|
|
using System.Net.Mime;
|
|
|
|
using System.Threading.Tasks;
|
2020-09-01 23:26:49 +00:00
|
|
|
using Jellyfin.Api.Attributes;
|
2020-06-27 10:26:43 +00:00
|
|
|
using Jellyfin.Api.Constants;
|
2020-08-09 23:20:14 +00:00
|
|
|
using Jellyfin.Api.Helpers;
|
2020-08-03 19:33:43 +00:00
|
|
|
using Jellyfin.Api.Models.MediaInfoDtos;
|
2020-09-10 12:16:41 +00:00
|
|
|
using MediaBrowser.Common.Extensions;
|
2020-06-27 10:26:43 +00:00
|
|
|
using MediaBrowser.Controller.Devices;
|
|
|
|
using MediaBrowser.Controller.Library;
|
|
|
|
using MediaBrowser.Controller.Net;
|
|
|
|
using MediaBrowser.Model.Dlna;
|
|
|
|
using MediaBrowser.Model.MediaInfo;
|
|
|
|
using Microsoft.AspNetCore.Authorization;
|
|
|
|
using Microsoft.AspNetCore.Http;
|
|
|
|
using Microsoft.AspNetCore.Mvc;
|
2020-12-10 18:36:58 +00:00
|
|
|
using Microsoft.AspNetCore.Mvc.ModelBinding;
|
2020-06-27 10:26:43 +00:00
|
|
|
using Microsoft.Extensions.Logging;
|
|
|
|
|
|
|
|
namespace Jellyfin.Api.Controllers
|
|
|
|
{
|
|
|
|
/// <summary>
|
|
|
|
/// The media info controller.
|
|
|
|
/// </summary>
|
2020-08-04 18:48:53 +00:00
|
|
|
[Route("")]
|
2020-06-27 10:26:43 +00:00
|
|
|
[Authorize(Policy = Policies.DefaultAuthorization)]
|
|
|
|
public class MediaInfoController : BaseJellyfinApiController
|
|
|
|
{
|
|
|
|
private readonly IMediaSourceManager _mediaSourceManager;
|
|
|
|
private readonly IDeviceManager _deviceManager;
|
|
|
|
private readonly ILibraryManager _libraryManager;
|
|
|
|
private readonly IAuthorizationContext _authContext;
|
2020-08-03 19:16:29 +00:00
|
|
|
private readonly ILogger<MediaInfoController> _logger;
|
2020-08-09 23:20:14 +00:00
|
|
|
private readonly MediaInfoHelper _mediaInfoHelper;
|
2020-06-27 10:26:43 +00:00
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Initializes a new instance of the <see cref="MediaInfoController"/> class.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="mediaSourceManager">Instance of the <see cref="IMediaSourceManager"/> interface.</param>
|
|
|
|
/// <param name="deviceManager">Instance of the <see cref="IDeviceManager"/> interface.</param>
|
|
|
|
/// <param name="libraryManager">Instance of the <see cref="ILibraryManager"/> interface.</param>
|
|
|
|
/// <param name="authContext">Instance of the <see cref="IAuthorizationContext"/> interface.</param>
|
|
|
|
/// <param name="logger">Instance of the <see cref="ILogger{MediaInfoController}"/> interface.</param>
|
2020-08-09 23:20:14 +00:00
|
|
|
/// <param name="mediaInfoHelper">Instance of the <see cref="MediaInfoHelper"/>.</param>
|
2020-06-27 10:26:43 +00:00
|
|
|
public MediaInfoController(
|
|
|
|
IMediaSourceManager mediaSourceManager,
|
|
|
|
IDeviceManager deviceManager,
|
|
|
|
ILibraryManager libraryManager,
|
|
|
|
IAuthorizationContext authContext,
|
|
|
|
ILogger<MediaInfoController> logger,
|
2020-08-09 23:20:14 +00:00
|
|
|
MediaInfoHelper mediaInfoHelper)
|
2020-06-27 10:26:43 +00:00
|
|
|
{
|
|
|
|
_mediaSourceManager = mediaSourceManager;
|
|
|
|
_deviceManager = deviceManager;
|
|
|
|
_libraryManager = libraryManager;
|
|
|
|
_authContext = authContext;
|
|
|
|
_logger = logger;
|
2020-08-09 23:20:14 +00:00
|
|
|
_mediaInfoHelper = mediaInfoHelper;
|
2020-06-27 10:26:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Gets live playback media info for an item.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="itemId">The item id.</param>
|
|
|
|
/// <param name="userId">The user id.</param>
|
|
|
|
/// <response code="200">Playback info returned.</response>
|
|
|
|
/// <returns>A <see cref="Task"/> containing a <see cref="PlaybackInfoResponse"/> with the playback information.</returns>
|
2020-08-04 18:48:53 +00:00
|
|
|
[HttpGet("Items/{itemId}/PlaybackInfo")]
|
2020-06-27 10:26:43 +00:00
|
|
|
[ProducesResponseType(StatusCodes.Status200OK)]
|
2020-09-09 20:28:30 +00:00
|
|
|
public async Task<ActionResult<PlaybackInfoResponse>> GetPlaybackInfo([FromRoute, Required] Guid itemId, [FromQuery, Required] Guid userId)
|
2020-06-27 10:26:43 +00:00
|
|
|
{
|
2020-08-09 23:20:14 +00:00
|
|
|
return await _mediaInfoHelper.GetPlaybackInfo(
|
|
|
|
itemId,
|
|
|
|
userId)
|
|
|
|
.ConfigureAwait(false);
|
2020-06-27 10:26:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Gets live playback media info for an item.
|
|
|
|
/// </summary>
|
2020-11-23 16:49:42 +00:00
|
|
|
/// <remarks>
|
|
|
|
/// For backwards compatibility parameters can be sent via Query or Body, with Query having higher precedence.
|
|
|
|
/// </remarks>
|
2020-06-27 10:26:43 +00:00
|
|
|
/// <param name="itemId">The item id.</param>
|
|
|
|
/// <param name="userId">The user id.</param>
|
|
|
|
/// <param name="maxStreamingBitrate">The maximum streaming bitrate.</param>
|
|
|
|
/// <param name="startTimeTicks">The start time in ticks.</param>
|
|
|
|
/// <param name="audioStreamIndex">The audio stream index.</param>
|
|
|
|
/// <param name="subtitleStreamIndex">The subtitle stream index.</param>
|
|
|
|
/// <param name="maxAudioChannels">The maximum number of audio channels.</param>
|
|
|
|
/// <param name="mediaSourceId">The media source id.</param>
|
|
|
|
/// <param name="liveStreamId">The livestream id.</param>
|
|
|
|
/// <param name="autoOpenLiveStream">Whether to auto open the livestream.</param>
|
|
|
|
/// <param name="enableDirectPlay">Whether to enable direct play. Default: true.</param>
|
|
|
|
/// <param name="enableDirectStream">Whether to enable direct stream. Default: true.</param>
|
|
|
|
/// <param name="enableTranscoding">Whether to enable transcoding. Default: true.</param>
|
|
|
|
/// <param name="allowVideoStreamCopy">Whether to allow to copy the video stream. Default: true.</param>
|
|
|
|
/// <param name="allowAudioStreamCopy">Whether to allow to copy the audio stream. Default: true.</param>
|
2020-11-23 16:49:42 +00:00
|
|
|
/// <param name="playbackInfoDto">The playback info.</param>
|
2020-06-27 10:26:43 +00:00
|
|
|
/// <response code="200">Playback info returned.</response>
|
|
|
|
/// <returns>A <see cref="Task"/> containing a <see cref="PlaybackInfoResponse"/> with the playback info.</returns>
|
2020-08-04 18:48:53 +00:00
|
|
|
[HttpPost("Items/{itemId}/PlaybackInfo")]
|
2020-06-27 10:26:43 +00:00
|
|
|
[ProducesResponseType(StatusCodes.Status200OK)]
|
|
|
|
public async Task<ActionResult<PlaybackInfoResponse>> GetPostedPlaybackInfo(
|
2020-09-06 15:07:27 +00:00
|
|
|
[FromRoute, Required] Guid itemId,
|
2020-07-07 15:10:51 +00:00
|
|
|
[FromQuery] Guid? userId,
|
2020-10-31 08:09:22 +00:00
|
|
|
[FromQuery] int? maxStreamingBitrate,
|
2020-06-27 10:26:43 +00:00
|
|
|
[FromQuery] long? startTimeTicks,
|
|
|
|
[FromQuery] int? audioStreamIndex,
|
|
|
|
[FromQuery] int? subtitleStreamIndex,
|
|
|
|
[FromQuery] int? maxAudioChannels,
|
2020-07-07 15:10:51 +00:00
|
|
|
[FromQuery] string? mediaSourceId,
|
|
|
|
[FromQuery] string? liveStreamId,
|
2020-11-23 16:49:42 +00:00
|
|
|
[FromQuery] bool? autoOpenLiveStream,
|
|
|
|
[FromQuery] bool? enableDirectPlay,
|
|
|
|
[FromQuery] bool? enableDirectStream,
|
|
|
|
[FromQuery] bool? enableTranscoding,
|
|
|
|
[FromQuery] bool? allowVideoStreamCopy,
|
|
|
|
[FromQuery] bool? allowAudioStreamCopy,
|
2020-12-10 18:36:58 +00:00
|
|
|
[FromBody(EmptyBodyBehavior = EmptyBodyBehavior.Allow)] PlaybackInfoDto? playbackInfoDto)
|
2020-06-27 10:26:43 +00:00
|
|
|
{
|
|
|
|
var authInfo = _authContext.GetAuthorizationInfo(Request);
|
|
|
|
|
2020-11-23 17:50:18 +00:00
|
|
|
var profile = playbackInfoDto?.DeviceProfile;
|
2020-06-27 10:26:43 +00:00
|
|
|
_logger.LogInformation("GetPostedPlaybackInfo profile: {@Profile}", profile);
|
|
|
|
|
|
|
|
if (profile == null)
|
|
|
|
{
|
|
|
|
var caps = _deviceManager.GetCapabilities(authInfo.DeviceId);
|
|
|
|
if (caps != null)
|
|
|
|
{
|
|
|
|
profile = caps.DeviceProfile;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-23 16:49:42 +00:00
|
|
|
// Copy params from posted body
|
2020-11-23 16:51:24 +00:00
|
|
|
// TODO clean up when breaking API compatibility.
|
2020-11-23 16:49:42 +00:00
|
|
|
userId ??= playbackInfoDto?.UserId;
|
|
|
|
maxStreamingBitrate ??= playbackInfoDto?.MaxStreamingBitrate;
|
|
|
|
startTimeTicks ??= playbackInfoDto?.StartTimeTicks;
|
|
|
|
audioStreamIndex ??= playbackInfoDto?.AudioStreamIndex;
|
|
|
|
subtitleStreamIndex ??= playbackInfoDto?.SubtitleStreamIndex;
|
|
|
|
maxAudioChannels ??= playbackInfoDto?.MaxAudioChannels;
|
|
|
|
mediaSourceId ??= playbackInfoDto?.MediaSourceId;
|
|
|
|
liveStreamId ??= playbackInfoDto?.LiveStreamId;
|
|
|
|
autoOpenLiveStream ??= playbackInfoDto?.AutoOpenLiveStream ?? false;
|
|
|
|
enableDirectPlay ??= playbackInfoDto?.EnableDirectPlay ?? true;
|
|
|
|
enableDirectStream ??= playbackInfoDto?.EnableDirectStream ?? true;
|
|
|
|
enableTranscoding ??= playbackInfoDto?.EnableTranscoding ?? true;
|
|
|
|
allowVideoStreamCopy ??= playbackInfoDto?.AllowVideoStreamCopy ?? true;
|
|
|
|
allowAudioStreamCopy ??= playbackInfoDto?.AllowAudioStreamCopy ?? true;
|
|
|
|
|
2020-08-09 23:20:14 +00:00
|
|
|
var info = await _mediaInfoHelper.GetPlaybackInfo(
|
|
|
|
itemId,
|
|
|
|
userId,
|
|
|
|
mediaSourceId,
|
|
|
|
liveStreamId)
|
|
|
|
.ConfigureAwait(false);
|
2020-06-27 10:26:43 +00:00
|
|
|
|
|
|
|
if (profile != null)
|
|
|
|
{
|
|
|
|
// set device specific data
|
|
|
|
var item = _libraryManager.GetItemById(itemId);
|
|
|
|
|
|
|
|
foreach (var mediaSource in info.MediaSources)
|
|
|
|
{
|
2020-08-09 23:20:14 +00:00
|
|
|
_mediaInfoHelper.SetDeviceSpecificData(
|
2020-06-27 10:26:43 +00:00
|
|
|
item,
|
|
|
|
mediaSource,
|
|
|
|
profile,
|
|
|
|
authInfo,
|
|
|
|
maxStreamingBitrate ?? profile.MaxStreamingBitrate,
|
|
|
|
startTimeTicks ?? 0,
|
2020-07-07 15:10:51 +00:00
|
|
|
mediaSourceId ?? string.Empty,
|
2020-06-27 10:26:43 +00:00
|
|
|
audioStreamIndex,
|
|
|
|
subtitleStreamIndex,
|
|
|
|
maxAudioChannels,
|
|
|
|
info!.PlaySessionId!,
|
2020-07-07 15:10:51 +00:00
|
|
|
userId ?? Guid.Empty,
|
2020-11-23 16:49:42 +00:00
|
|
|
enableDirectPlay.Value,
|
|
|
|
enableDirectStream.Value,
|
|
|
|
enableTranscoding.Value,
|
|
|
|
allowVideoStreamCopy.Value,
|
|
|
|
allowAudioStreamCopy.Value,
|
2020-09-10 12:16:41 +00:00
|
|
|
Request.HttpContext.GetNormalizedRemoteIp());
|
2020-06-27 10:26:43 +00:00
|
|
|
}
|
|
|
|
|
2020-08-09 23:20:14 +00:00
|
|
|
_mediaInfoHelper.SortMediaSources(info, maxStreamingBitrate);
|
2020-06-27 10:26:43 +00:00
|
|
|
}
|
|
|
|
|
2020-11-23 16:49:42 +00:00
|
|
|
if (autoOpenLiveStream.Value)
|
2020-06-27 10:26:43 +00:00
|
|
|
{
|
|
|
|
var mediaSource = string.IsNullOrWhiteSpace(mediaSourceId) ? info.MediaSources[0] : info.MediaSources.FirstOrDefault(i => string.Equals(i.Id, mediaSourceId, StringComparison.Ordinal));
|
|
|
|
|
|
|
|
if (mediaSource != null && mediaSource.RequiresOpening && string.IsNullOrWhiteSpace(mediaSource.LiveStreamId))
|
|
|
|
{
|
2020-08-09 23:20:14 +00:00
|
|
|
var openStreamResult = await _mediaInfoHelper.OpenMediaSource(
|
|
|
|
Request,
|
|
|
|
new LiveStreamRequest
|
|
|
|
{
|
|
|
|
AudioStreamIndex = audioStreamIndex,
|
2020-11-23 17:50:18 +00:00
|
|
|
DeviceProfile = playbackInfoDto?.DeviceProfile,
|
2020-11-23 16:49:42 +00:00
|
|
|
EnableDirectPlay = enableDirectPlay.Value,
|
|
|
|
EnableDirectStream = enableDirectStream.Value,
|
2020-08-09 23:20:14 +00:00
|
|
|
ItemId = itemId,
|
|
|
|
MaxAudioChannels = maxAudioChannels,
|
|
|
|
MaxStreamingBitrate = maxStreamingBitrate,
|
|
|
|
PlaySessionId = info.PlaySessionId,
|
|
|
|
StartTimeTicks = startTimeTicks,
|
|
|
|
SubtitleStreamIndex = subtitleStreamIndex,
|
|
|
|
UserId = userId ?? Guid.Empty,
|
|
|
|
OpenToken = mediaSource.OpenToken
|
|
|
|
}).ConfigureAwait(false);
|
2020-06-27 10:26:43 +00:00
|
|
|
|
|
|
|
info.MediaSources = new[] { openStreamResult.MediaSource };
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (info.MediaSources != null)
|
|
|
|
{
|
|
|
|
foreach (var mediaSource in info.MediaSources)
|
|
|
|
{
|
2020-08-09 23:20:14 +00:00
|
|
|
_mediaInfoHelper.NormalizeMediaSourceContainer(mediaSource, profile!, DlnaProfileType.Video);
|
2020-06-27 10:26:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return info;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Opens a media source.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="openToken">The open token.</param>
|
|
|
|
/// <param name="userId">The user id.</param>
|
|
|
|
/// <param name="playSessionId">The play session id.</param>
|
|
|
|
/// <param name="maxStreamingBitrate">The maximum streaming bitrate.</param>
|
|
|
|
/// <param name="startTimeTicks">The start time in ticks.</param>
|
|
|
|
/// <param name="audioStreamIndex">The audio stream index.</param>
|
|
|
|
/// <param name="subtitleStreamIndex">The subtitle stream index.</param>
|
|
|
|
/// <param name="maxAudioChannels">The maximum number of audio channels.</param>
|
|
|
|
/// <param name="itemId">The item id.</param>
|
2020-08-03 19:33:43 +00:00
|
|
|
/// <param name="openLiveStreamDto">The open live stream dto.</param>
|
2020-06-27 10:26:43 +00:00
|
|
|
/// <param name="enableDirectPlay">Whether to enable direct play. Default: true.</param>
|
|
|
|
/// <param name="enableDirectStream">Whether to enable direct stream. Default: true.</param>
|
|
|
|
/// <response code="200">Media source opened.</response>
|
|
|
|
/// <returns>A <see cref="Task"/> containing a <see cref="LiveStreamResponse"/>.</returns>
|
2020-08-04 18:48:53 +00:00
|
|
|
[HttpPost("LiveStreams/Open")]
|
2020-06-27 10:26:43 +00:00
|
|
|
[ProducesResponseType(StatusCodes.Status200OK)]
|
|
|
|
public async Task<ActionResult<LiveStreamResponse>> OpenLiveStream(
|
2020-07-07 15:10:51 +00:00
|
|
|
[FromQuery] string? openToken,
|
|
|
|
[FromQuery] Guid? userId,
|
|
|
|
[FromQuery] string? playSessionId,
|
2020-10-31 08:09:22 +00:00
|
|
|
[FromQuery] int? maxStreamingBitrate,
|
2020-06-27 10:26:43 +00:00
|
|
|
[FromQuery] long? startTimeTicks,
|
|
|
|
[FromQuery] int? audioStreamIndex,
|
|
|
|
[FromQuery] int? subtitleStreamIndex,
|
|
|
|
[FromQuery] int? maxAudioChannels,
|
2020-07-07 15:10:51 +00:00
|
|
|
[FromQuery] Guid? itemId,
|
2020-08-03 19:33:43 +00:00
|
|
|
[FromBody] OpenLiveStreamDto openLiveStreamDto,
|
2020-06-27 10:26:43 +00:00
|
|
|
[FromQuery] bool enableDirectPlay = true,
|
|
|
|
[FromQuery] bool enableDirectStream = true)
|
|
|
|
{
|
|
|
|
var request = new LiveStreamRequest
|
|
|
|
{
|
|
|
|
OpenToken = openToken,
|
2020-07-07 15:10:51 +00:00
|
|
|
UserId = userId ?? Guid.Empty,
|
2020-06-27 10:26:43 +00:00
|
|
|
PlaySessionId = playSessionId,
|
|
|
|
MaxStreamingBitrate = maxStreamingBitrate,
|
|
|
|
StartTimeTicks = startTimeTicks,
|
|
|
|
AudioStreamIndex = audioStreamIndex,
|
|
|
|
SubtitleStreamIndex = subtitleStreamIndex,
|
|
|
|
MaxAudioChannels = maxAudioChannels,
|
2020-07-07 15:10:51 +00:00
|
|
|
ItemId = itemId ?? Guid.Empty,
|
2020-08-03 19:33:43 +00:00
|
|
|
DeviceProfile = openLiveStreamDto?.DeviceProfile,
|
2020-06-27 10:26:43 +00:00
|
|
|
EnableDirectPlay = enableDirectPlay,
|
|
|
|
EnableDirectStream = enableDirectStream,
|
2020-08-03 19:33:43 +00:00
|
|
|
DirectPlayProtocols = openLiveStreamDto?.DirectPlayProtocols ?? new[] { MediaProtocol.Http }
|
2020-06-27 10:26:43 +00:00
|
|
|
};
|
2020-08-09 23:20:14 +00:00
|
|
|
return await _mediaInfoHelper.OpenMediaSource(Request, request).ConfigureAwait(false);
|
2020-06-27 10:26:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Closes a media source.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="liveStreamId">The livestream id.</param>
|
|
|
|
/// <response code="204">Livestream closed.</response>
|
|
|
|
/// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
|
2020-08-04 18:48:53 +00:00
|
|
|
[HttpPost("LiveStreams/Close")]
|
2020-06-27 10:26:43 +00:00
|
|
|
[ProducesResponseType(StatusCodes.Status204NoContent)]
|
2020-09-09 20:28:30 +00:00
|
|
|
public async Task<ActionResult> CloseLiveStream([FromQuery, Required] string liveStreamId)
|
2020-06-27 10:26:43 +00:00
|
|
|
{
|
2020-08-21 20:06:06 +00:00
|
|
|
await _mediaSourceManager.CloseLiveStream(liveStreamId).ConfigureAwait(false);
|
2020-06-27 10:26:43 +00:00
|
|
|
return NoContent();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Tests the network with a request with the size of the bitrate.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="size">The bitrate. Defaults to 102400.</param>
|
|
|
|
/// <response code="200">Test buffer returned.</response>
|
|
|
|
/// <response code="400">Size has to be a numer between 0 and 10,000,000.</response>
|
|
|
|
/// <returns>A <see cref="FileResult"/> with specified bitrate.</returns>
|
2020-08-04 18:48:53 +00:00
|
|
|
[HttpGet("Playback/BitrateTest")]
|
2020-06-27 10:26:43 +00:00
|
|
|
[ProducesResponseType(StatusCodes.Status200OK)]
|
|
|
|
[ProducesResponseType(StatusCodes.Status400BadRequest)]
|
|
|
|
[Produces(MediaTypeNames.Application.Octet)]
|
2020-09-01 23:26:49 +00:00
|
|
|
[ProducesFile(MediaTypeNames.Application.Octet)]
|
2020-06-27 10:26:43 +00:00
|
|
|
public ActionResult GetBitrateTestBytes([FromQuery] int size = 102400)
|
|
|
|
{
|
|
|
|
const int MaxSize = 10_000_000;
|
|
|
|
|
|
|
|
if (size <= 0)
|
|
|
|
{
|
|
|
|
return BadRequest($"The requested size ({size}) is equal to or smaller than 0.");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (size > MaxSize)
|
|
|
|
{
|
|
|
|
return BadRequest($"The requested size ({size}) is larger than the max allowed value ({MaxSize}).");
|
|
|
|
}
|
|
|
|
|
|
|
|
byte[] buffer = ArrayPool<byte>.Shared.Rent(size);
|
|
|
|
try
|
|
|
|
{
|
|
|
|
new Random().NextBytes(buffer);
|
|
|
|
return File(buffer, MediaTypeNames.Application.Octet);
|
|
|
|
}
|
|
|
|
finally
|
|
|
|
{
|
|
|
|
ArrayPool<byte>.Shared.Return(buffer);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|