File: ResponseBodyPipeWriter.cs
Web Access
Project: src\src\Hosting\TestHost\src\Microsoft.AspNetCore.TestHost.csproj (Microsoft.AspNetCore.TestHost)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
 
using System.Diagnostics.Contracts;
using System.IO.Pipelines;
 
namespace Microsoft.AspNetCore.TestHost;
 
internal sealed class ResponseBodyPipeWriter : PipeWriter
{
    private readonly Func<Task> _onFirstWriteAsync;
    private readonly Pipe _pipe;
 
    private bool _firstWrite;
    private bool _complete;
 
    internal ResponseBodyPipeWriter(Pipe pipe, Func<Task> onFirstWriteAsync)
    {
        _pipe = pipe ?? throw new ArgumentNullException(nameof(pipe));
        _onFirstWriteAsync = onFirstWriteAsync ?? throw new ArgumentNullException(nameof(onFirstWriteAsync));
        _firstWrite = true;
    }
 
    public override async ValueTask<FlushResult> FlushAsync(CancellationToken cancellationToken)
    {
        cancellationToken.ThrowIfCancellationRequested();
        CheckNotComplete();
 
        await FirstWriteAsync();
        return await _pipe.Writer.FlushAsync(cancellationToken);
    }
 
    private Task FirstWriteAsync()
    {
        if (_firstWrite)
        {
            _firstWrite = false;
            return _onFirstWriteAsync();
        }
        return Task.CompletedTask;
    }
 
    internal void Abort(Exception innerException)
    {
        Contract.Requires(innerException != null);
        _complete = true;
        _pipe.Writer.Complete(new IOException(string.Empty, innerException));
    }
 
    internal void Complete()
    {
        if (_complete)
        {
            return;
        }
 
        // Throw for further writes, but not reads. Allow reads to drain the buffered data and then return 0 for further reads.
        _complete = true;
        _pipe.Writer.Complete();
    }
 
    private void CheckNotComplete()
    {
        if (_complete)
        {
            throw new IOException("The request was aborted or the pipeline has finished.");
        }
    }
 
    public override void Complete(Exception? exception = null)
    {
        // No-op in the non-error case
        if (exception != null)
        {
            Abort(exception);
        }
    }
 
    public override void CancelPendingFlush() => _pipe.Writer.CancelPendingFlush();
 
    public override void Advance(int bytes)
    {
        CheckNotComplete();
        _pipe.Writer.Advance(bytes);
    }
 
    public override Memory<byte> GetMemory(int sizeHint = 0)
    {
        CheckNotComplete();
        return _pipe.Writer.GetMemory(sizeHint);
    }
 
    public override Span<byte> GetSpan(int sizeHint = 0)
    {
        CheckNotComplete();
        return _pipe.Writer.GetSpan(sizeHint);
    }
 
    public override bool CanGetUnflushedBytes => _pipe.Writer.CanGetUnflushedBytes;
 
    public override long UnflushedBytes => _pipe.Writer.UnflushedBytes;
}