536cd135cc
Former-commit-id: 5624ac747d633e885131e8349322922b6a59baaa
1058 lines
41 KiB
C#
1058 lines
41 KiB
C#
//------------------------------------------------------------------------------
|
|
// <copyright file="OdbcCommand.cs" company="Microsoft">
|
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
// </copyright>
|
|
// <owner current="true" primary="true">Microsoft</owner>
|
|
// <owner current="true" primary="false">Microsoft</owner>
|
|
//------------------------------------------------------------------------------
|
|
|
|
using System;
|
|
using System.ComponentModel; //Component
|
|
using System.Data;
|
|
using System.Data.Common;
|
|
using System.Data.ProviderBase;
|
|
using System.Diagnostics;
|
|
using System.Runtime.CompilerServices;
|
|
using System.Runtime.InteropServices;
|
|
using System.Threading;
|
|
|
|
// todo:
|
|
// There may be two ways to improve performance:
|
|
// 1. pool statements on the connection object
|
|
// 2. Do not create a datareader object for non-datareader returning command execution.
|
|
//
|
|
// We do not want to do the effort unless we have to squeze performance.
|
|
|
|
|
|
|
|
namespace System.Data.Odbc {
|
|
|
|
[
|
|
DefaultEvent("RecordsAffected"),
|
|
ToolboxItem(true),
|
|
Designer("Microsoft.VSDesigner.Data.VS.OdbcCommandDesigner, " + AssemblyRef.MicrosoftVSDesigner)
|
|
]
|
|
public sealed class OdbcCommand : DbCommand, ICloneable {
|
|
private static int _objectTypeCount; // Bid counter
|
|
internal readonly int ObjectID = System.Threading.Interlocked.Increment(ref _objectTypeCount);
|
|
|
|
private string _commandText;
|
|
private CommandType _commandType;
|
|
private int _commandTimeout = ADP.DefaultCommandTimeout;
|
|
private UpdateRowSource _updatedRowSource = UpdateRowSource.Both;
|
|
private bool _designTimeInvisible;
|
|
private bool _isPrepared; // true if the command is prepared
|
|
|
|
private OdbcConnection _connection;
|
|
private OdbcTransaction _transaction;
|
|
|
|
private WeakReference weakDataReaderReference;
|
|
|
|
private CMDWrapper _cmdWrapper;
|
|
|
|
private OdbcParameterCollection _parameterCollection; // Parameter collection
|
|
|
|
private ConnectionState cmdState;
|
|
|
|
public OdbcCommand() : base() {
|
|
GC.SuppressFinalize(this);
|
|
}
|
|
|
|
public OdbcCommand(string cmdText) : this() {
|
|
// note: arguments are assigned to properties so we do not have to trace them.
|
|
// We still need to include them into the argument list of the definition!
|
|
CommandText = cmdText;
|
|
}
|
|
|
|
public OdbcCommand(string cmdText, OdbcConnection connection) : this() {
|
|
CommandText = cmdText;
|
|
Connection = connection;
|
|
}
|
|
|
|
public OdbcCommand(string cmdText, OdbcConnection connection, OdbcTransaction transaction) : this() {
|
|
CommandText = cmdText;
|
|
Connection = connection;
|
|
Transaction = transaction;
|
|
}
|
|
|
|
private void DisposeDeadDataReader() {
|
|
if (ConnectionState.Fetching == cmdState) {
|
|
if (null != this.weakDataReaderReference && !this.weakDataReaderReference.IsAlive) {
|
|
if (_cmdWrapper != null) {
|
|
_cmdWrapper.FreeKeyInfoStatementHandle(ODBC32.STMT.CLOSE);
|
|
_cmdWrapper.FreeStatementHandle(ODBC32.STMT.CLOSE);
|
|
}
|
|
CloseFromDataReader();
|
|
}
|
|
}
|
|
}
|
|
|
|
private void DisposeDataReader() {
|
|
if (null != this.weakDataReaderReference) {
|
|
IDisposable reader = (IDisposable) this.weakDataReaderReference.Target;
|
|
if ((null != reader) && this.weakDataReaderReference.IsAlive) {
|
|
((IDisposable) reader).Dispose();
|
|
}
|
|
CloseFromDataReader();
|
|
}
|
|
}
|
|
|
|
internal void DisconnectFromDataReaderAndConnection () {
|
|
// get a reference to the datareader if it is alive
|
|
OdbcDataReader liveReader = null;
|
|
if (this.weakDataReaderReference != null){
|
|
OdbcDataReader reader;
|
|
reader = (OdbcDataReader)this.weakDataReaderReference.Target;
|
|
if (this.weakDataReaderReference.IsAlive) {
|
|
liveReader = reader;
|
|
}
|
|
}
|
|
|
|
// remove reference to this from the live datareader
|
|
if (liveReader != null) {
|
|
liveReader.Command = null;
|
|
}
|
|
|
|
_transaction = null;
|
|
|
|
if (null != _connection) {
|
|
_connection.RemoveWeakReference(this);
|
|
_connection = null;
|
|
}
|
|
|
|
// if the reader is dead we have to dismiss the statement
|
|
if (liveReader == null){
|
|
CloseCommandWrapper();
|
|
}
|
|
// else DataReader now has exclusive ownership
|
|
_cmdWrapper = null;
|
|
}
|
|
|
|
override protected void Dispose(bool disposing) { // MDAC 65459
|
|
if (disposing) {
|
|
// release mananged objects
|
|
// in V1.0, V1.1 the Connection,Parameters,CommandText,Transaction where reset
|
|
this.DisconnectFromDataReaderAndConnection ();
|
|
_parameterCollection = null;
|
|
CommandText = null;
|
|
}
|
|
_cmdWrapper = null; // let go of the CommandWrapper
|
|
_isPrepared = false;
|
|
|
|
base.Dispose(disposing); // notify base classes
|
|
}
|
|
|
|
internal bool Canceling {
|
|
get {
|
|
return _cmdWrapper.Canceling;
|
|
}
|
|
}
|
|
|
|
[
|
|
ResCategoryAttribute(Res.DataCategory_Data),
|
|
DefaultValue(""),
|
|
RefreshProperties(RefreshProperties.All), // MDAC 67707
|
|
ResDescriptionAttribute(Res.DbCommand_CommandText),
|
|
Editor("Microsoft.VSDesigner.Data.Odbc.Design.OdbcCommandTextEditor, " + AssemblyRef.MicrosoftVSDesigner, "System.Drawing.Design.UITypeEditor, " + AssemblyRef.SystemDrawing)
|
|
]
|
|
override public string CommandText {
|
|
get {
|
|
string value = _commandText;
|
|
return ((null != value) ? value : ADP.StrEmpty);
|
|
}
|
|
set {
|
|
if (Bid.TraceOn) {
|
|
Bid.Trace("<odbc.OdbcCommand.set_CommandText|API> %d#, '", ObjectID);
|
|
Bid.PutStr(value); // Use PutStr to write out entire string
|
|
Bid.Trace("'\n");
|
|
}
|
|
if (0 != ADP.SrcCompare(_commandText, value)) {
|
|
PropertyChanging();
|
|
_commandText = value;
|
|
}
|
|
}
|
|
}
|
|
|
|
[
|
|
ResCategoryAttribute(Res.DataCategory_Data),
|
|
ResDescriptionAttribute(Res.DbCommand_CommandTimeout),
|
|
]
|
|
override public int CommandTimeout { // V1.2.3300, XXXCommand V1.0.5000
|
|
get {
|
|
return _commandTimeout;
|
|
}
|
|
set {
|
|
Bid.Trace("<odbc.OdbcCommand.set_CommandTimeout|API> %d#, %d\n", ObjectID, value);
|
|
if (value < 0) {
|
|
throw ADP.InvalidCommandTimeout(value);
|
|
}
|
|
if (value != _commandTimeout) {
|
|
PropertyChanging();
|
|
_commandTimeout = value;
|
|
}
|
|
}
|
|
}
|
|
|
|
public void ResetCommandTimeout() { // V1.2.3300
|
|
if (ADP.DefaultCommandTimeout != _commandTimeout) {
|
|
PropertyChanging();
|
|
_commandTimeout = ADP.DefaultCommandTimeout;
|
|
}
|
|
}
|
|
|
|
private bool ShouldSerializeCommandTimeout() { // V1.2.3300
|
|
return (ADP.DefaultCommandTimeout != _commandTimeout);
|
|
}
|
|
|
|
[
|
|
DefaultValue(System.Data.CommandType.Text),
|
|
RefreshProperties(RefreshProperties.All),
|
|
ResCategoryAttribute(Res.DataCategory_Data),
|
|
ResDescriptionAttribute(Res.DbCommand_CommandType),
|
|
]
|
|
override public CommandType CommandType {
|
|
get {
|
|
CommandType cmdType = _commandType;
|
|
return ((0 != cmdType) ? cmdType : CommandType.Text);
|
|
}
|
|
set {
|
|
switch(value) { // @perfnote: Enum.IsDefined
|
|
case CommandType.Text:
|
|
case CommandType.StoredProcedure:
|
|
PropertyChanging();
|
|
_commandType = value;
|
|
break;
|
|
case CommandType.TableDirect:
|
|
throw ODBC.NotSupportedCommandType(value);
|
|
default:
|
|
throw ADP.InvalidCommandType(value);
|
|
}
|
|
}
|
|
}
|
|
|
|
// This will establish a relationship between the command and the connection
|
|
[
|
|
DefaultValue(null),
|
|
ResCategoryAttribute(Res.DataCategory_Behavior),
|
|
ResDescriptionAttribute(Res.DbCommand_Connection),
|
|
Editor("Microsoft.VSDesigner.Data.Design.DbConnectionEditor, " + AssemblyRef.MicrosoftVSDesigner, "System.Drawing.Design.UITypeEditor, " + AssemblyRef.SystemDrawing),
|
|
]
|
|
new public OdbcConnection Connection {
|
|
get {
|
|
return _connection;
|
|
}
|
|
set {
|
|
if (value != _connection) {
|
|
PropertyChanging();
|
|
this.DisconnectFromDataReaderAndConnection();
|
|
Debug.Assert(null == _cmdWrapper, "has CMDWrapper when setting connection");
|
|
_connection = value;
|
|
//OnSchemaChanged();
|
|
}
|
|
}
|
|
}
|
|
|
|
override protected DbConnection DbConnection { // V1.2.3300
|
|
get {
|
|
return Connection;
|
|
}
|
|
set {
|
|
Connection = (OdbcConnection)value;
|
|
}
|
|
}
|
|
|
|
override protected DbParameterCollection DbParameterCollection { // V1.2.3300
|
|
get {
|
|
return Parameters;
|
|
}
|
|
}
|
|
|
|
override protected DbTransaction DbTransaction { // V1.2.3300
|
|
get {
|
|
return Transaction;
|
|
}
|
|
set {
|
|
Transaction = (OdbcTransaction)value;
|
|
}
|
|
}
|
|
|
|
// @devnote: By default, the cmd object is visible on the design surface (i.e. VS7 Server Tray)
|
|
// to limit the number of components that clutter the design surface,
|
|
// when the DataAdapter design wizard generates the insert/update/delete commands it will
|
|
// set the DesignTimeVisible property to false so that cmds won't appear as individual objects
|
|
[
|
|
DefaultValue(true),
|
|
DesignOnly(true),
|
|
Browsable(false),
|
|
EditorBrowsableAttribute(EditorBrowsableState.Never),
|
|
]
|
|
public override bool DesignTimeVisible { // V1.2.3300, XXXCommand V1.0.5000
|
|
get {
|
|
return !_designTimeInvisible;
|
|
}
|
|
set {
|
|
_designTimeInvisible = !value;
|
|
TypeDescriptor.Refresh(this); // VS7 208845
|
|
}
|
|
}
|
|
|
|
internal bool HasParameters {
|
|
get {
|
|
return (null != _parameterCollection);
|
|
}
|
|
}
|
|
|
|
[
|
|
DesignerSerializationVisibility(DesignerSerializationVisibility.Content),
|
|
ResCategoryAttribute(Res.DataCategory_Data),
|
|
ResDescriptionAttribute(Res.DbCommand_Parameters),
|
|
]
|
|
new public OdbcParameterCollection Parameters {
|
|
get {
|
|
if (null == _parameterCollection) {
|
|
_parameterCollection = new OdbcParameterCollection();
|
|
}
|
|
return _parameterCollection;
|
|
}
|
|
}
|
|
|
|
[
|
|
Browsable(false),
|
|
DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden),
|
|
ResDescriptionAttribute(Res.DbCommand_Transaction),
|
|
]
|
|
new public OdbcTransaction Transaction {
|
|
get {
|
|
if ((null != _transaction) && (null == _transaction.Connection)) {
|
|
_transaction = null; // Dawn of the Dead
|
|
}
|
|
return _transaction;
|
|
}
|
|
set {
|
|
if (_transaction != value) {
|
|
PropertyChanging(); // fire event before value is validated
|
|
_transaction = value;
|
|
}
|
|
}
|
|
}
|
|
|
|
[
|
|
DefaultValue(System.Data.UpdateRowSource.Both),
|
|
ResCategoryAttribute(Res.DataCategory_Update),
|
|
ResDescriptionAttribute(Res.DbCommand_UpdatedRowSource),
|
|
]
|
|
override public UpdateRowSource UpdatedRowSource { // V1.2.3300, XXXCommand V1.0.5000
|
|
get {
|
|
return _updatedRowSource;
|
|
}
|
|
set {
|
|
switch(value) { // @perfnote: Enum.IsDefined
|
|
case UpdateRowSource.None:
|
|
case UpdateRowSource.OutputParameters:
|
|
case UpdateRowSource.FirstReturnedRecord:
|
|
case UpdateRowSource.Both:
|
|
_updatedRowSource = value;
|
|
break;
|
|
default:
|
|
throw ADP.InvalidUpdateRowSource(value);
|
|
}
|
|
}
|
|
}
|
|
|
|
internal OdbcDescriptorHandle GetDescriptorHandle(ODBC32.SQL_ATTR attribute) {
|
|
return _cmdWrapper.GetDescriptorHandle(attribute);
|
|
}
|
|
|
|
|
|
// GetStatementHandle
|
|
// ------------------
|
|
// Try to return a cached statement handle.
|
|
//
|
|
// Creates a CmdWrapper object if necessary
|
|
// If no handle is available a handle will be allocated.
|
|
// Bindings will be unbound if a handle is cached and the bindings are invalid.
|
|
//
|
|
internal CMDWrapper GetStatementHandle () {
|
|
// update the command wrapper object, allocate buffer
|
|
// create reader object
|
|
//
|
|
if (_cmdWrapper==null) {
|
|
_cmdWrapper = new CMDWrapper(_connection);
|
|
|
|
Debug.Assert(null != _connection, "GetStatementHandle without connection?");
|
|
_connection.AddWeakReference(this, OdbcReferenceCollection.CommandTag);
|
|
}
|
|
|
|
if (_cmdWrapper._dataReaderBuf == null) {
|
|
_cmdWrapper._dataReaderBuf = new CNativeBuffer(4096);
|
|
}
|
|
|
|
// if there is already a statement handle we need to do some cleanup
|
|
//
|
|
if (null == _cmdWrapper.StatementHandle) {
|
|
_isPrepared = false;
|
|
_cmdWrapper.CreateStatementHandle();
|
|
}
|
|
else if ((null != _parameterCollection) && _parameterCollection.RebindCollection) {
|
|
_cmdWrapper.FreeStatementHandle(ODBC32.STMT.RESET_PARAMS);
|
|
}
|
|
return _cmdWrapper;
|
|
}
|
|
|
|
// OdbcCommand.Cancel()
|
|
//
|
|
// In ODBC3.0 ... a call to SQLCancel when no processing is done has no effect at all
|
|
// (ODBC Programmer's Reference ...)
|
|
//
|
|
|
|
override public void Cancel() {
|
|
CMDWrapper wrapper = _cmdWrapper;
|
|
if (null != wrapper) {
|
|
wrapper.Canceling = true;
|
|
OdbcStatementHandle stmt = wrapper.StatementHandle;
|
|
if (null != stmt) {
|
|
lock (stmt) {
|
|
// Cancel the statement
|
|
ODBC32.RetCode retcode = stmt.Cancel();
|
|
|
|
// copy of StatementErrorHandler, because stmt may become null
|
|
switch(retcode) {
|
|
case ODBC32.RetCode.SUCCESS:
|
|
case ODBC32.RetCode.SUCCESS_WITH_INFO:
|
|
// don't fire info message events on cancel
|
|
break;
|
|
default:
|
|
throw wrapper.Connection.HandleErrorNoThrow(stmt, retcode);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
object ICloneable.Clone() {
|
|
OdbcCommand clone = new OdbcCommand();
|
|
Bid.Trace("<odbc.OdbcCommand.Clone|API> %d#, clone=%d#\n", ObjectID, clone.ObjectID);
|
|
clone.CommandText = CommandText;
|
|
clone.CommandTimeout = this.CommandTimeout;
|
|
clone.CommandType = CommandType;
|
|
clone.Connection = this.Connection;
|
|
clone.Transaction = this.Transaction;
|
|
clone.UpdatedRowSource = UpdatedRowSource;
|
|
|
|
if ((null != _parameterCollection) && (0 < Parameters.Count)) {
|
|
OdbcParameterCollection parameters = clone.Parameters;
|
|
foreach(ICloneable parameter in Parameters) {
|
|
parameters.Add(parameter.Clone());
|
|
}
|
|
}
|
|
return clone;
|
|
}
|
|
|
|
internal bool RecoverFromConnection() {
|
|
DisposeDeadDataReader();
|
|
return (ConnectionState.Closed == cmdState);
|
|
}
|
|
|
|
private void CloseCommandWrapper() {
|
|
CMDWrapper wrapper = _cmdWrapper;
|
|
if (null != wrapper) {
|
|
try {
|
|
wrapper.Dispose();
|
|
|
|
if (null != _connection) {
|
|
_connection.RemoveWeakReference(this);
|
|
}
|
|
}
|
|
finally {
|
|
_cmdWrapper = null;
|
|
}
|
|
}
|
|
}
|
|
|
|
internal void CloseFromConnection () {
|
|
if (null != _parameterCollection) {
|
|
_parameterCollection.RebindCollection = true;
|
|
}
|
|
DisposeDataReader();
|
|
CloseCommandWrapper();
|
|
_isPrepared = false;
|
|
_transaction = null;
|
|
}
|
|
|
|
internal void CloseFromDataReader() {
|
|
this.weakDataReaderReference = null;
|
|
this.cmdState = ConnectionState.Closed;
|
|
}
|
|
|
|
new public OdbcParameter CreateParameter() {
|
|
return new OdbcParameter();
|
|
}
|
|
|
|
override protected DbParameter CreateDbParameter() {
|
|
return CreateParameter();
|
|
}
|
|
|
|
override protected DbDataReader ExecuteDbDataReader(CommandBehavior behavior) {
|
|
return ExecuteReader(behavior);
|
|
}
|
|
|
|
override public int ExecuteNonQuery() {
|
|
OdbcConnection.ExecutePermission.Demand();
|
|
using (OdbcDataReader reader = ExecuteReaderObject(0, ADP.ExecuteNonQuery, false)) {
|
|
reader.Close();
|
|
return reader.RecordsAffected;
|
|
}
|
|
}
|
|
|
|
new public OdbcDataReader ExecuteReader() {
|
|
return ExecuteReader(0/*CommandBehavior*/);
|
|
}
|
|
|
|
|
|
new public OdbcDataReader ExecuteReader(CommandBehavior behavior) {
|
|
OdbcConnection.ExecutePermission.Demand();
|
|
return ExecuteReaderObject(behavior, ADP.ExecuteReader, true);
|
|
}
|
|
|
|
internal OdbcDataReader ExecuteReaderFromSQLMethod(object[] methodArguments,
|
|
ODBC32.SQL_API method){
|
|
|
|
return ExecuteReaderObject(CommandBehavior.Default,method.ToString(),true,methodArguments,method);
|
|
|
|
}
|
|
|
|
private OdbcDataReader ExecuteReaderObject(CommandBehavior behavior, string method, bool needReader) { // MDAC 68324
|
|
|
|
if ((CommandText == null) || (CommandText.Length == 0)) {
|
|
throw (ADP.CommandTextRequired(method));
|
|
}
|
|
// using all functions to tell ExecuteReaderObject that
|
|
return ExecuteReaderObject(behavior,method,needReader,null,ODBC32.SQL_API.SQLEXECDIRECT);
|
|
}
|
|
|
|
private OdbcDataReader ExecuteReaderObject(CommandBehavior behavior,
|
|
string method,
|
|
bool needReader,
|
|
object[] methodArguments,
|
|
ODBC32.SQL_API odbcApiMethod) { // MDAC 68324
|
|
|
|
OdbcDataReader localReader = null;
|
|
try {
|
|
DisposeDeadDataReader(); // this is a no-op if cmdState is not Fetching
|
|
ValidateConnectionAndTransaction(method); // cmdState will change to Executing
|
|
|
|
if(0 != (CommandBehavior.SingleRow & behavior)) {
|
|
// CommandBehavior.SingleRow implies CommandBehavior.SingleResult
|
|
behavior |= CommandBehavior.SingleResult;
|
|
}
|
|
|
|
ODBC32.RetCode retcode;
|
|
|
|
OdbcStatementHandle stmt = GetStatementHandle().StatementHandle;
|
|
_cmdWrapper.Canceling = false;
|
|
|
|
if(null != weakDataReaderReference) {
|
|
if(weakDataReaderReference.IsAlive) {
|
|
object target = weakDataReaderReference.Target;
|
|
if(null != target && weakDataReaderReference.IsAlive) {
|
|
if(!((OdbcDataReader)target).IsClosed) {
|
|
throw ADP.OpenReaderExists(); // MDAC 66411
|
|
}
|
|
}
|
|
}
|
|
}
|
|
localReader = new OdbcDataReader(this, _cmdWrapper, behavior);
|
|
|
|
//Set command properties
|
|
//Not all drivers support timeout. So fail silently if error
|
|
if(!Connection.ProviderInfo.NoQueryTimeout) {
|
|
TrySetStatementAttribute(stmt,
|
|
ODBC32.SQL_ATTR.QUERY_TIMEOUT,
|
|
(IntPtr)this.CommandTimeout);
|
|
}
|
|
|
|
// todo: If we remember the state we can omit a lot of SQLSetStmtAttrW calls ...
|
|
// if we do not create a reader we do not even need to do that
|
|
if(needReader) {
|
|
if(Connection.IsV3Driver) {
|
|
if(!Connection.ProviderInfo.NoSqlSoptSSNoBrowseTable && !Connection.ProviderInfo.NoSqlSoptSSHiddenColumns) {
|
|
// Need to get the metadata information
|
|
|
|
//SQLServer actually requires browse info turned on ahead of time...
|
|
//Note: We ignore any failures, since this is SQLServer specific
|
|
//We won't specialcase for SQL Server but at least for non-V3 drivers
|
|
if(localReader.IsBehavior(CommandBehavior.KeyInfo)) {
|
|
if(!_cmdWrapper._ssKeyInfoModeOn) {
|
|
TrySetStatementAttribute(stmt, (ODBC32.SQL_ATTR)ODBC32.SQL_SOPT_SS.NOBROWSETABLE, (IntPtr)ODBC32.SQL_NB.ON);
|
|
TrySetStatementAttribute(stmt, (ODBC32.SQL_ATTR)ODBC32.SQL_SOPT_SS.HIDDEN_COLUMNS, (IntPtr)ODBC32.SQL_HC.ON);
|
|
_cmdWrapper._ssKeyInfoModeOff = false;
|
|
_cmdWrapper._ssKeyInfoModeOn = true;
|
|
}
|
|
}
|
|
else {
|
|
if(!_cmdWrapper._ssKeyInfoModeOff) {
|
|
TrySetStatementAttribute(stmt, (ODBC32.SQL_ATTR)ODBC32.SQL_SOPT_SS.NOBROWSETABLE, (IntPtr)ODBC32.SQL_NB.OFF);
|
|
TrySetStatementAttribute(stmt, (ODBC32.SQL_ATTR)ODBC32.SQL_SOPT_SS.HIDDEN_COLUMNS, (IntPtr)ODBC32.SQL_HC.OFF);
|
|
_cmdWrapper._ssKeyInfoModeOff = true;
|
|
_cmdWrapper._ssKeyInfoModeOn = false;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if(localReader.IsBehavior(CommandBehavior.KeyInfo) ||
|
|
localReader.IsBehavior(CommandBehavior.SchemaOnly)) {
|
|
|
|
retcode = stmt.Prepare(CommandText);
|
|
|
|
if(ODBC32.RetCode.SUCCESS != retcode) {
|
|
_connection.HandleError(stmt, retcode);
|
|
}
|
|
}
|
|
|
|
bool mustRelease = false;
|
|
CNativeBuffer parameterBuffer = _cmdWrapper._nativeParameterBuffer;
|
|
|
|
RuntimeHelpers.PrepareConstrainedRegions();
|
|
try {
|
|
//Handle Parameters
|
|
//Note: We use the internal variable as to not instante a new object collection,
|
|
//for the the common case of using no parameters.
|
|
if((null != _parameterCollection) && (0 < _parameterCollection.Count)) {
|
|
int parameterBufferSize = _parameterCollection.CalcParameterBufferSize(this);
|
|
|
|
if(null == parameterBuffer || parameterBuffer.Length < parameterBufferSize) {
|
|
if (null != parameterBuffer) {
|
|
parameterBuffer.Dispose();
|
|
}
|
|
parameterBuffer = new CNativeBuffer(parameterBufferSize);
|
|
_cmdWrapper._nativeParameterBuffer = parameterBuffer;
|
|
}
|
|
else {
|
|
parameterBuffer.ZeroMemory();
|
|
}
|
|
|
|
parameterBuffer.DangerousAddRef(ref mustRelease);
|
|
|
|
_parameterCollection.Bind(this, _cmdWrapper, parameterBuffer);
|
|
}
|
|
|
|
if(!localReader.IsBehavior(CommandBehavior.SchemaOnly)) {
|
|
|
|
// Can't get the KeyInfo after command execution (SQL Server only since it does not support multiple
|
|
// results on the same connection). Stored procedures (SP) do not return metadata before actual execution
|
|
// Need to check the column count since the command type may not be set to SP for a SP.
|
|
if((localReader.IsBehavior(CommandBehavior.KeyInfo) || localReader.IsBehavior(CommandBehavior.SchemaOnly))
|
|
&& (CommandType != CommandType.StoredProcedure)) {
|
|
Int16 cColsAffected;
|
|
retcode = stmt.NumberOfResultColumns(out cColsAffected);
|
|
if(retcode == ODBC32.RetCode.SUCCESS || retcode == ODBC32.RetCode.SUCCESS_WITH_INFO) {
|
|
if(cColsAffected > 0) {
|
|
localReader.GetSchemaTable();
|
|
}
|
|
}
|
|
else if(retcode == ODBC32.RetCode.NO_DATA) {
|
|
// do nothing
|
|
}
|
|
else {
|
|
// any other returncode indicates an error
|
|
_connection.HandleError(stmt, retcode);
|
|
}
|
|
}
|
|
|
|
switch(odbcApiMethod) {
|
|
case ODBC32.SQL_API.SQLEXECDIRECT:
|
|
if(localReader.IsBehavior(CommandBehavior.KeyInfo) || _isPrepared) {
|
|
//Already prepared, so use SQLExecute
|
|
retcode = stmt.Execute();
|
|
// Build metadata here
|
|
// localReader.GetSchemaTable();
|
|
}
|
|
else {
|
|
#if DEBUG
|
|
//if (AdapterSwitches.OleDbTrace.TraceInfo) {
|
|
// ADP.DebugWriteLine("SQLExecDirectW: " + CommandText);
|
|
//}
|
|
#endif
|
|
//SQLExecDirect
|
|
retcode = stmt.ExecuteDirect(CommandText);
|
|
}
|
|
break;
|
|
|
|
case ODBC32.SQL_API.SQLTABLES:
|
|
retcode = stmt.Tables((string)methodArguments[0], //TableCatalog
|
|
(string)methodArguments[1], //TableSchema,
|
|
(string)methodArguments[2], //TableName
|
|
(string)methodArguments[3]); //TableType
|
|
break;
|
|
|
|
case ODBC32.SQL_API.SQLCOLUMNS:
|
|
retcode = stmt.Columns((string)methodArguments[0], //TableCatalog
|
|
(string)methodArguments[1], //TableSchema
|
|
(string)methodArguments[2], //TableName
|
|
(string)methodArguments[3]); //ColumnName
|
|
break;
|
|
|
|
case ODBC32.SQL_API.SQLPROCEDURES:
|
|
retcode = stmt.Procedures((string)methodArguments[0], //ProcedureCatalog
|
|
(string)methodArguments[1], //ProcedureSchema
|
|
(string)methodArguments[2]); //procedureName
|
|
break;
|
|
|
|
case ODBC32.SQL_API.SQLPROCEDURECOLUMNS:
|
|
retcode = stmt.ProcedureColumns((string)methodArguments[0], //ProcedureCatalog
|
|
(string)methodArguments[1], //ProcedureSchema
|
|
(string)methodArguments[2], //procedureName
|
|
(string)methodArguments[3]); //columnName
|
|
break;
|
|
|
|
case ODBC32.SQL_API.SQLSTATISTICS:
|
|
retcode = stmt.Statistics((string)methodArguments[0], //TableCatalog
|
|
(string)methodArguments[1], //TableSchema
|
|
(string)methodArguments[2], //TableName
|
|
(Int16)methodArguments[3], //IndexTrpe
|
|
(Int16)methodArguments[4]); //Accuracy
|
|
break;
|
|
|
|
case ODBC32.SQL_API.SQLGETTYPEINFO:
|
|
retcode = stmt.GetTypeInfo((Int16)methodArguments[0]); //SQL Type
|
|
break;
|
|
|
|
default:
|
|
// this should NEVER happen
|
|
Debug.Assert(false, "ExecuteReaderObjectcalled with unsupported ODBC API method.");
|
|
throw ADP.InvalidOperation(method.ToString());
|
|
}
|
|
|
|
//Note: Execute will return NO_DATA for Update/Delete non-row returning queries
|
|
if((ODBC32.RetCode.SUCCESS != retcode) && (ODBC32.RetCode.NO_DATA != retcode)) {
|
|
_connection.HandleError(stmt, retcode);
|
|
}
|
|
} // end SchemaOnly
|
|
}
|
|
finally {
|
|
if(mustRelease) {
|
|
parameterBuffer.DangerousRelease();
|
|
}
|
|
}
|
|
|
|
this.weakDataReaderReference = new WeakReference(localReader);
|
|
|
|
// XXXCommand.Execute should position reader on first row returning result
|
|
// any exceptions in the initial non-row returning results should be thrown
|
|
// from from ExecuteXXX not the DataReader
|
|
if(!localReader.IsBehavior(CommandBehavior.SchemaOnly)) {
|
|
localReader.FirstResult();
|
|
}
|
|
cmdState = ConnectionState.Fetching;
|
|
}
|
|
finally {
|
|
if(ConnectionState.Fetching != cmdState) {
|
|
if(null != localReader) {
|
|
// clear bindings so we don't grab output parameters on a failed execute
|
|
if(null != _parameterCollection) {
|
|
_parameterCollection.ClearBindings();
|
|
}
|
|
((IDisposable)localReader).Dispose();
|
|
}
|
|
if(ConnectionState.Closed != cmdState) {
|
|
cmdState = ConnectionState.Closed;
|
|
}
|
|
}
|
|
}
|
|
return localReader;
|
|
}
|
|
|
|
override public object ExecuteScalar() {
|
|
OdbcConnection.ExecutePermission.Demand();
|
|
|
|
object value = null;
|
|
using(IDataReader reader = ExecuteReaderObject(0, ADP.ExecuteScalar, false)) {
|
|
if (reader.Read() && (0 < reader.FieldCount)) {
|
|
value = reader.GetValue(0);
|
|
}
|
|
reader.Close();
|
|
}
|
|
return value;
|
|
}
|
|
|
|
internal string GetDiagSqlState() {
|
|
return _cmdWrapper.GetDiagSqlState();
|
|
}
|
|
|
|
private void PropertyChanging() {
|
|
_isPrepared = false;
|
|
}
|
|
|
|
// Prepare
|
|
//
|
|
// if the CommandType property is set to TableDirect Prepare does nothing.
|
|
// if the CommandType property is set to StoredProcedure Prepare should succeed but result
|
|
// in a no-op
|
|
//
|
|
// throw InvalidOperationException
|
|
// if the connection is not set
|
|
// if the connection is not open
|
|
//
|
|
override public void Prepare() {
|
|
OdbcConnection.ExecutePermission.Demand();
|
|
ODBC32.RetCode retcode;
|
|
|
|
ValidateOpenConnection(ADP.Prepare);
|
|
|
|
if (0 != (ConnectionState.Fetching & _connection.InternalState)) {
|
|
throw ADP.OpenReaderExists();
|
|
}
|
|
|
|
if (CommandType == CommandType.TableDirect) {
|
|
return; // do nothing
|
|
}
|
|
|
|
DisposeDeadDataReader();
|
|
GetStatementHandle();
|
|
|
|
OdbcStatementHandle stmt = _cmdWrapper.StatementHandle;
|
|
|
|
retcode = stmt.Prepare(CommandText);
|
|
|
|
|
|
if (ODBC32.RetCode.SUCCESS != retcode) {
|
|
_connection.HandleError(stmt, retcode);
|
|
}
|
|
_isPrepared = true;
|
|
}
|
|
|
|
|
|
|
|
void TrySetStatementAttribute (OdbcStatementHandle stmt, ODBC32.SQL_ATTR stmtAttribute, IntPtr value) {
|
|
|
|
ODBC32.RetCode retcode = stmt.SetStatementAttribute(
|
|
stmtAttribute,
|
|
value,
|
|
ODBC32.SQL_IS.UINTEGER);
|
|
|
|
if (retcode == ODBC32.RetCode.ERROR) {
|
|
|
|
string sqlState;
|
|
stmt.GetDiagnosticField(out sqlState);
|
|
|
|
if ((sqlState == "HYC00") || (sqlState == "HY092")) {
|
|
Connection.FlagUnsupportedStmtAttr(stmtAttribute);
|
|
}
|
|
else {
|
|
// now what? Should we throw?
|
|
}
|
|
}
|
|
}
|
|
|
|
private void ValidateOpenConnection(string methodName) {
|
|
// see if we have a connection
|
|
OdbcConnection connection = Connection;
|
|
|
|
if (null == connection) {
|
|
throw ADP.ConnectionRequired(methodName);
|
|
}
|
|
|
|
// must have an open and available connection
|
|
ConnectionState state = connection.State;
|
|
|
|
if (ConnectionState.Open != state) {
|
|
throw ADP.OpenConnectionRequired(methodName, state);
|
|
}
|
|
}
|
|
|
|
private void ValidateConnectionAndTransaction(string method) {
|
|
if (null == _connection) {
|
|
throw ADP.ConnectionRequired(method);
|
|
}
|
|
_transaction = _connection.SetStateExecuting(method, Transaction);
|
|
cmdState = ConnectionState.Executing;
|
|
}
|
|
|
|
}
|
|
sealed internal class CMDWrapper {
|
|
|
|
private OdbcStatementHandle _stmt; // hStmt
|
|
private OdbcStatementHandle _keyinfostmt; // hStmt for keyinfo
|
|
|
|
internal OdbcDescriptorHandle _hdesc; // hDesc
|
|
|
|
internal CNativeBuffer _nativeParameterBuffer; // Native memory for internal memory management
|
|
// (Performance optimization)
|
|
|
|
internal CNativeBuffer _dataReaderBuf; // Reusable DataReader buffer
|
|
|
|
private readonly OdbcConnection _connection; // Connection
|
|
private bool _canceling; // true if the command is canceling
|
|
internal bool _hasBoundColumns;
|
|
internal bool _ssKeyInfoModeOn; // tells us if the SqlServer specific options are on
|
|
internal bool _ssKeyInfoModeOff; // a tri-state value would be much better ...
|
|
|
|
internal CMDWrapper (OdbcConnection connection) {
|
|
_connection = connection;
|
|
}
|
|
|
|
internal bool Canceling {
|
|
get {
|
|
return _canceling;
|
|
}
|
|
set {
|
|
_canceling = value;
|
|
}
|
|
}
|
|
|
|
internal OdbcConnection Connection {
|
|
get {
|
|
return _connection;
|
|
}
|
|
}
|
|
|
|
internal bool HasBoundColumns {
|
|
// get {
|
|
// return _hasBoundColumns;
|
|
// }
|
|
set {
|
|
_hasBoundColumns = value;
|
|
}
|
|
}
|
|
|
|
internal OdbcStatementHandle StatementHandle {
|
|
get { return _stmt; }
|
|
}
|
|
|
|
internal OdbcStatementHandle KeyInfoStatement {
|
|
get {
|
|
return _keyinfostmt;
|
|
}
|
|
}
|
|
|
|
internal void CreateKeyInfoStatementHandle() {
|
|
DisposeKeyInfoStatementHandle();
|
|
_keyinfostmt = _connection.CreateStatementHandle();
|
|
}
|
|
|
|
internal void CreateStatementHandle() {
|
|
DisposeStatementHandle();
|
|
_stmt = _connection.CreateStatementHandle();
|
|
}
|
|
|
|
internal void Dispose() {
|
|
if (null != _dataReaderBuf) {
|
|
_dataReaderBuf.Dispose();
|
|
_dataReaderBuf = null;
|
|
}
|
|
DisposeStatementHandle();
|
|
|
|
CNativeBuffer buffer = _nativeParameterBuffer;
|
|
_nativeParameterBuffer = null;
|
|
if (null != buffer) {
|
|
buffer.Dispose();
|
|
}
|
|
_ssKeyInfoModeOn = false;
|
|
_ssKeyInfoModeOff = false;
|
|
}
|
|
|
|
private void DisposeDescriptorHandle() {
|
|
OdbcDescriptorHandle handle = _hdesc;
|
|
if (null != handle) {
|
|
_hdesc = null;
|
|
handle.Dispose();
|
|
}
|
|
}
|
|
internal void DisposeStatementHandle() {
|
|
DisposeKeyInfoStatementHandle();
|
|
DisposeDescriptorHandle();
|
|
|
|
OdbcStatementHandle handle = _stmt;
|
|
if (null != handle) {
|
|
_stmt = null;
|
|
handle.Dispose();
|
|
}
|
|
}
|
|
|
|
internal void DisposeKeyInfoStatementHandle() {
|
|
OdbcStatementHandle handle = _keyinfostmt;
|
|
if (null != handle) {
|
|
_keyinfostmt = null;
|
|
handle.Dispose();
|
|
}
|
|
}
|
|
|
|
internal void FreeStatementHandle(ODBC32.STMT stmt) {
|
|
DisposeDescriptorHandle();
|
|
|
|
OdbcStatementHandle handle = _stmt;
|
|
if (null != handle) {
|
|
try {
|
|
ODBC32.RetCode retcode;
|
|
retcode = handle.FreeStatement(stmt);
|
|
StatementErrorHandler(retcode);
|
|
}
|
|
catch (Exception e) {
|
|
//
|
|
if (ADP.IsCatchableExceptionType(e)) {
|
|
_stmt = null;
|
|
handle.Dispose();
|
|
}
|
|
|
|
throw;
|
|
}
|
|
}
|
|
}
|
|
|
|
internal void FreeKeyInfoStatementHandle(ODBC32.STMT stmt) {
|
|
OdbcStatementHandle handle = _keyinfostmt;
|
|
if (null != handle) {
|
|
try {
|
|
handle.FreeStatement(stmt);
|
|
}
|
|
catch (Exception e) {
|
|
//
|
|
if (ADP.IsCatchableExceptionType(e)) {
|
|
_keyinfostmt = null;
|
|
handle.Dispose();
|
|
}
|
|
|
|
throw;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Get the Descriptor Handle for the current statement
|
|
//
|
|
internal OdbcDescriptorHandle GetDescriptorHandle(ODBC32.SQL_ATTR attribute) {
|
|
OdbcDescriptorHandle hdesc = _hdesc;
|
|
if (null == _hdesc) {
|
|
_hdesc = hdesc = new OdbcDescriptorHandle(_stmt, attribute);
|
|
}
|
|
return hdesc;
|
|
}
|
|
|
|
internal string GetDiagSqlState () {
|
|
string sqlstate;
|
|
_stmt.GetDiagnosticField(out sqlstate);
|
|
return sqlstate;
|
|
}
|
|
|
|
internal void StatementErrorHandler(ODBC32.RetCode retcode) {
|
|
switch(retcode) {
|
|
case ODBC32.RetCode.SUCCESS:
|
|
case ODBC32.RetCode.SUCCESS_WITH_INFO:
|
|
_connection.HandleErrorNoThrow(_stmt, retcode);
|
|
break;
|
|
default:
|
|
throw _connection.HandleErrorNoThrow(_stmt, retcode);
|
|
}
|
|
}
|
|
|
|
internal void UnbindStmtColumns() {
|
|
if (_hasBoundColumns) {
|
|
FreeStatementHandle(ODBC32.STMT.UNBIND);
|
|
_hasBoundColumns = false;
|
|
}
|
|
}
|
|
}
|
|
}
|