2016-08-03 10:59:49 +00:00
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
// <copyright file="OdbcCommandBuilder.cs" company="Microsoft">
|
|
|
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
|
|
// </copyright>
|
2017-08-21 15:34:15 +00:00
|
|
|
// <owner current="true" primary="true">Microsoft</owner>
|
|
|
|
// <owner current="true" primary="false">Microsoft</owner>
|
2016-08-03 10:59:49 +00:00
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
using System;
|
|
|
|
using System.Collections;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.ComponentModel;
|
|
|
|
using System.Data;
|
|
|
|
using System.Data.Common;
|
|
|
|
using System.Data.ProviderBase;
|
|
|
|
using System.Diagnostics;
|
|
|
|
using System.Runtime.InteropServices;
|
|
|
|
using System.Threading;
|
|
|
|
using System.Globalization;
|
|
|
|
using System.Text;
|
|
|
|
|
|
|
|
|
|
|
|
namespace System.Data.Odbc {
|
|
|
|
|
|
|
|
public sealed class OdbcCommandBuilder : DbCommandBuilder {
|
|
|
|
|
|
|
|
public OdbcCommandBuilder() : base() {
|
|
|
|
GC.SuppressFinalize(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
public OdbcCommandBuilder(OdbcDataAdapter adapter) : this() {
|
|
|
|
DataAdapter = adapter;
|
|
|
|
}
|
|
|
|
|
|
|
|
[
|
|
|
|
DefaultValue(null),
|
|
|
|
ResCategoryAttribute(Res.DataCategory_Update),
|
|
|
|
ResDescriptionAttribute(Res.OdbcCommandBuilder_DataAdapter), // MDAC 60524
|
|
|
|
]
|
|
|
|
new public OdbcDataAdapter DataAdapter {
|
|
|
|
get {
|
|
|
|
return (base.DataAdapter as OdbcDataAdapter);
|
|
|
|
}
|
|
|
|
set {
|
|
|
|
base.DataAdapter = value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void OdbcRowUpdatingHandler(object sender, OdbcRowUpdatingEventArgs ruevent) {
|
|
|
|
RowUpdatingHandler(ruevent);
|
|
|
|
}
|
|
|
|
|
|
|
|
new public OdbcCommand GetInsertCommand() {
|
|
|
|
return (OdbcCommand) base.GetInsertCommand();
|
|
|
|
}
|
|
|
|
new public OdbcCommand GetInsertCommand(bool useColumnsForParameterNames) {
|
|
|
|
return (OdbcCommand) base.GetInsertCommand(useColumnsForParameterNames);
|
|
|
|
}
|
|
|
|
|
|
|
|
new public OdbcCommand GetUpdateCommand() {
|
|
|
|
return (OdbcCommand) base.GetUpdateCommand();
|
|
|
|
}
|
|
|
|
new public OdbcCommand GetUpdateCommand(bool useColumnsForParameterNames) {
|
|
|
|
return (OdbcCommand) base.GetUpdateCommand(useColumnsForParameterNames);
|
|
|
|
}
|
|
|
|
|
|
|
|
new public OdbcCommand GetDeleteCommand() {
|
|
|
|
return (OdbcCommand) base.GetDeleteCommand();
|
|
|
|
}
|
|
|
|
new public OdbcCommand GetDeleteCommand(bool useColumnsForParameterNames) {
|
|
|
|
return (OdbcCommand) base.GetDeleteCommand(useColumnsForParameterNames);
|
|
|
|
}
|
|
|
|
|
|
|
|
override protected string GetParameterName(int parameterOrdinal) {
|
|
|
|
return "p" + parameterOrdinal.ToString(System.Globalization.CultureInfo.InvariantCulture);
|
|
|
|
}
|
|
|
|
override protected string GetParameterName(string parameterName) {
|
|
|
|
return parameterName;
|
|
|
|
}
|
|
|
|
|
|
|
|
override protected string GetParameterPlaceholder(int parameterOrdinal) {
|
|
|
|
return "?";
|
|
|
|
}
|
|
|
|
|
|
|
|
override protected void ApplyParameterInfo(DbParameter parameter, DataRow datarow, StatementType statementType, bool whereClause) {
|
|
|
|
OdbcParameter p = (OdbcParameter) parameter;
|
|
|
|
object valueType = datarow[SchemaTableColumn.ProviderType];
|
|
|
|
p.OdbcType = (OdbcType) valueType;
|
|
|
|
|
|
|
|
object bvalue = datarow[SchemaTableColumn.NumericPrecision];
|
|
|
|
if (DBNull.Value != bvalue) {
|
|
|
|
byte bval = (byte)(short)bvalue;
|
|
|
|
p.PrecisionInternal = ((0xff != bval) ? bval : (byte)0);
|
|
|
|
}
|
|
|
|
|
|
|
|
bvalue = datarow[SchemaTableColumn.NumericScale];
|
|
|
|
if (DBNull.Value != bvalue) {
|
|
|
|
byte bval = (byte)(short)bvalue;
|
|
|
|
p.ScaleInternal = ((0xff != bval) ? bval : (byte)0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static public void DeriveParameters(OdbcCommand command) {
|
|
|
|
// MDAC 65927
|
|
|
|
OdbcConnection.ExecutePermission.Demand();
|
|
|
|
|
|
|
|
if (null == command) {
|
|
|
|
throw ADP.ArgumentNull("command");
|
|
|
|
}
|
|
|
|
switch (command.CommandType) {
|
|
|
|
case System.Data.CommandType.Text:
|
|
|
|
throw ADP.DeriveParametersNotSupported(command);
|
|
|
|
case System.Data.CommandType.StoredProcedure:
|
|
|
|
break;
|
|
|
|
case System.Data.CommandType.TableDirect:
|
|
|
|
// CommandType.TableDirect - do nothing, parameters are not supported
|
|
|
|
throw ADP.DeriveParametersNotSupported(command);
|
|
|
|
default:
|
|
|
|
throw ADP.InvalidCommandType(command.CommandType);
|
|
|
|
}
|
|
|
|
if (ADP.IsEmpty(command.CommandText)) {
|
|
|
|
throw ADP.CommandTextRequired(ADP.DeriveParameters);
|
|
|
|
}
|
|
|
|
|
|
|
|
OdbcConnection connection = command.Connection;
|
|
|
|
|
|
|
|
if (null == connection) {
|
|
|
|
throw ADP.ConnectionRequired(ADP.DeriveParameters);
|
|
|
|
}
|
|
|
|
|
|
|
|
ConnectionState state = connection.State;
|
|
|
|
|
|
|
|
if (ConnectionState.Open != state) {
|
|
|
|
throw ADP.OpenConnectionRequired(ADP.DeriveParameters, state);
|
|
|
|
}
|
|
|
|
|
|
|
|
OdbcParameter[] list = DeriveParametersFromStoredProcedure(connection, command);
|
|
|
|
|
|
|
|
OdbcParameterCollection parameters = command.Parameters;
|
|
|
|
parameters.Clear();
|
|
|
|
|
|
|
|
int count = list.Length;
|
|
|
|
if (0 < count) {
|
|
|
|
for(int i = 0; i < list.Length; ++i) {
|
|
|
|
parameters.Add(list[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// DeriveParametersFromStoredProcedure (
|
|
|
|
// OdbcConnection connection,
|
|
|
|
// OdbcCommand command);
|
|
|
|
//
|
|
|
|
// Uses SQLProcedureColumns to create an array of OdbcParameters
|
|
|
|
//
|
|
|
|
|
|
|
|
static private OdbcParameter[] DeriveParametersFromStoredProcedure(OdbcConnection connection, OdbcCommand command) {
|
|
|
|
List<OdbcParameter> rParams = new List<OdbcParameter>();
|
|
|
|
|
|
|
|
// following call ensures that the command has a statement handle allocated
|
|
|
|
CMDWrapper cmdWrapper = command.GetStatementHandle();
|
|
|
|
OdbcStatementHandle hstmt = cmdWrapper.StatementHandle;
|
|
|
|
int cColsAffected;
|
|
|
|
|
|
|
|
// maps an enforced 4-part qualified string as follows
|
|
|
|
// parts[0] = null - ignored but removal would be a run-time breaking change from V1.0
|
|
|
|
// parts[1] = CatalogName (optional, may be null)
|
|
|
|
// parts[2] = SchemaName (optional, may be null)
|
|
|
|
// parts[3] = ProcedureName
|
|
|
|
//
|
|
|
|
string quote = connection.QuoteChar(ADP.DeriveParameters);
|
|
|
|
string[] parts = MultipartIdentifier.ParseMultipartIdentifier(command.CommandText, quote, quote, '.', 4, true, Res.ODBC_ODBCCommandText, false);
|
|
|
|
if (null == parts[3]) { // match everett behavior, if the commandtext is nothing but whitespace set the command text to the whitespace
|
|
|
|
parts[3] = command.CommandText;
|
|
|
|
}
|
|
|
|
// note: native odbc appears to ignore all but the procedure name
|
|
|
|
ODBC32.RetCode retcode = hstmt.ProcedureColumns(parts[1], parts[2], parts[3], null);
|
|
|
|
|
|
|
|
// Note: the driver does not return an error if the given stored procedure does not exist
|
|
|
|
// therefore we cannot handle that case and just return not parameters.
|
|
|
|
|
|
|
|
if (ODBC32.RetCode.SUCCESS != retcode) {
|
|
|
|
connection.HandleError(hstmt, retcode);
|
|
|
|
}
|
|
|
|
|
|
|
|
using (OdbcDataReader reader = new OdbcDataReader(command, cmdWrapper, CommandBehavior.Default)) {
|
|
|
|
reader.FirstResult();
|
|
|
|
cColsAffected = reader.FieldCount;
|
|
|
|
|
|
|
|
// go through the returned rows and filter out relevant parameter data
|
|
|
|
//
|
|
|
|
while (reader.Read()) {
|
|
|
|
// devnote: column types are specified in the ODBC Programmer's Reference
|
|
|
|
// COLUMN_TYPE Smallint 16bit
|
|
|
|
// COLUMN_SIZE Integer 32bit
|
|
|
|
// DECIMAL_DIGITS Smallint 16bit
|
|
|
|
// NUM_PREC_RADIX Smallint 16bit
|
|
|
|
|
|
|
|
OdbcParameter parameter = new OdbcParameter();
|
|
|
|
|
|
|
|
parameter.ParameterName = reader.GetString(ODBC32.COLUMN_NAME-1);
|
|
|
|
switch ((ODBC32.SQL_PARAM)reader.GetInt16(ODBC32.COLUMN_TYPE-1)){
|
|
|
|
case ODBC32.SQL_PARAM.INPUT:
|
|
|
|
parameter.Direction = ParameterDirection.Input;
|
|
|
|
break;
|
|
|
|
case ODBC32.SQL_PARAM.OUTPUT:
|
|
|
|
parameter.Direction = ParameterDirection.Output;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ODBC32.SQL_PARAM.INPUT_OUTPUT:
|
|
|
|
parameter.Direction = ParameterDirection.InputOutput;
|
|
|
|
break;
|
|
|
|
case ODBC32.SQL_PARAM.RETURN_VALUE:
|
|
|
|
parameter.Direction = ParameterDirection.ReturnValue;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
Debug.Assert(false, "Unexpected Parametertype while DeriveParamters");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
parameter.OdbcType = TypeMap.FromSqlType((ODBC32.SQL_TYPE)reader.GetInt16(ODBC32.DATA_TYPE-1))._odbcType;
|
|
|
|
parameter.Size = (int)reader.GetInt32(ODBC32.COLUMN_SIZE-1);
|
|
|
|
switch(parameter.OdbcType){
|
|
|
|
case OdbcType.Decimal:
|
|
|
|
case OdbcType.Numeric:
|
|
|
|
parameter.ScaleInternal = (Byte)reader.GetInt16(ODBC32.DECIMAL_DIGITS-1);
|
|
|
|
parameter.PrecisionInternal = (Byte)reader.GetInt16(ODBC32.NUM_PREC_RADIX-1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
rParams.Add (parameter);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
retcode = hstmt.CloseCursor();
|
|
|
|
return rParams.ToArray();;
|
|
|
|
}
|
|
|
|
|
|
|
|
public override string QuoteIdentifier(string unquotedIdentifier){
|
|
|
|
return QuoteIdentifier(unquotedIdentifier, null /* use DataAdapter.SelectCommand.Connection if available */);
|
|
|
|
}
|
|
|
|
public string QuoteIdentifier( string unquotedIdentifier, OdbcConnection connection){
|
|
|
|
|
|
|
|
ADP.CheckArgumentNull(unquotedIdentifier, "unquotedIdentifier");
|
|
|
|
|
|
|
|
// if the user has specificed a prefix use the user specified prefix and suffix
|
|
|
|
// otherwise get them from the provider
|
|
|
|
string quotePrefix = QuotePrefix;
|
|
|
|
string quoteSuffix = QuoteSuffix;
|
|
|
|
if (ADP.IsEmpty(quotePrefix) == true) {
|
|
|
|
if (connection == null) {
|
|
|
|
// VSTFDEVDIV 479567: use the adapter's connection if QuoteIdentifier was called from
|
|
|
|
// DbCommandBuilder instance (which does not have an overload that gets connection object)
|
|
|
|
connection = base.GetConnection() as OdbcConnection;
|
|
|
|
if (connection == null) {
|
|
|
|
throw ADP.QuotePrefixNotSet(ADP.QuoteIdentifier);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
quotePrefix = connection.QuoteChar(ADP.QuoteIdentifier);
|
|
|
|
quoteSuffix = quotePrefix;
|
|
|
|
}
|
|
|
|
|
|
|
|
// by the ODBC spec "If the data source does not support quoted identifiers, a blank is returned."
|
|
|
|
// So if a blank is returned the string is returned unchanged. Otherwise the returned string is used
|
|
|
|
// to quote the string
|
|
|
|
if ((ADP.IsEmpty(quotePrefix) == false) && (quotePrefix != " ")) {
|
|
|
|
return ADP.BuildQuotedString(quotePrefix,quoteSuffix,unquotedIdentifier);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return unquotedIdentifier;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
override protected void SetRowUpdatingHandler(DbDataAdapter adapter) {
|
|
|
|
Debug.Assert(adapter is OdbcDataAdapter, "!OdbcDataAdapter");
|
|
|
|
if (adapter == base.DataAdapter) { // removal case
|
|
|
|
((OdbcDataAdapter)adapter).RowUpdating -= OdbcRowUpdatingHandler;
|
|
|
|
}
|
|
|
|
else { // adding case
|
|
|
|
((OdbcDataAdapter)adapter).RowUpdating += OdbcRowUpdatingHandler;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public override string UnquoteIdentifier( string quotedIdentifier){
|
|
|
|
return UnquoteIdentifier(quotedIdentifier, null /* use DataAdapter.SelectCommand.Connection if available */);
|
|
|
|
}
|
|
|
|
|
|
|
|
public string UnquoteIdentifier(string quotedIdentifier, OdbcConnection connection){
|
|
|
|
|
|
|
|
ADP.CheckArgumentNull(quotedIdentifier, "quotedIdentifier");
|
|
|
|
|
|
|
|
|
|
|
|
// if the user has specificed a prefix use the user specified prefix and suffix
|
|
|
|
// otherwise get them from the provider
|
|
|
|
string quotePrefix = QuotePrefix;
|
|
|
|
string quoteSuffix = QuoteSuffix;
|
|
|
|
if (ADP.IsEmpty(quotePrefix) == true) {
|
|
|
|
if (connection == null) {
|
|
|
|
// VSTFDEVDIV 479567: use the adapter's connection if UnquoteIdentifier was called from
|
|
|
|
// DbCommandBuilder instance (which does not have an overload that gets connection object)
|
|
|
|
connection = base.GetConnection() as OdbcConnection;
|
|
|
|
if (connection == null) {
|
|
|
|
throw ADP.QuotePrefixNotSet(ADP.UnquoteIdentifier);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
quotePrefix = connection.QuoteChar(ADP.UnquoteIdentifier);
|
|
|
|
quoteSuffix = quotePrefix;
|
|
|
|
}
|
|
|
|
|
|
|
|
String unquotedIdentifier;
|
|
|
|
// by the ODBC spec "If the data source does not support quoted identifiers, a blank is returned."
|
|
|
|
// So if a blank is returned the string is returned unchanged. Otherwise the returned string is used
|
|
|
|
// to unquote the string
|
|
|
|
if ((ADP.IsEmpty(quotePrefix) == false) || (quotePrefix != " ")) {
|
|
|
|
// ignoring the return value because it is acceptable for the quotedString to not be quoted in this
|
|
|
|
// context.
|
|
|
|
ADP.RemoveStringQuotes(quotePrefix, quoteSuffix, quotedIdentifier, out unquotedIdentifier);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
unquotedIdentifier = quotedIdentifier;
|
|
|
|
}
|
|
|
|
return unquotedIdentifier;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|