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.

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