三者関係を維持するシステムを構築しようとしています。しかし、関係の 2 番目のインスタンスに入ると、例外が発生します。
ここに基本的なコードがあります
public class Role
{
protected Role() { }
public Role(string id, string description) : this()
{
Id = id;
Description = description;
}
public string Id { get; set; }
public string Description { get; set; }
}
public abstract class Entity
{
public Guid EntityId { get; private set; }
}
public class Project : Entity
{
protected Project()
{
UserProjectRoles = new Collection<UserProjectRole>();
}
public Project(string id, string description, string connectionString)
{
Id = id;
Description = description;
ConnectionString = connectionString;
}
public string Id { get; private set; }
public string Description { get; private set; }
public string ConnectionString { get; private set; }
public virtual ICollection<UserProjectRole> UserProjectRoles { get; private set; }
public IEnumerable<User> Users { get { return UserProjectRoles.Select(x => x.User).Distinct(); } }
}
public class User : Entity
{
protected User()
{
UserProjectRoles = new Collection<UserProjectRole>();
}
public User(string uniqueName)
: this()
{
UniqueName = uniqueName;
}
public string UniqueName { get; set; }
public virtual ICollection<UserProjectRole> UserProjectRoles { get; private set; }
public IEnumerable<Project> Projects { get { return UserProjectRoles.Select(x => x.Project).Distinct(); } }
}
public class UserProjectRole
{
protected UserProjectRole() { }
public UserProjectRole(Project project, User user, Role role)
{
Project = project;
User = user;
Role = role;
}
public Guid ProjectId { get; private set; }
public Guid UserId { get; private set; }
public string RoleId { get; private set; }
public Project Project { get; private set; }
public User User { get; private set; }
public Role Role { get; private set; }
}
public class MasterContext : DbContext
{
public MasterContext(string connectionString) : base(connectionString) { }
public IDbSet<User> Users { get; set; }
protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
modelBuilder.Entity<Entity>().HasKey(e => e.EntityId);
modelBuilder.Entity<Entity>().Property(e => e.EntityId).HasDatabaseGeneratedOption(DatabaseGeneratedOption.Identity);
modelBuilder.Entity<Project>().Map(u => u.MapInheritedProperties()).ToTable("Projects");
modelBuilder.Entity<User>().Map(p => p.MapInheritedProperties()).ToTable("Users");
modelBuilder.Entity<Role>().HasKey(r => r.Id);
modelBuilder.Entity<UserProjectRole>().HasKey(x => new { x.ProjectId, x.UserId, x.RoleId });
modelBuilder.Entity<UserProjectRole>().HasRequired(x => x.User).WithMany(u => u.UserProjectRoles).HasForeignKey(x => x.UserId).WillCascadeOnDelete(false);
modelBuilder.Entity<UserProjectRole>().HasRequired(x => x.Project).WithMany(p => p.UserProjectRoles).HasForeignKey(x => x.ProjectId).WillCascadeOnDelete(false);
modelBuilder.Entity<UserProjectRole>().HasRequired(x => x.Role).WithMany().HasForeignKey(x => x.RoleId).WillCascadeOnDelete(false);
}
}
エラーを示すテスト ケースを次に示します。
[TestFixture]
public class EFTest
{
private const string MasterConnectionString =
"Data Source=localhost;Initial Catalog=TheDatabase;Integrated Security=True;";
private static readonly Role DataCollector = new Role("1", "Data Collector");
private static readonly Role Manager = new Role("2", "Manager");
[Test]
public void TheTest()
{
var user = new User("Tim the Enchanter");
var project = new Project("Foo", "Bar", "Another Connection string");
Database.SetInitializer(new DropCreateDatabaseAlways<MasterContext>());
using (var masterContext = new MasterContext(MasterConnectionString))
{
user.UserProjectRoles.Add(new UserProjectRole(project, user, DataCollector));
masterContext.Users.Add(user);
masterContext.SaveChanges();
}
Database.SetInitializer(new CreateDatabaseIfNotExists<MasterContext>());
using (var masterContext = new MasterContext(MasterConnectionString))
{
var u = masterContext.Users.First();
var p = u.Projects.First();
u.UserProjectRoles.Add(new UserProjectRole(p, u, Manager));
masterContext.SaveChanges();
}
}
}
そのテストを実行すると、次のエラーが表示されます。
System.Data.Entity.Infrastructure.DbUpdateException : エントリの更新中にエラーが発生しました。詳細については、内部例外を参照してください。----> System.Data.UpdateException : エントリの更新中にエラーが発生しました。詳細については、内部例外を参照してください。----> System.Data.SqlClient.SqlException : INSERT ステートメントが FOREIGN KEY 制約 "FK_dbo.UserProjectRoles_dbo.Projects_ProjectId" と競合しました。データベース「Dev_Local_Master」、テーブル「dbo.Projects」、列「EntityId」で競合が発生しました。ステートメントは終了されました。