File: StreamHelperExtensions.cs
Web Access
Project: src\src\Http\WebUtilities\src\Microsoft.AspNetCore.WebUtilities.csproj (Microsoft.AspNetCore.WebUtilities)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
 
using System.Buffers;
 
namespace Microsoft.AspNetCore.WebUtilities;
 
/// <summary>
/// HTTP extension methods for <see cref="Stream"/>.
/// </summary>
public static class StreamHelperExtensions
{
    private const int _maxReadBufferSize = 1024 * 4;
 
    /// <summary>
    /// Reads the specified <paramref name="stream"/> to the end.
    /// <para>
    /// This API is effective when used in conjunction with buffering. It allows
    /// a buffered request stream to be synchronously read after it has been completely drained.
    /// </para>
    /// </summary>
    /// <param name="stream">The <see cref="Stream"/> to completely read.</param>
    /// <param name="cancellationToken">The token to monitor for cancellation requests.</param>
    public static Task DrainAsync(this Stream stream, CancellationToken cancellationToken)
    {
        return stream.DrainAsync(ArrayPool<byte>.Shared, null, cancellationToken);
    }
 
    /// <summary>
    /// Reads the specified <paramref name="stream"/> to the end.
    /// <para>
    /// This API is effective when used in conjunction with buffering. It allows
    /// a buffered request stream to be synchronously read after it has been completely drained.
    /// </para>
    /// </summary>
    /// <param name="stream">The <see cref="Stream"/> to completely read.</param>
    /// <param name="limit">The maximum number of bytes to read. Throws if the <see cref="Stream"/> is larger than this limit.</param>
    /// <param name="cancellationToken">The token to monitor for cancellation requests.</param>
    public static Task DrainAsync(this Stream stream, long? limit, CancellationToken cancellationToken)
    {
        return stream.DrainAsync(ArrayPool<byte>.Shared, limit, cancellationToken);
    }
 
    /// <summary>
    /// Reads the specified <paramref name="stream"/> to the end.
    /// <para>
    /// This API is effective when used in conjunction with buffering. It allows
    /// a buffered request stream to be synchronously read after it has been completely drained.
    /// </para>
    /// </summary>
    /// <param name="stream">The <see cref="Stream"/> to completely read.</param>
    /// <param name="bytePool">The byte array pool to use.</param>
    /// <param name="limit">The maximum number of bytes to read. Throws if the <see cref="Stream"/> is larger than this limit.</param>
    /// <param name="cancellationToken">The token to monitor for cancellation requests.</param>
    public static async Task DrainAsync(this Stream stream, ArrayPool<byte> bytePool, long? limit, CancellationToken cancellationToken)
    {
        cancellationToken.ThrowIfCancellationRequested();
        var buffer = bytePool.Rent(_maxReadBufferSize);
        long total = 0;
        try
        {
            var read = await stream.ReadAsync(buffer.AsMemory(), cancellationToken);
            while (read > 0)
            {
                // Not all streams support cancellation directly.
                cancellationToken.ThrowIfCancellationRequested();
                if (limit.HasValue && limit.GetValueOrDefault() - total < read)
                {
                    throw new InvalidDataException($"The stream exceeded the data limit {limit.GetValueOrDefault()}.");
                }
                total += read;
                read = await stream.ReadAsync(buffer.AsMemory(), cancellationToken);
            }
        }
        finally
        {
            bytePool.Return(buffer);
        }
    }
}