ASP.NET Core Web API / Razor 페이지에서 발생할 수 있는 async void 메서드의 부작용
지난 글에서 async void 메서드의 부작용을 설명했었는데요,
C# - (async Task가 아닌) async void 사용 시의 부작용
; https://www.sysnet.pe.kr/2/0/13884
이런 문제가 ASP.NET Core Web Application 기반의 Web API나 Razor 페이지에서 발생할 수 있습니다. 하나씩 예를 들어 볼까요? ^^
가령, Web API 프로젝트를 생성한 경우 기본 제공되는 WeatherForecastController는,
using Microsoft.AspNetCore.Mvc;
namespace WebApplication1.Controllers
{
[ApiController]
[Route("[controller]")]
public class WeatherForecastController : ControllerBase
{
private static readonly string[] Summaries = new[]
{
"Freezing", "Bracing", "Chilly", "Cool", "Mild", "Warm", "Balmy", "Hot", "Sweltering", "Scorching"
};
private readonly ILogger<WeatherForecastController> _logger;
public WeatherForecastController(ILogger<WeatherForecastController> logger)
{
_logger = logger;
}
[HttpGet]
public IEnumerable<WeatherForecast> Get()
{
return Enumerable.Range(1, 5).Select(index => new WeatherForecast
{
Date = DateOnly.FromDateTime(DateTime.Now.AddDays(index)),
TemperatureC = Random.Shared.Next(-20, 55),
Summary = Summaries[Random.Shared.Next(Summaries.Length)]
})
.ToArray();
}
}
}
Get 메서드가 동기 함수인데요, 다행히 이런 경우에는 (반드시 값을 반환해야 하기 때문에) async void로 작성하는 실수를 할 여지가 없습니다.
public async IEnumerable<WeatherForecast> Get()
[내부에서 await 호출을 하고 싶다면]
==> public async Task<IEnumerable<WeatherForecast>> Get()
하지만 Post 등의 메서드를 추가했을 때,
[HttpPost]
public void Post()
{
Thread.Sleep(1000);
}
이 메서드 호출을 curl로 테스트하면 1초 후에 응답이 오지만,
c:\temp> curl -v -X POST http://localhost:5252/weatherforecast
async void로 작성하면,
[HttpPost]
public async void Post()
{
await Task.Delay(1000);
throw new Exception("Error");
}
curl 호출 시 1초를 기다리지 않고 (심지어 위의 코드에서 예외도 발생시키고 있는데) 그냥 "HTTP/1.1 200 OK" 응답을 받습니다. (그 이유는 이미
지난 글에서 설명했기 때문에 생략합니다.)
당연히, ASP.NET Core Web App (razor) 프로젝트를 만들어도 이와 유사한 현상을 겪을 수 있습니다.
예를 들어 기본 생성한 Razor 프로젝트의 Index.cshtml/cs 파일을 async Task OnGet으로 변경해 다음과 같이 코딩하면,
@page
@model IndexModel
@{
ViewData["Title"] = "Home page";
}
<div class="text-center">
<h1 class="display-4">Welcome</h1>
<p>Learn about <a href="https://learn.microsoft.com/aspnet/core?WT.mc_id=DT-MVP-4038148">building Web apps with ASP.NET Core</a>.</p>
<p>@ViewData["Text"]</p>
</div>
using Microsoft.AspNetCore.Mvc;
using Microsoft.AspNetCore.Mvc.RazorPages;
namespace WebApplication2.Pages;
public class IndexModel : PageModel
{
private readonly ILogger<IndexModel> _logger;
public IndexModel(ILogger<IndexModel> logger)
{
_logger = logger;
}
public async Task OnGet()
{
await Task.Delay(1000);
this.ViewData["Text"] = "My Async Test";
}
}
화면에는 정상적으로 "My Async Test"가 출력되지만, async void로 작성하면,
public async void OnGet()
{
await Task.Delay(1000);
this.ViewData["Text"] = "My Async Test";
}
"My Async Test"는 화면에 출력되지 않습니다. 결국, 컴파일이 된다고 해서 그것이 정상적으로 동작한다고 기대해서는 안 됩니다.
[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]