おそらく明らかな何かが欠けていますが、SDF ベースの ASP.NET MVC 4 Web アプリを新しい単純なメンバーシップで動作させることができません。他の初心者の参考になるように、私の手順を詳しく説明します。
まず、新しいメンバーシップ システムの紹介が非常に便利であることがわかりました。 new-asp-net-4-5-web-forms-and-asp-net-mvc-4-templates.aspx . 既存のデータベース (SDF は本格的な既存の SQL Server db の一時的なプレースホルダーです) を使用したコード ファーストでの私の手順は次のとおりです。
- VS 2012 で新しいインターネット アプリを作成しました。
- 新しい SDF ファイルを App_Data (
Accounts.sdf
) に追加し、そこにユーザーとロールのテーブルを作成しました。 - に新しい接続文字列を追加しました
web.config
:
<connectionStrings>
<clear/>
<add name="AccountsContext" connectionString="Data Source=|DataDirectory|\Accounts.sdf;Persist Security Info=False" providerName="System.Data.SqlServerCe.4.0" />
</connectionStrings>
- 中間データ層でホストされて
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");
});
}
}
LocalSqlServer
Web アプリを実行すると、接続名が見つからないという例外がすぐに発生します。これは、次のエントリを見つけることができる 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.Data
とWebMatrix.WebData
(両方のバージョン 2) への参照があり、VS テンプレートによって既に設定されています。では、どうすればこれを機能させることができますか?