ASP.NET MVC POST首次尝试无限期等待

时间:2017-07-18 14:32:21

标签: c# asp.net asp.net-mvc async-await taskcompletionsource

我有一个用Html.BeginForm制作的标准表单,它发布到控制器中的异步操作。它看起来像这样(这是一个大纲,而不是实际的代码):

    [HttpPost]
    public async Task<ActionResult> Index(UserCreds creds)
    {
        try
        {
            if (ModelState.IsValid)
            {
                var user = await loginRep.Login(creds.Username, creds.Password);

                if (user != null)
                {
                     _context.SetAuthenticationToken(user);

                     return RedirectToAction("Index", "Landing");
                } 
                else
                {
                    ModelState.AddModelError("", "Login failed.");
                }
            }
            else
            {
                _logger.Debug(string.Format("User failed authentication."));
            }

        }
        catch (Exception ex)
        {
            throw new HttpException(500, string.Format("An error occured during the execution of the action {0} from Controller {1}", "Index", "Login"), ex);
        }

        return View();
    }

首次提交表单后,浏览器将无限期地等待响应,即使可以看到调试程序踩到了RedirectToAction。如果然后在浏览器中停止请求,然后再次提交,则会发生重定向。所有后续登录尝试也会成功重定向。在第一次尝试期间,也会以某种方式设置身份验证令牌。

这可能与loginRep.Login内的委托使用有关。在里面它最终会做这样的事情:

private async Task<LoginResponse> SendLoginRequest(string username, string password)
{
    TaskCompletionSource<LoginResponse> tcs = new TaskCompletionSource<LoginResponse>();

    LoginResponseCallback callback = null;
    callback = new LoginResponseHandler(delegate (response) {
       securityService.OnLoginResponse -= callback;
       tcs.SetResult(response);        
    });

    securityService.OnLoginResponse += callback;

    securityService.SendLoginRequest(username, password);

    return await tcs.Task;
}

有谁知道发生了什么?如果它是一个死锁,我不希望看到调试器到达重定向,我也不希望登录适用于除第一个之外的所有尝试。

请注意,如果只是跳过发送登录请求并且只是硬编码成功的响应,那么表单会第一次起作用。

1 个答案:

答案 0 :(得分:2)

确定。问题得到了解决。我展示的两个代码示例没有任何问题。错误在于设置我的安全服务,所以不幸的是,这个应用程序非常具体。

也就是说,无限期等待是因为Global.asax.cs中的Application_Error实际上吞噬了某些异常。一旦我将其更改为始终重定向到我的错误页面,无论如何,至少它会在问题发生时立即重定向到错误页面,而不是从用户的角度出发。