22

Visual Studio 2013 を使用して、ASP.NET 4.5 (Microsoft.AspNet.Identity) に新しい Identity 機能のカスタム バージョンを実装しようとしています。エラーなしで実行するための努力。以下にコードをリストしました。ローカル登録を行うと、データベース テーブルが作成されますが、CreateLocalUser メソッドは失敗します。誰かが必要な変更を特定するのを手伝ってくれることを願っています。

モデル/MembershipModel.cs

using Microsoft.AspNet.Identity;
using Microsoft.AspNet.Identity.EntityFramework;
using System;
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations;
using System.ComponentModel.DataAnnotations.Schema;
using System.Data.Entity;
using System.Linq;
using System.Web;

namespace thePulse.web.Models
{
    public class PulseUser : IUser
    {
        public PulseUser() { }
        public PulseUser(string userName) 
        {
            UserName = userName;
        }

        [Key]
        public string Id { get; set; }
        [Required]
        [StringLength(20)]
        public string UserName { get; set; }
        [StringLength(100)]
        public string Email { get; set; }
        [Column(TypeName = "Date")]
        public DateTime? BirthDate { get; set; }
        [StringLength(1)]
        public string Gender { get; set; }
    }

    public class PulseUserClaim : IUserClaim
    {
        public PulseUserClaim() { }

        [Key]
        public string Key { get; set; }
        public string UserId { get; set; }
        public string ClaimType { get; set; }
        public string ClaimValue { get; set; }

    }

    public class PulseUserSecret : IUserSecret
    {
        public PulseUserSecret() { }
        public PulseUserSecret(string userName, string secret)
        {
            UserName = userName;
            Secret = secret;
        }

        [Key]
        public string UserName { get; set; }
        public string Secret { get; set; }

    }

    public class PulseUserLogin : IUserLogin
    {
        public PulseUserLogin() { }
        public PulseUserLogin(string userId, string loginProvider, string providerKey) 
        {
            LoginProvider = LoginProvider;
            ProviderKey = providerKey;
            UserId = userId;
        }

        [Key, Column(Order = 0)]
        public string LoginProvider { get; set; }
        [Key, Column(Order = 1)]
        public string ProviderKey { get; set; }
        public string UserId { get; set; }
    }

    public class PulseRole : IRole
    {
        public PulseRole() { }
        public PulseRole(string roleId)
        {
            Id = roleId;
        }

        [Key]
        public string Id { get; set; }
    }

    public class PulseUserRole : IUserRole
    {
        public PulseUserRole() { }

        [Key, Column(Order = 0)]
        public string RoleId { get; set; }
        [Key, Column(Order = 1)]
        public string UserId { get; set; }
    }

    public class PulseUserContext : IdentityStoreContext
    {
        public PulseUserContext(DbContext db) : base(db)
        {
            Users = new UserStore<PulseUser>(db);
            Logins = new UserLoginStore<PulseUserLogin>(db);
            Roles = new RoleStore<PulseRole, PulseUserRole>(db);
            Secrets = new UserSecretStore<PulseUserSecret>(db);
            UserClaims = new UserClaimStore<PulseUserClaim>(db);
        }
    }

    public class PulseDbContext : IdentityDbContext<PulseUser, PulseUserClaim, PulseUserSecret, PulseUserLogin, PulseRole, PulseUserRole>
    {
    }
}

Controllers/AccountController.cs への変更

public AccountController() 
{

    IdentityStore = new IdentityStoreManager(new PulseUserContext(new PulseDbContext()));
    AuthenticationManager = new IdentityAuthenticationManager(IdentityStore);
}

