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.

621 lines
24 KiB

4 years ago
4 years ago
4 years ago
4 years ago
3 years ago
4 years ago
3 years ago
4 years ago
3 years ago
4 years ago
4 years ago
4 years ago
4 years ago
3 years ago
4 years ago
3 years ago
4 years ago
2 years ago
4 years ago
4 years ago
4 years ago
3 years ago
4 years ago
3 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
3 years ago
4 years ago
4 years ago
3 years ago
3 years ago
3 years ago
4 years ago
4 years ago
3 years ago
3 years ago
3 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
3 years ago
4 years ago
3 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
3 years ago
4 years ago
4 years ago
3 years ago
4 years ago
3 years ago
4 years ago
3 years ago
4 years ago
4 years ago
4 years ago
3 years ago
4 years ago
3 years ago
4 years ago
4 years ago
4 years ago
3 years ago
4 years ago
3 years ago
4 years ago
3 years ago
4 years ago
4 years ago
3 years ago
4 years ago
4 years ago
3 years ago
3 years ago
4 years ago
4 years ago
3 years ago
4 years ago
4 years ago
3 years ago
4 years ago
4 years ago
4 years ago
4 years ago
3 years ago
4 years ago
4 years ago
4 years ago
4 years ago
3 years ago
4 years ago
4 years ago
4 years ago
3 years ago
4 years ago
3 years ago
4 years ago
4 years ago
3 years ago
3 years ago
4 years ago
3 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
3 years ago
4 years ago
4 years ago
4 years ago
2 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
3 years ago
4 years ago
3 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
  1. #if MYSQL_6_9 || MYSQL_6_10
  2. using Externals.MySql.Data.MySqlClient;
  3. using System;
  4. using System.Collections.Generic;
  5. using System.Data;
  6. using System.Text;
  7. using static Apewer.Source.SourceUtility;
  8. namespace Apewer.Source
  9. {
  10. /// <summary>连接 MySQL 数据库的客户端。</summary>
  11. public class MySql : DbClient
  12. {
  13. #region connection
  14. private MySqlConnection _conn = null;
  15. private string _connstr = null;
  16. /// <summary>使用连接字符串创建数据库连接实例。</summary>
  17. /// <exception cref="ArgumentNullException"></exception>
  18. /// <exception cref="ArgumentException"></exception>
  19. public MySql(IDbConnection connection, Timeout timeout = null) : base(timeout)
  20. {
  21. if (connection == null) throw new ArgumentNullException(nameof(connection), "指定的连接无效。");
  22. var conn = connection as MySqlConnection;
  23. if (conn == null) throw new ArgumentException(nameof(connection), "指定的连接不是支持的类型。");
  24. _conn = conn;
  25. _connstr = conn.ConnectionString;
  26. }
  27. /// <summary>创建实例。</summary>
  28. public MySql(string connnectionString, Timeout timeout = default) : base(timeout)
  29. {
  30. _connstr = connnectionString;
  31. }
  32. /// <summary>构建连接字符串以创建实例。</summary>
  33. /// <exception cref="ArgumentNullException"></exception>
  34. public MySql(string address, string store, string user, string pass, Timeout timeout = null) : base(timeout)
  35. {
  36. if (string.IsNullOrEmpty(address)) throw new ArgumentNullException(nameof(address));
  37. if (string.IsNullOrEmpty(store)) store = "mysql";
  38. if (string.IsNullOrEmpty(user)) user = "root";
  39. var cs = $"server={address}; database={store}; uid={user}; pwd={pass ?? ""}; ";
  40. _connstr = cs;
  41. }
  42. /// <summary>构建连接字符串以创建实例。</summary>
  43. /// <exception cref="ArgumentNullException"></exception>
  44. public MySql(string address, int port, string store, string user, string pass, Timeout timeout = null) : base(timeout)
  45. {
  46. if (string.IsNullOrEmpty(address)) throw new ArgumentNullException(nameof(address));
  47. if (port < 1 || port > 65535) throw new ArgumentOutOfRangeException(nameof(port));
  48. if (string.IsNullOrEmpty(store)) store = "mysql";
  49. if (string.IsNullOrEmpty(user)) user = "root";
  50. var cs = $"server={address}; port={port}; database={store}; uid={user}; pwd={pass ?? ""}; ";
  51. _connstr = cs;
  52. }
  53. #endregion
  54. #region override
  55. /// <summary></summary>
  56. public override string ConnectionString { get => _connstr; }
  57. /// <summary></summary>
  58. protected override IDataAdapter CreateDataAdapter(IDbCommand command) => new MySqlDataAdapter((MySqlCommand)command);
  59. /// <summary></summary>
  60. protected override IDbConnection GetConnection()
  61. {
  62. if (_conn == null) _conn = new MySqlConnection(_connstr);
  63. return _conn;
  64. }
  65. /// <summary></summary>
  66. protected override IDataParameter CreateParameter() => new MySqlParameter();
  67. /// <summary></summary>
  68. public override string[] StoreNames()
  69. {
  70. throw new NotImplementedException();
  71. }
  72. /// <summary></summary>
  73. public override string[] TableNames()
  74. {
  75. var store = StoreName();
  76. var sql = $"select table_name from information_schema.tables where table_schema='{store}' and table_type='base table'";
  77. return QueryStrings(sql);
  78. }
  79. /// <summary></summary>
  80. public override string[] ColumnNames(string tableName)
  81. {
  82. var store = StoreName();
  83. var table = TextUtility.AntiInject(tableName);
  84. var sql = $"select column_name from information_schema.columns where table_schema='{store}' and table_name='{table}'";
  85. return QueryStrings(sql);
  86. }
  87. /// <summary></summary>
  88. protected override string Initialize(TableStructure structure, string table)
  89. {
  90. if (string.IsNullOrEmpty(table)) table = structure.TableName;
  91. // 检查现存表。
  92. var exists = false;
  93. var tables = TableNames();
  94. if (tables.Length > 0)
  95. {
  96. var lower = table.ToLower();
  97. foreach (var tn in tables)
  98. {
  99. if (TextUtility.IsEmpty(tn)) continue;
  100. if (tn.ToLower() == lower)
  101. {
  102. exists = true;
  103. break;
  104. }
  105. }
  106. }
  107. if (exists)
  108. {
  109. var columns = ColumnNames(table);
  110. if (columns.Length > 0)
  111. {
  112. var lower = new List<string>(columns.Length);
  113. var added = 0;
  114. foreach (var column in columns)
  115. {
  116. if (TextUtility.IsBlank(column)) continue;
  117. lower.Add(column.ToLower());
  118. added++;
  119. }
  120. lower.Capacity = added;
  121. columns = lower.ToArray();
  122. }
  123. var sqlsb = new StringBuilder();
  124. foreach (var column in structure.Columns)
  125. {
  126. // 检查 Independent 特性。
  127. if (structure.Independent && column.Independent) continue;
  128. // 去重。
  129. var lower = column.Field.ToLower();
  130. if (columns.Contains(lower)) continue;
  131. var type = Declaration(column);
  132. if (type.IsEmpty()) return TextUtility.Merge("类型 ", column.Type.ToString(), " 不受支持。");
  133. // alter table `_record` add column `_index` bigint;
  134. sqlsb.Append("alter table `", table, "` add column ", type, "; ");
  135. }
  136. var sql = sqlsb.ToString();
  137. if (sql.IsEmpty()) return null;
  138. var execute = Execute(sql, null, false);
  139. if (!execute.Success) return execute.Message;
  140. // 设置主键。
  141. foreach (var column in structure.Columns)
  142. {
  143. if (column.PrimaryKey)
  144. {
  145. var sql_pk = $"alter table {structure.TableName} add primary key {column.Field}";
  146. Execute(sql_pk, null, false);
  147. }
  148. }
  149. return null;
  150. }
  151. else
  152. {
  153. // create table _record (`_index` bigint, `_key` varchar(255), `_text` longtext) engine=innodb default charset=utf8mb4
  154. var columns = new List<string>(structure.Columns.Length);
  155. var columnsAdded = 0;
  156. var primarykeys = new List<string>();
  157. foreach (var column in structure.Columns)
  158. {
  159. // 检查 Independent 特性。
  160. if (structure.Independent && column.Independent) continue;
  161. // 字段。
  162. var type = Declaration(column);
  163. if (type.IsEmpty()) return TextUtility.Merge("类型 ", column.Type.ToString(), " 不受支持。");
  164. if (!column.Independent)
  165. {
  166. if (column.PrimaryKey) primarykeys.Add(column.Field);
  167. if (column.Incremental) type += " auto_increment";
  168. }
  169. columns.Add(type);
  170. columnsAdded++;
  171. }
  172. columns.Capacity = columnsAdded;
  173. var joined = string.Join(", ", columns);
  174. // 设置主键。
  175. string sql;
  176. var sqlPrimaryKey = primarykeys.Count < 1 ? "" : (", primary key (" + string.Join(", ", primarykeys.ToArray()) + ")");
  177. sql = TextUtility.Merge("create table `", table, "`(", joined, sqlPrimaryKey, ") engine=innodb default charset=utf8mb4; ");
  178. var execute = Execute(sql, null, false);
  179. return execute.Success ? null : execute.Message;
  180. }
  181. }
  182. /// <summary>插入记录。返回错误信息。</summary>
  183. public override string Insert(object record, string table = null, bool adjust = true)
  184. {
  185. if (record == null) return "参数无效。";
  186. if (adjust) SourceUtility.FixProperties(record);
  187. var structure = TableStructure.Parse(record.GetType());
  188. if (structure == null) return "无法解析记录模型。";
  189. if (string.IsNullOrEmpty(table)) table = structure.TableName;
  190. if (string.IsNullOrEmpty(table)) return "表名称无效。";
  191. // 排除字段。
  192. var excluded = new List<string>();
  193. foreach (var ca in structure.Columns)
  194. {
  195. // 排除不使用 ORM 的属性。
  196. if (ca.Independent || ca.Incremental) excluded.Add(ca.Field);
  197. }
  198. var ps = structure.CreateParameters(record, Parameter, excluded);
  199. var psc = ps.Length;
  200. if (psc < 1) return "数据模型不包含字段。";
  201. var names = new List<string>(psc);
  202. var values = new List<string>(psc);
  203. foreach (var p in ps)
  204. {
  205. var pn = p.ParameterName;
  206. names.Add("`" + p + "`");
  207. values.Add("@" + p);
  208. }
  209. var ns = string.Join(", ", names);
  210. var vs = string.Join(", ", values);
  211. var sql = $"insert into {table} ({ns}) values ({vs}); ";
  212. var execute = Execute(sql, ps, false);
  213. if (execute.Success) return TextUtility.Empty;
  214. return execute.Message;
  215. }
  216. /// <summary>更新记录,实体中的 Key 属性不被更新。返回错误信息。</summary>
  217. /// <remarks>无法更新带有 Independent 特性的模型(缺少 Key 属性)。</remarks>
  218. public override string Update(IRecord record, string table = null, bool adjust = true)
  219. {
  220. if (record == null) return "参数无效。";
  221. if (adjust)
  222. {
  223. FixProperties(record);
  224. SetUpdated(record);
  225. }
  226. var structure = TableStructure.Parse(record.GetType());
  227. if (structure == null) return "无法解析记录模型。";
  228. if (structure.Independent) return "无法更新带有 Independent 特性的模型。";
  229. if (string.IsNullOrEmpty(table)) table = structure.TableName;
  230. if (string.IsNullOrEmpty(table)) return "表名称无效。";
  231. // 排除字段。
  232. var excluded = new List<string>();
  233. if (structure.Key != null) excluded.Add(structure.Key.Field);
  234. foreach (var ca in structure.Columns)
  235. {
  236. // 排除不使用 ORM 的属性、自增属性和主键属性。
  237. if (ca.Independent || ca.Incremental || ca.PrimaryKey || ca.NoUpdate) excluded.Add(ca.Field);
  238. }
  239. var ps = structure.CreateParameters(record, Parameter, excluded);
  240. var psc = ps.Length;
  241. if (psc < 1) return "数据模型不包含字段。";
  242. var items = new List<string>(psc);
  243. foreach (var p in ps)
  244. {
  245. var pn = p.ParameterName;
  246. items.Add(TextUtility.Merge("`", pn, "` = @", pn));
  247. }
  248. var key = record.Key.SafeKey();
  249. var sql = $"update {table} set {string.Join(", ", items)} where `{structure.Key.Field}`='{key}'; ";
  250. var execute = Execute(sql, ps, false);
  251. if (execute.Success) return TextUtility.Empty;
  252. return execute.Message;
  253. }
  254. /// <summary></summary>
  255. protected override string Keys(string tableName, string keyField, string flagField, long flagValue)
  256. {
  257. if (flagValue == 0) return $"select `{keyField}` from `{tableName}`";
  258. else return $"select `{keyField}` from `{tableName}` where `{flagField}` = {flagValue}";
  259. }
  260. /// <summary></summary>
  261. protected override string Get(string tableName, string keyField, string keyValue, string flagField, long flagValue)
  262. {
  263. if (flagValue == 0) return $"select * from `{tableName}` where `{keyField}` = '{keyValue}' limit 1";
  264. else return $"select * from `{tableName}` where `{keyField}` = '{keyValue}' and `{flagField}` = {flagValue} limit 1";
  265. }
  266. /// <summary></summary>
  267. protected override string List(string tableName, string flagField, long flagValue)
  268. {
  269. if (flagValue == 0) return $"select * from `{tableName}`";
  270. else return $"select * from `{tableName}` where `{flagField}` = {flagValue}";
  271. }
  272. #endregion
  273. #region special
  274. string StoreName() => Internals.TextHelper.ParseConnectionString(_connstr).GetValue("database") ?? "";
  275. /// <summary>获取所有视图的名称。</summary>
  276. public string[] ViewNames()
  277. {
  278. var store = StoreName();
  279. var sql = $"select table_name from information_schema.tables where table_schema='{store}' and table_type='view'";
  280. return QueryStrings(sql);
  281. }
  282. /// <summary>获取记录。</summary>
  283. /// <param name="model">填充的记录模型。</param>
  284. /// <param name="skip">要跳过的记录数,可用最小值为 0。</param>
  285. /// <param name="count">要获取的记录数,可用最小值为 1。</param>
  286. /// <exception cref="ArgumentNullException"></exception>
  287. /// <exception cref="ArgumentOutOfRangeException"></exception>
  288. /// <exception cref="ModelException"></exception>
  289. public T[] Range<T>(Type model, int skip, int count) where T : class, new()
  290. {
  291. if (model == null) throw new ArgumentNullException(nameof(model));
  292. if (skip < 0) throw new ArgumentOutOfRangeException(nameof(skip));
  293. if (count < 1) throw new ArgumentOutOfRangeException(nameof(count));
  294. var ts = TableStructure.Parse(model);
  295. if (ts.TableName.IsEmpty()) throw ModelException.InvalidTableName(model);
  296. var sql = $"select * from `{ts.TableName}` limit {skip}, {count}";
  297. using (var query = Query(sql))
  298. {
  299. if (!query.Success) throw new SqlException(query, sql);
  300. var array = query.Fill<T>();
  301. return array;
  302. }
  303. }
  304. /// <summary>对表添加列,返回错误信息。</summary>
  305. /// <typeparam name="T">记录类型。</typeparam>
  306. /// <param name="column">列名称。</param>
  307. /// <param name="type">字段类型。</param>
  308. /// <param name="length">字段长度,仅对 VarChar 和 NVarChar 类型有效。</param>
  309. /// <returns></returns>
  310. public string AddColumn<T>(string column, ColumnType type, int length = 0) where T : class, IRecord
  311. {
  312. var columnName = column.SafeName();
  313. if (columnName.IsEmpty()) return "列名无效。";
  314. var ta = TableAttribute.Parse(typeof(T));
  315. if (ta == null) return "无法解析记录模型。";
  316. var tableName = ta.Name;
  317. var columeType = "";
  318. switch (type)
  319. {
  320. case ColumnType.Integer:
  321. columeType = "bigint";
  322. break;
  323. case ColumnType.Float:
  324. columeType = "double";
  325. break;
  326. case ColumnType.Bytes:
  327. columeType = "longblob";
  328. break;
  329. case ColumnType.DateTime:
  330. columeType = "datetime";
  331. break;
  332. case ColumnType.VarChar:
  333. case ColumnType.NVarChar:
  334. columeType = $"varchar({length})";
  335. break;
  336. case ColumnType.VarChar191:
  337. case ColumnType.NVarChar191:
  338. columeType = "varchar(191)";
  339. break;
  340. case ColumnType.VarCharMax:
  341. case ColumnType.NVarCharMax:
  342. columeType = "varchar(max)";
  343. break;
  344. case ColumnType.Text:
  345. columeType = "longtext";
  346. break;
  347. }
  348. if (columeType.IsEmpty()) return "类型不支持。";
  349. var sql = $"alter table `{tableName}` add {columnName} {columeType}; ";
  350. var execute = Execute(sql, null, false) as Execute;
  351. var error = execute.Message;
  352. return error;
  353. }
  354. #endregion
  355. #region static
  356. /// <summary>清空与指定连接关联的连接池。</summary>
  357. public static void ClearPool(IDbConnection connection)
  358. {
  359. var special = connection as MySqlConnection;
  360. if (special != null)
  361. {
  362. try
  363. {
  364. MySqlConnection.ClearPool(special);
  365. }
  366. catch { }
  367. }
  368. }
  369. /// <summary>清空与指定连接关联的连接池。</summary>
  370. public static void ClearPool(IDbAdo instance)
  371. {
  372. var connection = instance?.Connection;
  373. if (connection != null) ClearPool(connection);
  374. }
  375. /// <summary>清空连接池。</summary>
  376. public static void ClearAllPools()
  377. {
  378. try
  379. {
  380. MySqlConnection.ClearAllPools();
  381. }
  382. catch { }
  383. }
  384. /// <summary></summary>
  385. /// <exception cref="ArgumentNullException"></exception>
  386. /// <exception cref="InvalidOperationException"></exception>
  387. internal static MySqlParameter Parameter(Parameter parameter)
  388. {
  389. if (parameter == null) throw new InvalidOperationException("参数无效。");
  390. return Parameter(parameter.Name, parameter.Type, parameter.Size, parameter.Value);
  391. }
  392. /// <summary></summary>
  393. internal static MySqlParameter Parameter(string name, ColumnType type, Int32 size, object value)
  394. {
  395. if (TextUtility.IsBlank(name)) return null;
  396. var dbtype = MySqlDbType.Int64;
  397. switch (type)
  398. {
  399. case ColumnType.Bytes:
  400. dbtype = MySqlDbType.LongBlob;
  401. break;
  402. case ColumnType.Integer:
  403. dbtype = MySqlDbType.Int64;
  404. break;
  405. case ColumnType.Float:
  406. dbtype = MySqlDbType.Double;
  407. break;
  408. case ColumnType.DateTime:
  409. dbtype = MySqlDbType.DateTime;
  410. break;
  411. case ColumnType.VarChar:
  412. case ColumnType.VarChar191:
  413. case ColumnType.VarCharMax:
  414. case ColumnType.NVarChar:
  415. case ColumnType.NVarChar191:
  416. case ColumnType.NVarCharMax:
  417. dbtype = MySqlDbType.VarChar;
  418. break;
  419. case ColumnType.Text:
  420. case ColumnType.NText:
  421. dbtype = MySqlDbType.LongText;
  422. break;
  423. default:
  424. throw new InvalidOperationException(TextUtility.Merge("类型 ", type.ToString(), " 不受支持。"));
  425. }
  426. switch (type)
  427. {
  428. case ColumnType.VarChar:
  429. case ColumnType.NVarChar:
  430. size = NumberUtility.Restrict(size, 0, 65535);
  431. break;
  432. case ColumnType.VarChar191:
  433. case ColumnType.NVarChar191:
  434. size = NumberUtility.Restrict(size, 0, 191);
  435. break;
  436. default:
  437. size = 0;
  438. break;
  439. }
  440. if (value is string && value != null && size > 0)
  441. {
  442. value = TextUtility.Left((string)value, size);
  443. }
  444. var parameter = new MySqlParameter();
  445. parameter.ParameterName = name;
  446. parameter.MySqlDbType = dbtype;
  447. parameter.Value = value ?? DBNull.Value;
  448. if (size > 0) parameter.Size = size;
  449. return parameter;
  450. }
  451. /// <summary></summary>
  452. internal static MySqlParameter Parameter(string name, MySqlDbType type, object value, Int32 size = 0)
  453. {
  454. var parameter = new MySqlParameter();
  455. parameter.ParameterName = name;
  456. parameter.MySqlDbType = type;
  457. parameter.Value = value ?? DBNull.Value;
  458. if (size > 0) parameter.Size = size;
  459. return parameter;
  460. }
  461. private static string Declaration(ColumnAttribute column)
  462. {
  463. var type = TextUtility.Empty;
  464. var length = Math.Max(0, (int)column.Length);
  465. switch (column.Type)
  466. {
  467. case ColumnType.Integer:
  468. type = "bigint";
  469. break;
  470. case ColumnType.Float:
  471. type = "double";
  472. break;
  473. case ColumnType.Bytes:
  474. type = "longblob";
  475. break;
  476. case ColumnType.DateTime:
  477. type = "datetime";
  478. break;
  479. case ColumnType.VarChar:
  480. type = TextUtility.Merge("varchar(", Math.Max(65535, length).ToString(), ")");
  481. break;
  482. case ColumnType.VarChar191:
  483. type = TextUtility.Merge("varchar(191)");
  484. break;
  485. case ColumnType.VarCharMax:
  486. type = TextUtility.Merge("varchar(max)");
  487. break;
  488. case ColumnType.Text:
  489. type = TextUtility.Merge("longtext");
  490. break;
  491. case ColumnType.NVarChar:
  492. type = TextUtility.Merge("varchar(", Math.Min(65535, length).ToString(), ")");
  493. break;
  494. case ColumnType.NVarChar191:
  495. type = TextUtility.Merge("varchar(191)");
  496. break;
  497. case ColumnType.NVarCharMax:
  498. type = TextUtility.Merge("varchar(65535)");
  499. break;
  500. case ColumnType.NText:
  501. type = TextUtility.Merge("longtext");
  502. break;
  503. default:
  504. return TextUtility.Empty;
  505. }
  506. return TextUtility.Merge("`", (string)column.Field, "` ", type);
  507. }
  508. /// <summary>获取每个数据库中,每个表的容量,单位为字节。</summary>
  509. public static Dictionary<string, Dictionary<string, long>> GetTablesCapacity(MySql source)
  510. {
  511. var result = new Dictionary<string, Dictionary<string, long>>();
  512. if (source != null && source.Connect().IsEmpty())
  513. {
  514. var sql = "select `table_schema`, `table_name`, `engine`, `data_length`, `index_length` from `information_schema`.tables order by `table_schema`, `table_name`";
  515. using (var query = (Query)source.Query(sql))
  516. {
  517. for (var r = 0; r < query.Rows; r++)
  518. {
  519. var store = query.Text(r, "table_schema");
  520. var table = query.Text(r, "table_name");
  521. var engine = query.Text(r, "engine");
  522. var capacity = NumberUtility.Int64(query.Text(r, "data_length")) + NumberUtility.Int64(query.Text(r, "index_length"));
  523. if (store == "mysql") continue;
  524. if (store == "information_schema") continue;
  525. if (store == "performance_schema") continue;
  526. if (engine != "MyISAM" && engine != "InnoDB") continue;
  527. if (!result.ContainsKey(store)) result.Add(store, new Dictionary<string, long>());
  528. if (!result[store].ContainsKey(table)) result[store].Add(table, 0L);
  529. result[store][table] = capacity;
  530. }
  531. }
  532. }
  533. return result;
  534. }
  535. #endregion
  536. }
  537. }
  538. #endif