Не удается сопоставить модель с парой отношений один-к-одному с Entity Framework Core - PullRequest
1 голос
/ 23 марта 2019

Я разрабатываю свой первый API с ASP.NET Core и использую Entity Framework Core для моделирования моих данных.Я не могу отобразить 2 вложенных объекта класса в родительском.

Я прочитал один к одному и один ко многим, но ничего не получаю, когда ищу несколько сопоставлений один к одному, за исключением некоторых постов.это мне не помогло.

Я видел это:

Таблица Entity Framework с несколькими необязательными отношениями один к одному

https://github.com/aspnet/EntityFrameworkCore/issues/5344

Это моя модель

Класс Contact владеет 2 адресами свойств класса и контактным телефоном.Затем в моем dbcontext я указываю отношения один к одному.Я использую:

Инструменты Entity Framework Core .NET Командная строка 2.2.3-servicing-35854

Сервер SQL 13.0.40001

public class Contact
{
    public long Id { get; set; }
    [MaxLength(40)]
    [Required(ErrorMessage = "Contact name is required ")]
    public string Name { get; set; }
    [Required(ErrorMessage = "Company name is required ")]
    public string Company { get; set; }
    public string Pro`enter code here`fileImageURL { get; set; }
    [Required(ErrorMessage = "Contact Emil is required ")]
    public string Email { get; set; }
    [Required(ErrorMessage = "Birthday is required ")]
    public DateTime BirthDate { get; set; }
    [Required(ErrorMessage = "Contact phone is required ")]
    public ContactPhone ContactPhone { get; set; }
    public Address Address { get; set; }
}

public class Address
{
    public long id { get; set; }
    public string AddressLine1 { get; set; }
    public string AddressLine2 { get; set; }
    public string City { get; set; }
    public string State { get; set; }
    public Contact Contact { get; set; }
    public long AddresssContactForeignKey { get; set; }
}

public class ContactPhone
{
    public long ContactPhoneid { get; set; }
    public string PersonalPhone { get; set; }
    public string WorkPhone { get; set; }
    public Contact Contact { get; set; }
    public long ContactPhonesContactForeignKey { get; set; }
}

Моя БДконтекст:

public class ContactDbContext : DbContext
{
    public DbSet<Contact> Contacts { get; set; }
    public DbSet<ContactPhone> ContactPhones { get; set; }
    public DbSet<Address> ContactAddress { get; set; }

    public ContactDbContext(DbContextOptions<ContactDbContext> options)
        : base(options)
    {
    }

    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        modelBuilder.Entity<Contact>(e =>
        {
            e.HasOne(x => x.ContactPhone)
                .WithOne(y => y.Contact)
                .HasForeignKey<ContactPhone>(z => 
                z.ContactPhonesContactForeignKey);

            e.HasOne(x => x.Address)
                .WithOne(y => y.Contact)
                .HasForeignKey<Address>(z => 
                z.AddresssContactForeignKey);
        });
    }
}

Затем я применяю миграцию:

PM> Add-Migration Migration2

Microsoft.EntityFrameworkCore.Infrastructure [10403] Entity Framework Core 2.2.3-servicing-35854 инициализировал «ContactDbContext» с использованием провайдера «Microsoft.EntityFrameworkCore.SqlServer» с параметрами: Нет Операция была помечена, что может привести к потере данных.Пожалуйста, проверьте точность миграции.Чтобы отменить это действие, используйте Remove-Migration.

Кажется, что нет ошибок, затем я обновляю базу данных

