我有一个用 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;
}
有谁明白发生了什么?如果这是一个死锁,我不会期望看到调试器到达重定向,我也不会期望登录对除第一次以外的所有尝试都有效。
请注意,如果一个人只是跳过发送登录请求而只是硬编码成功响应的样子,则该表单第一次工作。