我使用.Net代码中的npgsql进行PostgreSQL批量插入的代码。
try
{
var items = GetSourceData(task);
connection.Open();
var command = new NpgsqlCommand(null, connection);
BeforeDestinationCommandExecution(task, command);
command.CommandText = string.Format("COPY {0} FROM STDIN", task.DestinationTable);
command.CommandTimeout = 3600;
var cin = new NpgsqlCopyIn(command, connection);
var rowCount = 0;
try
{
cin.Start();
foreach (var item in items)
{
var b = StreamEncoding.GetBytes(ConvertSourceData(item));
cin.CopyStream.Write(b, 0, b.Length);
++rowCount;
}
cin.End();
log.Debug(string.Format("Table {0} contained {1:N0} records", task.DestinationTable, rowCount));
}
catch (Exception e)
{
log.ErrorException("Exception caught in inner try block - MigrateWithCopyMode", e);
try
{
// send CopyFail to server
cin.Cancel("Undo copy");
}
catch (Exception cancelException)
{
// we should get an error in response to our cancel request:
if (!cancelException.ToString().Contains("Undo copy"))
{
throw new Exception("Failed to cancel COPY: " + cancelException + " upon failure: " + e);
}
}
throw;
}
finally
{
_migrationCounts.Add(task.DestinationTable, rowCount);
}
在过去的两天里,我在执行代码时遇到了未处理的异常。经过一些调查并将代码附加到UnhandledException事件。
System.AppDomain.CurrentDomain.UnhandledException += unhandledException;
我发现问题出在数据处理中。
Error [16] [HubAdapterMsSqlPostgres] Unhandled exception Npgsql.NpgsqlException:
null value in column "name_ru" violates not-null constraint
Severity: ERROR
Code: 23502
at Npgsql.NpgsqlState.<ProcessBackendResponses_Ver_3>d__a.MoveNext()
at Npgsql.NpgsqlState.IterateThroughAllResponses(IEnumerable`1 ienum)
at Npgsql.NpgsqlConnector.NpgsqlContextHolder.ProcessServerMessages()
at System.Threading.ExecutionContext.RunInternal(ExecutionContext executionContext, ContextCallback callback, Object state, Boolean preserveSyncCtx)
at System.Threading.ExecutionContext.Run(ExecutionContext executionContext, ContextCallback callback, Object state, Boolean preserveSyncCtx)
at System.Threading.ExecutionContext.Run(ExecutionContext executionContext, ContextCallback callback, Object state)
at System.Threading.ThreadHelper.ThreadStart()
此错误是在不同的线程上生成的。我猜之后从前面的代码示例中调用了cin.CopyStream.Write(b, 0, b.Length);
。
我的问题是
处理此类错误的建议方法是,可以省略具有错误值的行并继续批量插入操作
谢谢
答案 0 :(得分:0)
您尝试导入包含与您的表定义不兼容的行的数据 - 它们缺少非可空列的数据。 COPY是一个全有或全无的过程:要么整个过程成功,要么完全失败。因此,您必须: *首先清理您的输入,删除有问题的行(这将涉及解析您的输入,这可能很困难),或 *暂时删除表上的非空约束,执行COPY,从表中删除有问题的行并恢复约束。根据您的使用情况,这可能适合也可能不适合。
无论如何,您使用的是Npgsql 2.x,它现在已经很老了,而且没有维护。强烈建议您升级到最新版本的Npgsql。