SqlParameter.cs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630
  1. //
  2. // System.Data.SqlClient.SqlParameter.cs
  3. //
  4. // Author:
  5. // Rodrigo Moya ([email protected])
  6. // Daniel Morgan ([email protected])
  7. // Tim Coleman ([email protected])
  8. //
  9. // (C) Ximian, Inc. 2002
  10. // Copyright (C) Tim Coleman, 2002
  11. //
  12. using Mono.Data.TdsClient.Internal;
  13. using System;
  14. using System.ComponentModel;
  15. using System.Data;
  16. using System.Data.Common;
  17. using System.Runtime.InteropServices;
  18. using System.Text;
  19. namespace System.Data.SqlClient {
  20. /// <summary>
  21. /// Represents a parameter to a Command object, and optionally,
  22. /// its mapping to DataSet columns; and is implemented by .NET
  23. /// data providers that access data sources.
  24. /// </summary>
  25. public sealed class SqlParameter : MarshalByRefObject, IDbDataParameter, IDataParameter, ICloneable
  26. {
  27. #region Fields
  28. SqlDbType sqlDbType;
  29. DbType dbType;
  30. string typeName;
  31. bool isSizeSet = false;
  32. bool isTypeSet = false;
  33. string parmName;
  34. object objValue;
  35. int size;
  36. string sourceColumn;
  37. ParameterDirection direction = ParameterDirection.Input;
  38. bool isNullable;
  39. byte precision;
  40. byte scale;
  41. DataRowVersion sourceVersion;
  42. int offset;
  43. SqlParameterCollection container = null;
  44. #endregion // Fields
  45. #region Constructors
  46. public SqlParameter ()
  47. : this (String.Empty, SqlDbType.NVarChar, 0, ParameterDirection.Input, false, 0, 0, String.Empty, DataRowVersion.Current, null)
  48. {
  49. }
  50. public SqlParameter (string parameterName, object value)
  51. {
  52. this.parmName = parameterName;
  53. this.objValue = value;
  54. this.sourceVersion = DataRowVersion.Current;
  55. InferSqlType (value);
  56. }
  57. public SqlParameter (string parameterName, SqlDbType dbType)
  58. : this (parameterName, dbType, 0, ParameterDirection.Input, false, 0, 0, String.Empty, DataRowVersion.Current, null)
  59. {
  60. }
  61. public SqlParameter (string parameterName, SqlDbType dbType, int size)
  62. : this (parameterName, dbType, size, ParameterDirection.Input, false, 0, 0, String.Empty, DataRowVersion.Current, null)
  63. {
  64. }
  65. public SqlParameter (string parameterName, SqlDbType dbType, int size, string sourceColumn)
  66. : this (parameterName, dbType, size, ParameterDirection.Input, false, 0, 0, sourceColumn, DataRowVersion.Current, null)
  67. {
  68. }
  69. [EditorBrowsable (EditorBrowsableState.Advanced)]
  70. public SqlParameter (string parameterName, SqlDbType dbType, int size, ParameterDirection direction, bool isNullable, byte precision, byte scale, string sourceColumn, DataRowVersion sourceVersion, object value)
  71. {
  72. SqlDbType = dbType;
  73. Size = size;
  74. Value = value;
  75. ParameterName = parameterName;
  76. Direction = direction;
  77. IsNullable = isNullable;
  78. Precision = precision;
  79. Scale = scale;
  80. SourceColumn = sourceColumn;
  81. SourceVersion = sourceVersion;
  82. }
  83. internal SqlParameter (object[] dbValues)
  84. {
  85. precision = 0;
  86. scale = 0;
  87. direction = ParameterDirection.Input;
  88. parmName = (string) dbValues[3];
  89. switch ((short) dbValues[5]) {
  90. case 1:
  91. direction = ParameterDirection.Input;
  92. break;
  93. case 2:
  94. direction = ParameterDirection.Output;
  95. break;
  96. case 3:
  97. direction = ParameterDirection.InputOutput;
  98. break;
  99. case 4:
  100. direction = ParameterDirection.ReturnValue;
  101. break;
  102. }
  103. isNullable = (bool) dbValues[8];
  104. if (dbValues[12] != null)
  105. precision = (byte) ((short) dbValues[12]);
  106. if (dbValues[13] != null)
  107. scale = (byte) ((short) dbValues[13]);
  108. SetDbTypeName ((string) dbValues[16]);
  109. }
  110. #endregion // Constructors
  111. #region Properties
  112. // Used to ensure that only one collection can contain this
  113. // parameter
  114. internal SqlParameterCollection Container {
  115. get { return container; }
  116. set { container = value; }
  117. }
  118. [Browsable (false)]
  119. [DataCategory ("Data")]
  120. [DataSysDescription ("The parameter generic type.")]
  121. [DesignerSerializationVisibility (DesignerSerializationVisibility.Hidden)]
  122. [RefreshProperties (RefreshProperties.All)]
  123. public DbType DbType {
  124. get { return dbType; }
  125. set {
  126. SetDbType (value);
  127. isTypeSet = true;
  128. }
  129. }
  130. [DataCategory ("Data")]
  131. [DataSysDescription ("Input, output, or bidirectional parameter.")]
  132. [DefaultValue (ParameterDirection.Input)]
  133. public ParameterDirection Direction {
  134. get { return direction; }
  135. set { direction = value; }
  136. }
  137. string IDataParameter.ParameterName {
  138. get { return parmName; }
  139. set { parmName = value; }
  140. }
  141. [Browsable (false)]
  142. [DataSysDescription ("a design-time property used for strongly typed code-generation.")]
  143. [DefaultValue (false)]
  144. [DesignOnly (true)]
  145. [EditorBrowsable (EditorBrowsableState.Advanced)]
  146. public bool IsNullable {
  147. get { return isNullable; }
  148. set { isNullable = value; }
  149. }
  150. [Browsable (false)]
  151. [DataCategory ("Data")]
  152. [DataSysDescription ("Offset in variable length data types.")]
  153. [DefaultValue (0)]
  154. public int Offset {
  155. get { return offset; }
  156. set { offset = value; }
  157. }
  158. [DataSysDescription ("Name of the parameter, like '@p1'")]
  159. [DefaultValue ("")]
  160. public string ParameterName {
  161. get { return parmName; }
  162. set { parmName = value; }
  163. }
  164. [DataCategory ("Data")]
  165. [DataSysDescription ("For decimal, numeric, varnumeric DBTypes.")]
  166. [DefaultValue (0)]
  167. public byte Precision {
  168. get { return precision; }
  169. set { precision = value; }
  170. }
  171. [DataCategory ("Data")]
  172. [DataSysDescription ("For decimal, numeric, varnumeric DBTypes.")]
  173. [DefaultValue (0)]
  174. public byte Scale {
  175. get { return scale; }
  176. set { scale = value; }
  177. }
  178. [DataCategory ("Data")]
  179. [DataSysDescription ("Size of variable length datatypes (strings & arrays).")]
  180. [DefaultValue (0)]
  181. public int Size {
  182. get { return size; }
  183. set {
  184. size = value;
  185. isSizeSet = true;
  186. }
  187. }
  188. [DataCategory ("Data")]
  189. [DataSysDescription ("When used by a DataAdapter.Update, the source column name that is used to find the DataSetColumn name in the ColumnMappings. This is to copy a value between the parameter and a datarow.")]
  190. [DefaultValue ("")]
  191. public string SourceColumn {
  192. get { return sourceColumn; }
  193. set { sourceColumn = value; }
  194. }
  195. [DataCategory ("Data")]
  196. [DataSysDescription ("When used by a DataAdapter.Update (UpdateCommand only), the version of the DataRow value that is used to update the data source.")]
  197. [DefaultValue (DataRowVersion.Current)]
  198. public DataRowVersion SourceVersion {
  199. get { return sourceVersion; }
  200. set { sourceVersion = value; }
  201. }
  202. [DataCategory ("Data")]
  203. [DataSysDescription ("The parameter native type.")]
  204. [DefaultValue (SqlDbType.NVarChar)]
  205. [RefreshProperties (RefreshProperties.All)]
  206. public SqlDbType SqlDbType {
  207. get { return sqlDbType; }
  208. set {
  209. SetSqlDbType (value);
  210. isTypeSet = true;
  211. }
  212. }
  213. [DataCategory ("Data")]
  214. [DataSysDescription ("Value of the parameter.")]
  215. [DefaultValue (null)]
  216. public object Value {
  217. get { return objValue; }
  218. set {
  219. if (!isTypeSet)
  220. InferSqlType (value);
  221. objValue = value;
  222. }
  223. }
  224. #endregion // Properties
  225. #region Methods
  226. object ICloneable.Clone ()
  227. {
  228. return new SqlParameter (ParameterName, SqlDbType, Size, Direction, IsNullable, Precision, Scale, SourceColumn, SourceVersion, Value);
  229. }
  230. private void InferSqlType (object value)
  231. {
  232. Type type = value.GetType ();
  233. string exception = String.Format ("The parameter data type of {0} is invalid.", type.Name);
  234. switch (type.FullName) {
  235. case "System.Int64":
  236. SetSqlDbType (SqlDbType.BigInt);
  237. break;
  238. case "System.Boolean":
  239. SetSqlDbType (SqlDbType.Bit);
  240. break;
  241. case "System.String":
  242. SetSqlDbType (SqlDbType.NVarChar);
  243. break;
  244. case "System.DateTime":
  245. SetSqlDbType (SqlDbType.DateTime);
  246. break;
  247. case "System.Decimal":
  248. SetSqlDbType (SqlDbType.Decimal);
  249. break;
  250. case "System.Double":
  251. SetSqlDbType (SqlDbType.Float);
  252. break;
  253. case "System.Byte[]":
  254. SetSqlDbType (SqlDbType.VarBinary);
  255. break;
  256. case "System.Byte":
  257. SetSqlDbType (SqlDbType.TinyInt);
  258. break;
  259. case "System.Int32":
  260. SetSqlDbType (SqlDbType.Int);
  261. break;
  262. case "System.Single":
  263. SetSqlDbType (SqlDbType.Real);
  264. break;
  265. case "System.Int16":
  266. SetSqlDbType (SqlDbType.SmallInt);
  267. break;
  268. case "System.Guid":
  269. SetSqlDbType (SqlDbType.UniqueIdentifier);
  270. break;
  271. case "System.Object":
  272. SetSqlDbType (SqlDbType.Variant);
  273. break;
  274. default:
  275. throw new ArgumentException (exception);
  276. }
  277. }
  278. internal string Prepare (string name)
  279. {
  280. StringBuilder result = new StringBuilder ();
  281. result.Append (name);
  282. result.Append (" ");
  283. result.Append (typeName);
  284. switch (sqlDbType) {
  285. case SqlDbType.VarBinary :
  286. case SqlDbType.NVarChar :
  287. case SqlDbType.VarChar :
  288. if (!isSizeSet || size == 0)
  289. throw new InvalidOperationException ("All variable length parameters must have an explicitly set non-zero size.");
  290. result.Append (String.Format ("({0})", size));
  291. break;
  292. case SqlDbType.NChar :
  293. case SqlDbType.Char :
  294. case SqlDbType.Binary :
  295. if (size > 0)
  296. result.Append (String.Format ("({0})", size));
  297. break;
  298. case SqlDbType.Decimal :
  299. result.Append (String.Format ("({0},{1})", precision, scale));
  300. break;
  301. default:
  302. break;
  303. }
  304. return result.ToString ();
  305. }
  306. private void SetDbType (DbType type)
  307. {
  308. string exception = String.Format ("No mapping exists from DbType {0} to a known SqlDbType.", type);
  309. switch (type) {
  310. case DbType.AnsiString:
  311. typeName = "varchar";
  312. sqlDbType = SqlDbType.VarChar;
  313. break;
  314. case DbType.AnsiStringFixedLength:
  315. typeName = "char";
  316. sqlDbType = SqlDbType.Char;
  317. break;
  318. case DbType.Binary:
  319. typeName = "varbinary";
  320. sqlDbType = SqlDbType.VarBinary;
  321. break;
  322. case DbType.Boolean:
  323. typeName = "bit";
  324. sqlDbType = SqlDbType.Bit;
  325. break;
  326. case DbType.Byte:
  327. typeName = "tinyint";
  328. sqlDbType = SqlDbType.TinyInt;
  329. break;
  330. case DbType.Currency:
  331. sqlDbType = SqlDbType.Money;
  332. typeName = "money";
  333. break;
  334. case DbType.Date:
  335. case DbType.DateTime:
  336. typeName = "datetime";
  337. sqlDbType = SqlDbType.DateTime;
  338. break;
  339. case DbType.Decimal:
  340. typeName = "decimal";
  341. sqlDbType = SqlDbType.Decimal;
  342. break;
  343. case DbType.Double:
  344. typeName = "float";
  345. sqlDbType = SqlDbType.Float;
  346. break;
  347. case DbType.Guid:
  348. typeName = "uniqueidentifier";
  349. sqlDbType = SqlDbType.UniqueIdentifier;
  350. break;
  351. case DbType.Int16:
  352. typeName = "smallint";
  353. sqlDbType = SqlDbType.SmallInt;
  354. break;
  355. case DbType.Int32:
  356. typeName = "int";
  357. sqlDbType = SqlDbType.Int;
  358. break;
  359. case DbType.Int64:
  360. typeName = "bigint";
  361. sqlDbType = SqlDbType.BigInt;
  362. break;
  363. case DbType.Object:
  364. typeName = "sql_variant";
  365. sqlDbType = SqlDbType.Variant;
  366. break;
  367. case DbType.Single:
  368. typeName = "real";
  369. sqlDbType = SqlDbType.Real;
  370. break;
  371. case DbType.String:
  372. typeName = "nvarchar";
  373. sqlDbType = SqlDbType.NVarChar;
  374. break;
  375. case DbType.StringFixedLength:
  376. typeName = "nchar";
  377. sqlDbType = SqlDbType.NChar;
  378. break;
  379. case DbType.Time:
  380. typeName = "datetime";
  381. sqlDbType = SqlDbType.DateTime;
  382. break;
  383. default:
  384. throw new ArgumentException (exception);
  385. }
  386. dbType = type;
  387. }
  388. // Used by internal constructor which has a SQL Server typename
  389. private void SetDbTypeName (string dbTypeName)
  390. {
  391. switch (dbTypeName.ToLower ()) {
  392. case "bigint":
  393. SqlDbType = SqlDbType.BigInt;
  394. break;
  395. case "binary":
  396. SqlDbType = SqlDbType.Binary;
  397. break;
  398. case "bit":
  399. SqlDbType = SqlDbType.Bit;
  400. break;
  401. case "char":
  402. SqlDbType = SqlDbType.Char;
  403. break;
  404. case "datetime":
  405. SqlDbType = SqlDbType.DateTime;
  406. break;
  407. case "decimal":
  408. SqlDbType = SqlDbType.Decimal;
  409. break;
  410. case "float":
  411. SqlDbType = SqlDbType.Float;
  412. break;
  413. case "image":
  414. SqlDbType = SqlDbType.Image;
  415. break;
  416. case "int":
  417. SqlDbType = SqlDbType.Int;
  418. break;
  419. case "money":
  420. SqlDbType = SqlDbType.Money;
  421. break;
  422. case "nchar":
  423. SqlDbType = SqlDbType.NChar;
  424. break;
  425. case "ntext":
  426. SqlDbType = SqlDbType.NText;
  427. break;
  428. case "nvarchar":
  429. SqlDbType = SqlDbType.NVarChar;
  430. break;
  431. case "real":
  432. SqlDbType = SqlDbType.Real;
  433. break;
  434. case "smalldatetime":
  435. SqlDbType = SqlDbType.SmallDateTime;
  436. break;
  437. case "smallint":
  438. SqlDbType = SqlDbType.SmallInt;
  439. break;
  440. case "smallmoney":
  441. SqlDbType = SqlDbType.SmallMoney;
  442. break;
  443. case "text":
  444. SqlDbType = SqlDbType.Text;
  445. break;
  446. case "timestamp":
  447. SqlDbType = SqlDbType.Timestamp;
  448. break;
  449. case "tinyint":
  450. SqlDbType = SqlDbType.TinyInt;
  451. break;
  452. case "uniqueidentifier":
  453. SqlDbType = SqlDbType.UniqueIdentifier;
  454. break;
  455. case "varbinary":
  456. SqlDbType = SqlDbType.VarBinary;
  457. break;
  458. case "varchar":
  459. SqlDbType = SqlDbType.VarChar;
  460. break;
  461. default:
  462. SqlDbType = SqlDbType.Variant;
  463. break;
  464. }
  465. }
  466. private void SetSqlDbType (SqlDbType type)
  467. {
  468. string exception = String.Format ("No mapping exists from SqlDbType {0} to a known DbType.", type);
  469. switch (type) {
  470. case SqlDbType.BigInt:
  471. typeName = "bigint";
  472. dbType = DbType.Int64;
  473. break;
  474. case SqlDbType.Binary:
  475. typeName = "binary";
  476. dbType = DbType.Binary;
  477. break;
  478. case SqlDbType.Timestamp:
  479. typeName = "timestamp";
  480. dbType = DbType.Binary;
  481. break;
  482. case SqlDbType.VarBinary:
  483. typeName = "varbinary";
  484. dbType = DbType.Binary;
  485. break;
  486. case SqlDbType.Bit:
  487. typeName = "bit";
  488. dbType = DbType.Boolean;
  489. break;
  490. case SqlDbType.Char:
  491. typeName = "char";
  492. dbType = DbType.AnsiStringFixedLength;
  493. break;
  494. case SqlDbType.DateTime:
  495. typeName = "datetime";
  496. dbType = DbType.DateTime;
  497. break;
  498. case SqlDbType.SmallDateTime:
  499. typeName = "smalldatetime";
  500. dbType = DbType.DateTime;
  501. break;
  502. case SqlDbType.Decimal:
  503. typeName = "decimal";
  504. dbType = DbType.Decimal;
  505. break;
  506. case SqlDbType.Float:
  507. typeName = "float";
  508. dbType = DbType.Double;
  509. break;
  510. case SqlDbType.Image:
  511. typeName = "image";
  512. dbType = DbType.Binary;
  513. break;
  514. case SqlDbType.Int:
  515. typeName = "int";
  516. dbType = DbType.Int32;
  517. break;
  518. case SqlDbType.Money:
  519. typeName = "money";
  520. dbType = DbType.Currency;
  521. break;
  522. case SqlDbType.SmallMoney:
  523. typeName = "smallmoney";
  524. dbType = DbType.Currency;
  525. break;
  526. case SqlDbType.NChar:
  527. typeName = "nchar";
  528. dbType = DbType.StringFixedLength;
  529. break;
  530. case SqlDbType.NText:
  531. typeName = "ntext";
  532. dbType = DbType.String;
  533. break;
  534. case SqlDbType.NVarChar:
  535. typeName = "nvarchar";
  536. dbType = DbType.String;
  537. break;
  538. case SqlDbType.Real:
  539. typeName = "real";
  540. dbType = DbType.Single;
  541. break;
  542. case SqlDbType.SmallInt:
  543. typeName = "smallint";
  544. dbType = DbType.Int16;
  545. break;
  546. case SqlDbType.Text:
  547. typeName = "text";
  548. dbType = DbType.AnsiString;
  549. break;
  550. case SqlDbType.VarChar:
  551. typeName = "varchar";
  552. dbType = DbType.AnsiString;
  553. break;
  554. case SqlDbType.TinyInt:
  555. typeName = "tinyint";
  556. dbType = DbType.Byte;
  557. break;
  558. case SqlDbType.UniqueIdentifier:
  559. typeName = "uniqueidentifier";
  560. dbType = DbType.Guid;
  561. break;
  562. case SqlDbType.Variant:
  563. typeName = "sql_variant";
  564. dbType = DbType.Object;
  565. break;
  566. default:
  567. throw new ArgumentException (exception);
  568. }
  569. sqlDbType = type;
  570. }
  571. public override string ToString()
  572. {
  573. return parmName;
  574. }
  575. #endregion // Methods
  576. }
  577. }