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.

678 lines
26 KiB

3 years ago
3 years ago
4 years ago
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
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
4 years ago
4 years ago
3 years ago
4 years ago
3 years ago
3 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
3 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
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
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
3 years ago
4 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
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
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
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
3 years ago
4 years ago
4 years ago
4 years ago
3 years ago
3 years ago
4 years ago
3 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
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
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
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Data;
  4. using System.Data.SqlClient;
  5. using System.Text;
  6. using static Apewer.Source.SourceUtility;
  7. #if NETFRAMEWORK
  8. using System.Data.Sql;
  9. #endif
  10. namespace Apewer.Source
  11. {
  12. /// <summary>连接 SQL Server 数据库的客户端。</summary>
  13. [Serializable]
  14. public class SqlClient : DbClient
  15. {
  16. #region connection
  17. private SqlConnection _conn = null;
  18. private string _connstr = "";
  19. /// <summary>使用连接字符串创建数据库连接实例。</summary>
  20. /// <exception cref="ArgumentNullException"></exception>
  21. /// <exception cref="ArgumentException"></exception>
  22. public SqlClient(IDbConnection connection, Timeout timeout = null) : base(timeout)
  23. {
  24. if (connection == null) throw new ArgumentNullException(nameof(connection), "指定的连接无效。");
  25. var conn = connection as SqlConnection;
  26. if (conn == null) throw new ArgumentException(nameof(connection), "指定的连接不是支持的类型。");
  27. _conn = conn;
  28. _connstr = conn.ConnectionString;
  29. }
  30. /// <summary>使用连接字符串创建数据库连接实例。</summary>
  31. public SqlClient(string connectionString, Timeout timeout = null) : base(timeout)
  32. {
  33. _connstr = connectionString ?? "";
  34. }
  35. /// <summary>使用连接凭据创建数据库连接实例。</summary>
  36. /// <exception cref="ArgumentNullException"></exception>
  37. public SqlClient(string address, string store, string user, string pass, Timeout timeout = null) : base(timeout)
  38. {
  39. if (address.IsEmpty()) throw new ArgumentNullException(nameof(address));
  40. if (store.IsEmpty()) store = "master";
  41. var cs = $"data source = {address ?? ""}; initial catalog = {store}; ";
  42. if (string.IsNullOrEmpty(user)) cs += "integrated security = sspi; ";
  43. else
  44. {
  45. cs += $"user id = {user}; ";
  46. if (!string.IsNullOrEmpty(pass)) cs += $"password = {pass}; ";
  47. }
  48. if (timeout != null) cs += $"connection timeout = {timeout.Connect}; ";
  49. _connstr = cs;
  50. }
  51. /// <summary></summary>
  52. protected override void ClearPool(bool all = false)
  53. {
  54. var conn = Connection as SqlConnection;
  55. if (conn != null) SqlConnection.ClearPool(conn);
  56. if (all) SqlConnection.ClearAllPools();
  57. }
  58. #endregion
  59. #region override
  60. /// <summary>查询数据库中的所有表名。</summary>
  61. public override string[] TableNames() => QueryStrings("select [name] from [sysobjects] where [type] = 'u' order by [name]");
  62. /// <summary>查询数据库实例中的所有数据库名。</summary>
  63. public override string[] StoreNames() => QueryStrings("select [name] from [master]..[sysdatabases] order by [name]", new string[] { "master", "model", "msdb", "tempdb" });
  64. /// <summary>查询表中的所有列名。</summary>
  65. public override string[] ColumnNames(string tableName) => QueryStrings($"select [name] from [syscolumns] where [id] = object_id('{TextUtility.AntiInject(tableName)}')");
  66. /// <summary>创建表,当表不存在时创建表,当现存表中缺少模型中属性对应的列时增加列。成功时返回空字符串,发生异常时返回异常信息。</summary>
  67. protected override string Initialize(TableStructure structure, string table)
  68. {
  69. var model = structure.Model;
  70. if (string.IsNullOrEmpty(table)) table = structure.TableName;
  71. // 检查现存表。
  72. var exists = false;
  73. var tables = TableNames();
  74. if (tables.Length > 0)
  75. {
  76. var lower = table.ToLower();
  77. foreach (var tn in tables)
  78. {
  79. if (TextUtility.IsEmpty(tn)) continue;
  80. if (tn.ToLower() == lower)
  81. {
  82. exists = true;
  83. break;
  84. }
  85. }
  86. }
  87. if (exists)
  88. {
  89. // 获取已存在的列名。
  90. var columns = ColumnNames(table);
  91. if (columns.Length > 0)
  92. {
  93. var lower = new List<string>();
  94. foreach (var column in columns)
  95. {
  96. if (TextUtility.IsBlank(column)) continue;
  97. lower.Add(column.ToLower());
  98. }
  99. columns = lower.ToArray();
  100. }
  101. // 增加列。
  102. foreach (var column in structure.Columns)
  103. {
  104. // 检查 Independent 特性。
  105. if (structure.Independent && column.Independent) continue;
  106. // 去重。
  107. var lower = column.Field.ToLower();
  108. if (columns.Contains(lower)) continue;
  109. var type = Declaration(column);
  110. if (string.IsNullOrEmpty(type)) return TextUtility.Merge("类型 ", column.Type.ToString(), " 不受支持。");
  111. var sql = TextUtility.Merge("alter table [", table, "] add ", type, "; ");
  112. var execute = Execute(sql, null, false);
  113. if (execute.Success == false) return execute.Message;
  114. }
  115. return TextUtility.Empty;
  116. }
  117. else
  118. {
  119. var sqlcolumns = new List<string>();
  120. foreach (var column in structure.Columns)
  121. {
  122. // 检查 Independent 特性。
  123. if (structure.Independent && column.Independent) continue;
  124. var type = Declaration(column);
  125. if (!column.Independent)
  126. {
  127. if (column.PrimaryKey) type = type + " primary key";
  128. if (column.Incremental) type += " identity";
  129. }
  130. if (string.IsNullOrEmpty(type)) return TextUtility.Merge("类型 ", column.Type.ToString(), " 不受支持。");
  131. sqlcolumns.Add(type);
  132. }
  133. if (sqlcolumns.Count < 1) return $"无法对类型 {model.FullName} 创建表:没有定义任何字段。";
  134. var sql = TextUtility.Merge("create table [", table, "](", string.Join(", ", sqlcolumns.ToArray()), "); ");
  135. var execute = Execute(sql, null, false);
  136. if (execute.Success) return TextUtility.Empty;
  137. return execute.Message;
  138. }
  139. }
  140. /// <summary>插入记录。返回错误信息。</summary>
  141. public override string Insert(object record, string table = null, bool adjust = true)
  142. {
  143. if (record == null) return "参数无效。";
  144. if (adjust) FixProperties(record);
  145. var structure = TableStructure.Parse(record.GetType());
  146. if (structure == null) return "无法解析记录模型。";
  147. if (string.IsNullOrEmpty(table)) table = structure.TableName;
  148. if (string.IsNullOrEmpty(table)) return "表名称无效。";
  149. // 排除字段。
  150. var excluded = new List<string>();
  151. foreach (var ca in structure.Columns)
  152. {
  153. // 排除不使用 ORM 的属性。
  154. if (ca.Independent || ca.Incremental) excluded.Add(ca.Field);
  155. }
  156. var ps = structure.CreateParameters(record, Parameter, excluded);
  157. var psc = ps.Length;
  158. if (psc < 1) return "数据模型不包含字段。";
  159. var names = new List<string>(psc);
  160. var values = new List<string>(psc);
  161. foreach (var column in ps)
  162. {
  163. //names.Add(TextGenerator.Merge("[", column, "]"));
  164. names.Add(TextUtility.Merge(column));
  165. values.Add("@" + column);
  166. }
  167. var sb = new StringBuilder();
  168. sb.Append("insert into ", table, "(", string.Join(", ", names.ToArray()), ") ");
  169. sb.Append("values(", string.Join(", ", values.ToArray()), "); ");
  170. var sql = sb.ToString();
  171. var execute = Execute(sql, ps, false);
  172. if (execute.Success) return TextUtility.Empty;
  173. return execute.Message;
  174. }
  175. /// <summary>更新记录,实体中的 Key 属性不被更新。返回错误信息。</summary>
  176. /// <remarks>不更新带有 Independent 特性的模型(缺少 Key 属性)。</remarks>
  177. public override string Update(IRecord record, string table = null, bool adjust = true)
  178. {
  179. if (record == null) return "参数无效。";
  180. if (adjust)
  181. {
  182. FixProperties(record);
  183. SetUpdated(record);
  184. }
  185. var structure = TableStructure.Parse(record.GetType());
  186. if (structure == null) return "无法解析记录模型。";
  187. if (structure.Independent) return "无法更新带有 Independent 特性的模型。";
  188. if (string.IsNullOrEmpty(table)) table = structure.TableName;
  189. if (string.IsNullOrEmpty(table)) return "表名称无效。";
  190. // 排除字段。
  191. var excluded = new List<string>();
  192. if (structure.Key != null) excluded.Add(structure.Key.Field);
  193. foreach (var ca in structure.Columns)
  194. {
  195. // 排除不使用 ORM 的属性、自增属性和主键属性。
  196. if (ca.Independent || ca.Incremental || ca.PrimaryKey || ca.NoUpdate) 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 items = new List<string>();
  202. foreach (var p in ps)
  203. {
  204. var pn = p.ParameterName;
  205. items.Add(TextUtility.Merge("[", pn, "] = @", pn));
  206. }
  207. var key = record.Key.SafeKey();
  208. var sql = $"update {table} set {string.Join(", ", items.ToArray())} where [{structure.Key.Field}]='{key}'";
  209. var execute = Execute(sql, ps, false);
  210. if (execute.Success) return TextUtility.Empty;
  211. return execute.Message;
  212. }
  213. /// <summary></summary>
  214. public override string ConnectionString => _connstr;
  215. /// <summary></summary>
  216. protected override IDataAdapter CreateDataAdapter(IDbCommand command) => new SqlDataAdapter((SqlCommand)command);
  217. /// <summary></summary>
  218. protected override IDbConnection GetConnection()
  219. {
  220. if (_conn == null) _conn = new SqlConnection(_connstr);
  221. return _conn;
  222. }
  223. /// <summary></summary>
  224. protected override IDataParameter CreateParameter() => new SqlParameter();
  225. /// <summary></summary>
  226. protected override string Keys(string tableName, string keyField, string flagField, long flagValue)
  227. {
  228. if (flagValue == 0) return $"select [{keyField}] from [{tableName}]";
  229. else return $"select [{keyField}] from [{tableName}] where [{flagField}] = {flagValue}";
  230. }
  231. /// <summary></summary>
  232. protected override string Get(string tableName, string keyField, string keyValue, string flagField, long flagValue)
  233. {
  234. if (flagValue == 0) return $"select top 1 * from [{tableName}] where [{keyField}] = '{keyValue}'";
  235. else return $"select top 1 * from [{tableName}] where [{keyField}] = '{keyValue}' and [{flagField}] = {flagValue}";
  236. }
  237. /// <summary></summary>
  238. protected override string List(string tableName, string flagField, long flagValue)
  239. {
  240. if (flagValue == 0) return $"select * from [{tableName}]";
  241. else return $"select * from [{tableName}] where [{flagField}] = {flagValue}";
  242. }
  243. #endregion
  244. #region special
  245. /// <summary>获取列信息。</summary>
  246. public ColumnInfo[] ColumnsInfo(string tableName)
  247. {
  248. if (tableName.IsEmpty()) throw new ArgumentNullException(nameof(tableName));
  249. var sql = $"select name, xtype, length from syscolumns where id = object_id('{tableName}') ";
  250. using (var query = Query(sql))
  251. {
  252. var ab = new ArrayBuilder<ColumnInfo>();
  253. for (var i = 0; i < query.Rows; i++)
  254. {
  255. var info = new ColumnInfo();
  256. info.Name = query.Text(i, "name");
  257. info.Type = XType(query.Int32(i, "xtype"));
  258. info.Length = query.Int32(i, "length");
  259. ab.Add(info);
  260. }
  261. return ab.Export();
  262. }
  263. }
  264. /// <summary>批量插入,必须在 DataTable 中指定表名,或指定 tableName 参数。</summary>
  265. /// <exception cref="ArgumentNullException"></exception>
  266. /// <exception cref="ArgumentException"></exception>
  267. /// <exception cref="Exception"></exception>
  268. public void BulkCopy(DataTable table, string tableName = null)
  269. {
  270. // 检查 table 参数。
  271. if (table == null) throw new ArgumentNullException(nameof(table));
  272. if (table.Rows.Count < 1) return;
  273. // 检查 tableName 参数。
  274. if (tableName.IsEmpty()) tableName = table.TableName;
  275. if (tableName.IsEmpty()) throw new ArgumentException("未指定表名。");
  276. // 连接数据库。
  277. var connect = Connect();
  278. if (connect.NotEmpty()) throw new Exception(connect);
  279. // 准备参数。
  280. var options = SqlBulkCopyOptions.Default;
  281. var trans = Transaction as SqlTransaction;
  282. if (trans == null) options |= SqlBulkCopyOptions.UseInternalTransaction;
  283. // 批量插入。
  284. var bc = null as SqlBulkCopy;
  285. try
  286. {
  287. bc = new SqlBulkCopy((SqlConnection)Connection, options, trans);
  288. bc.DestinationTableName = tableName;
  289. bc.BatchSize = table.Rows.Count;
  290. bc.WriteToServer(table);
  291. try { bc.Close(); } catch { }
  292. }
  293. catch (Exception ex)
  294. {
  295. try { bc.Close(); } catch { }
  296. throw ex;
  297. }
  298. }
  299. /// <summary>创建数据库,返回错误信息。</summary>
  300. /// <param name="name">数据库名称。</param>
  301. /// <param name="logMaxSizeMB">日志文件的最大 MB 数,指定非正数将不限制增长。</param>
  302. /// <returns>成功时候返回 NULL 值,失败时返回错误信息。</returns>
  303. public string CreateStore(string name, int logMaxSizeMB = 1024)
  304. {
  305. var store = name.Escape().ToTrim();
  306. if (store.IsEmpty()) return "创建失败:未指定数据库名称。";
  307. if (ConnectionString.IsEmpty()) return "创建失败:未指定数据库连接方式。";
  308. using (var source = new SqlClient(ConnectionString))
  309. {
  310. var connect = source.Connect();
  311. if (connect.NotEmpty()) return "创建失败:" + connect;
  312. var schema = source.Cell("select default_schema_name from sys.database_principals where type = 'S' and name=user_name()");
  313. if (schema.IsEmpty()) return "创建失败:无法获取默认模式名称。";
  314. var refPath = source.Cell(@"select f.physical_name path from sys.filegroups g left join sys.database_files f on f.data_space_id = g.data_space_id where g.name = 'PRIMARY' and g.type = 'FG' and g.is_default = 1 and g.filegroup_guid is null");
  315. if (refPath.IsEmpty()) return "创建失败:无法获取文件路径。";
  316. var win = refPath.Substring(1, 2) == ":\\";
  317. var nix = refPath.StartsWith("/");
  318. if (!win && !nix) return "创建失败:暂不支持该服务器。";
  319. var mdfPath = store + ".mdf";
  320. var ldfPath = store + ".ldf";
  321. if (win)
  322. {
  323. var refSplit = refPath.Split('\\');
  324. var dir = string.Join("\\", refSplit, 0, refSplit.Length - 1);
  325. mdfPath = dir + "\\" + mdfPath;
  326. ldfPath = dir + "\\" + ldfPath;
  327. }
  328. if (nix)
  329. {
  330. var refSplit = refPath.Split('/');
  331. var dir = string.Join("/", refSplit, 0, refSplit.Length - 1);
  332. mdfPath = dir + "/" + mdfPath;
  333. ldfPath = dir + "/" + ldfPath;
  334. }
  335. // 创建库。
  336. var maxLog = logMaxSizeMB > 0 ? $"{logMaxSizeMB}MB" : "UNLIMITED";
  337. var sql1 = $@"
  338. CREATE DATABASE [{store}]
  339. ON PRIMARY
  340. (
  341. NAME = N'{store}',
  342. FILENAME = N'{mdfPath}',
  343. SIZE = 4MB,
  344. MAXSIZE = UNLIMITED,
  345. FILEGROWTH = 4MB
  346. )
  347. LOG ON
  348. (
  349. NAME = N'{store}_log',
  350. FILENAME = N'{ldfPath}',
  351. SIZE = 1MB,
  352. MAXSIZE = {maxLog},
  353. FILEGROWTH = 1MB
  354. )
  355. COLLATE Chinese_PRC_CI_AS
  356. ";
  357. var create = source.Execute(sql1, null, false);
  358. if (!create.Success) return TextUtility.Merge("创建失败:", create.Message);
  359. // 设置兼容性级别。
  360. var sql2 = $"alter database [{store}] set compatibility_level = 0";
  361. source.Execute(sql2, null, false);
  362. // 设置恢复默认为“简单”
  363. var sql3 = $"alter database [{store}] set recovery simple";
  364. source.Execute(sql3, null, false);
  365. return null;
  366. }
  367. }
  368. static string XType(int xtype)
  369. {
  370. switch (xtype)
  371. {
  372. case 34: return "image";
  373. case 35: return "text";
  374. case 36: return "uniqueidentifier";
  375. case 48: return "tinyint";
  376. case 52: return "smallint";
  377. case 56: return "int";
  378. case 58: return "smalldatetime";
  379. case 59: return "real";
  380. case 60: return "money";
  381. case 61: return "datetime";
  382. case 62: return "float";
  383. case 98: return "sql_variant";
  384. case 99: return "ntext";
  385. case 104: return "bit";
  386. case 106: return "decimal";
  387. case 108: return "numeric";
  388. case 122: return "smallmoney";
  389. case 127: return "bigint";
  390. case 165: return "varbinary";
  391. case 167: return "varchar";
  392. case 173: return "binary";
  393. case 175: return "char";
  394. case 189: return "timestamp";
  395. case 231: return "nvarchar";
  396. case 239: return "nchar";
  397. case 241: return "xml";
  398. }
  399. return null;
  400. }
  401. #if NET20 || NET40
  402. /// <summary>枚举本地网络中的 SQL Server 实例的信息。</summary>
  403. public static Source[] EnumerateSources()
  404. {
  405. var list = new List<Source>();
  406. // 表中列名:ServerName、InstanceName、IsClustered、Version。
  407. using (var table = SqlDataSourceEnumerator.Instance.GetDataSources())
  408. {
  409. var query = new Query(table);
  410. var rows = query.Rows;
  411. list.Capacity = rows;
  412. for (int i = 0; i < rows; i++)
  413. {
  414. var item = new Source();
  415. item.ServerName = query.Text(i, "ServerName");
  416. list.Add(item);
  417. }
  418. }
  419. return list.ToArray();
  420. }
  421. /// <summary>SQL Server 实例的信息。</summary>
  422. public sealed class Source
  423. {
  424. /// <summary></summary>
  425. public string ServerName { get; set; }
  426. /// <summary></summary>
  427. public string InstanceName { get; set; }
  428. /// <summary></summary>
  429. public string IsClustered { get; set; }
  430. /// <summary></summary>
  431. public string Version { get; set; }
  432. }
  433. #endif
  434. /// <summary>创建参数。</summary>
  435. /// <exception cref="ArgumentNullException"></exception>
  436. /// <exception cref="InvalidOperationException"></exception>
  437. static SqlParameter Parameter(Parameter parameter)
  438. {
  439. if (parameter == null) throw new InvalidOperationException("参数无效。");
  440. return Parameter(parameter.Name, parameter.Type, parameter.Size, parameter.Value);
  441. }
  442. /// <summary>创建参数。</summary>
  443. public static SqlParameter Parameter(string name, ColumnType type, int size, object value)
  444. {
  445. var vname = TextUtility.Trim(name);
  446. if (TextUtility.IsBlank(vname)) return null;
  447. var vtype = SqlDbType.BigInt;
  448. switch (type)
  449. {
  450. case ColumnType.Boolean:
  451. vtype = SqlDbType.Bit;
  452. break;
  453. case ColumnType.Bytes:
  454. vtype = SqlDbType.Image;
  455. break;
  456. case ColumnType.Integer:
  457. vtype = SqlDbType.BigInt;
  458. break;
  459. case ColumnType.Float:
  460. vtype = SqlDbType.Float;
  461. break;
  462. case ColumnType.DateTime:
  463. vtype = SqlDbType.DateTime;
  464. break;
  465. case ColumnType.VarChar:
  466. case ColumnType.VarChar191:
  467. case ColumnType.VarCharMax:
  468. vtype = SqlDbType.VarChar;
  469. break;
  470. case ColumnType.NVarChar:
  471. case ColumnType.NVarChar191:
  472. case ColumnType.NVarCharMax:
  473. vtype = SqlDbType.NVarChar;
  474. break;
  475. case ColumnType.Text:
  476. vtype = SqlDbType.Text;
  477. break;
  478. case ColumnType.NText:
  479. vtype = SqlDbType.NText;
  480. break;
  481. default:
  482. throw new InvalidOperationException(TextUtility.Merge("类型 ", type.ToString(), " 不受支持。"));
  483. }
  484. var vsize = size;
  485. switch (type)
  486. {
  487. case ColumnType.VarChar:
  488. vsize = NumberUtility.Restrict(vsize, 0, 8000);
  489. break;
  490. case ColumnType.NVarChar:
  491. vsize = NumberUtility.Restrict(vsize, 0, 4000);
  492. break;
  493. case ColumnType.VarChar191:
  494. case ColumnType.NVarChar191:
  495. vsize = NumberUtility.Restrict(vsize, 0, 191);
  496. break;
  497. default:
  498. vsize = 0;
  499. break;
  500. }
  501. var vvalue = value ?? DBNull.Value;
  502. if (vvalue is string && vvalue != null && vsize > 0)
  503. {
  504. vvalue = TextUtility.Left((string)vvalue, vsize);
  505. }
  506. var parameter = new SqlParameter();
  507. parameter.ParameterName = vname;
  508. parameter.SqlDbType = vtype;
  509. parameter.Value = vvalue;
  510. if (vsize > 0) parameter.Size = vsize;
  511. return parameter;
  512. }
  513. /// <summary>创建参数。</summary>
  514. public static SqlParameter Parameter(string name, SqlDbType type, int size, object value)
  515. {
  516. if (value is string && value != null && size > 0)
  517. {
  518. value = TextUtility.Left((string)value, (int)size);
  519. }
  520. var p = new SqlParameter();
  521. p.ParameterName = name ?? "";
  522. p.SqlDbType = type;
  523. p.Size = size;
  524. p.Value = value ?? DBNull.Value;
  525. return p;
  526. }
  527. /// <summary>创建参数。</summary>
  528. public static SqlParameter Parameter(string name, SqlDbType type, object value)
  529. {
  530. var p = new SqlParameter();
  531. p.ParameterName = name ?? "";
  532. p.SqlDbType = type;
  533. p.Value = value ?? DBNull.Value;
  534. return p;
  535. }
  536. static string Declaration(ColumnAttribute column)
  537. {
  538. var type = TextUtility.Empty;
  539. var vcolumn = column;
  540. var length = Math.Max(0, vcolumn.Length);
  541. switch (vcolumn.Type)
  542. {
  543. case ColumnType.Boolean:
  544. type = "bit";
  545. break;
  546. case ColumnType.Integer:
  547. type = "bigint";
  548. break;
  549. case ColumnType.Float:
  550. type = "float";
  551. break;
  552. case ColumnType.Bytes:
  553. type = "image";
  554. break;
  555. case ColumnType.DateTime:
  556. type = "datetime";
  557. break;
  558. case ColumnType.VarChar:
  559. type = TextUtility.Merge("varchar(", Math.Min(8000, length).ToString(), ")");
  560. break;
  561. case ColumnType.VarChar191:
  562. type = TextUtility.Merge("varchar(191)");
  563. break;
  564. case ColumnType.VarCharMax:
  565. type = TextUtility.Merge("varchar(max)");
  566. break;
  567. case ColumnType.Text:
  568. type = TextUtility.Merge("text");
  569. break;
  570. case ColumnType.NVarChar:
  571. type = TextUtility.Merge("nvarchar(", Math.Min(4000, length).ToString(), ")");
  572. break;
  573. case ColumnType.NVarChar191:
  574. type = TextUtility.Merge("nvarchar(191)");
  575. break;
  576. case ColumnType.NVarCharMax:
  577. type = TextUtility.Merge("nvarchar(max)");
  578. break;
  579. case ColumnType.NText:
  580. type = TextUtility.Merge("ntext");
  581. break;
  582. default:
  583. return TextUtility.Empty;
  584. }
  585. return TextUtility.Merge("[", vcolumn.Field, "] ", type);
  586. }
  587. #endregion
  588. }
  589. }