userManager.CreateAsync System.ObjectDisposedException 未处理
userManager.CreateAsync System.ObjectDisposedException was unhandled
我们正在尝试在 ASP.Net 5 项目中使用 Microsoft 帐户身份验证。我们不需要本地身份验证,也不需要用户名。
在 Web 应用程序的 ASP.Net 5 模板中,在使用外部提供商登录后,在 AccountController 中控制 returns 到 ExternalLoginCallback
。
如果用户没有在本地注册 ExternalLoginCallback
returns 用户到注册屏幕。我试图修改 ExternalLoginCallback
以自动注册新用户,如下所示。
public async Task<IActionResult> ExternalLoginCallback(string returnUrl = null)
{
var info = await _signInManager.GetExternalLoginInfoAsync();
if (info == null)
{
return RedirectToAction(nameof(Login));
}
// Sign in the user with this external login provider if the user already has a login.
var result = await _signInManager.ExternalLoginSignInAsync(info.LoginProvider, info.ProviderKey, isPersistent: false);
if (result.Succeeded)
{
_logger.LogInformation(5, "User logged in with {Name} provider.", info.LoginProvider);
return RedirectToLocal(returnUrl);
}
if (result.RequiresTwoFactor)
{
return RedirectToAction(nameof(SendCode), new { ReturnUrl = returnUrl });
}
if (result.IsLockedOut)
{
return View("Lockout");
}
else
{
// If the user does not have an account, then ask the user to create an account.
//ViewData["ReturnUrl"] = returnUrl;
//ViewData["LoginProvider"] = info.LoginProvider;
//var email = info.ExternalPrincipal.FindFirstValue(ClaimTypes.Email);
//return View("ExternalLoginConfirmation", new ExternalLoginConfirmationViewModel { Email = email });
// The user has not previously logged in with an external provider. Create a new user.
CreateUser(info);
return RedirectToLocal(returnUrl);
}
}
CreateUser 执行从 ExternalLoginConfirmation
复制的代码,因为它出现在 ASP.Net 5 网络应用程序模板中。
private async void CreateUser(ExternalLoginInfo info)
{
var email = info.ExternalPrincipal.FindFirstValue(ClaimTypes.Email);
var user = new ApplicationUser { UserName = email, Email = email };
var result = await _userManager.CreateAsync(user);
if (result.Succeeded)
{
result = await _userManager.AddLoginAsync(user, info);
if (result.Succeeded)
{
await _signInManager.SignInAsync(user, isPersistent: false);
_logger.LogInformation(6, "User created an account using {Name} provider.", info.LoginProvider);
}
}
AddErrors(result);
}
线上报错
var result = await _userManager.CreateAsync(user);
错误是
System.ObjectDisposedException was unhandled Message: An unhandled exception of type 'System.ObjectDisposedException' occurred in mscorlib.dll. Additional information: Cannot access a disposed object.
我已经重新启动了我的机器以防万一 'just one of those things',但错误再次出现。
使用 async void
方法很少是一个好主意,它是引入奇怪的竞争条件的最佳方式,就像您正在经历的那样:因为您的 CreateUser
方法没有 return 一个任务,它不能被 ExternalLoginCallback
等待并且请求在 CreateUser
有时间执行数据库操作之前完成(当请求完成时,DI 系统调用 Dispose
在作用域依赖项上,例如您的 EF 上下文)。
将您的 CreateUser
方法更新为 return a Task
并从 ExternalLoginCallback
等待它,它应该可以工作:
await CreateUser(info);
我们正在尝试在 ASP.Net 5 项目中使用 Microsoft 帐户身份验证。我们不需要本地身份验证,也不需要用户名。
在 Web 应用程序的 ASP.Net 5 模板中,在使用外部提供商登录后,在 AccountController 中控制 returns 到 ExternalLoginCallback
。
如果用户没有在本地注册 ExternalLoginCallback
returns 用户到注册屏幕。我试图修改 ExternalLoginCallback
以自动注册新用户,如下所示。
public async Task<IActionResult> ExternalLoginCallback(string returnUrl = null)
{
var info = await _signInManager.GetExternalLoginInfoAsync();
if (info == null)
{
return RedirectToAction(nameof(Login));
}
// Sign in the user with this external login provider if the user already has a login.
var result = await _signInManager.ExternalLoginSignInAsync(info.LoginProvider, info.ProviderKey, isPersistent: false);
if (result.Succeeded)
{
_logger.LogInformation(5, "User logged in with {Name} provider.", info.LoginProvider);
return RedirectToLocal(returnUrl);
}
if (result.RequiresTwoFactor)
{
return RedirectToAction(nameof(SendCode), new { ReturnUrl = returnUrl });
}
if (result.IsLockedOut)
{
return View("Lockout");
}
else
{
// If the user does not have an account, then ask the user to create an account.
//ViewData["ReturnUrl"] = returnUrl;
//ViewData["LoginProvider"] = info.LoginProvider;
//var email = info.ExternalPrincipal.FindFirstValue(ClaimTypes.Email);
//return View("ExternalLoginConfirmation", new ExternalLoginConfirmationViewModel { Email = email });
// The user has not previously logged in with an external provider. Create a new user.
CreateUser(info);
return RedirectToLocal(returnUrl);
}
}
CreateUser 执行从 ExternalLoginConfirmation
复制的代码,因为它出现在 ASP.Net 5 网络应用程序模板中。
private async void CreateUser(ExternalLoginInfo info)
{
var email = info.ExternalPrincipal.FindFirstValue(ClaimTypes.Email);
var user = new ApplicationUser { UserName = email, Email = email };
var result = await _userManager.CreateAsync(user);
if (result.Succeeded)
{
result = await _userManager.AddLoginAsync(user, info);
if (result.Succeeded)
{
await _signInManager.SignInAsync(user, isPersistent: false);
_logger.LogInformation(6, "User created an account using {Name} provider.", info.LoginProvider);
}
}
AddErrors(result);
}
线上报错
var result = await _userManager.CreateAsync(user);
错误是
System.ObjectDisposedException was unhandled Message: An unhandled exception of type 'System.ObjectDisposedException' occurred in mscorlib.dll. Additional information: Cannot access a disposed object.
我已经重新启动了我的机器以防万一 'just one of those things',但错误再次出现。
使用 async void
方法很少是一个好主意,它是引入奇怪的竞争条件的最佳方式,就像您正在经历的那样:因为您的 CreateUser
方法没有 return 一个任务,它不能被 ExternalLoginCallback
等待并且请求在 CreateUser
有时间执行数据库操作之前完成(当请求完成时,DI 系统调用 Dispose
在作用域依赖项上,例如您的 EF 上下文)。
将您的 CreateUser
方法更新为 return a Task
并从 ExternalLoginCallback
等待它,它应该可以工作:
await CreateUser(info);