You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

480 lines
19 KiB

4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
  1. #if DEBUG
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Data;
  5. using System.Data.Common;
  6. using System.Text;
  7. namespace Apewer.Source
  8. {
  9. /// <summary>数据库客户端基类。</summary>
  10. public abstract class DbClient : IDbClient
  11. {
  12. /// <summary></summary>
  13. public virtual Logger Logger { get; set; }
  14. /// <summary></summary>
  15. public DbClient(Timeout timeout) { _timeout = timeout ?? Timeout.Default; }
  16. #region connection
  17. Timeout _timeout = null;
  18. IDbConnection _conn = null;
  19. string _str = null;
  20. /// <summary></summary>
  21. public Timeout Timeout { get => _timeout; set => _timeout = value ?? Timeout.Default; }
  22. /// <summary>获取当前的 SqlConnection 对象。</summary>
  23. public IDbConnection Connection { get => _conn; }
  24. /// <summary></summary>
  25. public bool Online { get => _conn == null ? false : (_conn.State == ConnectionState.Open); }
  26. /// <summary>连接字符串。</summary>
  27. public string ConnectionString { get => _str; }
  28. /// <summary>连接数据库,若未连接则尝试连接,获取连接成功的状态。</summary>
  29. public string Connect()
  30. {
  31. if (_conn == null)
  32. {
  33. _str = NewConnectionString();
  34. _conn = NewConnection();
  35. _conn.ConnectionString = _str;
  36. }
  37. else
  38. {
  39. if (_conn.State == ConnectionState.Open) return null;
  40. }
  41. try
  42. {
  43. _conn.Open();
  44. if (_conn.State == ConnectionState.Open) return null;
  45. var message = $"连接后状态为 {_conn.State},无法验证打开状态。";
  46. Logger.Error(this, "Connect", message, _str);
  47. return "连接失败," + message;
  48. }
  49. catch (Exception ex)
  50. {
  51. Logger.Error(this, "Connect", ex.GetType().Name, ex.Message, _str);
  52. Close();
  53. return ex.Message;
  54. }
  55. }
  56. /// <summary>关闭连接,并释放对象所占用的系统资源。</summary>
  57. public void Close()
  58. {
  59. if (_conn != null)
  60. {
  61. if (_transaction != null)
  62. {
  63. if (_autocommit) Commit();
  64. else Rollback();
  65. }
  66. _conn.Close();
  67. _conn.Dispose();
  68. _conn = null;
  69. }
  70. }
  71. /// <summary>关闭连接,释放对象所占用的系统资源,并清除连接信息。</summary>
  72. public void Dispose() { Close(); }
  73. #endregion
  74. #region transaction
  75. private IDbTransaction _transaction = null;
  76. private bool _autocommit = false;
  77. /// <summary>
  78. /// <para>启动事务,可指定事务锁定行为。</para>
  79. /// <para>Chaos<br />无法覆盖隔离级别更高的事务中的挂起的更改。</para>
  80. /// <para>ReadCommitted<br />在正在读取数据时保持共享锁,以避免脏读,但是在事务结束之前可以更改数据,从而导致不可重复的读取或幻像数据。</para>
  81. /// <para>ReadUncommitted<br />可以进行脏读,意思是说,不发布共享锁,也不接受独占锁。</para>
  82. /// <para>RepeatableRead<br />在查询中使用的所有数据上放置锁,以防止其他用户更新这些数据。 防止不可重复的读取,但是仍可以有幻像行。</para>
  83. /// <para>Serializable<br />在 System.Data.DataSet 上放置范围锁,以防止在事务完成之前由其他用户更新行或向数据集中插入行。</para>
  84. /// <para>Snapshot<br />通过在一个应用程序正在修改数据时存储另一个应用程序可以读取的相同数据版本来减少阻止。 表示您无法从一个事务中看到在其他事务中进行的更改,即便重新查询也是如此。</para>
  85. /// <para>Unspecified = -1<br />正在使用与指定隔离级别不同的隔离级别,但是无法确定该级别。</para>
  86. /// </summary>
  87. /// <param name="commit">在连接的生命周期结束时未结束事务,指定 TRUE 将自动提交,指定 FALSE 将自动回滚。</param>
  88. /// <param name="isolation">指定事务锁定行为,不指定时将使用默认值。</param>
  89. public string Begin(bool commit = false, Class<IsolationLevel> isolation = null)
  90. {
  91. if (_transaction != null) return "已存在未完成的事务,无法再次启动。";
  92. var connect = Connect();
  93. if (connect.NotEmpty()) return $"无法启动事务:连接失败。(${connect})";
  94. try
  95. {
  96. _transaction = isolation ? _conn.BeginTransaction(isolation.Value) : _conn.BeginTransaction();
  97. _autocommit = commit;
  98. return null;
  99. }
  100. catch (Exception ex)
  101. {
  102. Logger.Error(this, "Begin", ex.Message());
  103. return ex.Message();
  104. }
  105. }
  106. /// <summary>提交事务。</summary>
  107. public string Commit()
  108. {
  109. if (_transaction == null) return "事务不存在。";
  110. try
  111. {
  112. _transaction.Commit();
  113. RuntimeUtility.Dispose(_transaction);
  114. _transaction = null;
  115. return null;
  116. }
  117. catch (Exception ex)
  118. {
  119. RuntimeUtility.Dispose(_transaction);
  120. _transaction = null;
  121. Logger.Error(this, "Commit", ex.Message());
  122. return ex.Message();
  123. }
  124. }
  125. /// <summary>从挂起状态回滚事务。</summary>
  126. public string Rollback()
  127. {
  128. if (_transaction == null) return "事务不存在。";
  129. try
  130. {
  131. _transaction.Rollback();
  132. RuntimeUtility.Dispose(_transaction);
  133. _transaction = null;
  134. return null;
  135. }
  136. catch (Exception ex)
  137. {
  138. RuntimeUtility.Dispose(_transaction);
  139. _transaction = null;
  140. Logger.Error(this, "Rollback", ex.Message());
  141. return ex.Message();
  142. }
  143. }
  144. #endregion
  145. #region ado
  146. /// <summary>查询。</summary>
  147. public IQuery Query(string sql) => Query(sql, null);
  148. /// <summary>查询。</summary>
  149. public IQuery Query(string sql, IEnumerable<IDataParameter> parameters)
  150. {
  151. if (TextUtility.IsEmpty(sql)) return new Query(false, "语句无效。");
  152. var connected = Connect();
  153. if (connected.NotEmpty()) return new Query(false, connected);
  154. try
  155. {
  156. using (var command = NewCommand())
  157. {
  158. command.Connection = _conn;
  159. if (_timeout != null) command.CommandTimeout = Timeout.Query;
  160. command.CommandText = sql;
  161. if (parameters != null)
  162. {
  163. foreach (var parameter in parameters)
  164. {
  165. if (parameter != null) command.Parameters.Add(parameter);
  166. }
  167. }
  168. var ex = null as Exception;
  169. var da = null as IDataAdapter;
  170. try { da = NewDataAdapter(command); }
  171. catch (Exception adapterEx) { ex = adapterEx; }
  172. if (ex == null)
  173. {
  174. using (var ds = new DataSet())
  175. {
  176. da.Fill(ds);
  177. if (ds.Tables.Count > 0)
  178. {
  179. var tables = new DataTable[ds.Tables.Count];
  180. ds.Tables.CopyTo(tables, 0);
  181. return new Query(tables, true);
  182. }
  183. else
  184. {
  185. Logger.Error(this, "Query", "查询结果不包含任何数据表。", sql);
  186. return new Query(false, "查询结果不包含任何数据表。");
  187. }
  188. }
  189. }
  190. else
  191. {
  192. Logger.Error(this, "Query", ex.GetType().FullName, ex.Message, sql);
  193. return new Query(ex);
  194. }
  195. }
  196. }
  197. catch (Exception exception)
  198. {
  199. Logger.Error(this, "Query", exception.GetType().FullName, exception.Message, sql);
  200. return new Query(exception);
  201. }
  202. }
  203. /// <summary>执行。</summary>
  204. public IExecute Execute(string sql) => Execute(sql, null);
  205. /// <summary>执行单条 Transact-SQL 语句,并加入参数。</summary>
  206. public IExecute Execute(string sql, IEnumerable<IDataParameter> parameters)
  207. {
  208. if (TextUtility.IsEmpty(sql)) return new Execute(false, "语句无效。");
  209. var connected = Connect();
  210. if (connected.NotEmpty()) return new Execute(false, connected);
  211. var inTransaction = _transaction != null;
  212. if (!inTransaction) Begin();
  213. try
  214. {
  215. using (var command = NewCommand())
  216. {
  217. command.Connection = _conn;
  218. command.Transaction = (DbTransaction)_transaction;
  219. if (Timeout != null) command.CommandTimeout = Timeout.Execute;
  220. command.CommandText = sql;
  221. if (parameters != null)
  222. {
  223. foreach (var parameter in parameters)
  224. {
  225. if (parameter != null) command.Parameters.Add(parameter);
  226. }
  227. }
  228. var rows = command.ExecuteNonQuery();
  229. if (!inTransaction) Commit(); // todo 此处应该检查事务提交产生的错误。
  230. return new Execute(true, rows);
  231. }
  232. }
  233. catch (Exception exception)
  234. {
  235. Logger.Error(this, "Execute", exception, sql);
  236. if (!inTransaction) Rollback();
  237. return new Execute(exception);
  238. }
  239. }
  240. /// <summary>输出查询结果的首列数据。</summary>
  241. protected string[] TextColumn(string sql, string[] excluded = null)
  242. {
  243. if (Connect().NotEmpty()) return new string[0];
  244. using (var query = Query(sql))
  245. {
  246. var rows = query.Rows;
  247. var list = new List<string>(rows);
  248. for (int r = 0; r < query.Rows; r++)
  249. {
  250. var cell = query.Text(r, 0);
  251. if (TextUtility.IsEmpty(cell)) continue;
  252. if (excluded != null && excluded.Contains(cell)) continue;
  253. list.Add(cell);
  254. }
  255. return list.ToArray();
  256. }
  257. }
  258. #endregion
  259. #region orm
  260. /// <summary>使用指定语句查询,获取查询结果。</summary>
  261. /// <param name="sql">要执行的 SQL 语句。</param>
  262. /// <param name="parameters">为 SQL 语句提供的参数。</param>
  263. public Result<T[]> Query<T>(string sql, IEnumerable<IDataParameter> parameters = null) where T : class, new() => SourceUtility.As<object, T>(Query(typeof(T), sql, parameters));
  264. /// <summary>使用指定语句查询,获取查询结果。</summary>
  265. /// <param name="model">目标记录的类型。</param>
  266. /// <param name="sql">要执行的 SQL 语句。</param>
  267. /// <param name="parameters">为 SQL 语句提供的参数。</param>
  268. public Result<object[]> Query(Type model, string sql, IEnumerable<IDataParameter> parameters = null)
  269. {
  270. if (_conn == null) return new Result<object[]>("连接无效。");
  271. if (model == null) return new Result<object[]>("数据模型类型无效。");
  272. if (string.IsNullOrEmpty(sql)) return new Result<object[]>("SQL 语句无效。");
  273. using (var query = Query(sql, parameters))
  274. {
  275. var result = null as Result<object[]>;
  276. if (query.Success)
  277. {
  278. try
  279. {
  280. var array = SourceUtility.Fill(query, model);
  281. result = new Result<object[]>(array);
  282. }
  283. catch (Exception ex)
  284. {
  285. result = new Result<object[]>(ex);
  286. }
  287. }
  288. else
  289. {
  290. result = new Result<object[]>(query.Message);
  291. }
  292. return result;
  293. }
  294. }
  295. #endregion
  296. #region static
  297. /// <summary>对文本转义,符合 SQL 安全性。可根据字段类型限制 UTF-8 字节数,默认为 0 时不限制字节数。</summary>
  298. public static string Escape(string text, int bytes = 0)
  299. {
  300. if (text.IsEmpty()) return "";
  301. var t = text ?? "";
  302. t = t.Replace("\\", "\\\\");
  303. t = t.Replace("'", "\\'");
  304. t = t.Replace("\n", "\\n");
  305. t = t.Replace("\r", "\\r");
  306. t = t.Replace("\b", "\\b");
  307. t = t.Replace("\t", "\\t");
  308. t = t.Replace("\f", "\\f");
  309. if (bytes > 5)
  310. {
  311. if (t.Bytes(Encoding.UTF8).Length > bytes)
  312. {
  313. while (true)
  314. {
  315. t = t.Substring(0, t.Length - 1);
  316. if (t.Bytes(Encoding.UTF8).Length <= (bytes - 4)) break;
  317. }
  318. t = t + " ...";
  319. }
  320. }
  321. return t;
  322. }
  323. /// <summary>获取表名。</summary>
  324. protected static string Table<T>() => Table(typeof(T));
  325. /// <summary>获取表名。</summary>
  326. protected static string Table(Type model)
  327. {
  328. if (model == null) throw new Exception($"无法从无效类型获取表名。");
  329. var ts = TableStructure.Parse(model);
  330. if (ts == null) throw new Exception($"无法从类型 {model.FullName} 获取表名。");
  331. return ts.Name;
  332. }
  333. #endregion
  334. #region derived
  335. /// <summary>为 Ado 创建连接字符串。</summary>
  336. protected abstract string NewConnectionString();
  337. /// <summary>为 Ado 创建 IDbConnection 对象。</summary>
  338. protected abstract IDbConnection NewConnection();
  339. /// <summary>为 Ado 创建 IDbCommand 对象。</summary>
  340. protected abstract IDbCommand NewCommand();
  341. /// <summary>为 Ado 创建 IDataAdapter 对象。</summary>
  342. protected abstract IDataAdapter NewDataAdapter(IDbCommand command);
  343. // /// <summary>为 Ado 创建 IDataParameter 对象。</summary>
  344. // protected abstract IDataParameter NewDataParameter();
  345. #endregion
  346. #region initialization
  347. /// <summary>查询数据库中的所有表名。</summary>
  348. public abstract string[] TableNames();
  349. /// <summary>查询数据库实例中的所有数据库名。</summary>
  350. public abstract string[] StoreNames();
  351. /// <summary>查询表中的所有列名。</summary>
  352. public abstract string[] ColumnNames(string tableName);
  353. /// <summary>获取列信息。</summary>
  354. public abstract ColumnInfo[] ColumnsInfo(string tableName);
  355. /// <summary>初始化指定类型,以创建表或增加字段。</summary>
  356. /// <returns>错误信息。当成功时候返回空字符串。</returns>
  357. public string Initialize<T>() where T : class, new() => Initialize(typeof(T));
  358. #endregion
  359. #region IDbClientOrm
  360. /// <summary>初始化指定类型,以创建表或增加字段。</summary>
  361. /// <param name="model">要初始化的类型。</param>
  362. /// <returns>错误信息。当成功时候返回空字符串。</returns>
  363. public abstract string Initialize(Type model);
  364. /// <summary>插入记录。</summary>
  365. /// <param name="record">要插入的记录实体。</param>
  366. /// <param name="table">插入到指定表。当不指定时,由 record 类型决定。</param>
  367. /// <returns>错误信息。当成功时候返回空字符串。</returns>
  368. public abstract string Insert(object record, string table = null);
  369. /// <summary>更新记录。</summary>
  370. /// <param name="record">要插入的记录实体。</param>
  371. /// <param name="table">插入到指定表。当不指定时,由 record 类型决定。</param>
  372. /// <returns>错误信息。当成功时候返回空字符串。</returns>
  373. public abstract string Update(IRecord record, string table = null);
  374. /// <summary>获取指定类型的主键,按 Flag 属性筛选。</summary>
  375. /// <param name="model">要查询的类型。</param>
  376. /// <param name="flag">要求目标记录具有的 Flag 属性,当指定 0 时忽略此要求。</param>
  377. public abstract Result<string[]> Keys(Type model, long flag = 0);
  378. /// <summary>获取指定类型的主键,按 Flag 属性筛选。</summary>
  379. /// <param name="flag">要求目标记录具有的 Flag 属性,当指定 0 时忽略此要求。</param>
  380. public abstract Result<string[]> Keys<T>(long flag = 0) where T : class, IRecord, new();
  381. /// <summary>获取具有指定 Key 的记录,并要求记录具有指定的 Flag 属性。</summary>
  382. /// <param name="model">目标记录的类型。</param>
  383. /// <param name="key">目标记录的主键。</param>
  384. /// <param name="flag">要求目标记录具有的 Flag 属性,当指定 0 时忽略此要求。</param>
  385. public abstract Result<object> Get(Type model, string key, long flag = 0);
  386. /// <summary>获取具有指定 Key 的记录,并要求记录具有指定的 Flag 属性。</summary>
  387. /// <param name="key">目标记录的主键。</param>
  388. /// <param name="flag">要求目标记录具有的 Flag 属性,当指定 0 时忽略此要求。</param>
  389. public abstract Result<T> Get<T>(string key, long flag = 0) where T : class, IRecord, new();
  390. /// <summary>查询所有记录。</summary>
  391. /// <param name="model">目标记录的类型。</param>
  392. /// <param name="flag">要求目标记录具有的 Flag 属性,当指定 0 时忽略此要求。</param>
  393. public abstract Result<object[]> Query(Type model, long flag = 0);
  394. /// <summary>查询所有记录。</summary>
  395. /// <param name="flag">要求目标记录具有的 Flag 属性,当指定 0 时忽略此要求。</param>
  396. public abstract Result<T[]> Query<T>(long flag = 0) where T : class, IRecord, new();
  397. #endregion
  398. }
  399. }
  400. #endif