PM> update-database Microsoft.EntityFrameworkCore.Infrastructure[10403]
  Entity Framework Core 2.2.3-servicing-35854 initialized 'ContactDbContext' using provider 'Microsoft.EntityFrameworkCore.SqlServer' with options: None Microsoft.EntityFrameworkCore.Database.Command[20101]
  Executed DbCommand (29ms) [Parameters=[], CommandType='Text', CommandTimeout='30']
  SELECT OBJECT_ID(N'[__EFMigrationsHistory]'); Microsoft.EntityFrameworkCore.Database.Command[20101]
  Executed DbCommand (1ms) [Parameters=[], CommandType='Text', CommandTimeout='30']
  SELECT OBJECT_ID(N'[__EFMigrationsHistory]'); Microsoft.EntityFrameworkCore.Database.Command[20101]
  Executed DbCommand (28ms) [Parameters=[], CommandType='Text', CommandTimeout='30']
  SELECT [MigrationId], [ProductVersion]
  FROM [__EFMigrationsHistory]
  ORDER BY [MigrationId]; Microsoft.EntityFrameworkCore.Migrations[20402]
  Applying migration '20190323155442_Migration2'. Applying migration '20190323155442_Migration2'. Microsoft.EntityFrameworkCore.Database.Command[20101]
  Executed DbCommand (42ms) [Parameters=[], CommandType='Text', CommandTimeout='30']
  ALTER TABLE [ContactPhones] DROP CONSTRAINT [FK_ContactPhones_Contacts_ContactForeignKey]; Microsoft.EntityFrameworkCore.Database.Command[20101]
  Executed DbCommand (40ms) [Parameters=[], CommandType='Text', CommandTimeout='30']
  DROP INDEX [IX_ContactPhones_ContactForeignKey] ON [ContactPhones]; Microsoft.EntityFrameworkCore.Database.Command[20101]
  Executed DbCommand (385ms) [Parameters=[], CommandType='Text', CommandTimeout='30']
  DECLARE @var0 sysname;
  SELECT @var0 = [d].[name]
  FROM [sys].[default_constraints] [d]
  INNER JOIN [sys].[columns] [c] ON [d].[parent_column_id] = [c].[column_id] AND [d].[parent_object_id] = [c].[object_id]
  WHERE ([d].[parent_object_id] = OBJECT_ID(N'[Contacts]') AND [c].[name] = N'Address');
  IF @var0 IS NOT NULL EXEC(N'ALTER TABLE [Contacts] DROP CONSTRAINT [' + @var0 + '];');
  ALTER TABLE [Contacts] DROP COLUMN [Address]; Microsoft.EntityFrameworkCore.Database.Command[20101]
  Executed DbCommand (17ms) [Parameters=[], CommandType='Text', CommandTimeout='30']
  DECLARE @var1 sysname;
  SELECT @var1 = [d].[name]
  FROM [sys].[default_constraints] [d]
  INNER JOIN [sys].[columns] [c] ON [d].[parent_column_id] = [c].[column_id] AND [d].[parent_object_id] = [c].[object_id]
  WHERE ([d].[parent_object_id] = OBJECT_ID(N'[ContactPhones]') AND [c].[name] = N'ContactForeignKey');
  IF @var1 IS NOT NULL EXEC(N'ALTER TABLE [ContactPhones] DROP CONSTRAINT [' + @var1 + '];');
  ALTER TABLE [ContactPhones] DROP COLUMN [ContactForeignKey]; fail: Microsoft.EntityFrameworkCore.Database.Command[20102]
  Failed executing DbCommand (51ms) [Parameters=[], CommandType='Text', CommandTimeout='30']
  DECLARE @var2 sysname;
  SELECT @var2 = [d].[name]
  FROM [sys].[default_constraints] [d]
  INNER JOIN [sys].[columns] [c] ON [d].[parent_column_id] = [c].[column_id] AND [d].[parent_object_id] = [c].[object_id]
  WHERE ([d].[parent_object_id] = OBJECT_ID(N'[Contacts]') AND [c].[name] = N'Id');
  IF @var2 IS NOT NULL EXEC(N'ALTER TABLE [Contacts] DROP CONSTRAINT [' + @var2 + '];');
  ALTER TABLE [Contacts] ALTER COLUMN [Id] bigint NOT NULL; System.Data.SqlClient.SqlException (0x80131904): The object 'PK_Contacts' is dependent on column 'Id'. ALTER TABLE ALTER COLUMN Id failed because one or more objects access this column.    at System.Data.SqlClient.SqlConnection.OnError(SqlException exception, Boolean breakConnection, Action`1 wrapCloseInAction)    at System.Data.SqlClient.SqlInternalConnection.OnError(SqlException exception, Boolean breakConnection, Action`1 wrapCloseInAction)    at System.Data.SqlClient.TdsParser.ThrowExceptionAndWarning(TdsParserStateObject stateObj, Boolean callerHasConnectionLock, Boolean asyncClose)    at System.Data.SqlClient.TdsParser.TryRun(RunBehavior runBehavior, SqlCommand cmdHandler, SqlDataReader dataStream, BulkCopySimpleResultSet bulkCopyHandler, TdsParserStateObject stateObj, Boolean& dataReady)    at System.Data.SqlClient.SqlCommand.RunExecuteNonQueryTds(String methodName, Boolean async, Int32 timeout, Boolean asyncWrite)    at System.Data.SqlClient.SqlCommand.InternalExecuteNonQuery(TaskCompletionSource`1 completion, Boolean sendToPipe, Int32 timeout, Boolean asyncWrite, String methodName)    at System.Data.SqlClient.SqlCommand.ExecuteNonQuery()    at Microsoft.EntityFrameworkCore.Storage.Internal.RelationalCommand.Execute(IRelationalConnection connection, DbCommandMethod executeMethod, IReadOnlyDictionary`2 parameterValues) ClientConnectionId:7bf3ba67-b8dc-4d76-a11b-c9b5ef584fad Error Number:5074,State:1,Class:16 Failed executing DbCommand (51ms) [Parameters=[], CommandType='Text', CommandTimeout='30'] DECLARE @var2 sysname; SELECT @var2 = [d].[name] FROM [sys].[default_constraints] [d] INNER JOIN [sys].[columns] [c] ON [d].[parent_column_id] = [c].[column_id] AND [d].[parent_object_id] = [c].[object_id] WHERE ([d].[parent_object_id] = OBJECT_ID(N'[Contacts]') AND [c].[name] = N'Id'); IF @var2 IS NOT NULL EXEC(N'ALTER TABLE [Contacts] DROP CONSTRAINT [' + @var2 + '];'); ALTER TABLE [Contacts] ALTER COLUMN [Id] bigint NOT NULL; System.Data.SqlClient.SqlException (0x80131904): The object 'PK_Contacts' is dependent on column 'Id'. ALTER TABLE ALTER COLUMN Id failed because one or more objects access this column. at System.Data.SqlClient.SqlConnection.OnError(SqlException exception, Boolean breakConnection, Action`1 wrapCloseInAction)    at System.Data.SqlClient.SqlInternalConnection.OnError(SqlException exception, Boolean breakConnection, Action`1 wrapCloseInAction)    at System.Data.SqlClient.TdsParser.ThrowExceptionAndWarning(TdsParserStateObject stateObj, Boolean callerHasConnectionLock, Boolean asyncClose)    at System.Data.SqlClient.TdsParser.TryRun(RunBehavior runBehavior, SqlCommand cmdHandler, SqlDataReader dataStream, BulkCopySimpleResultSet bulkCopyHandler, TdsParserStateObject stateObj, Boolean& dataReady)    at System.Data.SqlClient.SqlCommand.RunExecuteNonQueryTds(String methodName, Boolean async, Int32 timeout, Boolean asyncWrite)    at System.Data.SqlClient.SqlCommand.InternalExecuteNonQuery(TaskCompletionSource`1 completion, Boolean sendToPipe, Int32 timeout, Boolean asyncWrite, String methodName)    at System.Data.SqlClient.SqlCommand.ExecuteNonQuery()    at Microsoft.EntityFrameworkCore.Storage.Internal.RelationalCommand.Execute(IRelationalConnection connection, DbCommandMethod executeMethod, IReadOnlyDictionary`2 parameterValues)    at Microsoft.EntityFrameworkCore.Storage.Internal.RelationalCommand.ExecuteNonQuery(IRelationalConnection connection, IReadOnlyDictionary`2 parameterValues)    at Microsoft.EntityFrameworkCore.Migrations.MigrationCommand.ExecuteNonQuery(IRelationalConnection connection, IReadOnlyDictionary`2 parameterValues)    at Microsoft.EntityFrameworkCore.Migrations.Internal.MigrationCommandExecutor.ExecuteNonQuery(IEnumerable`1 migrationCommands, IRelationalConnection connection)    at Microsoft.EntityFrameworkCore.Migrations.Internal.Migrator.Migrate(String targetMigration)    at Microsoft.EntityFrameworkCore.Design.Internal.MigrationsOperations.UpdateDatabase(String targetMigration, String contextType)    at Microsoft.EntityFrameworkCore.Design.OperationExecutor.UpdateDatabase.<>c__DisplayClass0_1.<.ctor>b__0() at Microsoft.EntityFrameworkCore.Design.OperationExecutor.OperationBase.Execute(Action action) ClientConnectionId:7bf3ba67-b8dc-4d76-a11b-c9b5ef584fad Error Number:5074,State:1,Class:16 The object 'PK_Contacts' is dependent on column 'Id'. ALTER TABLE ALTER COLUMN Id failed because one or more objects access this column.

И это в результате Миграция

public partial class Migration2 : Migration
 {
     protected override void Up(MigrationBuilder migrationBuilder)
     {
         migrationBuilder.DropForeignKey(
             name: "FK_ContactPhones_Contacts_ContactForeignKey",
             table: "ContactPhones");

         migrationBuilder.DropIndex(
             name: "IX_ContactPhones_ContactForeignKey",
             table: "ContactPhones");

         migrationBuilder.DropColumn(
             name: "Address",
             table: "Contacts");

         migrationBuilder.DropColumn(
             name: "ContactForeignKey",
             table: "ContactPhones");

         migrationBuilder.AlterColumn<long>(
             name: "Id",
             table: "Contacts",
             nullable: false,
             oldClrType: typeof(int))
             .Annotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn)
             .OldAnnotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn);

         migrationBuilder.AlterColumn<long>(
             name: "ContactPhoneid",
             table: "ContactPhones",
             nullable: false,
             oldClrType: typeof(int))
             .Annotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn)
             .OldAnnotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn);

         migrationBuilder.AddColumn<long>(
             name: "ContactPhonesContactForeignKey",
             table: "ContactPhones",
             nullable: false,
             defaultValue: 0L);

         migrationBuilder.CreateTable(
             name: "ContactAddress",
             columns: table => new
             {
                 id = table.Column<long>(nullable: false)
                     .Annotation("SqlServer:ValueGenerationStrategy",
SqlServerValueGenerationStrategy.IdentityColumn),
                 AddressLine1 = table.Column<string>(nullable: true),
                 AddressLine2 = table.Column<string>(nullable: true),
                 City = table.Column<string>(nullable: true),
                 State = table.Column<string>(nullable: true),
                 AddresssContactForeignKey = table.Column<long>(nullable: false)
             },
             constraints: table =>
             {
                 table.PrimaryKey("PK_ContactAddress", x => x.id);
                 table.ForeignKey(
                     name: "FK_ContactAddress_Contacts_AddresssContactForeignKey",
                     column: x => x.AddresssContactForeignKey,
                     principalTable: "Contacts",
                     principalColumn: "Id",
                     onDelete: ReferentialAction.Cascade);
             });

         migrationBuilder.CreateIndex(
             name: "IX_ContactPhones_ContactPhonesContactForeignKey",
             table: "ContactPhones",
             column: "ContactPhonesContactForeignKey",
             unique: true);

         migrationBuilder.CreateIndex(
             name: "IX_ContactAddress_AddresssContactForeignKey",
             table: "ContactAddress",
             column: "AddresssContactForeignKey",
             unique: true);

         migrationBuilder.AddForeignKey(
             name: "FK_ContactPhones_Contacts_ContactPhonesContactForeignKey",
             table: "ContactPhones",
             column: "ContactPhonesContactForeignKey",
             principalTable: "Contacts",
             principalColumn: "Id",
             onDelete: ReferentialAction.Cascade);
     }

     protected override void Down(MigrationBuilder migrationBuilder)
     {
         migrationBuilder.DropForeignKey(
             name: "FK_ContactPhones_Contacts_ContactPhonesContactForeignKey",
             table: "ContactPhones");

         migrationBuilder.DropTable(
             name: "ContactAddress");

         migrationBuilder.DropIndex(
             name: "IX_ContactPhones_ContactPhonesContactForeignKey",
             table: "ContactPhones");

         migrationBuilder.DropColumn(
             name: "ContactPhonesContactForeignKey",
             table: "ContactPhones");

         migrationBuilder.AlterColumn<int>(
             name: "Id",
             table: "Contacts",
             nullable: false,
             oldClrType: typeof(long))
             .Annotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn)
             .OldAnnotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn);

         migrationBuilder.AddColumn<string>(
             name: "Address",
             table: "Contacts",
             nullable: false,
             defaultValue: "");

         migrationBuilder.AlterColumn<int>(
             name: "ContactPhoneid",
             table: "ContactPhones",
             nullable: false,
             oldClrType: typeof(long))
             .Annotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn)
             .OldAnnotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn);

         migrationBuilder.AddColumn<int>(
             name: "ContactForeignKey",
             table: "ContactPhones",
             nullable: false,
             defaultValue: 0);

         migrationBuilder.CreateIndex(
             name: "IX_ContactPhones_ContactForeignKey",
             table: "ContactPhones",
             column: "ContactForeignKey",
             unique: true);

         migrationBuilder.AddForeignKey(
             name: "FK_ContactPhones_Contacts_ContactForeignKey",
             table: "ContactPhones",
             column: "ContactForeignKey",
             principalTable: "Contacts",
             principalColumn: "Id",
             onDelete: ReferentialAction.Cascade);
     }
 }

Наконец, консоль выдает следующее:

Номер ошибки: 5074, Состояние: 1, Класс: 16
Объект 'PK_Contacts' зависит от столбца 'Id'.
ALTER TABLE ALTER COLUMN Id не выполнен, поскольку один или несколько объектов обращаются к этому столбцу.

Я ожидаю, что сопоставление будет 1: 1

UPDATE

Я нашел проблему.Раньше у всех моих сущностей были идентификаторы beint int , затем я изменил их на long Я запустил новый проект с написанным кодом и запустил новую базу данных с первой миграцией, и она работалаСпасибо всем, кто был так рад помочь мне в этом !!

1 Ответ

0 голосов
/ 23 марта 2019

ОБНОВЛЕНИЕ

Я нашел проблему.Раньше у меня были все идентификаторы сущностей beint int , затем я изменил их на long Я запустил новый проект с написанным кодом и запустил новую базу данных с первой миграцией, и она работалаСпасибо всем, кто был так рад помочь мне в этом !!

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...