Moja aplikacja na Daj Się Poznać powoli się rozwija. Wraz z tym ciągłym rozwojem musiałem dokonać modyfikacji struktury bazy danych. Do tabeli zadań konieczne było dodanie kolumny z ID użytkownika, któremu zlecamy zadanie (będzie kluczem obcym tabeli użytkowników). Istnieje już podobne powiązanie tych samych tabel – dla użytkownika tworzącego zadanie. Dodam jeszcze, że korzystam z możliwości code-first w Entity Framework.
Zmodyfikowałem więc szybko model „Zadania”. Początkowo miałem tylko jedno powiązanie:
1 2 3 4 5 | [MaxLength(128)] [ForeignKey("UserID")] public string TaskOwnerId { get; set; } public virtual ApplicationUser UserID { get; set; } |
Dostawiłem drugie:
1 2 3 4 5 6 7 8 9 | [MaxLength(128)] [ForeignKey("UserID")] public string TaskOwnerId { get; set; } [MaxLength(128)] [ForeignKey("UserID")] public string TaskExecutorId { get; set; } public virtual ApplicationUser UserID { get; set; } |
Przystąpiłem to przygotowania migracji za pomocą konsoli NuGet:
1 | PM> Add-Migration Task |
Pudło! Otrzymałem taki komunikat:
Unable to determine a composite foreign key ordering for foreign key on type todo.team.Domain.Entities.Task. When using the ForeignKey data annotation on composite foreign key properties ensure order is specified by using the Column data annotation or the fluent API.
Szybki research i gotowe. Po drobnej modyfikacji mojej encji migracja przeszła poprawnie…
1 2 3 4 5 6 7 8 9 10 | [MaxLength(128)] [ForeignKey("OwnerUserID")] public string TaskOwnerId { get; set; } [MaxLength(128)] [ForeignKey("ExecutorUserID")] public string TaskExecutorId { get; set; } public virtual ApplicationUser OwnerUserID { get; set; } public virtual ApplicationUser ExecutorUserID { get; set; } |
…, a baza została zaktualizowana za pomocą automatycznie wygenerowanego kodu.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 | CreateTable( "dbo.Tasks", c => new { TaskOwnerId = c.String(maxLength: 128), TaskExecutorId = c.String(maxLength: 128), TaskId = c.Int(nullable: false, identity: true), TaskSubject = c.String(nullable: false, maxLength: 128), TaskBody = c.String(), TaskCreateDate = c.DateTime(nullable: false), TaskStartDate = c.DateTime(), TaskEndDate = c.DateTime(), TaskScheduleTypeId = c.Int(), TaskStatusId = c.Int(), }) .PrimaryKey(t => t.TaskId) .ForeignKey("dbo.AspNetUsers", t => t.TaskExecutorId) .ForeignKey("dbo.AspNetUsers", t => t.TaskOwnerId) .Index(t => t.TaskOwnerId) .Index(t => t.TaskExecutorId); |
Nie za bardzo rozumiałem wcześniej tworzenie relacji między tabelami w podejściu code-first, stąd ta pomyłka. Wklejam rozwiązanie, gdyby ktoś miał kiedyś podobny przypadek. Więcej o tworzeniu bazy na podstawie modelu aplikacji w nadchodzącym wpisie na blogu.
Stay tuned!
Tagi: asp.net • code-first • Daj Się Poznać 2017 • entity framework
Czy moższesz podać kod kontrolera i widoku?