我有一个用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;
}
有谁知道发生了什么?如果它是一个死锁,我不希望看到调试器到达重定向,我也不希望登录适用于除第一个之外的所有尝试。
请注意,如果只是跳过发送登录请求并且只是硬编码成功的响应,那么表单会第一次起作用。
答案 0 :(得分:2)
确定。问题得到了解决。我展示的两个代码示例没有任何问题。错误在于设置我的安全服务,所以不幸的是,这个应用程序非常具体。
也就是说,无限期等待是因为Global.asax.cs中的Application_Error实际上吞噬了某些异常。一旦我将其更改为始终重定向到我的错误页面,无论如何,至少它会在问题发生时立即重定向到错误页面,而不是从用户的角度出发。