PostgreSQLProvider.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Data;
  4. using System.Data.Common;
  5. using System.Linq;
  6. using System.Text;
  7. using System.Threading.Tasks;
  8. using Npgsql;
  9. using NpgsqlTypes;
  10. namespace SqlSugar
  11. {
  12. public partial class PostgreSQLProvider : AdoProvider
  13. {
  14. public PostgreSQLProvider()
  15. {
  16. if (StaticConfig.AppContext_ConvertInfinityDateTime == false)
  17. {
  18. AppContext.SetSwitch("Npgsql.EnableLegacyTimestampBehavior", true);
  19. AppContext.SetSwitch("Npgsql.DisableDateTimeInfinityConversions", true);
  20. }
  21. }
  22. public override IDbConnection Connection
  23. {
  24. get
  25. {
  26. if (base._DbConnection == null)
  27. {
  28. try
  29. {
  30. var npgsqlConnectionString = base.Context.CurrentConnectionConfig.ConnectionString;
  31. base._DbConnection = new NpgsqlConnection(npgsqlConnectionString);
  32. }
  33. catch (Exception ex)
  34. {
  35. Check.Exception(true, ErrorMessage.ConnnectionOpen, ex.Message);
  36. }
  37. }
  38. return base._DbConnection;
  39. }
  40. set
  41. {
  42. base._DbConnection = value;
  43. }
  44. }
  45. public override void BeginTran(string transactionName)
  46. {
  47. base.BeginTran();
  48. }
  49. /// <summary>
  50. /// Only SqlServer
  51. /// </summary>
  52. /// <param name="iso"></param>
  53. /// <param name="transactionName"></param>
  54. public override void BeginTran(IsolationLevel iso, string transactionName)
  55. {
  56. base.BeginTran(iso);
  57. }
  58. public override IDataAdapter GetAdapter()
  59. {
  60. return new NpgsqlDataAdapter();
  61. }
  62. public override DbCommand GetCommand(string sql, SugarParameter[] parameters)
  63. {
  64. NpgsqlCommand sqlCommand = new NpgsqlCommand(sql, (NpgsqlConnection)this.Connection);
  65. sqlCommand.CommandType = this.CommandType;
  66. sqlCommand.CommandTimeout = this.CommandTimeOut;
  67. if (this.Transaction != null)
  68. {
  69. sqlCommand.Transaction = (NpgsqlTransaction)this.Transaction;
  70. }
  71. if (parameters.HasValue())
  72. {
  73. IDataParameter[] ipars = ToIDbDataParameter(parameters);
  74. sqlCommand.Parameters.AddRange((NpgsqlParameter[])ipars);
  75. }
  76. CheckConnection();
  77. return sqlCommand;
  78. }
  79. public override void SetCommandToAdapter(IDataAdapter dataAdapter, DbCommand command)
  80. {
  81. ((NpgsqlDataAdapter)dataAdapter).SelectCommand = (NpgsqlCommand)command;
  82. }
  83. /// <summary>
  84. /// if mysql return MySqlParameter[] pars
  85. /// if sqlerver return SqlParameter[] pars ...
  86. /// </summary>
  87. /// <param name="parameters"></param>
  88. /// <returns></returns>
  89. public override IDataParameter[] ToIDbDataParameter(params SugarParameter[] parameters)
  90. {
  91. if (parameters == null || parameters.Length == 0) return null;
  92. NpgsqlParameter[] result = new NpgsqlParameter[parameters.Length];
  93. int index = 0;
  94. var isVarchar = this.Context.IsVarchar();
  95. foreach (var parameter in parameters)
  96. {
  97. UNumber(parameter);
  98. if (parameter.Value == null) parameter.Value = DBNull.Value;
  99. if (parameter.Value is System.Data.SqlTypes.SqlDateTime && parameter.DbType == System.Data.DbType.AnsiString)
  100. {
  101. parameter.DbType = System.Data.DbType.DateTime;
  102. parameter.Value = DBNull.Value;
  103. }
  104. var sqlParameter = new NpgsqlParameter();
  105. sqlParameter.ParameterName = parameter.ParameterName;
  106. sqlParameter.Size = parameter.Size;
  107. sqlParameter.Value = parameter.Value;
  108. sqlParameter.DbType = parameter.DbType;
  109. sqlParameter.Direction = parameter.Direction;
  110. if (parameter.IsJson)
  111. {
  112. sqlParameter.NpgsqlDbType = NpgsqlDbType.Json;
  113. }
  114. if (parameter.IsArray)
  115. {
  116. Array(parameter, sqlParameter);
  117. }
  118. if (sqlParameter.Direction == 0)
  119. {
  120. sqlParameter.Direction = ParameterDirection.Input;
  121. }
  122. result[index] = sqlParameter;
  123. if (sqlParameter.Direction.IsIn(ParameterDirection.Output, ParameterDirection.InputOutput, ParameterDirection.ReturnValue))
  124. {
  125. if (this.OutputParameters == null) this.OutputParameters = new List<IDataParameter>();
  126. this.OutputParameters.RemoveAll(it => it.ParameterName == sqlParameter.ParameterName);
  127. this.OutputParameters.Add(sqlParameter);
  128. }
  129. if (isVarchar && sqlParameter.DbType == System.Data.DbType.String)
  130. {
  131. sqlParameter.DbType = System.Data.DbType.AnsiString;
  132. }
  133. else if (sqlParameter.Value is DateTime && sqlParameter.DbType == System.Data.DbType.AnsiString)
  134. {
  135. sqlParameter.DbType = System.Data.DbType.DateTime;
  136. }
  137. ++index;
  138. if (parameter.CustomDbType != null&& parameter.CustomDbType is NpgsqlDbType)
  139. {
  140. sqlParameter.NpgsqlDbType =((NpgsqlDbType)parameter.CustomDbType);
  141. }
  142. }
  143. return result;
  144. }
  145. //private static void ConvertUNumber(SugarParameter parameter)
  146. //{
  147. // if (parameter.DbType == System.Data.DbType.UInt32)
  148. // {
  149. // parameter.DbType = System.Data.DbType.Int32;
  150. // }
  151. // else if (parameter.DbType == System.Data.DbType.UInt64)
  152. // {
  153. // parameter.DbType = System.Data.DbType.UInt64;
  154. // }
  155. //}
  156. private static void Array(SugarParameter parameter, NpgsqlParameter sqlParameter)
  157. {
  158. // sqlParameter.Value = this.Context.Utilities.SerializeObject(sqlParameter.Value);
  159. var type = sqlParameter.Value.GetType();
  160. if (ArrayMapping.ContainsKey(type))
  161. {
  162. sqlParameter.NpgsqlDbType = ArrayMapping[type] | NpgsqlDbType.Array;
  163. }
  164. else if (type == DBNull.Value.GetType())
  165. {
  166. DbNullParametrerArray(parameter, sqlParameter);
  167. }
  168. else
  169. {
  170. Check.Exception(true, sqlParameter.Value.GetType().Name + " No Support");
  171. }
  172. }
  173. private static void DbNullParametrerArray(SugarParameter parameter, NpgsqlParameter sqlParameter)
  174. {
  175. if (parameter.DbType.IsIn(System.Data.DbType.Int32))
  176. {
  177. sqlParameter.NpgsqlDbType = NpgsqlDbType.Integer | NpgsqlDbType.Array;
  178. }
  179. else if (parameter.DbType.IsIn(System.Data.DbType.Int16))
  180. {
  181. sqlParameter.NpgsqlDbType = NpgsqlDbType.Smallint | NpgsqlDbType.Array;
  182. }
  183. else if (parameter.DbType.IsIn(System.Data.DbType.Int64))
  184. {
  185. sqlParameter.NpgsqlDbType = NpgsqlDbType.Bigint | NpgsqlDbType.Array;
  186. }
  187. else if (parameter.DbType.IsIn(System.Data.DbType.Guid))
  188. {
  189. sqlParameter.NpgsqlDbType = NpgsqlDbType.Uuid | NpgsqlDbType.Array;
  190. }
  191. else
  192. {
  193. sqlParameter.NpgsqlDbType = NpgsqlDbType.Text | NpgsqlDbType.Array;
  194. }
  195. }
  196. private static void UNumber(SugarParameter parameter)
  197. {
  198. if (parameter.DbType == System.Data.DbType.UInt16)
  199. {
  200. parameter.DbType = System.Data.DbType.Int16;
  201. parameter.Value = Convert.ToInt16(parameter.Value);
  202. }
  203. else if (parameter.DbType == System.Data.DbType.UInt32)
  204. {
  205. parameter.DbType = System.Data.DbType.Int32;
  206. parameter.Value = Convert.ToInt32(parameter.Value);
  207. }
  208. else if (parameter.DbType == System.Data.DbType.UInt64)
  209. {
  210. parameter.DbType = System.Data.DbType.Int64;
  211. parameter.Value = Convert.ToInt64(parameter.Value);
  212. }
  213. if (parameter.Value is uint)
  214. {
  215. parameter.Value = Convert.ToInt32(parameter.Value);
  216. }
  217. else if (parameter.Value is ulong)
  218. {
  219. parameter.Value = Convert.ToInt64(parameter.Value);
  220. }
  221. }
  222. public override Action<SqlSugarException> ErrorEvent => it =>
  223. {
  224. if (base.ErrorEvent != null)
  225. {
  226. base.ErrorEvent(it);
  227. }
  228. if (it.Message != null && it.Message.StartsWith("42883: function uuid_generate_v4() does not exist"))
  229. {
  230. Check.ExceptionEasy(it.Message, $"使用uuid_generate_v4()函数需要创建 CREATE EXTENSION IF NOT EXISTS pgcrypto;CREATE EXTENSION IF NOT EXISTS \"uuid-ossp\" ");
  231. }
  232. };
  233. static readonly Dictionary<Type, NpgsqlDbType> ArrayMapping = new Dictionary<Type, NpgsqlDbType>()
  234. {
  235. { typeof(int[]),NpgsqlDbType.Integer},
  236. { typeof(short[]),NpgsqlDbType.Smallint},
  237. { typeof(long[]),NpgsqlDbType.Bigint},
  238. { typeof(decimal[]),NpgsqlDbType.Numeric},
  239. { typeof(char[]),NpgsqlDbType.Text},
  240. { typeof(byte[]),NpgsqlDbType.Bytea},
  241. { typeof(bool[]),NpgsqlDbType.Boolean},
  242. {typeof(DateTime[]),NpgsqlDbType.Date},
  243. {typeof(float[]),NpgsqlDbType.Real},
  244. {typeof(Guid[]),NpgsqlDbType.Uuid},
  245. { typeof(int?[]),NpgsqlDbType.Integer},
  246. { typeof(short?[]),NpgsqlDbType.Smallint},
  247. { typeof(long?[]),NpgsqlDbType.Bigint},
  248. { typeof(decimal?[]),NpgsqlDbType.Numeric},
  249. { typeof(char?[]),NpgsqlDbType.Text},
  250. { typeof(byte?[]),NpgsqlDbType.Bytea},
  251. { typeof(bool?[]),NpgsqlDbType.Boolean},
  252. {typeof(DateTime?[]),NpgsqlDbType.Date},
  253. {typeof(Guid?[]),NpgsqlDbType.Uuid},
  254. { typeof(string[]), NpgsqlDbType.Text},
  255. {typeof(float?[]),NpgsqlDbType.Real},
  256. };
  257. }
  258. }