EF Core, недопустимое исключение имени столбца после обновления с netcore2.2 до netcore3.1 - PullRequest
1 голос
/ 03 февраля 2020

Поскольку я обновил с .netcore2.2 до .netcore3.1, я столкнулся с новым странным исключением "Неверное имя столбца 'TenantTemplateTypeID 1 '". Пожалуйста, не «1», это не должно быть здесь. В коде НЕТ TenantTemplateTypeID1, поэтому я предполагаю, что он сгенерирован ядром EF.

Версия ядра EF: 3.1.1

Исключение:

Exception: Invalid column name 'TenantTemplateTypeID1'.
Invalid column name 'TenantTemplateTypeID1'.
Invalid column name 'TenantTemplateTypeID1'.
Invalid column name 'TenantTemplateTypeID1'.
StactTrace:    at Microsoft.Data.SqlClient.SqlConnection.OnError(SqlException exception, Boolean breakConnection, Action`1 wrapCloseInAction)
   at Microsoft.Data.SqlClient.SqlInternalConnection.OnError(SqlException exception, Boolean breakConnection, Action`1 wrapCloseInAction)
   at Microsoft.Data.SqlClient.TdsParser.ThrowExceptionAndWarning(TdsParserStateObject stateObj, Boolean callerHasConnectionLock, Boolean asyncClose)
   at Microsoft.Data.SqlClient.TdsParser.TryRun(RunBehavior runBehavior, SqlCommand cmdHandler, SqlDataReader dataStream, BulkCopySimpleResultSet bulkCopyHandler, TdsParserStateObject stateObj, Boolean& dataReady)
   at Microsoft.Data.SqlClient.SqlDataReader.TryConsumeMetaData()
   at Microsoft.Data.SqlClient.SqlDataReader.get_MetaData()

Запрос:

var t = engineCtx.TenantTemplates
                 .AsNoTracking()
                 .Include(tt => tt.TenantTemplateParameters)
                 .Include(tt => tt.TenantTemplateStyles)
                 .FirstOrDefault(tt => tt.TenantTemplateTypeID == tenantTemplateTypeID);

DbContext

modelBuilder.Entity<Db.Model.TenantTemplate>(entity =>
            {
                entity.HasKey(e => e.TenantTemplateTypeID)
                      .HasName("PK_TenantTemplate_TenantTemplateTypeID");

                entity.ToTable("TenantTemplates", "templateEngine");

                entity.Property(e => e.TenantTemplateTypeID)
                      .HasColumnName("TenantTemplateTypeId");
                //... not related rows removed
                entity.HasMany(e => e.TenantTemplateParameters)
                      .WithOne(e => e.TenantTemplate)
                      .HasConstraintName("FK_TenantTemplate_TenantTemplateParameters")
                      .OnDelete(DeleteBehavior.Restrict);

                entity.HasMany(e => e.TenantTemplateStyles)
                      .WithOne(e => e.TenantTemplate)
                      .HasConstraintName("FK_TenantTemplate_TenantTemplateStyles")
                      .OnDelete(DeleteBehavior.Restrict);
            });

Модель БД содержит только свойства без каких-либо атрибутов.

public partial class TenantTemplate
    {
         public long TenantTemplateTypeID { get; set; }
         // not related rows removed
         public virtual ICollection<TenantTemplateParameter> TenantTemplateParameters { get; set; }
         public virtual ICollection<TenantTemplateStyle> TenantTemplateStyles { get; set; }
    }
// TenantTemplateStyle is exacly the same (just PK has a different name)
public class TenantTemplateParameter
   {
        public long TenantTemplateParameterID { get; set; }
        public long TenantTemplateTypeID { get; set; }
        // rows removed
        public virtual TenantTemplate TenantTemplate { get; set; }
   }

Кто-нибудь знает что-то сверх? Спасибо ...

1 Ответ

2 голосов
/ 04 февраля 2020

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

В соответствии с этим правило, TenantTemplateTypeID в TenantTemplateParameter не считается именем FK.

Странно то, что в то же время это стандартное имя FK по умолчанию в соответствии с правилами, описанными в Тень внешнего ключа properties :

Свойству будет присвоено имя <navigation property name><principal key property name> (для именования используется навигация по зависимому объекту, который указывает на главный объект). Если имя свойства основного ключа включает имя свойства навигации, тогда оно будет просто <principal key property name>. Если у зависимого объекта нет свойства навигации, то имя основного типа используется в его имени. place.

Так как имя уже зарезервировано свойством "не FK", построитель условных имен добавляет индекс к условному имени по умолчанию, отсюда и наблюдаемое поведение.

Я хотел бы рассмотреть это как ошибка / дефект текущей реализации EF Core. Обходное решение / решение, конечно, заключается в явном сопоставлении свойства FK, например,

entity.HasMany(e => e.TenantTemplateParameters)
      .WithOne(e => e.TenantTemplate)
      .HasForeignKey(e => e.TenantTemplateTypeID) // <--
      .HasConstraintName("FK_TenantTemplate_TenantTemplateParameters")
      .OnDelete(DeleteBehavior.Restrict);

и аналогичных для других отношений (с TenantTemplateStyle), если используется то же соглашение об именах.

...