3

我有一个用 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;
}

有谁明白发生了什么?如果这是一个死锁,我不会期望看到调试器到达重定向,我也不会期望登录对除第一次以外的所有尝试都有效。

请注意,如果一个人只是跳过发送登录请求而只是硬编码成功响应的样子,则该表单第一次工作。

4

1 回答 1

2

好的。问题已解决。我展示的两个代码示例没有任何问题。错误在于设置我的安全服务,所以不幸的是它是特定于这个应用程序的。

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

于 2017-07-19T16:09:26.560 回答