jellyfin/Emby.Server.Implementations/HttpServer/StreamWriter.cs

121 lines
3.6 KiB
C#
Raw Normal View History

using System;
using System.Collections.Generic;
using System.Globalization;
using System.IO;
2016-10-25 19:02:04 +00:00
using System.Threading;
2016-07-14 19:13:52 +00:00
using System.Threading.Tasks;
2016-10-25 19:02:04 +00:00
using MediaBrowser.Model.Services;
using Microsoft.Net.Http.Headers;
2016-11-04 01:18:51 +00:00
namespace Emby.Server.Implementations.HttpServer
{
/// <summary>
2019-11-01 17:38:54 +00:00
/// Class StreamWriter.
/// </summary>
2016-10-25 19:02:04 +00:00
public class StreamWriter : IAsyncStreamWriter, IHasHeaders
{
/// <summary>
/// The options.
/// </summary>
private readonly IDictionary<string, string> _options = new Dictionary<string, string>();
2014-08-30 14:26:29 +00:00
/// <summary>
/// Initializes a new instance of the <see cref="StreamWriter" /> class.
/// </summary>
/// <param name="source">The source.</param>
/// <param name="contentType">Type of the content.</param>
2019-02-09 10:53:07 +00:00
public StreamWriter(Stream source, string contentType)
{
if (string.IsNullOrEmpty(contentType))
{
throw new ArgumentNullException(nameof(contentType));
}
SourceStream = source;
Headers["Content-Type"] = contentType;
if (source.CanSeek)
{
Headers[HeaderNames.ContentLength] = source.Length.ToString(CultureInfo.InvariantCulture);
}
Headers[HeaderNames.ContentType] = contentType;
}
2013-03-25 02:41:27 +00:00
/// <summary>
/// Initializes a new instance of the <see cref="StreamWriter"/> class.
/// </summary>
/// <param name="source">The source.</param>
/// <param name="contentType">Type of the content.</param>
2019-11-01 17:38:54 +00:00
/// <param name="contentLength">The content length.</param>
public StreamWriter(byte[] source, string contentType, int contentLength)
2013-03-25 02:41:27 +00:00
{
2016-10-06 18:55:01 +00:00
if (string.IsNullOrEmpty(contentType))
{
throw new ArgumentNullException(nameof(contentType));
2016-10-06 18:55:01 +00:00
}
2016-11-12 06:58:50 +00:00
SourceBytes = source;
2016-10-06 18:55:01 +00:00
Headers[HeaderNames.ContentLength] = contentLength.ToString(CultureInfo.InvariantCulture);
Headers[HeaderNames.ContentType] = contentType;
2013-03-25 02:41:27 +00:00
}
2019-11-01 17:38:54 +00:00
/// <summary>
/// Gets or sets the source stream.
/// </summary>
/// <value>The source stream.</value>
private Stream SourceStream { get; set; }
private byte[] SourceBytes { get; set; }
/// <summary>
/// Gets the options.
/// </summary>
/// <value>The options.</value>
public IDictionary<string, string> Headers => _options;
/// <summary>
/// Fires when complete.
/// </summary>
public Action OnComplete { get; set; }
/// <summary>
/// Fires when an error occours.
/// </summary>
public Action OnError { get; set; }
/// <inheritdoc />
2016-10-25 19:02:04 +00:00
public async Task WriteToAsync(Stream responseStream, CancellationToken cancellationToken)
{
2013-03-11 04:04:08 +00:00
try
{
2016-11-12 06:58:50 +00:00
var bytes = SourceBytes;
if (bytes != null)
2016-10-06 18:55:01 +00:00
{
2020-08-07 15:38:01 +00:00
await responseStream.WriteAsync(bytes, 0, bytes.Length, cancellationToken).ConfigureAwait(false);
2016-10-06 18:55:01 +00:00
}
else
2013-03-11 04:04:08 +00:00
{
2016-10-06 18:55:01 +00:00
using (var src = SourceStream)
{
2020-08-07 15:38:01 +00:00
await src.CopyToAsync(responseStream, cancellationToken).ConfigureAwait(false);
2016-10-06 18:55:01 +00:00
}
2013-03-11 04:04:08 +00:00
}
}
2018-09-12 17:26:21 +00:00
catch
2013-03-11 04:04:08 +00:00
{
2019-11-01 17:38:54 +00:00
OnError?.Invoke();
2016-07-14 19:13:52 +00:00
2013-03-11 04:04:08 +00:00
throw;
}
2014-09-03 02:30:24 +00:00
finally
{
2019-11-01 17:38:54 +00:00
OnComplete?.Invoke();
2014-09-03 02:30:24 +00:00
}
}
}
}