KdbndpFastBuilder.cs 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144
  1. using Npgsql;
  2. using NpgsqlTypes;
  3. using System;
  4. using System.Collections.Generic;
  5. using System.Data;
  6. using System.Linq;
  7. using System.Text;
  8. using System.Threading.Tasks;
  9. namespace SqlSugar
  10. {
  11. public class KdbndpFastBuilder : FastBuilder, IFastBuilder
  12. {
  13. public static Dictionary<string , NpgsqlDbType> PgSqlType = UtilMethods.EnumToDictionary<NpgsqlDbType>();
  14. public KdbndpFastBuilder()
  15. {
  16. }
  17. public override string UpdateSql { get; set; } = @"UPDATE {1} SET {0} FROM {2} AS TE WHERE {3}
  18. ";
  19. //public virtual async Task<int> UpdateByTempAsync(string tableName, string tempName, string[] updateColumns, string[] whereColumns)
  20. //{
  21. // Check.ArgumentNullException(!updateColumns.Any(), "update columns count is 0");
  22. // Check.ArgumentNullException(!whereColumns.Any(), "where columns count is 0");
  23. // var sets = string.Join(",", updateColumns.Select(it => $"TM.{it}=TE.{it}"));
  24. // var wheres = string.Join(",", whereColumns.Select(it => $"TM.{it}=TE.{it}"));
  25. // string sql = string.Format(UpdateSql, sets, tableName, tempName, wheres);
  26. // return await this.Context.Ado.ExecuteCommandAsync(sql);
  27. //}
  28. public async Task<int> ExecuteBulkCopyAsync(DataTable dt)
  29. {
  30. List<string> lsColNames = new List<string>();
  31. for (int i = 0; i < dt.Columns.Count; i++)
  32. {
  33. lsColNames.Add($"\"{dt.Columns[i].ColumnName}\"");
  34. }
  35. string copyString = $"COPY {dt.TableName} ( {string.Join(",", lsColNames) } ) FROM STDIN (FORMAT BINARY)";
  36. Kdbndp.KdbndpConnection conn = (Kdbndp.KdbndpConnection)this.Context.Ado.Connection;
  37. var columns = this.Context.DbMaintenance.GetColumnInfosByTableName(this.FastEntityInfo.DbTableName);
  38. try
  39. {
  40. var identityColumnInfo = this.FastEntityInfo.Columns.FirstOrDefault(it => it.IsIdentity);
  41. if (identityColumnInfo != null)
  42. {
  43. throw new Exception("PgSql bulkcopy no support identity");
  44. }
  45. BulkCopy(dt, copyString, conn, columns);
  46. }
  47. catch (Exception ex)
  48. {
  49. throw ex;
  50. }
  51. finally
  52. {
  53. base.CloseDb();
  54. }
  55. return await Task.FromResult(dt.Rows.Count);
  56. }
  57. private void BulkCopy(DataTable dt, string copyString, Kdbndp.KdbndpConnection conn, List<DbColumnInfo> columns)
  58. {
  59. if (conn.State == ConnectionState.Closed)
  60. conn.Open();
  61. List<ColumnView> columnViews = new List<ColumnView>();
  62. foreach (DataColumn item in dt.Columns)
  63. {
  64. ColumnView result = new ColumnView();
  65. result.DbColumnInfo = columns.FirstOrDefault(it => it.DbColumnName.EqualCase(item.ColumnName));
  66. result.DataColumn = item;
  67. result.EntityColumnInfo=this.FastEntityInfo.Columns.FirstOrDefault(it => it.DbColumnName.EqualCase(item.ColumnName));
  68. var key = result.DbColumnInfo?.DataType?.ToLower();
  69. if (result.DbColumnInfo == null)
  70. {
  71. result.Type = null;
  72. }
  73. else if (PgSqlType.ContainsKey(key))
  74. {
  75. result.Type = PgSqlType[key];
  76. }
  77. else if (key?.First() == '_')
  78. {
  79. var type = PgSqlType[key.Substring(1)];
  80. result.Type = NpgsqlDbType.Array | type;
  81. }
  82. else
  83. {
  84. result.Type = null;
  85. }
  86. columnViews.Add(result);
  87. }
  88. using (var writer = conn.BeginBinaryImport(copyString))
  89. {
  90. foreach (DataRow row in dt.Rows)
  91. {
  92. writer.StartRow();
  93. foreach (var column in columnViews)
  94. {
  95. var value = row[column.DataColumn.ColumnName];
  96. if (value == null)
  97. {
  98. value = DBNull.Value;
  99. }
  100. if (column.Type == null)
  101. {
  102. writer.Write(value);
  103. }
  104. else
  105. {
  106. writer.Write(value);
  107. }
  108. }
  109. }
  110. writer.Complete();
  111. }
  112. }
  113. public override async Task<int> UpdateByTempAsync(string tableName, string tempName, string[] updateColumns, string[] whereColumns)
  114. {
  115. var sqlquerybulder= this.Context.Queryable<object>().SqlBuilder;
  116. Check.ArgumentNullException(!updateColumns.Any(), "update columns count is 0");
  117. Check.ArgumentNullException(!whereColumns.Any(), "where columns count is 0");
  118. var sets = string.Join(",", updateColumns.Select(it => $"{sqlquerybulder.GetTranslationColumnName(it)}=TE.{sqlquerybulder.GetTranslationColumnName(it)}"));
  119. var wheres = string.Join(" AND ", whereColumns.Select(it => $"{tableName}.{sqlquerybulder.GetTranslationColumnName(it)}=TE.{sqlquerybulder.GetTranslationColumnName(it)}"));
  120. string sql = string.Format(UpdateSql, sets, tableName, tempName, wheres);
  121. return await this.Context.Ado.ExecuteCommandAsync(sql);
  122. }
  123. public override async Task CreateTempAsync<T>(DataTable dt)
  124. {
  125. await this.Context.Queryable<T>().Where(it => false).AS(dt.TableName).Select(" * into temp mytemptable").ToListAsync();
  126. dt.TableName = "mytemptable";
  127. }
  128. public class ColumnView
  129. {
  130. public DataColumn DataColumn { get; set; }
  131. public EntityColumnInfo EntityColumnInfo { get; set; }
  132. public DbColumnInfo DbColumnInfo { get; set; }
  133. public NpgsqlDbType? Type { get; set; }
  134. }
  135. }
  136. }