2

おそらく明らかな何かが欠けていますが、SDF ベースの ASP.NET MVC 4 Web アプリを新しい単純なメンバーシップで動作させることができません。他の初心者の参考になるように、私の手順を詳しく説明します。

まず、新しいメンバーシップ システムの紹介が非常に便利であることがわかりました。 new-asp-net-4-5-web-forms-and-asp-net-mvc-4-templates.aspx . 既存のデータベース (SDF は本格的な既存の SQL Server db の一時的なプレースホルダーです) を使用したコード ファーストでの私の手順は次のとおりです。

  1. VS 2012 で新しいインターネット アプリを作成しました。
  2. 新しい SDF ファイルを App_Data ( Accounts.sdf) に追加し、そこにユーザーとロールのテーブルを作成しました。
  3. に新しい接続文字列を追加しましたweb.config:
<connectionStrings>
      <clear/>
      <add name="AccountsContext" connectionString="Data Source=|DataDirectory|\Accounts.sdf;Persist Security Info=False" providerName="System.Data.SqlServerCe.4.0" />
    </connectionStrings>
  1. 中間データ層でホストされてInitializeSimpleMembershipAttribute.csいる独自の を使用するようにファイルを変更しました。datacontextここに、テンプレート コードに加えたいくつかの関連する変更を貼り付けます。
...
    public SimpleMembershipInitializer()
    {
        Database.SetInitializer(null);
        try
        {
            using (AccountsContext context = new AccountsContext())
            {
                if (!context.Database.Exists())
                {
                    ((IObjectContextAdapter)context).ObjectContext.CreateDatabase();
                }
            }
            WebSecurity.InitializeDatabaseConnection("AccountsContext", "User", "Id", "Name", autoCreateTables: true);
            // seed data here...
        }
    ...

これがデータ コンテキストです (デフォルトの接続文字列名に注意してctorください)。

public sealed class AccountsContext : DbContext
{
    public DbSet<User> Users { get; set; }
    public DbSet<Role> Roles { get; set; }

    public AccountsContext() : base("Name=AccountsContext")
    {
        Database.Initialize(false);
    }

    public AccountsContext(string nameOrConnectionString) : base(nameOrConnectionString)
    {
        Database.Initialize(false);
    }

    public AccountsContext(DbConnection connection, bool contextOwnsConnection) : 
        base(connection, contextOwnsConnection)
    {
        Database.Initialize(false);
    }

    protected override void OnModelCreating(DbModelBuilder modelBuilder)
    {
        base.OnModelCreating(modelBuilder);

        modelBuilder.Conventions.Remove<PluralizingTableNameConvention>();

        // user
        modelBuilder.Entity<User>().Property(u => u.Id)
            .HasDatabaseGeneratedOption(DatabaseGeneratedOption.Identity);

        // role
        modelBuilder.Entity<Role>().Property(r => r.Id)
            .HasDatabaseGeneratedOption(DatabaseGeneratedOption.Identity);
        modelBuilder.Entity<Role>().ToTable("webpages_Roles");
        modelBuilder.Entity<Role>().Property(r => r.Id).HasColumnName("RoleId");
        modelBuilder.Entity<Role>().Property(r => r.Name).HasColumnName("RoleName");

        // user-role
        modelBuilder.Entity<User>()
            .HasMany(u => u.Roles)
            .WithMany(r => r.Users)
            .Map(m =>
            {
                m.MapLeftKey("UserId");
                m.MapRightKey("RoleId");
                m.ToTable("webpages_UsersInRoles");
            });
    }
}

LocalSqlServerWeb アプリを実行すると、接続名が見つからないという例外がすぐに発生します。これは、次のエントリを見つけることができる machine.config に属します。

...
<membership>
<providers>
  <add name="AspNetSqlMembershipProvider" type="System.Web.Security.SqlMembershipProvider, System.Web, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a" connectionStringName="LocalSqlServer" .../>
  <add name="MySQLMembershipProvider" type="MySql.Web.Security.MySQLMembershipProvider, MySql.Web, Version=6.5.4.0, Culture=neutral, PublicKeyToken=c5687fc88969c44d" connectionStringName="LocalMySqlServer" ... autogenerateschema="true"/>
</providers>
...

したがって、これらの行を my に追加して、これらのエントリをオーバーライドしようとしましたweb.config:

<roleManager enabled="true" defaultProvider="SimpleRoleProvider">
  <providers>
    <clear/>
    <add name="SimpleRoleProvider" type="WebMatrix.WebData.SimpleRoleProvider, WebMatrix.WebData"/>
  </providers>
</roleManager>
<membership defaultProvider="SimpleRoleProvider">
  <providers>
    <clear/>
    <add name="SimpleMembershipProvider" type="WebMatrix.WebData.SimpleMembershipProvider, WebMatrix.WebData"/>        
  </providers>
</membership>

今アプリを実行すると、次のような例外が発生します。

System.Configuration.ConfigurationErrorsException
Message=Default Membership Provider could not be found.
Source=System.Web
BareMessage=Default Membership Provider could not be found.

ただし、Web アプリにはWebMatrix.DataWebMatrix.WebData(両方のバージョン 2) への参照があり、VS テンプレートによって既に設定されています。では、どうすればこれを機能させることができますか?

4

2 に答える 2

2

mssqlを使用した現在のmvc 4プロジェクトでは、単純なものなので、非常に単純なメンバーシッププロバイダーが必要で、InitializeSimpleMembershipAttributeを無効にしました

[Authorize]
//[InitializeSimpleMembership]
public partial class AccountController : Controller

このコードを Application_Start の下の global.asax に追加したので、もう SimpleMembershipInitializer は必要ありません

WebSecurity.InitializeDatabaseConnection(
             connectionStringName: "DefaultConnection",
             userTableName: "UserProfile",
             userIdColumn: "UserID",
             userNameColumn: "UserName",
             autoCreateTables: true);

私のSQLデータベースでは、アプリケーションはそれらのいくつかのテーブルを作成しましたロールであり、UserInRolesは管理者、顧客などの必要なロールを追加しました...データベースで同じことを行うか、ロールとメンバーシップを管理するためのインターフェイスを構築できます. このコードを追加して、一部のコントローラーまたはアクションへのアクセスを制限します

[Authorize(Roles = "Admin")]
public class MessagesController : Controller
于 2015-01-28T22:18:45.090 に答える
1

これは明らかだとは言えませんが、機能します。

SimpleMembeship を SQL Compact と連携させるには、最初に Nuget に "EntityFramework.SqlServerCompact" と "Microsoft ASP.NET Universal Providers Core Library" を追加する必要があります。

接続文字列の名前は「AccountsContext」です。モデルにも同じ名前を付ける必要があります。

   public class UsersContext : DbContext
    {
        public UsersContext()
            : base("AccountsContext")
        {
        }

        public DbSet<UserProfile> UserProfiles { get; set; }
        public DbSet<webpages_Membership> Membership { get; set; }
    }

次に、新しい SimpleMembership ではなく、ASP セキュリティのためであるため、セクションに SQL に関連するものを含めないでください。これが私の見た目です:

<membership defaultProvider="simple">
      <providers>
        <clear />
        <add name="simple" type="WebMatrix.WebData.SimpleMembershipProvider, WebMatrix.WebData" />
      </providers>
    </membership>
于 2013-09-30T16:14:25.327 に答える