dotnet / systemweb-adapters

MIT License
339 stars 61 forks source link

Add async APIs to push users aways from sync over async patterns #164

Open twsouthwick opened 2 years ago

twsouthwick commented 2 years ago

Summary

A number of APIs in .NET Core are async aware and probably should have been in .NET Framework. However, when adapting the behavior of core behind the APIs framework had ends up with a number of sync over async patterns. We should provide a way forward here that will guide people to using async.

Motivation and goals

We should provide async methods that end up calling into async pathways on .NET Core. Currently, this is handled either by invoking required async pathways in middleware or by calling GetAwaiter().GetResult().

In scope

This would ideally provide APIs that users on framework, core, and standard could all use. Depending on where things run, it would either be true async (i.e. on ASP.NET Core) or just return a completed task (i.e. on ASP.NET Framework).

Initial APIs identified:

probably some more (especially those that rely on any of these APIs).

Out of scope

Risks / unknowns

Examples

Currently:

public void SomeFunc(HttpResponse response)
{
  response.TransmitFile("some-path");
}

after:

public async Task SomeFunc(HttpResponse response)
{
  response.TransmitFileAsync("some-path");
}

Design considerations

davidfowl commented 2 years ago

I don't think we should do this. We don't want to change/extend this type at all.

GerardSmit commented 1 year ago

Since this isn't planned: when the interfaces like IHttpResponseEndFeature are publicly available (which been suggested for mocking https://github.com/dotnet/systemweb-adapters/issues/332#issuecomment-1526099263 and has been added to the milestone 1.3) we could make our own extension methods:

using System.Threading.Tasks;
using HttpResponse = System.Web.HttpResponse;

#if NET8_0_OR_GREATER

// ASP.NET Core implementation
using System.Runtime.CompilerServices;
using Microsoft.AspNetCore.Http.Features;
using HttpResponseCore = Microsoft.AspNetCore.Http.HttpResponse;

namespace SystemWeb
{
    public static class HttpResponseExtensions
    {
        [UnsafeAccessor(UnsafeAccessorKind.Field, Name = "_response")]
        private static extern HttpResponseCore GetResponse(HttpResponse @this);

        public static Task EndAsync(this HttpResponse @this)
            => GetResponse(@this).HttpContext.Features.GetRequiredFeature<IHttpResponseEndFeature>().EndAsync();
    }
}

#else

// ASP.NET implementation
namespace SystemWeb
{
    public static class HttpResponseExtensions
    {
        public static Task EndAsync(this HttpResponse @this)
        {
            @this.End();
            return Task.CompletedTask;
        }
    }
}

#endif