asp.net идентификатор 2 внешний вход в систему не сохранение пользовательского логина и не поиск пользователя после входа в систему провайдера

#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();