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.

526 lines
20 KiB

4 years ago
4 years ago
3 years ago
3 years ago
4 years ago
3 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
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
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
3 years ago
4 years ago
3 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
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
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
3 years ago
4 years ago
3 years ago
4 years ago
3 years ago
3 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
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
4 years ago
3 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
3 years ago
4 years ago
3 years ago
4 years ago
3 years ago
  1. using Apewer.Internals;
  2. #if NETFRAMEWORK
  3. using System;
  4. using System.Collections.Generic;
  5. using System.Data;
  6. using System.Data.OleDb;
  7. using System.IO;
  8. using System.Text;
  9. using static Apewer.Source.SourceUtility;
  10. #endif
  11. namespace Apewer.Source
  12. {
  13. /// <summary>连接 Access 数据库的客户端。</summary>
  14. public abstract partial class Access
  15. {
  16. /// <summary>尝试解析 Access 文件的密码。</summary>
  17. /// <remarks>当操作系统启用随机化内存分配时,此方法可能会产生异常。</remarks>
  18. public static string[] ParsePassword(string path) => AccessHelper.GetPassword(path);
  19. }
  20. #if NETFRAMEWORK
  21. public abstract partial class Access : DbClient
  22. {
  23. #region 连接
  24. OleDbConnection _conn = null;
  25. string _connstr = null;
  26. /// <summary>构造函数。</summary>
  27. internal Access(string connectrionString, Timeout timeout = null) : base(timeout)
  28. {
  29. _connstr = connectrionString;
  30. }
  31. /// <summary>使用连接字符串创建数据库连接实例。</summary>
  32. /// <exception cref="ArgumentNullException"></exception>
  33. /// <exception cref="ArgumentException"></exception>
  34. public Access(IDbConnection connection, Timeout timeout = null) : base(timeout)
  35. {
  36. if (connection == null) throw new ArgumentNullException(nameof(connection), "指定的连接无效。");
  37. var conn = connection as OleDbConnection;
  38. if (conn == null) throw new ArgumentException(nameof(connection), "指定的连接不是支持的类型。");
  39. _conn = conn;
  40. _connstr = conn.ConnectionString;
  41. }
  42. #endregion
  43. #region public
  44. /// <summary></summary>
  45. public override string[] ColumnNames(string tableName)
  46. {
  47. if (_conn == null) _conn = new OleDbConnection(_connstr);
  48. // OleDbConnection.GetOleDbSchemaTable(OleDbSchemaGuid.Tables, new object[] { null, null, null, "TABLE" });
  49. var table = _conn.GetOleDbSchemaTable(OleDbSchemaGuid.Columns, new object[] { null, null, tableName.ToString(), null });
  50. using (var query = new Query(table))
  51. {
  52. var names = new ArrayBuilder<string>();
  53. for (var i = 0; i < query.Rows; i++)
  54. {
  55. names.Add(query.Text(i, "COLUMN_NAME"));
  56. }
  57. return names.Export();
  58. }
  59. }
  60. /// <summary></summary>
  61. public override string[] StoreNames() => throw new InvalidOperationException();
  62. /// <summary></summary>
  63. public override string[] TableNames() => QueryStrings("select name from msysobjects where type=1 and flags = 0");
  64. /// <summary></summary>
  65. public override string Insert(object record, string table = null, bool adjust = true)
  66. {
  67. if (record == null) return "参数无效。";
  68. if (adjust) FixProperties(record);
  69. var structure = TableStructure.Parse(record.GetType());
  70. if (structure == null) return "无法解析记录模型。";
  71. if (string.IsNullOrEmpty(table)) table = structure.TableName;
  72. if (string.IsNullOrEmpty(table)) return "表名称无效。";
  73. // 排除字段。
  74. var excluded = new List<string>();
  75. foreach (var ca in structure.Columns)
  76. {
  77. // 排除不使用 ORM 的属性。
  78. if (ca.Independent || ca.Incremental) excluded.Add(ca.Field);
  79. }
  80. var ps = structure.CreateParameters(record, Parameter, excluded);
  81. var psc = ps.Length;
  82. if (psc < 1) return "数据模型不包含字段。";
  83. var names = new List<string>(psc);
  84. var values = new List<string>(psc);
  85. foreach (var column in ps)
  86. {
  87. //names.Add(TextGenerator.Merge("[", column, "]"));
  88. names.Add(TextUtility.Merge(column));
  89. values.Add("@" + column);
  90. }
  91. var sb = new StringBuilder();
  92. sb.Append("insert into ", table, "(", string.Join(", ", names.ToArray()), ") ");
  93. sb.Append("values(", string.Join(", ", values.ToArray()), "); ");
  94. var sql = sb.ToString();
  95. var execute = Execute(sql, ps, false);
  96. if (execute.Success) return TextUtility.Empty;
  97. return execute.Message;
  98. }
  99. /// <summary></summary>
  100. public override string Update(IRecord record, string table = null, bool adjust = true)
  101. {
  102. if (record == null) return "参数无效。";
  103. if (adjust)
  104. {
  105. FixProperties(record);
  106. SetUpdated(record);
  107. }
  108. var structure = TableStructure.Parse(record.GetType());
  109. if (structure == null) return "无法解析记录模型。";
  110. if (structure.Independent) return "无法更新带有 Independent 特性的模型。";
  111. if (string.IsNullOrEmpty(table)) table = structure.TableName;
  112. if (string.IsNullOrEmpty(table)) return "表名称无效。";
  113. // 排除字段。
  114. var excluded = new List<string>();
  115. if (structure.Key != null) excluded.Add(structure.Key.Field);
  116. foreach (var ca in structure.Columns)
  117. {
  118. // 排除不使用 ORM 的属性、自增属性和主键属性。
  119. if (ca.Independent || ca.Incremental || ca.PrimaryKey || ca.NoUpdate) excluded.Add(ca.Field);
  120. }
  121. var ps = structure.CreateParameters(record, Parameter, excluded);
  122. var psc = ps.Length;
  123. if (psc < 1) return "数据模型不包含字段。";
  124. var items = new List<string>();
  125. foreach (var p in ps)
  126. {
  127. var pn = p.ParameterName;
  128. items.Add(TextUtility.Merge("[", pn, "] = @", pn));
  129. }
  130. var key = record.Key.SafeKey();
  131. var sql = $"update {table} set {string.Join(", ", items.ToArray())} where [{structure.Key.Field}]='{key}'";
  132. var execute = Execute(sql, ps, false);
  133. if (execute.Success) return TextUtility.Empty;
  134. return execute.Message;
  135. }
  136. #endregion
  137. #region protected
  138. /// <summary></summary>
  139. public override string ConnectionString { get => _connstr; }
  140. /// <summary></summary>
  141. protected override IDbConnection GetConnection()
  142. {
  143. if (_conn == null) _conn = new OleDbConnection(_connstr);
  144. return _conn;
  145. }
  146. /// <summary></summary>
  147. protected override IDataAdapter CreateDataAdapter(IDbCommand command) => new OleDbDataAdapter((OleDbCommand)command);
  148. /// <summary></summary>
  149. protected override IDataParameter CreateParameter() => new OleDbParameter();
  150. /// <summary></summary>
  151. protected override string Initialize(TableStructure structure, string table)
  152. {
  153. var model = structure.Model;
  154. if (table.IsEmpty()) table = structure.TableName;
  155. // 检查现存表。
  156. var exists = false;
  157. var tables = TableNames();
  158. if (tables.Length > 0)
  159. {
  160. var lower = table.ToLower();
  161. foreach (var tn in tables)
  162. {
  163. if (TextUtility.IsEmpty(tn)) continue;
  164. if (tn.ToLower() == lower)
  165. {
  166. exists = true;
  167. break;
  168. }
  169. }
  170. }
  171. if (exists)
  172. {
  173. // 获取已存在的列名。
  174. var columns = ColumnNames(table);
  175. if (columns.Length > 0)
  176. {
  177. var lower = new List<string>();
  178. foreach (var column in columns)
  179. {
  180. if (TextUtility.IsBlank(column)) continue;
  181. lower.Add(column.ToLower());
  182. }
  183. columns = lower.ToArray();
  184. }
  185. // 增加列。
  186. foreach (var column in structure.Columns)
  187. {
  188. // 检查 Independent 特性。
  189. if (structure.Independent && column.Independent) continue;
  190. // 去重。
  191. var lower = column.Field.ToLower();
  192. if (columns.Contains(lower)) continue;
  193. var type = Declaration(column);
  194. if (string.IsNullOrEmpty(type)) return TextUtility.Merge("类型 ", column.Type.ToString(), " 不受支持。");
  195. var sql = TextUtility.Merge("alter table [", table, "] add ", type, "; ");
  196. var execute = Execute(sql, null, false);
  197. if (execute.Success == false) return execute.Message;
  198. }
  199. return TextUtility.Empty;
  200. }
  201. else
  202. {
  203. var sqlcolumns = new List<string>();
  204. foreach (var column in structure.Columns)
  205. {
  206. // 检查 Independent 特性。
  207. if (structure.Independent && column.Independent) continue;
  208. var type = Declaration(column);
  209. if (!column.Independent)
  210. {
  211. if (column.PrimaryKey) type = type + " primary key";
  212. if (column.Incremental) type += " identity";
  213. }
  214. if (string.IsNullOrEmpty(type)) return TextUtility.Merge("类型 ", column.Type.ToString(), " 不受支持。");
  215. sqlcolumns.Add(type);
  216. }
  217. if (sqlcolumns.Count < 1) return $"无法对类型 {model.FullName} 创建表:没有定义任何字段。";
  218. var sql = TextUtility.Merge("create table [", table, "](", string.Join(", ", sqlcolumns.ToArray()), "); ");
  219. var execute = Execute(sql, null, false);
  220. if (execute.Success) return TextUtility.Empty;
  221. return execute.Message;
  222. }
  223. }
  224. /// <summary></summary>
  225. protected override string Keys(string tableName, string keyField, string flagField, long flagValue)
  226. {
  227. if (flagValue == 0) return $"select [{keyField}] from [{tableName}]";
  228. else return $"select [{keyField}] from [{tableName}] where [{flagField}] = {flagValue}";
  229. }
  230. /// <summary></summary>
  231. protected override string Get(string tableName, string keyField, string keyValue, string flagField, long flagValue)
  232. {
  233. if (flagValue == 0) return $"select top 1 * from [{tableName}] where [{keyField}] = '{keyValue}'";
  234. else return $"select top 1 * from [{tableName}] where [{keyField}] = '{keyValue}' and [{flagField}] = {flagValue}";
  235. }
  236. /// <summary></summary>
  237. protected override string List(string tableName, string flagField, long flagValue)
  238. {
  239. if (flagValue == 0) return $"select * from [{tableName}]";
  240. else return $"select * from [{tableName}] where [{flagField}] = {flagValue}";
  241. }
  242. #endregion
  243. #region Parameter
  244. /// <summary>创建参数。</summary>
  245. /// <exception cref="ArgumentNullException"></exception>
  246. /// <exception cref="InvalidOperationException"></exception>
  247. static OleDbParameter Parameter(Parameter parameter)
  248. {
  249. if (parameter == null) throw new InvalidOperationException("参数无效。");
  250. return Parameter(parameter.Name, parameter.Type, parameter.Size, parameter.Value);
  251. }
  252. /// <summary>创建参数。</summary>
  253. public static OleDbParameter Parameter(string name, ColumnType type, int size, object value)
  254. {
  255. var vname = TextUtility.Trim(name);
  256. if (TextUtility.IsBlank(vname)) return null;
  257. var vtype = OleDbType.BigInt;
  258. switch (type)
  259. {
  260. case ColumnType.Boolean:
  261. vtype = OleDbType.Boolean;
  262. break;
  263. case ColumnType.Bytes:
  264. vtype = OleDbType.LongVarBinary;
  265. break;
  266. case ColumnType.Integer:
  267. vtype = OleDbType.Integer;
  268. break;
  269. case ColumnType.Float:
  270. vtype = OleDbType.Double;
  271. break;
  272. case ColumnType.DateTime:
  273. vtype = OleDbType.Date;
  274. break;
  275. case ColumnType.VarChar:
  276. case ColumnType.VarChar191:
  277. case ColumnType.VarCharMax:
  278. vtype = OleDbType.VarChar;
  279. break;
  280. case ColumnType.NVarChar:
  281. case ColumnType.NVarChar191:
  282. case ColumnType.NVarCharMax:
  283. vtype = OleDbType.VarWChar;
  284. break;
  285. case ColumnType.Text:
  286. vtype = OleDbType.LongVarChar;
  287. break;
  288. case ColumnType.NText:
  289. vtype = OleDbType.LongVarWChar;
  290. break;
  291. default:
  292. throw new InvalidOperationException(TextUtility.Merge("类型 ", type.ToString(), " 不受支持。"));
  293. }
  294. var vsize = size;
  295. switch (type)
  296. {
  297. case ColumnType.VarChar:
  298. vsize = NumberUtility.Restrict(vsize, 0, 8000);
  299. break;
  300. case ColumnType.NVarChar:
  301. vsize = NumberUtility.Restrict(vsize, 0, 4000);
  302. break;
  303. case ColumnType.VarChar191:
  304. case ColumnType.NVarChar191:
  305. vsize = NumberUtility.Restrict(vsize, 0, 191);
  306. break;
  307. default:
  308. vsize = 0;
  309. break;
  310. }
  311. var vvalue = value;
  312. if (vvalue is string && vvalue != null && vsize > 0)
  313. {
  314. vvalue = TextUtility.Left((string)vvalue, vsize);
  315. }
  316. var parameter = new OleDbParameter();
  317. parameter.ParameterName = vname;
  318. parameter.OleDbType = vtype;
  319. parameter.Value = vvalue;
  320. if (vsize > 0) parameter.Size = vsize;
  321. return parameter;
  322. }
  323. /// <summary>创建参数。</summary>
  324. public static IDbDataParameter CreateParameter(String field, DbType type, Int32 size, Object value)
  325. {
  326. var p = new OleDbParameter();
  327. p.ParameterName = field;
  328. p.DbType = type;
  329. p.Size = size;
  330. p.Value = value;
  331. return p;
  332. }
  333. /// <summary>创建参数。</summary>
  334. public static IDbDataParameter CreateParameter(String field, DbType type, Object value)
  335. {
  336. var p = new OleDbParameter();
  337. p.ParameterName = field;
  338. p.DbType = type;
  339. p.Value = value;
  340. return p;
  341. }
  342. #endregion
  343. #region protected
  344. /// <summary>获取或设置连接字符串。</summary>
  345. /// <exception cref="FileNotFoundException"></exception>
  346. internal protected static string GenerateCS(string provider, string path, string pass, string jo)
  347. {
  348. if (!File.Exists(path)) throw new FileNotFoundException("文件不存在。", path);
  349. var sb = new StringBuilder();
  350. sb.Append("provider=", provider, "; ");
  351. if (!string.IsNullOrEmpty(path)) sb.Append("data source=", path, "; ");
  352. if (string.IsNullOrEmpty(pass)) sb.Append("persist security info=false; ");
  353. else sb.Append("jet oledb:database password=\"", pass, "\"; ");
  354. // Microsoft Access Workgroup Information
  355. if (!string.IsNullOrEmpty(jo)) sb.Append("jet oledb:system database=", jo, "; ");
  356. return sb.ToString();
  357. }
  358. #endregion
  359. #region static
  360. static string Declaration(ColumnAttribute column)
  361. {
  362. var type = TextUtility.Empty;
  363. var vcolumn = column;
  364. var length = Math.Max(0, vcolumn.Length);
  365. switch (vcolumn.Type)
  366. {
  367. case ColumnType.Boolean:
  368. type = "bit";
  369. break;
  370. case ColumnType.Integer:
  371. type = "money";
  372. break;
  373. case ColumnType.Float:
  374. type = "float";
  375. break;
  376. case ColumnType.Bytes:
  377. type = "binary";
  378. break;
  379. case ColumnType.DateTime:
  380. type = "datetime";
  381. break;
  382. case ColumnType.VarChar:
  383. type = TextUtility.Merge("varchar(", Math.Min(8000, length).ToString(), ")");
  384. break;
  385. case ColumnType.VarChar191:
  386. type = TextUtility.Merge("varchar(191)");
  387. break;
  388. case ColumnType.VarCharMax:
  389. type = TextUtility.Merge("varchar(max)");
  390. break;
  391. case ColumnType.Text:
  392. type = TextUtility.Merge("text");
  393. break;
  394. case ColumnType.NVarChar:
  395. type = TextUtility.Merge("nvarchar(", Math.Min(4000, length).ToString(), ")");
  396. break;
  397. case ColumnType.NVarChar191:
  398. type = TextUtility.Merge("nvarchar(191)");
  399. break;
  400. case ColumnType.NVarCharMax:
  401. type = TextUtility.Merge("nvarchar(max)");
  402. break;
  403. case ColumnType.NText:
  404. type = TextUtility.Merge("ntext");
  405. break;
  406. default:
  407. return TextUtility.Empty;
  408. }
  409. return TextUtility.Merge("[", vcolumn.Field, "] ", type);
  410. }
  411. #endregion
  412. }
  413. /// <summary>使用 Microsoft.Jet.OLEDB.4.0 访问 Access 97 - 2003 数据库文件。</summary>
  414. public class AccessJet4 : Access
  415. {
  416. const string JetOleDB4 = "microsoft.jet.oledb.4.0";
  417. /// <summary>创建 Access 类的新实例。</summary>
  418. /// <exception cref="FileNotFoundException"></exception>
  419. public AccessJet4(string path, string pass = null, string jo = null, Timeout timeout = null)
  420. : base(GenerateCS(JetOleDB4, path, pass, jo), timeout) { }
  421. /// <summary>使用连接字符串创建数据库连接实例。</summary>
  422. /// <exception cref="ArgumentNullException"></exception>
  423. /// <exception cref="ArgumentException"></exception>
  424. public AccessJet4(IDbConnection connection, Timeout timeout = null) : base(connection, timeout) { }
  425. }
  426. /// <summary>使用 Microsoft.ACE.OLEDB.12.0 访问 Access 2007 数据库文件。</summary>
  427. public class AccessAce12 : Access
  428. {
  429. const string AceOleDB12 = "microsoft.ace.oledb.12.0";
  430. /// <summary>创建 Access 类的新实例。</summary>
  431. /// <exception cref="FileNotFoundException"></exception>
  432. public AccessAce12(string path, string pass = null, string jo = null, Timeout timeout = null)
  433. : base(GenerateCS(AceOleDB12, path, pass, jo), timeout) { }
  434. /// <summary>使用连接字符串创建数据库连接实例。</summary>
  435. /// <exception cref="ArgumentNullException"></exception>
  436. /// <exception cref="ArgumentException"></exception>
  437. public AccessAce12(IDbConnection connection, Timeout timeout = null) : base(connection, timeout) { }
  438. }
  439. #endif
  440. }