697 lines
22 KiB
C#
697 lines
22 KiB
C#
|
//
|
||
|
// System.Data.Common.DbCommandBuilder
|
||
|
//
|
||
|
// Author:
|
||
|
// Tim Coleman (tim@timcoleman.com)
|
||
|
//
|
||
|
// Copyright (C) Tim Coleman, 2003
|
||
|
//
|
||
|
|
||
|
//
|
||
|
// Copyright (C) 2004 Novell, Inc (http://www.novell.com)
|
||
|
//
|
||
|
// Permission is hereby granted, free of charge, to any person obtaining
|
||
|
// a copy of this software and associated documentation files (the
|
||
|
// "Software"), to deal in the Software without restriction, including
|
||
|
// without limitation the rights to use, copy, modify, merge, publish,
|
||
|
// distribute, sublicense, and/or sell copies of the Software, and to
|
||
|
// permit persons to whom the Software is furnished to do so, subject to
|
||
|
// the following conditions:
|
||
|
//
|
||
|
// The above copyright notice and this permission notice shall be
|
||
|
// included in all copies or substantial portions of the Software.
|
||
|
//
|
||
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||
|
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||
|
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||
|
//
|
||
|
|
||
|
|
||
|
using System.ComponentModel;
|
||
|
using System.Data;
|
||
|
using System.Globalization;
|
||
|
using System.Text;
|
||
|
|
||
|
namespace System.Data.Common {
|
||
|
public abstract class DbCommandBuilder : Component
|
||
|
{
|
||
|
bool _setAllValues;
|
||
|
bool _disposed;
|
||
|
|
||
|
DataTable _dbSchemaTable;
|
||
|
DbDataAdapter _dbDataAdapter;
|
||
|
private CatalogLocation _catalogLocation = CatalogLocation.Start;
|
||
|
private ConflictOption _conflictOption = ConflictOption.CompareAllSearchableValues;
|
||
|
|
||
|
private string _tableName;
|
||
|
private string _catalogSeparator;
|
||
|
private string _quotePrefix;
|
||
|
private string _quoteSuffix;
|
||
|
private string _schemaSeparator;
|
||
|
private DbCommand _dbCommand;
|
||
|
|
||
|
DbCommand _deleteCommand;
|
||
|
DbCommand _insertCommand;
|
||
|
DbCommand _updateCommand;
|
||
|
|
||
|
static readonly string SEPARATOR_DEFAULT = ".";
|
||
|
// Used to construct WHERE clauses
|
||
|
static readonly string clause1 = "({0} = 1 AND {1} IS NULL)";
|
||
|
static readonly string clause2 = "({0} = {1})";
|
||
|
|
||
|
protected DbCommandBuilder ()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
private void BuildCache (bool closeConnection)
|
||
|
{
|
||
|
DbCommand sourceCommand = SourceCommand;
|
||
|
if (sourceCommand == null)
|
||
|
throw new InvalidOperationException ("The DataAdapter.SelectCommand property needs to be initialized.");
|
||
|
DbConnection connection = sourceCommand.Connection;
|
||
|
if (connection == null)
|
||
|
throw new InvalidOperationException ("The DataAdapter.SelectCommand.Connection property needs to be initialized.");
|
||
|
|
||
|
if (_dbSchemaTable == null) {
|
||
|
if (connection.State == ConnectionState.Open)
|
||
|
closeConnection = false;
|
||
|
else
|
||
|
connection.Open ();
|
||
|
|
||
|
DbDataReader reader = sourceCommand.ExecuteReader (CommandBehavior.SchemaOnly | CommandBehavior.KeyInfo);
|
||
|
_dbSchemaTable = reader.GetSchemaTable ();
|
||
|
reader.Close ();
|
||
|
if (closeConnection)
|
||
|
connection.Close ();
|
||
|
BuildInformation (_dbSchemaTable);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private string QuotedTableName {
|
||
|
get { return GetQuotedString (_tableName); }
|
||
|
}
|
||
|
|
||
|
bool IsCommandGenerated {
|
||
|
get {
|
||
|
return (_insertCommand != null || _updateCommand != null || _deleteCommand != null);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private string GetQuotedString (string value)
|
||
|
{
|
||
|
if (value == String.Empty || value == null)
|
||
|
return value;
|
||
|
|
||
|
string prefix = QuotePrefix;
|
||
|
string suffix = QuoteSuffix;
|
||
|
|
||
|
if (prefix.Length == 0 && suffix.Length == 0)
|
||
|
return value;
|
||
|
return String.Format ("{0}{1}{2}", prefix, value, suffix);
|
||
|
}
|
||
|
|
||
|
private void BuildInformation (DataTable schemaTable)
|
||
|
{
|
||
|
_tableName = String.Empty;
|
||
|
foreach (DataRow schemaRow in schemaTable.Rows) {
|
||
|
if (schemaRow.IsNull ("BaseTableName") || (string) schemaRow ["BaseTableName"] == String.Empty)
|
||
|
continue;
|
||
|
|
||
|
if (_tableName == String.Empty)
|
||
|
_tableName = (string) schemaRow ["BaseTableName"];
|
||
|
else if (_tableName != (string) schemaRow["BaseTableName"])
|
||
|
throw new InvalidOperationException ("Dynamic SQL generation is not supported against multiple base tables.");
|
||
|
}
|
||
|
if (_tableName == String.Empty)
|
||
|
throw new InvalidOperationException ("Dynamic SQL generation is not supported with no base table.");
|
||
|
_dbSchemaTable = schemaTable;
|
||
|
}
|
||
|
|
||
|
private bool IncludedInInsert (DataRow schemaRow)
|
||
|
{
|
||
|
// If the parameter has one of these properties, then we don't include it in the insert:
|
||
|
// AutoIncrement, Hidden, Expression, RowVersion, ReadOnly
|
||
|
|
||
|
if (!schemaRow.IsNull ("IsAutoIncrement") && (bool) schemaRow ["IsAutoIncrement"])
|
||
|
return false;
|
||
|
// if (!schemaRow.IsNull ("IsHidden") && (bool) schemaRow ["IsHidden"])
|
||
|
// return false;
|
||
|
if (!schemaRow.IsNull ("IsExpression") && (bool) schemaRow ["IsExpression"])
|
||
|
return false;
|
||
|
if (!schemaRow.IsNull ("IsRowVersion") && (bool) schemaRow ["IsRowVersion"])
|
||
|
return false;
|
||
|
if (!schemaRow.IsNull ("IsReadOnly") && (bool) schemaRow ["IsReadOnly"])
|
||
|
return false;
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
private bool IncludedInUpdate (DataRow schemaRow)
|
||
|
{
|
||
|
// If the parameter has one of these properties, then we don't include it in the insert:
|
||
|
// AutoIncrement, Hidden, RowVersion
|
||
|
|
||
|
if (!schemaRow.IsNull ("IsAutoIncrement") && (bool) schemaRow ["IsAutoIncrement"])
|
||
|
return false;
|
||
|
// if (!schemaRow.IsNull ("IsHidden") && (bool) schemaRow ["IsHidden"])
|
||
|
// return false;
|
||
|
if (!schemaRow.IsNull ("IsRowVersion") && (bool) schemaRow ["IsRowVersion"])
|
||
|
return false;
|
||
|
if (!schemaRow.IsNull ("IsExpression") && (bool) schemaRow ["IsExpression"])
|
||
|
return false;
|
||
|
if (!schemaRow.IsNull ("IsReadOnly") && (bool) schemaRow ["IsReadOnly"])
|
||
|
return false;
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
private bool IncludedInWhereClause (DataRow schemaRow)
|
||
|
{
|
||
|
if ((bool) schemaRow ["IsLong"])
|
||
|
return false;
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
private DbCommand CreateDeleteCommand (bool option)
|
||
|
{
|
||
|
// If no table was found, then we can't do an delete
|
||
|
if (QuotedTableName == String.Empty)
|
||
|
return null;
|
||
|
|
||
|
CreateNewCommand (ref _deleteCommand);
|
||
|
|
||
|
string command = String.Format ("DELETE FROM {0}", QuotedTableName);
|
||
|
StringBuilder whereClause = new StringBuilder ();
|
||
|
bool keyFound = false;
|
||
|
int parmIndex = 1;
|
||
|
|
||
|
foreach (DataRow schemaRow in _dbSchemaTable.Rows) {
|
||
|
if (!schemaRow.IsNull ("IsExpression") && (bool)schemaRow["IsExpression"] == true)
|
||
|
continue;
|
||
|
if (!IncludedInWhereClause (schemaRow))
|
||
|
continue;
|
||
|
|
||
|
if (whereClause.Length > 0)
|
||
|
whereClause.Append (" AND ");
|
||
|
|
||
|
bool isKey = (bool) schemaRow ["IsKey"];
|
||
|
DbParameter parameter = null;
|
||
|
string sourceColumnName;
|
||
|
|
||
|
if (isKey)
|
||
|
keyFound = true;
|
||
|
|
||
|
//ms.net 1.1 generates the null check for columns even if AllowDBNull is false
|
||
|
//while ms.net 2.0 does not. Anyways, since both forms are logically equivalent
|
||
|
//following the 2.0 approach
|
||
|
bool allowNull = (bool) schemaRow ["AllowDBNull"];
|
||
|
if (!isKey && allowNull) {
|
||
|
parameter = _deleteCommand.CreateParameter ();
|
||
|
if (option) {
|
||
|
parameter.ParameterName = String.Format ("@IsNull_{0}",
|
||
|
schemaRow ["BaseColumnName"]);
|
||
|
} else {
|
||
|
parameter.ParameterName = String.Format ("@p{0}", parmIndex++);
|
||
|
}
|
||
|
parameter.Value = 1;
|
||
|
parameter.DbType = DbType.Int32;
|
||
|
// This should be set for nullcheckparam
|
||
|
sourceColumnName = (string) schemaRow ["BaseColumnName"];
|
||
|
parameter.SourceColumn = sourceColumnName;
|
||
|
parameter.SourceColumnNullMapping = true;
|
||
|
parameter.SourceVersion = DataRowVersion.Original;
|
||
|
_deleteCommand.Parameters.Add (parameter);
|
||
|
|
||
|
whereClause.Append ("(");
|
||
|
whereClause.Append (String.Format (clause1, parameter.ParameterName,
|
||
|
GetQuotedString (sourceColumnName)));
|
||
|
whereClause.Append (" OR ");
|
||
|
}
|
||
|
|
||
|
if (option)
|
||
|
parameter = CreateParameter (_deleteCommand, schemaRow, true);
|
||
|
else
|
||
|
parameter = CreateParameter (_deleteCommand, parmIndex++, schemaRow);
|
||
|
|
||
|
parameter.SourceVersion = DataRowVersion.Original;
|
||
|
ApplyParameterInfo (parameter, schemaRow, StatementType.Delete, true);
|
||
|
//parameter.IsNullable = allowNull;
|
||
|
|
||
|
whereClause.Append (String.Format (clause2, GetQuotedString (parameter.SourceColumn), parameter.ParameterName));
|
||
|
|
||
|
if (!isKey && allowNull)
|
||
|
whereClause.Append (")");
|
||
|
}
|
||
|
if (!keyFound)
|
||
|
throw new InvalidOperationException ("Dynamic SQL generation for the DeleteCommand is not supported against a SelectCommand that does not return any key column information.");
|
||
|
|
||
|
// We're all done, so bring it on home
|
||
|
string sql = String.Format ("{0} WHERE ({1})", command, whereClause.ToString ());
|
||
|
_deleteCommand.CommandText = sql;
|
||
|
_dbCommand = _deleteCommand;
|
||
|
return _deleteCommand;
|
||
|
}
|
||
|
|
||
|
private DbCommand CreateInsertCommand (bool option, DataRow row)
|
||
|
{
|
||
|
if (QuotedTableName == String.Empty)
|
||
|
return null;
|
||
|
|
||
|
CreateNewCommand (ref _insertCommand);
|
||
|
|
||
|
string command = String.Format ("INSERT INTO {0}", QuotedTableName);
|
||
|
string sql;
|
||
|
StringBuilder columns = new StringBuilder ();
|
||
|
StringBuilder values = new StringBuilder ();
|
||
|
|
||
|
int parmIndex = 1;
|
||
|
DbParameter parameter = null;
|
||
|
foreach (DataRow schemaRow in _dbSchemaTable.Rows) {
|
||
|
if (!IncludedInInsert (schemaRow))
|
||
|
continue;
|
||
|
|
||
|
if (columns.Length > 0) {
|
||
|
columns.Append (", ");
|
||
|
values.Append (", ");
|
||
|
}
|
||
|
|
||
|
if (option)
|
||
|
parameter = CreateParameter (_insertCommand, schemaRow, false);
|
||
|
else
|
||
|
parameter = CreateParameter (_insertCommand, parmIndex++, schemaRow);
|
||
|
parameter.SourceVersion = DataRowVersion.Current;
|
||
|
ApplyParameterInfo (parameter, schemaRow, StatementType.Insert, false);
|
||
|
|
||
|
columns.Append (GetQuotedString (parameter.SourceColumn));
|
||
|
|
||
|
// Workaround for columns that may have a default/bound value and for now,
|
||
|
// the framework, don't provide a mechanism to read these values yet
|
||
|
// AllowDBNull and DataRow is used to workaround #385028 by using DEFAULT
|
||
|
string colName = schemaRow ["ColumnName"] as string;
|
||
|
bool allowDBNull = !schemaRow.IsNull ("AllowDBNull") & (bool) schemaRow ["AllowDBNull"];
|
||
|
if (!allowDBNull && row != null &&
|
||
|
(row [colName] == DBNull.Value || row [colName] == null)) {
|
||
|
values.Append ("DEFAULT");
|
||
|
} else {
|
||
|
values.Append (parameter.ParameterName);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
sql = String.Format ("{0} ({1}) VALUES ({2})", command, columns.ToString (), values.ToString ());
|
||
|
_insertCommand.CommandText = sql;
|
||
|
_dbCommand = _insertCommand;
|
||
|
return _insertCommand;
|
||
|
}
|
||
|
|
||
|
private void CreateNewCommand (ref DbCommand command)
|
||
|
{
|
||
|
DbCommand sourceCommand = SourceCommand;
|
||
|
if (command == null) {
|
||
|
command = sourceCommand.Connection.CreateCommand ();
|
||
|
command.CommandTimeout = sourceCommand.CommandTimeout;
|
||
|
command.Transaction = sourceCommand.Transaction;
|
||
|
}
|
||
|
command.CommandType = CommandType.Text;
|
||
|
command.UpdatedRowSource = UpdateRowSource.None;
|
||
|
command.Parameters.Clear ();
|
||
|
}
|
||
|
|
||
|
private DbCommand CreateUpdateCommand (bool option)
|
||
|
{
|
||
|
// If no table was found, then we can't do an update
|
||
|
if (QuotedTableName == String.Empty)
|
||
|
return null;
|
||
|
|
||
|
CreateNewCommand (ref _updateCommand);
|
||
|
|
||
|
string command = String.Format ("UPDATE {0} SET ", QuotedTableName);
|
||
|
StringBuilder columns = new StringBuilder ();
|
||
|
StringBuilder whereClause = new StringBuilder ();
|
||
|
int parmIndex = 1;
|
||
|
bool keyFound = false;
|
||
|
DbParameter parameter = null;
|
||
|
|
||
|
// First, create the X=Y list for UPDATE
|
||
|
foreach (DataRow schemaRow in _dbSchemaTable.Rows) {
|
||
|
if (!IncludedInUpdate (schemaRow))
|
||
|
continue;
|
||
|
if (columns.Length > 0)
|
||
|
columns.Append (", ");
|
||
|
|
||
|
if (option)
|
||
|
parameter = CreateParameter (_updateCommand, schemaRow, false);
|
||
|
else
|
||
|
parameter = CreateParameter (_updateCommand, parmIndex++, schemaRow);
|
||
|
parameter.SourceVersion = DataRowVersion.Current;
|
||
|
ApplyParameterInfo (parameter, schemaRow, StatementType.Update, false);
|
||
|
//parameter.IsNullable = (bool) schemaRow ["AllowDBNull"];
|
||
|
columns.Append (String.Format ("{0} = {1}", GetQuotedString (parameter.SourceColumn), parameter.ParameterName));
|
||
|
}
|
||
|
|
||
|
// Now, create the WHERE clause. This may be optimizable, but it would be ugly to incorporate
|
||
|
// into the loop above. "Premature optimization is the root of all evil." -- Knuth
|
||
|
foreach (DataRow schemaRow in _dbSchemaTable.Rows) {
|
||
|
if (!schemaRow.IsNull ("IsExpression") && (bool) schemaRow ["IsExpression"] == true)
|
||
|
continue;
|
||
|
|
||
|
if (!IncludedInWhereClause (schemaRow))
|
||
|
continue;
|
||
|
|
||
|
if (whereClause.Length > 0)
|
||
|
whereClause.Append (" AND ");
|
||
|
|
||
|
bool isKey = (bool) schemaRow ["IsKey"];
|
||
|
if (isKey)
|
||
|
keyFound = true;
|
||
|
|
||
|
//ms.net 1.1 generates the null check for columns even if AllowDBNull is false
|
||
|
//while ms.net 2.0 does not. Anyways, since both forms are logically equivalent
|
||
|
//following the 2.0 approach
|
||
|
bool allowNull = (bool) schemaRow ["AllowDBNull"];
|
||
|
if (!isKey && allowNull) {
|
||
|
parameter = _updateCommand.CreateParameter ();
|
||
|
if (option) {
|
||
|
parameter.ParameterName = String.Format ("@IsNull_{0}",
|
||
|
schemaRow ["BaseColumnName"]);
|
||
|
} else {
|
||
|
parameter.ParameterName = String.Format ("@p{0}", parmIndex++);
|
||
|
}
|
||
|
parameter.DbType = DbType.Int32;
|
||
|
parameter.Value = 1;
|
||
|
parameter.SourceColumn = (string) schemaRow ["BaseColumnName"];
|
||
|
parameter.SourceColumnNullMapping = true;
|
||
|
parameter.SourceVersion = DataRowVersion.Original;
|
||
|
whereClause.Append ("(");
|
||
|
whereClause.Append (String.Format (clause1, parameter.ParameterName,
|
||
|
GetQuotedString ((string) schemaRow ["BaseColumnName"])));
|
||
|
whereClause.Append (" OR ");
|
||
|
_updateCommand.Parameters.Add (parameter);
|
||
|
}
|
||
|
|
||
|
if (option)
|
||
|
parameter = CreateParameter (_updateCommand, schemaRow, true);
|
||
|
else
|
||
|
parameter = CreateParameter (_updateCommand, parmIndex++, schemaRow);
|
||
|
parameter.SourceVersion = DataRowVersion.Original;
|
||
|
//parameter.IsNullable = allowNull;
|
||
|
ApplyParameterInfo (parameter, schemaRow, StatementType.Update, true);
|
||
|
|
||
|
whereClause.Append (String.Format (clause2, GetQuotedString (parameter.SourceColumn), parameter.ParameterName));
|
||
|
|
||
|
if (!isKey && allowNull)
|
||
|
whereClause.Append (")");
|
||
|
}
|
||
|
if (!keyFound)
|
||
|
throw new InvalidOperationException ("Dynamic SQL generation for the UpdateCommand is not supported against a SelectCommand that does not return any key column information.");
|
||
|
|
||
|
// We're all done, so bring it on home
|
||
|
string sql = String.Format ("{0}{1} WHERE ({2})", command, columns.ToString (), whereClause.ToString ());
|
||
|
_updateCommand.CommandText = sql;
|
||
|
_dbCommand = _updateCommand;
|
||
|
return _updateCommand;
|
||
|
}
|
||
|
|
||
|
private DbParameter CreateParameter (DbCommand _dbCommand, DataRow schemaRow, bool whereClause)
|
||
|
{
|
||
|
string sourceColumn = (string) schemaRow ["BaseColumnName"];
|
||
|
DbParameter parameter = _dbCommand.CreateParameter ();
|
||
|
if (whereClause)
|
||
|
parameter.ParameterName = GetParameterName ("Original_" + sourceColumn);
|
||
|
else
|
||
|
parameter.ParameterName = GetParameterName (sourceColumn);
|
||
|
parameter.SourceColumn = sourceColumn;
|
||
|
//parameter.Size = (int) schemaRow ["ColumnSize"];
|
||
|
_dbCommand.Parameters.Add (parameter);
|
||
|
return parameter;
|
||
|
}
|
||
|
|
||
|
private DbParameter CreateParameter (DbCommand _dbCommand, int paramIndex, DataRow schemaRow)
|
||
|
{
|
||
|
string sourceColumn = (string) schemaRow ["BaseColumnName"];
|
||
|
DbParameter parameter = _dbCommand.CreateParameter ();
|
||
|
parameter.ParameterName = GetParameterName (paramIndex);
|
||
|
parameter.SourceColumn = sourceColumn;
|
||
|
//parameter.Size = (int) schemaRow ["ColumnSize"];
|
||
|
_dbCommand.Parameters.Add (parameter);
|
||
|
return parameter;
|
||
|
}
|
||
|
|
||
|
[DefaultValue (CatalogLocation.Start)]
|
||
|
public virtual CatalogLocation CatalogLocation {
|
||
|
get { return _catalogLocation; }
|
||
|
set {
|
||
|
CheckEnumValue (typeof (CatalogLocation),
|
||
|
(int) value);
|
||
|
_catalogLocation = value;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
[DefaultValue (".")]
|
||
|
public virtual string CatalogSeparator {
|
||
|
get {
|
||
|
if (_catalogSeparator == null || _catalogSeparator.Length == 0)
|
||
|
return SEPARATOR_DEFAULT;
|
||
|
return _catalogSeparator;
|
||
|
}
|
||
|
set { _catalogSeparator = value; }
|
||
|
}
|
||
|
|
||
|
[DefaultValue (ConflictOption.CompareAllSearchableValues)]
|
||
|
public virtual ConflictOption ConflictOption {
|
||
|
get { return _conflictOption; }
|
||
|
set {
|
||
|
CheckEnumValue (typeof (ConflictOption),
|
||
|
(int) value);
|
||
|
_conflictOption = value;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
[DesignerSerializationVisibility (DesignerSerializationVisibility.Hidden)]
|
||
|
[Browsable (false)]
|
||
|
public DbDataAdapter DataAdapter {
|
||
|
get { return _dbDataAdapter; }
|
||
|
set { if (value != null)
|
||
|
SetRowUpdatingHandler (value);
|
||
|
_dbDataAdapter = value;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
[DefaultValue ("")]
|
||
|
public virtual string QuotePrefix {
|
||
|
get {
|
||
|
if (_quotePrefix == null)
|
||
|
return string.Empty;
|
||
|
return _quotePrefix;
|
||
|
}
|
||
|
set {
|
||
|
if (IsCommandGenerated)
|
||
|
throw new InvalidOperationException (
|
||
|
"QuotePrefix cannot be set after " +
|
||
|
"an Insert, Update or Delete command " +
|
||
|
"has been generated.");
|
||
|
_quotePrefix = value;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
[DefaultValue ("")]
|
||
|
public virtual string QuoteSuffix {
|
||
|
get {
|
||
|
if (_quoteSuffix == null)
|
||
|
return string.Empty;
|
||
|
return _quoteSuffix;
|
||
|
}
|
||
|
set {
|
||
|
if (IsCommandGenerated)
|
||
|
throw new InvalidOperationException (
|
||
|
"QuoteSuffix cannot be set after " +
|
||
|
"an Insert, Update or Delete command " +
|
||
|
"has been generated.");
|
||
|
_quoteSuffix = value;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
[DefaultValue (".")]
|
||
|
public virtual string SchemaSeparator {
|
||
|
get {
|
||
|
if (_schemaSeparator == null || _schemaSeparator.Length == 0)
|
||
|
return SEPARATOR_DEFAULT;
|
||
|
return _schemaSeparator;
|
||
|
}
|
||
|
set { _schemaSeparator = value; }
|
||
|
}
|
||
|
|
||
|
[DefaultValue (false)]
|
||
|
public bool SetAllValues {
|
||
|
get { return _setAllValues; }
|
||
|
set { _setAllValues = value; }
|
||
|
}
|
||
|
|
||
|
private DbCommand SourceCommand {
|
||
|
get {
|
||
|
if (_dbDataAdapter != null)
|
||
|
return _dbDataAdapter.SelectCommand;
|
||
|
return null;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
protected abstract void ApplyParameterInfo (DbParameter parameter,
|
||
|
DataRow row,
|
||
|
StatementType statementType,
|
||
|
bool whereClause);
|
||
|
|
||
|
protected override void Dispose (bool disposing)
|
||
|
{
|
||
|
if (!_disposed) {
|
||
|
if (disposing) {
|
||
|
if (_insertCommand != null)
|
||
|
_insertCommand.Dispose ();
|
||
|
if (_deleteCommand != null)
|
||
|
_deleteCommand.Dispose ();
|
||
|
if (_updateCommand != null)
|
||
|
_updateCommand.Dispose ();
|
||
|
if (_dbSchemaTable != null)
|
||
|
_dbSchemaTable.Dispose ();
|
||
|
}
|
||
|
_disposed = true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public DbCommand GetDeleteCommand ()
|
||
|
{
|
||
|
return GetDeleteCommand (false);
|
||
|
}
|
||
|
|
||
|
public DbCommand GetDeleteCommand (bool useColumnsForParameterNames)
|
||
|
{
|
||
|
BuildCache (true);
|
||
|
if (_deleteCommand == null || useColumnsForParameterNames)
|
||
|
return CreateDeleteCommand (useColumnsForParameterNames);
|
||
|
return _deleteCommand;
|
||
|
}
|
||
|
|
||
|
public DbCommand GetInsertCommand ()
|
||
|
{
|
||
|
return GetInsertCommand (false, null);
|
||
|
}
|
||
|
|
||
|
public DbCommand GetInsertCommand (bool useColumnsForParameterNames)
|
||
|
{
|
||
|
return GetInsertCommand (useColumnsForParameterNames, null);
|
||
|
}
|
||
|
|
||
|
internal DbCommand GetInsertCommand (bool useColumnsForParameterNames, DataRow row)
|
||
|
{
|
||
|
BuildCache (true);
|
||
|
if (_insertCommand == null || useColumnsForParameterNames)
|
||
|
return CreateInsertCommand (useColumnsForParameterNames, row);
|
||
|
return _insertCommand;
|
||
|
}
|
||
|
|
||
|
public DbCommand GetUpdateCommand ()
|
||
|
{
|
||
|
return GetUpdateCommand (false);
|
||
|
}
|
||
|
|
||
|
public DbCommand GetUpdateCommand (bool useColumnsForParameterNames)
|
||
|
{
|
||
|
BuildCache (true);
|
||
|
if (_updateCommand == null || useColumnsForParameterNames)
|
||
|
return CreateUpdateCommand (useColumnsForParameterNames);
|
||
|
return _updateCommand;
|
||
|
}
|
||
|
|
||
|
protected virtual DbCommand InitializeCommand (DbCommand command)
|
||
|
{
|
||
|
if (_dbCommand == null) {
|
||
|
_dbCommand = SourceCommand;
|
||
|
} else {
|
||
|
_dbCommand.CommandTimeout = 30;
|
||
|
_dbCommand.Transaction = null;
|
||
|
_dbCommand.CommandType = CommandType.Text;
|
||
|
_dbCommand.UpdatedRowSource = UpdateRowSource.None;
|
||
|
}
|
||
|
return _dbCommand;
|
||
|
|
||
|
}
|
||
|
|
||
|
public virtual string QuoteIdentifier (string unquotedIdentifier)
|
||
|
{
|
||
|
throw new NotSupportedException ();
|
||
|
}
|
||
|
|
||
|
public virtual string UnquoteIdentifier (string quotedIdentifier)
|
||
|
{
|
||
|
if (quotedIdentifier == null) {
|
||
|
throw new ArgumentNullException ("Quoted identifier parameter cannot be null");
|
||
|
}
|
||
|
string unquotedIdentifier = quotedIdentifier.Trim ();
|
||
|
if (unquotedIdentifier.StartsWith (this.QuotePrefix)) {
|
||
|
unquotedIdentifier = unquotedIdentifier.Remove (0, 1);
|
||
|
}
|
||
|
if (unquotedIdentifier.EndsWith (this.QuoteSuffix)) {
|
||
|
unquotedIdentifier = unquotedIdentifier.Remove (unquotedIdentifier.Length - 1, 1);
|
||
|
}
|
||
|
return unquotedIdentifier;
|
||
|
}
|
||
|
|
||
|
public virtual void RefreshSchema ()
|
||
|
{
|
||
|
_tableName = String.Empty;
|
||
|
_dbSchemaTable = null;
|
||
|
_deleteCommand = null;
|
||
|
_updateCommand = null;
|
||
|
_insertCommand = null;
|
||
|
}
|
||
|
|
||
|
protected void RowUpdatingHandler (RowUpdatingEventArgs rowUpdatingEvent)
|
||
|
{
|
||
|
if (rowUpdatingEvent.Command != null)
|
||
|
return;
|
||
|
try {
|
||
|
switch (rowUpdatingEvent.StatementType) {
|
||
|
case StatementType.Insert:
|
||
|
rowUpdatingEvent.Command = GetInsertCommand (false, rowUpdatingEvent.Row);
|
||
|
break;
|
||
|
case StatementType.Update:
|
||
|
rowUpdatingEvent.Command = GetUpdateCommand ();
|
||
|
break;
|
||
|
case StatementType.Delete:
|
||
|
rowUpdatingEvent.Command = GetDeleteCommand ();
|
||
|
break;
|
||
|
}
|
||
|
} catch (Exception e) {
|
||
|
rowUpdatingEvent.Errors = e;
|
||
|
rowUpdatingEvent.Status = UpdateStatus.ErrorsOccurred;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
protected abstract string GetParameterName (int parameterOrdinal);
|
||
|
protected abstract string GetParameterName (String parameterName);
|
||
|
protected abstract string GetParameterPlaceholder (int parameterOrdinal);
|
||
|
|
||
|
protected abstract void SetRowUpdatingHandler (DbDataAdapter adapter);
|
||
|
|
||
|
protected virtual DataTable GetSchemaTable (DbCommand sourceCommand)
|
||
|
{
|
||
|
using (DbDataReader rdr = sourceCommand.ExecuteReader ())
|
||
|
return rdr.GetSchemaTable ();
|
||
|
}
|
||
|
|
||
|
static void CheckEnumValue (Type type, int value)
|
||
|
{
|
||
|
if (Enum.IsDefined (type, value))
|
||
|
return;
|
||
|
|
||
|
string typename = type.Name;
|
||
|
string msg = string.Format (CultureInfo.CurrentCulture,
|
||
|
"Value {0} is not valid for {1}.", value,
|
||
|
typename);
|
||
|
throw new ArgumentOutOfRangeException (typename, msg);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|