In a new ASPNET MVC application you now get the AspIdentity goodies for free.
There's a harmless little line 'plug in your email service here'.
So I did:
public class EmailService : IIdentityMessageService
{
    private static My.Services.IEmailService _emailservice;
    public EmailService(Insolvency.Services.IEmailService emailservice)
    {
        _emailservice = emailservice;
    }
    public Task SendAsync(IdentityMessage message)
    {
        _emailservice.SendEmail(message);
       return Task.FromResult(0);
    }
}
and now the joy:
public class ApplicationUserManager : UserManager<ApplicationUser>
{
    private My.Services.IEmailService _emailservice;
    public ApplicationUserManager(IUserStore<ApplicationUser> store, My.Services.IEmailService emailservice): base(store)        
    {
        _emailservice = emailservice;
    }
    public static ApplicationUserManager Create(IdentityFactoryOptions<ApplicationUserManager> options, IOwinContext context)
    {
        var manager = new ApplicationUserManager(new UserStore<ApplicationUser>(context.Get<ApplicationDbContext>()), _emailservice);
       ...
as Owin kicks in it calls the Create on ApplicationUserManager in Startup.Auth.cs:
public partial class Startup 
   {
    public void ConfigureAuth(IAppBuilder app) 
     {
        ...
        app.CreatePerOwinContext<ApplicationUserManager>(ApplicationUserManager.Create);
and as I'm using AutoFac as my IoC container, in global.asax.cs
builder.RegisterType<My.Services.EmailService>().As<IEmailService>();
if the Create method is static so I get:_emailService is null
I've looked here:http://forums.asp.net/post/5293670.aspx, and How do i create an instance of UserManager and Using Autofac to provide types exported by static factory. but no luck.
If I change:
private My.Services.IEmailService _emailservice;
to be public non-static I feel IoC gods shaking their heads, and I can't build 'object reference required'
I fought with this for a long while today, and in the end, the easiest thing to do was this.
Inside of the
public static ApplicationUserManager Create(IdentityFactoryOptions<ApplicationUserManager> options, IOwinContext context)
method, find these lines of code.
manager.EmailService = new EmailService(); manager.SmsService = new SmsService();
And change it to this.
manager.EmailService = new EmailService(DependencyResolver.Current.GetService<Insolvency.Services.IEmailService>());
I don't like the Service Locator pattern, but all of the MVC templated code uses it and it's best not to fight the framework.
Today I was struggling with the same issue (still working on it since I'm new to Asp.Identity). I did it this way:
Startup.cs (use your own container)
public class Startup
{
    public void Configuration(IAppBuilder app)
    {
        var authConfigurator = new AuthConfigurator();
        authConfigurator.ConfigureAuth(app);
        var unityContainer = UnityConfig<MvcUnityDependencyContainer>.UseContainer();
        //Asp identity registration
        IDataProtectionProvider dataProtectionProvider = app.GetDataProtectionProvider();
        unityContainer.RegisterInstance(dataProtectionProvider);
        unityContainer.RegisterType<DbContext, ApplicationDbContext>(new HierarchicalLifetimeManager());
        unityContainer.RegisterType<UserManager<ApplicationUser, int>>(new HierarchicalLifetimeManager());
        unityContainer.RegisterType IIdentityMessageService, EmailService>();
        unityContainer.RegisterType<IUserStore<ApplicationUser, int>,
            UserStore<ApplicationUser, CustomRole, int, CustomUserLogin, CustomUserRole, CustomUserClaim>>(
            new InjectionConstructor(new ApplicationDbContext()));
        unityContainer.RegisterType<IIdentityMessageService, EmailService>();
    }
}
ApplicationUserManager (I removed static method Create):
public class ApplicationUserManagerService : UserManager<ApplicationUser, int>
{
    public ApplicationUserManagerService(IUserStore<ApplicationUser, int> store, 
                                         IIdentityMessageService emailService,
                                         IDataProtectionProvider dataProtectionProvider)
                                         : base(store)
    {
        UserTokenProvider = new EmailTokenProvider<ApplicationUser, int>();
        EmailService = emailService;
        Configure(dataProtectionProvider);
    }
    private void Configure(IDataProtectionProvider dataProtectionProvider)
    {
        // Configure validation logic for usernames
        UserValidator = new UserValidator<ApplicationUser, int>(this)
        {
            AllowOnlyAlphanumericUserNames = false,
            RequireUniqueEmail = true
        };
        // Configure validation logic for passwords
        PasswordValidator = new PasswordValidator
        {
            RequiredLength = 1,
            RequireNonLetterOrDigit = false,
            RequireDigit = false,
            RequireLowercase = false,
            RequireUppercase = false,
        };
        // Register two factor authentication providers. This application uses Phone and Emails as a step of receiving a code for verifying the user
        // You can write your own provider and plug in here.
        RegisterTwoFactorProvider("PhoneCode", new PhoneNumberTokenProvider<ApplicationUser, int>
        {
            MessageFormat = "Your security code is: {0}"
        });
        RegisterTwoFactorProvider("EmailCode", new EmailTokenProvider<ApplicationUser, int>
        {
            Subject = "Security Code",
            BodyFormat = "Your security code is: {0}"
        });
        if (dataProtectionProvider != null)
        {
            UserTokenProvider = new DataProtectorTokenProvider<ApplicationUser, int>(dataProtectionProvider.Create("ASP.NET Identity"));
        }
    }
}
Controller
public class AccountController : Controller
{
    private ApplicationUserManagerService _userManagerService;
    public AccountController(ApplicationUserManagerService userManagerService)
    {
        Contract.Requires(userManagerService != null);
        _userManagerService = userManagerService;
    }
    /*....*/
}
ApplicationUser
public class ApplicationUser : IdentityUser<int, CustomUserLogin, CustomUserRole, CustomUserClaim>
{
    public async Task<ClaimsIdentity> GenerateUserIdentityAsync(UserManager<ApplicationUser, int> manager)
    {
        var userIdentity = await manager.CreateIdentityAsync(this, DefaultAuthenticationTypes.ApplicationCookie);
        return userIdentity;
    }
}
public class CustomRole : IdentityRole<int, CustomUserRole>
{
    public CustomRole() { }
    public CustomRole(string name) { Name = name; }
}
public class CustomUserClaim : IdentityUserClaim<int> { }
public class CustomUserLogin : IdentityUserLogin<int> { }
public class CustomUserRole : IdentityUserRole<int> { }
This works for me but please feel free to suggest better solution.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With