/EventScavenger/Version5/EventReaper/DataAccess/GenericSQLServerDAL.cs

# · C# · 386 lines · 374 code · 10 blank · 2 comment · 34 complexity · 70fb6a949da9a9760753ee11a6a674de MD5 · raw file

  1. using System;
  2. using System.Data.SqlClient;
  3. using System.Data;
  4. using System.Diagnostics;
  5. namespace HenIT.Data.SqlClient
  6. {
  7. public delegate void RaiseDBAccessDelegate();
  8. public delegate void RaiseDBInsertsDelegate(int inserts);
  9. public delegate void DBAccessDurationDelegate(long milliseconds);
  10. public delegate void RaiseMessageDelegate(string message);
  11. public abstract class GenericSQLServerDAL
  12. {
  13. #region Events
  14. public event RaiseDBAccessDelegate RaiseDBAccess;
  15. protected void DoRaiseDBAccessEvent()
  16. {
  17. if (RaiseDBAccess != null)
  18. {
  19. RaiseDBAccess();
  20. }
  21. }
  22. public event RaiseDBAccessDelegate RaiseDBAccessErrors;
  23. protected void DoRaiseDBAccessErrorsEvent()
  24. {
  25. if (RaiseDBAccessErrors != null)
  26. {
  27. RaiseDBAccessErrors();
  28. }
  29. }
  30. public event RaiseDBInsertsDelegate RaiseDBAccessInserts;
  31. protected void DoRaiseDBAccessInsertsEvent(int inserts)
  32. {
  33. if (RaiseDBAccessInserts != null)
  34. {
  35. RaiseDBAccessInserts(inserts);
  36. }
  37. }
  38. public event RaiseDBAccessDelegate RaiseDBAccessInsertDuplicates;
  39. protected void DoRaiseDBAccessInsertDuplicates()
  40. {
  41. if (RaiseDBAccessInsertDuplicates != null)
  42. {
  43. RaiseDBAccessInsertDuplicates();
  44. }
  45. }
  46. public event DBAccessDurationDelegate DBAccessDurationInfoMessage;
  47. protected void RaiseDBAccessDurationInfoMessage(long milliseconds)
  48. {
  49. if (DBAccessDurationInfoMessage != null)
  50. {
  51. DBAccessDurationInfoMessage(milliseconds);
  52. }
  53. }
  54. public event RaiseMessageDelegate RaiseInfoMessage;
  55. protected void DoRaiseInfoMessage(string message)
  56. {
  57. if (RaiseInfoMessage != null)
  58. {
  59. RaiseInfoMessage(message);
  60. }
  61. }
  62. public event RaiseMessageDelegate RaiseErrorMessage;
  63. protected void DoRaiseErrorMessage(string message)
  64. {
  65. if (RaiseErrorMessage != null)
  66. {
  67. RaiseErrorMessage(message);
  68. }
  69. }
  70. public event RaiseMessageDelegate RaiseSyncInfoMessage;
  71. protected void DoRaiseSyncInfoMessage(string message)
  72. {
  73. if (RaiseSyncInfoMessage != null)
  74. {
  75. RaiseSyncInfoMessage(message);
  76. }
  77. }
  78. #endregion
  79. #region Connection details
  80. #region Connection properties
  81. private string server = string.Empty;
  82. public string Server
  83. {
  84. get { return server; }
  85. set { server = value; }
  86. }
  87. private string database = string.Empty;
  88. public string Database
  89. {
  90. get { return database; }
  91. set { database = value; }
  92. }
  93. private string userName = string.Empty;
  94. public string UserName
  95. {
  96. get { return userName; }
  97. set { userName = value; }
  98. }
  99. private string password = string.Empty;
  100. public string Password
  101. {
  102. protected get { return password; }
  103. set { password = value; }
  104. }
  105. private int commandTimeout = 120;
  106. public int CommandTimeout
  107. {
  108. get { return commandTimeout; }
  109. set { commandTimeout = value; }
  110. }
  111. public bool TrustServerCertificate { get; set; }
  112. #endregion
  113. protected string lastError = "";
  114. public string LastError
  115. {
  116. get { return lastError; }
  117. }
  118. protected string connectionString;
  119. public string ConnectionString
  120. {
  121. get { return connectionString; }
  122. set { connectionString = value; }
  123. }
  124. public void SetConnection()
  125. {
  126. if (server.Length > 0 && database.Length > 0)
  127. {
  128. SqlConnectionStringBuilder sqlbuilder = new SqlConnectionStringBuilder();
  129. sqlbuilder.DataSource = server;
  130. sqlbuilder.InitialCatalog = database;
  131. if (userName.Length > 0)
  132. {
  133. sqlbuilder.UserID = userName;
  134. sqlbuilder.Password = password;
  135. sqlbuilder.IntegratedSecurity = false;
  136. }
  137. else
  138. sqlbuilder.IntegratedSecurity = true;
  139. if (TrustServerCertificate)
  140. sqlbuilder.TrustServerCertificate = true;
  141. connectionString = sqlbuilder.ConnectionString;
  142. }
  143. else
  144. {
  145. throw new Exception("Connection settings not set properly!");
  146. }
  147. }
  148. public void SetConnection(string sqlServer, string database)
  149. {
  150. server = sqlServer;
  151. this.database = database;
  152. SetConnection();
  153. }
  154. public void SetConnection(string sqlServer, string database, string userName, string password)
  155. {
  156. server = sqlServer;
  157. this.database = database;
  158. this.userName = userName;
  159. this.password = password;
  160. SetConnection();
  161. }
  162. #endregion
  163. #region SQL Connection Opening
  164. public void OpenConnectionWithRetries(SqlConnection conn, int retriesLeft = 5)
  165. {
  166. try
  167. {
  168. conn.Open();
  169. }
  170. catch (Exception sqlex)
  171. {
  172. if (sqlex is System.Data.SqlClient.SqlException && sqlex.Message.Contains("A connection was successfully established with the server, but then an error occurred during the pre-login handshake"))
  173. {
  174. System.Threading.Thread.Sleep(999);
  175. if (retriesLeft > 0)
  176. {
  177. DoRaiseErrorMessage("Connection opening timed out. Retries left:" + (retriesLeft - 1).ToString());
  178. OpenConnectionWithRetries(conn, retriesLeft - 1);
  179. }
  180. else
  181. {
  182. throw new Exception("Open connection to database failed!");
  183. }
  184. }
  185. else if (sqlex.ToString().Contains("A connection was successfully established with the server, but then an error occurred during the login process"))
  186. {
  187. System.Threading.Thread.Sleep(999);
  188. if (retriesLeft > 0)
  189. {
  190. DoRaiseErrorMessage("Error opening connection during login. Retries left:" + (retriesLeft - 1).ToString());
  191. OpenConnectionWithRetries(conn, retriesLeft - 1);
  192. }
  193. else
  194. {
  195. throw new Exception("Error opening connection during login!");
  196. }
  197. }
  198. else
  199. throw;
  200. }
  201. }
  202. #endregion
  203. #region GetSingleValue
  204. public object GetSingleValue(string sql)
  205. {
  206. return GetSingleValue(sql, CommandType.StoredProcedure, new SqlParameter[] { });
  207. }
  208. public object GetSingleValue(string sql, CommandType commandType)
  209. {
  210. return GetSingleValue(sql, commandType, new SqlParameter[] { });
  211. }
  212. public object GetSingleValue(string sql, CommandType commandType, SqlParameter[] parms)
  213. {
  214. object returnValue = null;
  215. Stopwatch sw = new Stopwatch();
  216. sw.Start();
  217. DoRaiseDBAccessEvent();
  218. try
  219. {
  220. using (SqlConnection conn = new SqlConnection(connectionString))
  221. {
  222. OpenConnectionWithRetries(conn);
  223. using (SqlCommand cmnd = new SqlCommand(sql, conn))
  224. {
  225. cmnd.CommandTimeout = commandTimeout;
  226. cmnd.CommandType = commandType;
  227. if (parms != null)
  228. cmnd.Parameters.AddRange(parms);
  229. returnValue = cmnd.ExecuteScalar();
  230. }
  231. }
  232. }
  233. catch (Exception ex)
  234. {
  235. lastError = ex.Message;
  236. DoRaiseDBAccessErrorsEvent();
  237. throw;
  238. }
  239. sw.Stop();
  240. RaiseDBAccessDurationInfoMessage(sw.ElapsedMilliseconds);
  241. return returnValue;
  242. }
  243. #endregion
  244. #region GetDataSet
  245. public DataSet GetDataSet(string sql)
  246. {
  247. return GetDataSet(sql, CommandType.StoredProcedure, null);
  248. }
  249. public DataSet GetDataSet(string sql, CommandType commandType)
  250. {
  251. return GetDataSet(sql, commandType, null);
  252. }
  253. public DataSet GetDataSet(string sql, CommandType commandType, SqlParameter[] parms)
  254. {
  255. DataSet returnDS = new DataSet();
  256. Stopwatch sw = new Stopwatch();
  257. sw.Start();
  258. DoRaiseDBAccessEvent();
  259. try
  260. {
  261. using (SqlConnection conn = new SqlConnection(connectionString))
  262. {
  263. OpenConnectionWithRetries(conn);
  264. using (SqlCommand cmnd = new SqlCommand(sql, conn))
  265. {
  266. cmnd.CommandTimeout = commandTimeout;
  267. cmnd.CommandType = commandType;
  268. if (parms != null)
  269. cmnd.Parameters.AddRange(parms);
  270. using (SqlDataAdapter da = new SqlDataAdapter(cmnd))
  271. {
  272. da.Fill(returnDS);
  273. }
  274. }
  275. }
  276. }
  277. catch (Exception ex)
  278. {
  279. lastError = ex.Message;
  280. DoRaiseDBAccessErrorsEvent();
  281. throw;
  282. }
  283. sw.Stop();
  284. RaiseDBAccessDurationInfoMessage(sw.ElapsedMilliseconds);
  285. return returnDS;
  286. }
  287. #endregion
  288. #region Execute
  289. public void ExecuteSP(string spName)
  290. {
  291. ExecuteSP(spName, null, commandTimeout);
  292. }
  293. public void ExecuteSP(string spName, SqlParameter[] parms)
  294. {
  295. ExecuteSP(spName, parms, commandTimeout);
  296. }
  297. public void ExecuteSP(string sql, SqlParameter[] parms, int commandTimeout, int retry = 5)
  298. {
  299. Stopwatch sw = new Stopwatch();
  300. sw.Start();
  301. DoRaiseDBAccessEvent();
  302. try
  303. {
  304. using (SqlConnection conn = new SqlConnection(connectionString))
  305. {
  306. OpenConnectionWithRetries(conn);
  307. using (SqlCommand cmnd = new SqlCommand(sql, conn))
  308. {
  309. cmnd.CommandType = CommandType.StoredProcedure;
  310. cmnd.CommandTimeout = commandTimeout;
  311. if (parms != null)
  312. cmnd.Parameters.AddRange(parms);
  313. try
  314. {
  315. cmnd.ExecuteNonQuery();
  316. }
  317. catch (SqlException SeEx)
  318. {
  319. //System.Data.SqlClient.SqlException (0x80131904): A connection was successfully established with the server, but then an error occurred during the login process.
  320. //(provider: Named Pipes Provider, error: 0 - No process is on the other end of the pipe.) ---> System.ComponentModel.Win32Exception (0x80004005): No process is
  321. if (SeEx.ToString().Contains("A connection was successfully established with the server, but then an error occurred during the login process"))
  322. {
  323. if (retry > 0)
  324. {
  325. DoRaiseErrorMessage("A connection was successfully established with the server, but then an error occurred during the login process. Retrying (" + retry.ToString() + ")");
  326. System.Threading.Thread.Sleep(200);
  327. ExecuteSP(sql, parms, commandTimeout, retry - 1);
  328. }
  329. else
  330. throw;
  331. }
  332. else
  333. throw;
  334. }
  335. }
  336. }
  337. }
  338. catch (Exception ex)
  339. {
  340. lastError = ex.Message;
  341. DoRaiseDBAccessErrorsEvent();
  342. throw;
  343. }
  344. sw.Stop();
  345. RaiseDBAccessDurationInfoMessage(sw.ElapsedMilliseconds);
  346. }
  347. public void ExecuteSQL(string sql)
  348. {
  349. Stopwatch sw = new Stopwatch();
  350. sw.Start();
  351. DoRaiseDBAccessEvent();
  352. try
  353. {
  354. using (SqlConnection conn = new SqlConnection(connectionString))
  355. {
  356. OpenConnectionWithRetries(conn);
  357. using (SqlCommand cmnd = new SqlCommand(sql, conn))
  358. {
  359. cmnd.CommandType = CommandType.Text;
  360. cmnd.CommandTimeout = commandTimeout;
  361. cmnd.ExecuteNonQuery();
  362. }
  363. }
  364. }
  365. catch (Exception ex)
  366. {
  367. lastError = ex.Message;
  368. DoRaiseDBAccessErrorsEvent();
  369. throw;
  370. }
  371. sw.Stop();
  372. RaiseDBAccessDurationInfoMessage(sw.ElapsedMilliseconds);
  373. }
  374. #endregion
  375. }
  376. }