//
// POST: /Account/Register
[HttpPost]
[AllowAnonymous]
[ValidateAntiForgeryToken]
public async Task<ActionResult> Register(RegisterViewModel model)
{
    if (ModelState.IsValid)
    {
        try
        {
            // Create a profile, password, and link the local login before signing in the user
            PulseUser user = new PulseUser(model.UserName);
            if (await IdentityStore.CreateLocalUser(user, model.Password))
            {
                await AuthenticationManager.SignIn(HttpContext, user.Id, isPersistent: false);
                return RedirectToAction("Index", "Home");
            }
            else
            {
                ModelState.AddModelError("", "Failed to register user name: " + model.UserName);
            }
        }
        catch (IdentityException e)
        {
            ModelState.AddModelError("", e.Message);
        }
    }

    // If we got this far, something failed, redisplay form
    return View(model);
}

上で述べたように、この実装は、CreateLocalUser メソッド (Microsoft.AspNet.Identity.EntityFramework) が失敗すると失敗します。理由がわかりません。

4

3 に答える 3

12

ここでの問題は、IdentityStoreManager が ID EF モデルの既定の実装に強く依存していることです。たとえば、CreateLocalUser メソッドは UserSecret オブジェクトと UserLogin オブジェクトを作成してストアに保存しますが、ストアが既定のモデル タイプを使用していない場合、これは機能しません。そのため、モデル タイプをカスタマイズすると、IdentityStoreManager でスムーズに動作しなくなります。

IUser モデルのみをカスタマイズするため、コードを単純化して、既定の ID ユーザーからカスタム ユーザーを継承し、ID EF モデルから他のモデルを再利用します。

using Microsoft.AspNet.Identity;
using Microsoft.AspNet.Identity.EntityFramework;
using System;
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations;
using System.ComponentModel.DataAnnotations.Schema;
using System.Data.Entity;
using System.Linq;
using System.Web;

namespace WebApplication11.Models
{
    public class PulseUser : User
    {
        public PulseUser() { }
        public PulseUser(string userName) : base(userName)
        {
        }

        [StringLength(100)]
        public string Email { get; set; }
        [Column(TypeName = "Date")]
        public DateTime? BirthDate { get; set; }
        [StringLength(1)]
        public string Gender { get; set; }
    }

    public class PulseUserContext : IdentityStoreContext
    {
        public PulseUserContext(DbContext db) : base(db)
        {
            this.Users = new UserStore<PulseUser>(this.DbContext);
        }
    }

    public class PulseDbContext : IdentityDbContext<PulseUser, UserClaim, UserSecret, UserLogin, Role, UserRole>
    {
    }
}

上記のコードは、Identity API のプレビュー バージョンで動作するはずです。

今後のリリースの IdentityStoreManager API は、この問題を既に認識しており、すべての非 EF 依存コードを基本クラスに変更して、継承してカスタマイズできるようにします。ここですべての問題を解決する必要があります。ありがとう。

于 2013-07-29T18:47:51.140 に答える
1

PulseUser.Id は文字列として定義されていますが、値が設定されていないようです。Id に GUID を使用するつもりでしたか? その場合は、コンストラクターで初期化します。

    public PulseUser() : this(String.Empty) { }
    public PulseUser(string userName) 
    {
        UserName = userName;
        Id = Guid.NewGuid().ToString();
    }

また、ユーザー名がまだ存在していないことを確認する必要があります。PulseDbContext での DbEntityValidationResult のオーバーライドを見てください。VS2013 で新しい MVC プロジェクトを実行して、例を確認してください。

于 2013-07-29T12:23:11.113 に答える
0

RTM に移行する際にこれに多くの変更があるため、すべての ID サインインなどに WebApi コントローラーを使用する SPA テンプレートを更新しました。あなたがそれを見たことがないなら、それは本当にクールなテンプレートです。

すべてのコードをここに置きます: https://github.com/s093294/aspnet-identity-rtm/tree/master

(注意してください、それはインスピレーションのためだけです。私はそれを機能させただけで、それ以上のものはありません。適切にバグも1つか2つ持っています)。

于 2013-10-04T00:26:34.553 に答える