【问题标题】:EF Core - String or binary data would be truncated [duplicate]EF Core - 字符串或二进制数据将被截断[重复]
【发布时间】:2022-03-04 03:20:31
【问题描述】:

当您有 80(+/-) 列可供选择时,您如何确定哪一列是罪魁祸首?使用 .Net Core (netcoreapp2.2) 和 EF Core 2.2.4。

提取了一些现有代码,并尝试跟踪失败的列。但是,它不起作用。我在这里和其他地方查看了几十个示例,但没有找到在 EF Core 2.x 中执行此操作的方法。

public int GetColumnMaxLength(string table, EntityEntry entityEntry)
{
    // Just a rough to get the right data - always returns 0 for the moment...
    int result = 0;
    var modelContext = entityEntry.Context;
    var entityType = modelContext.Model.FindEntityType(table); // THIS IS ALWAYS NULL!

    if (entityType != null)
    {
        // Table info 
        var tableName = entityType.Relational().TableName;
        var tableSchema = entityType.Relational().Schema;

        // Column info 
        foreach (var property in entityType.GetProperties())
        {
            var columnName = property.Relational().ColumnName;
            var columnType = property.Relational().ColumnType;
            var isFixedLength = property.Relational().IsFixedLength;
        };
    }
    return result;
}

上面的代码被 db.SaveAsync(); 周围的 try/catch 的这个 catch 部分调用;声明。

catch (Exception ex)
{
    // -----------------------------------------
    // no idea what this was really trying to 
    // do as it barfs out all columns...
    // -----------------------------------------

    var dataInfo = new DataInfo();

    var strLargeValues = new List<Tuple<int, string, string, string>>();

    foreach (var entityEntry in _db.ChangeTracker.Entries().Where(et => et.State != EntityState.Unchanged))
    {
        // -----------------------------------------
        // try to get the column info for all 
        // columns on this table...
        // -----------------------------------------
        dataInfo.GetColumnMaxLength("Subscription", entityEntry);

        foreach (var entry in entityEntry.CurrentValues.Properties)
        {
            var value = entry.PropertyInfo.GetValue(entityEntry.Entity);
            if (value is string s)
            {
                strLargeValues.Add(Tuple.Create(s.Length, s, entry.Name, entityEntry.Entity.GetType().Name));
            }
        }
    }

    var l = strLargeValues.OrderByDescending(v => v.Item1).ToArray();

    foreach (var x in l.Take(100))
    {
        Trace.WriteLine(x.Item4 + " - " + x.Item3 + " - " + x.Item1 + ": " + x.Item2);
    }

    throw;
}

所以,问题的关键是:如何从 EF Core 获取 SQL 列定义?

我希望能够在incomingData.Length &gt; targetColumnDefinition.Length

时记录特定的

最终解决方案:

public override int SaveChanges()
{
    using (LogContext.PushProperty("DbContext:Override:Save", nameof(SaveChanges)))
    {
        try
        {
            return base.SaveChanges();
        }
        catch (Exception ex)
        {
            var errorMessage = String.Empty;
            var token = Environment.NewLine;

            foreach (var entityEntry in this.ChangeTracker.Entries().Where(et => et.State != EntityState.Unchanged))
            {
                foreach (var entry in entityEntry.CurrentValues.Properties)
                {
                    var result = entityEntry.GetDatabaseDefinition(entry.Name);
                    var value = entry.PropertyInfo.GetValue(entityEntry.Entity);
                    if (result.IsFixedLength && value.ToLength() > result.MaxLength)
                    {
                        errorMessage = $"{errorMessage}{token}ERROR!! <<< {result.TableName}.{result.ColumnName} {result.ColumnType.ToUpper()} :: {entry.Name}({value.ToLength()}) = {value} >>>";
                        Log.Warning("Cannot save data to SQL column {TableName}.{ColumnName}!  Max length is {LengthTarget} and you are trying to save something that is {LengthSource}.  Column definition is {ColumnType}"
                            , result.TableName
                            , result.ColumnName
                            , result.MaxLength
                            , value.ToLength()
                            , result.ColumnType);
                    }
                }
            }
            throw new Exception(errorMessage, ex);
        }
    }
}

【问题讨论】:

  • 使用纯 SQL 语句不可行吗?
  • 您使用的是哪个版本的 SQL Server? 2019(截至 CTP2)在消息中包含表名和列名。它还应该在CU12CU6 中向后移植到2017 和2016 SP2,需要跟踪标志460 才能启用它。如果您正在运行其中一个版本,希望您可以进行更新并启用该标志。
  • FindEntityType 需要完整的类型名称,否则需要类型本身,typeof(Subscription)
  • 您可以从这里开始 - 拥有EntityEntry 时无需查找实体类型,因为它由Metadata 属性提供,例如var entityType = entityEntry.Metadata;
  • 您是否尝试将以上内容更新到 v3.1?我想尝试上述方法,但找不到一些缺失的 defs。 GetDatabaseDefinition().

标签: c# entity-framework-core


【解决方案1】:

在 .NET Core 3.1 和 EFCore 5.0.2 上,此日志记录无需额外的扩展方法即可工作:

try
{
    await context.SaveChangesAsync();
}
catch(Exception ex)
{
    foreach (var entityEntry in context.ChangeTracker.Entries().Where(et => et.State != EntityState.Unchanged))
    {
        foreach (var entry in entityEntry.CurrentValues.Properties)
        { 
            var prop = entityEntry.Property(entry.Name).Metadata;
            var value = entry.PropertyInfo?.GetValue(entityEntry.Entity);
            var valueLength = value?.ToString()?.Length;
            var typemapping = prop.GetTypeMapping();
            var typeSize = ((Microsoft.EntityFrameworkCore.Storage.RelationalTypeMapping) typemapping).Size;
            if (typeSize.HasValue && valueLength > typeSize.Value)
            {
                Log.Error( $"Truncation will occur: {entityEntry.Metadata.GetTableName()}.{prop.GetColumnName()} {prop.GetColumnType()} :: {entry.Name}({valueLength}) = {value}");
            }
        }
    }
    throw ex;
}

【讨论】:

    【解决方案2】:

    如 cmets 中所述,您需要全名,这可以从元数据中读取。

    public int GetColumnMaxLength(EntityEntry entityEntry)
    {
        int result = 0;
    
        var table = entityEntry.Metadata.Model.FindEntityType(entityEntry.Metadata.ClrType);
    
        // Column info 
        foreach (var property in table.GetProperties())
        {
            var maxLength = property.GetMaxLength();
    
            // For sql info, e.g. ColumnType = nvarchar(255):
            var sqlInfo = property.SqlServer();
        };
        return result;
    }
    

    【讨论】:

    • 这让我很接近!将尝试记住发布我使用此信息创建的扩展方法。谢谢!
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2010-10-21
    • 2022-01-11
    相关资源
    最近更新 更多