#asp.net #identity
#asp.net #идентификатор
Вопрос:
Я использую asp.net идентификатор 2 в новом проекте mvc (я настроил его на использование идентификатора пользователя int вместо идентификатора строки по умолчанию). Он успешно выполняет обычный вход в систему с существующей учетной записью, но если вместо этого я использую внешний вход, используя тот же адрес электронной почты, что и существующая учетная запись, используя Google sign in, я могу войти в систему с помощью Google и разрешить приложение, но когда оно перенаправляет меня на действие ExternalLoginCallback, оно не может найтипользователь. GetExternalLoginInfoAsync() работает для получения loginInfo, но когда я передаю это ExternalSignIn, внутри него он не может найти пользователя при вызове var user = await UserManager .FindAsync(loginInfo.Вход в систему);. FindAsync возвращает значение null. Может ли это быть ошибкой в новой структуре идентификации? Пользователь с этим адресом электронной почты уже существует в базе данных, поэтому он должен быть в состоянии найти его (я проверил, что он тоже там).
Вот действие контроллера (которое точно такое же, как в asp.net проект образцов удостоверений личности):
public async Task<ActionResult> ExternalLoginCallback(string returnUrl)
{
var loginInfo = await AuthenticationManager.GetExternalLoginInfoAsync();
if (loginInfo == null)
{
return RedirectToAction("Login");
}
// Sign in the user with this external login provider if the user already has a login
var result = await SignInHelper.ExternalSignIn(loginInfo, isPersistent: false);
switch (result)
{
case SignInStatus.Success:
return RedirectToLocal(returnUrl);
case SignInStatus.LockedOut:
return View("Lockout");
case SignInStatus.RequiresTwoFactorAuthentication:
return RedirectToAction("SendCode", new { ReturnUrl = returnUrl });
case SignInStatus.Failure:
default:
// If the user does not have an account, then prompt the user to create an account
ViewBag.ReturnUrl = returnUrl;
ViewBag.LoginProvider = loginInfo.Login.LoginProvider;
return View("ExternalLoginConfirmation", new ExternalLoginConfirmationViewModel { Email = loginInfo.Email });
}
}
Вот метод входа, вызываемый в приведенном выше действии (также в проекте образцов идентификаторов):
public async Task<SignInStatus> ExternalSignIn(ExternalLoginInfo loginInfo, bool isPersistent)
{
var user = await UserManager.FindAsync(loginInfo.Login);
if (user == null)
{
return SignInStatus.Failure;
}
if (await UserManager.IsLockedOutAsync(user.Id))
{
return SignInStatus.LockedOut;
}
return await SignInOrTwoFactor(user, isPersistent);
}
Если я изменю FindAsync в приведенном выше методе на FindByEmailAsync, он затем найдет пользователя — почему бы просто не найти пользователя по электронной почте? т.е. var user = await UserManager .FindByEmailAsync(loginInfo.Email);
Просто провел еще несколько исследований, и, похоже, это происходит только тогда, когда вы меняете pk на int и должны создать новый пользовательский класс user, role и т. Д.
Комментарии:
1. Я вижу, что в таблице AspNetUserLogins ничего не сохраняется, что, возможно, имеет к этому какое-то отношение, но сохранение в это происходит за кулисами в рамках, и я все равно использую шаблонный код, как я уже упоминал, поэтому не уверен, почему это не работает как есть.
Ответ №1:
Вы пытаетесь извлечь данные из вновь установленного объекта идентификации, но контекст еще не обновлен.
Есть два способа обойти это:
-Подождите, пока контекст не будет обновлен, поскольку доступ к User.identity
информации еще не получен, дождитесь следующего вызова.
-Попробуйте получить контекст для сохранения обновления.
var store = new UserStore<ApplicationUser>(new MyDbContext());
var manager = new UserManager(store);
manager.UpdateAsync(user);
var ctx = store.context;
ctx.saveChanges();