Как вызвать метод IdentityServer UserManager из веб-API? - PullRequest
0 голосов
/ 24 сентября 2019

Мне нужно инициировать сброс пароля из моего веб-API.У меня есть приложение для идентификации сервера и приложения веб-API отдельно.Я использую клиент OIDC для связи с сервером идентификации.С этим клиентом я могу вызывать только вход, выход и некоторые стандартные методы.

Мне нужно сгенерировать токен сброса пароля и получить маркер сброса в API.

Я попытался включить 'Microsoft.Extensions.Identity.Core' в свой основной слой webapi, в котором есть все сущности.Но я получаю

Невозможно разрешить службу для типа Microsoft.AspNetCore.Identity.IUserStore1 [Application.Core.Entities.User] при попытке активировать Microsoft.AspNetCore.Identity.UserManager

код

private readonly UserManager<User> _userManager;

public AccountController(UserManager<User> userManager)
{
    _userManager = userManager;
}
private async Task<string> GeneratePasswordResetLinkAsync(User user)
{
    string token = await _userManager.GeneratePasswordResetTokenAsync(user);
    return token;
}

Я также пытался добавить приведенный ниже код в автозагрузку служб инъекций. AddScoped, UserManager> ();

Ответы [ 2 ]

0 голосов
/ 26 сентября 2019

Вам необходимо настроить идентификацию ядра asp.net в классе запуска веб-API.К сожалению, вы не можете просто вызвать services.AddIdentity(... в ConfigureServices, потому что за кулисами схема проверки подлинности на основе cookie регистрируется и устанавливается как схема вызова по умолчанию, как вы можете видеть в коде здесь .

Таким образом, решение, которое я получаю в итоге, состоит в том, чтобы скопировать и обновить метод AddIdentity следующим образом:

public static IdentityBuilder AddIdentityForWebApi<TUser, TRole>(
    this IServiceCollection services,
    Action<IdentityOptions> setupAction)
    where TUser : class
    where TRole : class
{
    // Hosting doesn't add IHttpContextAccessor by default
    services.AddHttpContextAccessor();

    // Identity services
    services.TryAddScoped<IUserValidator<TUser>, UserValidator<TUser>>();
    services.TryAddScoped<IPasswordValidator<TUser>, PasswordValidator<TUser>>();
    services.TryAddScoped<IPasswordHasher<TUser>, PasswordHasher<TUser>>();
    services.TryAddScoped<ILookupNormalizer, UpperInvariantLookupNormalizer>();
    services.TryAddScoped<IRoleValidator<TRole>, RoleValidator<TRole>>();

    // No interface for the error describer so we can add errors without rev'ing the interface
    services.TryAddScoped<IdentityErrorDescriber>();
    services.TryAddScoped<ISecurityStampValidator, SecurityStampValidator<TUser>>();
    services.TryAddScoped<ITwoFactorSecurityStampValidator, TwoFactorSecurityStampValidator<TUser>>();
    services.TryAddScoped<IUserClaimsPrincipalFactory<TUser>, UserClaimsPrincipalFactory<TUser, TRole>>();
    services.TryAddScoped<UserManager<TUser>>();
    services.TryAddScoped<SignInManager<TUser>>();
    services.TryAddScoped<RoleManager<TRole>>();

    if (setupAction != null)
    {
        services.Configure(setupAction);
    }

    return new IdentityBuilder(typeof(TUser), typeof(TRole), services);
}

Затем вам следует вызвать AddIdentityForWebApi при запуске приложения веб-API.Это зарегистрирует UserManager, и теперь он будет внедрен в конструктор вашего контроллера.

Затем необходимо правильно настроить API защиты данных (DPAPI), чтобы токен, генерируемый вашим веб-API (при вызове _userManager.GeneratePasswordResetTokenAsync... изВаш вопрос) может быть не защищено приложением сервера идентификации.

Поэтому я установил «Имя приложения DPAPI» для двух приложений, в обоих ConfigureServices методах:

services.AddDataProtection()
  .dataProtectionBuilder.SetApplicationName("YOUR_DPAPI_APPLICATION_NAME");

Для производствав среде веб-фермы вам придется делиться ключами DPAPI.В зависимости от вашей ситуации у вас есть несколько вариантов.

Более подробную информацию о настройке DPAPI см. В официальной документации .

Я тестировал на Kestrel под управлением Windows,Я не уверен насчет IIS.

0 голосов
/ 25 сентября 2019

Код впрыска не выглядит правильно.Попробуйте изменить его на

services.AddScoped<UserManager<Application.Core.Entities.User>>();
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...