1 /////////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     Implementation of the wxDb class.  The wxDb class represents a connection 
   4 //              to an ODBC data source.  The wxDb class allows operations on the data 
   5 //              source such as opening and closing the data source. 
   7 // Modified by: George Tasker 
   9 //              Mark Johnson, wxWindows@mj10777.de 
  11 //                -Added support for SQL statement logging and database cataloging 
  13 //                -Added QUERY_ONLY mode support to reduce default number of cursors 
  14 //                -Added additional SQL logging code 
  15 //                -Added DEBUG-ONLY tracking of wxTable objects to detect orphaned DB connections 
  16 //                -Set ODBC option to only read committed writes to the DB so all 
  17 //                   databases operate the same in that respect 
  20 // Copyright:   (c) 1996 Remstar International, Inc. 
  21 // Licence:     wxWindows licence 
  22 /////////////////////////////////////////////////////////////////////////////// 
  29 #include "wx/wxprec.h" 
  35 #ifdef DBDEBUG_CONSOLE 
  36     #include "wx/ioswrap.h" 
  40     #include "wx/string.h" 
  41     #include "wx/object.h" 
  46 #include "wx/filefn.h" 
  47 #include "wx/wxchar.h" 
  59 // DLL options compatibility check: 
  61 WX_CHECK_BUILD_OPTIONS("wxODBC") 
  63 WXDLLIMPEXP_DATA_ODBC(wxDbList
*) PtrBegDbList 
= 0; 
  65 wxChar 
const *SQL_LOG_FILENAME         
= wxT("sqllog.txt"); 
  66 wxChar 
const *SQL_CATALOG_FILENAME     
= wxT("catalog.txt"); 
  69     extern wxList TablesInUse
; 
  72 // SQL Log defaults to be used by GetDbConnection 
  73 wxDbSqlLogState SQLLOGstate 
= sqlLogOFF
; 
  75 static wxString SQLLOGfn 
= SQL_LOG_FILENAME
; 
  77 // The wxDb::errorList is copied to this variable when the wxDb object 
  78 // is closed.  This way, the error list is still available after the 
  79 // database object is closed.  This is necessary if the database 
  80 // connection fails so the calling application can show the operator 
  81 // why the connection failed.  Note: as each wxDb object is closed, it 
  82 // will overwrite the errors of the previously destroyed wxDb object in 
  83 // this variable.  NOTE: This occurs during a CLOSE, not a FREEing of the 
  85 wxChar DBerrorList
[DB_MAX_ERROR_HISTORY
][DB_MAX_ERROR_MSG_LEN
+1]; 
  88 // This type defines the return row-struct form 
  89 // SQLTablePrivileges, and is used by wxDB::TablePrivileges. 
  92    wxChar        tableQual
[128+1]; 
  93    wxChar        tableOwner
[128+1]; 
  94    wxChar        tableName
[128+1]; 
  95    wxChar        grantor
[128+1]; 
  96    wxChar        grantee
[128+1]; 
  97    wxChar        privilege
[128+1]; 
  98    wxChar        grantable
[3+1]; 
  99 } wxDbTablePrivilegeInfo
; 
 102 /********** wxDbConnectInf Constructor - form 1 **********/ 
 103 wxDbConnectInf::wxDbConnectInf() 
 106     freeHenvOnDestroy 
= false; 
 112 /********** wxDbConnectInf Constructor - form 2 **********/ 
 113 wxDbConnectInf::wxDbConnectInf(HENV henv
, const wxString 
&dsn
, const wxString 
&userID
, 
 114                        const wxString 
&password
, const wxString 
&defaultDir
, 
 115                        const wxString 
&fileType
, const wxString 
&description
) 
 118     freeHenvOnDestroy 
= false; 
 129     SetPassword(password
); 
 130     SetDescription(description
); 
 131     SetFileType(fileType
); 
 132     SetDefaultDir(defaultDir
); 
 133 }  // wxDbConnectInf Constructor 
 136 wxDbConnectInf::~wxDbConnectInf() 
 138     if (freeHenvOnDestroy
) 
 142 }  // wxDbConnectInf Destructor 
 146 /********** wxDbConnectInf::Initialize() **********/ 
 147 bool wxDbConnectInf::Initialize() 
 149     freeHenvOnDestroy 
= false; 
 151     if (freeHenvOnDestroy 
&& Henv
) 
 158     ConnectionStr
[0] = 0; 
 163     useConnectionStr 
= false; 
 166 }  // wxDbConnectInf::Initialize() 
 169 /********** wxDbConnectInf::AllocHenv() **********/ 
 170 bool wxDbConnectInf::AllocHenv() 
 172     // This is here to help trap if you are getting a new henv 
 173     // without releasing an existing henv 
 176     // Initialize the ODBC Environment for Database Operations 
 177     if (SQLAllocEnv(&Henv
) != SQL_SUCCESS
) 
 179         wxLogDebug(wxT("A problem occurred while trying to get a connection to the data source")); 
 183     freeHenvOnDestroy 
= true; 
 186 }  // wxDbConnectInf::AllocHenv() 
 189 void wxDbConnectInf::FreeHenv() 
 197     freeHenvOnDestroy 
= false; 
 199 }  // wxDbConnectInf::FreeHenv() 
 202 void wxDbConnectInf::SetDsn(const wxString 
&dsn
) 
 204     wxASSERT(dsn
.Length() < WXSIZEOF(Dsn
)); 
 206     wxStrncpy(Dsn
, dsn
, WXSIZEOF(Dsn
)-1); 
 207     Dsn
[WXSIZEOF(Dsn
)-1] = 0;  // Prevent buffer overrun 
 208 }  // wxDbConnectInf::SetDsn() 
 211 void wxDbConnectInf::SetUserID(const wxString 
&uid
) 
 213     wxASSERT(uid
.Length() < WXSIZEOF(Uid
)); 
 214     wxStrncpy(Uid
, uid
, WXSIZEOF(Uid
)-1); 
 215     Uid
[WXSIZEOF(Uid
)-1] = 0;  // Prevent buffer overrun 
 216 }  // wxDbConnectInf::SetUserID() 
 219 void wxDbConnectInf::SetPassword(const wxString 
&password
) 
 221     wxASSERT(password
.Length() < WXSIZEOF(AuthStr
)); 
 223     wxStrncpy(AuthStr
, password
, WXSIZEOF(AuthStr
)-1); 
 224     AuthStr
[WXSIZEOF(AuthStr
)-1] = 0;  // Prevent buffer overrun 
 225 }  // wxDbConnectInf::SetPassword() 
 227 void wxDbConnectInf::SetConnectionStr(const wxString 
&connectStr
) 
 229     wxASSERT(connectStr
.Length() < WXSIZEOF(ConnectionStr
)); 
 231     useConnectionStr 
= wxStrlen(connectStr
) > 0; 
 233     wxStrncpy(ConnectionStr
, connectStr
, WXSIZEOF(ConnectionStr
)-1); 
 234     ConnectionStr
[WXSIZEOF(ConnectionStr
)-1] = 0;  // Prevent buffer overrun 
 235 }  // wxDbConnectInf::SetConnectionStr() 
 238 /********** wxDbColFor Constructor **********/ 
 239 wxDbColFor::wxDbColFor() 
 242 }  // wxDbColFor::wxDbColFor() 
 245 /********** wxDbColFor::Initialize() **********/ 
 246 void wxDbColFor::Initialize() 
 256     i_Nation      
= 0;                     // 0=EU, 1=UK, 2=International, 3=US 
 259     Format(1,DB_DATA_TYPE_VARCHAR
,0,0,0);  // the Function that does the work 
 260 }  // wxDbColFor::Initialize() 
 263 /********** wxDbColFor::Format() **********/ 
 264 int wxDbColFor::Format(int Nation
, int dbDataType
, SWORD sqlDataType
, 
 265                        short columnLength
, short decimalDigits
) 
 267     // ---------------------------------------------------------------------------------------- 
 268     // -- 19991224 : mj10777 : Create 
 269     // There is still a lot of work to do here, but it is a start 
 270     // It handles all the basic data-types that I have run into up to now 
 271     // The main work will have be with Dates and float Formatting 
 272     //    (US 1,000.00 ; EU 1.000,00) 
 273     // There are wxWindow plans for locale support and the new wxDateTime.  If 
 274     //    they define some constants (wxEUROPEAN) that can be gloably used, 
 275     //    they should be used here. 
 276     // ---------------------------------------------------------------------------------------- 
 277     // There should also be a function to scan in a string to fill the variable 
 278     // ---------------------------------------------------------------------------------------- 
 280     i_Nation      
= Nation
;                                       // 0 = timestamp , 1=EU, 2=UK, 3=International, 4=US 
 281     i_dbDataType  
= dbDataType
; 
 282     i_sqlDataType 
= sqlDataType
; 
 283     s_Field
.Printf(wxT("%s%d"),s_Amount
[1].c_str(),i_Amount
[1]);  // OK for VARCHAR, INTEGER and FLOAT 
 285     if (i_dbDataType 
== 0)                                        // Filter unsupported dbDataTypes 
 287         if ((i_sqlDataType 
== SQL_VARCHAR
) 
 289     #if defined(SQL_WCHAR) 
 290             || (i_sqlDataType 
== SQL_WCHAR
)  
 292     #if defined(SQL_WVARCHAR) 
 293             || (i_sqlDataType 
== SQL_WVARCHAR
) 
 296             || (i_sqlDataType 
== SQL_LONGVARCHAR
)) 
 297             i_dbDataType 
= DB_DATA_TYPE_VARCHAR
; 
 298         if ((i_sqlDataType 
== SQL_C_DATE
) || (i_sqlDataType 
== SQL_C_TIMESTAMP
)) 
 299             i_dbDataType 
= DB_DATA_TYPE_DATE
; 
 300         if (i_sqlDataType 
== SQL_C_BIT
) 
 301             i_dbDataType 
= DB_DATA_TYPE_INTEGER
; 
 302         if (i_sqlDataType 
== SQL_NUMERIC
) 
 303             i_dbDataType 
= DB_DATA_TYPE_VARCHAR
;   // glt - ??? is this right? 
 304         if (i_sqlDataType 
== SQL_REAL
) 
 305             i_dbDataType 
= DB_DATA_TYPE_FLOAT
; 
 306         if (i_sqlDataType 
== SQL_C_BINARY
) 
 307             i_dbDataType 
= DB_DATA_TYPE_BLOB
; 
 310     if ((i_dbDataType 
== DB_DATA_TYPE_INTEGER
) && (i_sqlDataType 
== SQL_C_DOUBLE
)) 
 312         i_dbDataType 
= DB_DATA_TYPE_FLOAT
; 
 315     switch(i_dbDataType
)     // TBD: Still a lot of proper formatting to do 
 317         case DB_DATA_TYPE_VARCHAR
: 
 320         case DB_DATA_TYPE_INTEGER
: 
 323         case DB_DATA_TYPE_FLOAT
: 
 324             if (decimalDigits 
== 0) 
 326             tempStr
.Printf(wxT("%%%d.%d"), columnLength
, decimalDigits
); 
 327             s_Field
.Printf(wxT("%sf"), tempStr
.c_str()); 
 329         case DB_DATA_TYPE_DATE
: 
 330             if (i_Nation 
== 0)      // timestamp       YYYY-MM-DD HH:MM:SS.SSS (tested for SYBASE) 
 332                 s_Field 
= wxT("%04d-%02d-%02d %02d:%02d:%02d.%03d"); 
 334             if (i_Nation 
== 1)      // European        DD.MM.YYYY HH:MM:SS.SSS 
 336                 s_Field 
= wxT("%02d.%02d.%04d %02d:%02d:%02d.%03d"); 
 338             if (i_Nation 
== 2)      // UK              DD/MM/YYYY HH:MM:SS.SSS 
 340                 s_Field 
= wxT("%02d/%02d/%04d %02d:%02d:%02d.%03d"); 
 342             if (i_Nation 
== 3)      // International   YYYY-MM-DD HH:MM:SS.SSS 
 344                 s_Field 
= wxT("%04d-%02d-%02d %02d:%02d:%02d.%03d"); 
 346             if (i_Nation 
== 4)      // US              MM/DD/YYYY HH:MM:SS.SSS 
 348                 s_Field 
= wxT("%02d/%02d/%04d %02d:%02d:%02d.%03d"); 
 351           case DB_DATA_TYPE_BLOB
: 
 352             s_Field
.Printf(wxT("Unable to format(%d)-SQL(%d)"), dbDataType
,sqlDataType
);        // 
 355             s_Field
.Printf(wxT("Unknown Format(%d)-SQL(%d)"), dbDataType
,sqlDataType
);        // 
 359 }  // wxDbColFor::Format() 
 362 /********** wxDbColInf Constructor **********/ 
 363 wxDbColInf::wxDbColInf() 
 366 }  // wxDbColInf::wxDbColInf() 
 369 /********** wxDbColInf Destructor ********/ 
 370 wxDbColInf::~wxDbColInf() 
 375 }  // wxDbColInf::~wxDbColInf() 
 378 bool wxDbColInf::Initialize() 
 400 }  // wxDbColInf::Initialize() 
 403 /********** wxDbTableInf Constructor ********/ 
 404 wxDbTableInf::wxDbTableInf() 
 407 }  // wxDbTableInf::wxDbTableInf() 
 410 /********** wxDbTableInf Constructor ********/ 
 411 wxDbTableInf::~wxDbTableInf() 
 416 }  // wxDbTableInf::~wxDbTableInf() 
 419 bool wxDbTableInf::Initialize() 
 428 }  // wxDbTableInf::Initialize() 
 431 /********** wxDbInf Constructor *************/ 
 435 }  // wxDbInf::wxDbInf() 
 438 /********** wxDbInf Destructor *************/ 
 444 }  // wxDbInf::~wxDbInf() 
 447 /********** wxDbInf::Initialize() *************/ 
 448 bool wxDbInf::Initialize() 
 456 }  // wxDbInf::Initialize() 
 459 /********** wxDb Constructor **********/ 
 460 wxDb::wxDb(const HENV 
&aHenv
, bool FwdOnlyCursors
) 
 462     // Copy the HENV into the db class 
 464     fwdOnlyCursors 
= FwdOnlyCursors
; 
 470 /********** wxDb Destructor **********/ 
 473     wxASSERT_MSG(!IsCached(),wxT("Cached connections must not be manually deleted, use\nwxDbFreeConnection() or wxDbCloseConnections().")); 
 483 /********** PRIVATE! wxDb::initialize PRIVATE! **********/ 
 484 /********** wxDb::initialize() **********/ 
 485 void wxDb::initialize() 
 487  * Private member function that sets all wxDb member variables to 
 488  * known values at creation of the wxDb 
 493     fpSqlLog      
= 0;            // Sql Log file pointer 
 494     sqlLogState   
= sqlLogOFF
;    // By default, logging is turned off 
 496     dbmsType      
= dbmsUNIDENTIFIED
; 
 498     wxStrcpy(sqlState
,wxEmptyString
); 
 499     wxStrcpy(errorMsg
,wxEmptyString
); 
 500     nativeError 
= cbErrorMsg 
= 0; 
 501     for (i 
= 0; i 
< DB_MAX_ERROR_HISTORY
; i
++) 
 502         wxStrcpy(errorList
[i
], wxEmptyString
); 
 504     // Init typeInf structures 
 505     typeInfVarchar
.TypeName
.Empty(); 
 506     typeInfVarchar
.FsqlType      
= 0; 
 507     typeInfVarchar
.Precision     
= 0; 
 508     typeInfVarchar
.CaseSensitive 
= 0; 
 509     typeInfVarchar
.MaximumScale  
= 0; 
 511     typeInfInteger
.TypeName
.Empty(); 
 512     typeInfInteger
.FsqlType      
= 0; 
 513     typeInfInteger
.Precision     
= 0; 
 514     typeInfInteger
.CaseSensitive 
= 0; 
 515     typeInfInteger
.MaximumScale  
= 0; 
 517     typeInfFloat
.TypeName
.Empty(); 
 518     typeInfFloat
.FsqlType      
= 0; 
 519     typeInfFloat
.Precision     
= 0; 
 520     typeInfFloat
.CaseSensitive 
= 0; 
 521     typeInfFloat
.MaximumScale  
= 0; 
 523     typeInfDate
.TypeName
.Empty(); 
 524     typeInfDate
.FsqlType      
= 0; 
 525     typeInfDate
.Precision     
= 0; 
 526     typeInfDate
.CaseSensitive 
= 0; 
 527     typeInfDate
.MaximumScale  
= 0; 
 529     typeInfBlob
.TypeName
.Empty(); 
 530     typeInfBlob
.FsqlType      
= 0; 
 531     typeInfBlob
.Precision     
= 0; 
 532     typeInfBlob
.CaseSensitive 
= 0; 
 533     typeInfBlob
.MaximumScale  
= 0; 
 535     typeInfMemo
.TypeName
.Empty(); 
 536     typeInfMemo
.FsqlType      
= 0; 
 537     typeInfMemo
.Precision     
= 0; 
 538     typeInfMemo
.CaseSensitive 
= 0; 
 539     typeInfMemo
.MaximumScale  
= 0; 
 541     // Error reporting is turned OFF by default 
 544     // Allocate a data source connection handle 
 545     if (SQLAllocConnect(henv
, &hdbc
) != SQL_SUCCESS
) 
 548     // Initialize the db status flag 
 551     // Mark database as not open as of yet 
 554     dbOpenedWithConnectionString 
= false; 
 555 }  // wxDb::initialize() 
 558 /********** PRIVATE! wxDb::convertUserID PRIVATE! **********/ 
 560 // NOTE: Return value from this function MUST be copied 
 561 //       immediately, as the value is not good after 
 562 //       this function has left scope. 
 564 const wxChar 
*wxDb::convertUserID(const wxChar 
*userID
, wxString 
&UserID
) 
 568         if (!wxStrlen(userID
)) 
 576     // dBase does not use user names, and some drivers fail if you try to pass one 
 577     if ( Dbms() == dbmsDBASE
 
 578          || Dbms() == dbmsXBASE_SEQUITER 
) 
 581     // Some databases require user names to be specified in uppercase, 
 582     // so force the name to uppercase 
 583     if ((Dbms() == dbmsORACLE
) || 
 584         (Dbms() == dbmsMAXDB
)) 
 585         UserID 
= UserID
.Upper(); 
 587     return UserID
.c_str(); 
 588 }  // wxDb::convertUserID() 
 591 bool wxDb::determineDataTypes(bool failOnDataTypeUnsupported
) 
 595     // These are the possible SQL types we check for use against the datasource we are connected 
 596     // to for the purpose of determining which data type to use for the basic character strings 
 599     // NOTE: The first type in this enumeration that is determined to be supported by the 
 600     //       datasource/driver is the one that will be used. 
 601     SWORD PossibleSqlCharTypes
[] = { 
 602 #if wxUSE_UNICODE && defined(SQL_WVARCHAR) 
 606 #if wxUSE_UNICODE && defined(SQL_WVARCHAR) 
 612     // These are the possible SQL types we check for use against the datasource we are connected 
 613     // to for the purpose of determining which data type to use for the basic non-floating point 
 616     // NOTE: The first type in this enumeration that is determined to be supported by the 
 617     //       datasource/driver is the one that will be used. 
 618     SWORD PossibleSqlIntegerTypes
[] = { 
 622     // These are the possible SQL types we check for use against the datasource we are connected 
 623     // to for the purpose of determining which data type to use for the basic floating point number 
 626     // NOTE: The first type in this enumeration that is determined to be supported by the 
 627     //       datasource/driver is the one that will be used. 
 628     SWORD PossibleSqlFloatTypes
[] = { 
 636     // These are the possible SQL types we check for use agains the datasource we are connected 
 637     // to for the purpose of determining which data type to use for the date/time column types 
 639     // NOTE: The first type in this enumeration that is determined to be supported by the 
 640     //       datasource/driver is the one that will be used. 
 641     SWORD PossibleSqlDateTypes
[] = { 
 649     // These are the possible SQL types we check for use agains the datasource we are connected 
 650     // to for the purpose of determining which data type to use for the BLOB column types. 
 652     // NOTE: The first type in this enumeration that is determined to be supported by the 
 653     //       datasource/driver is the one that will be used. 
 654     SWORD PossibleSqlBlobTypes
[] = { 
 659     // These are the possible SQL types we check for use agains the datasource we are connected 
 660     // to for the purpose of determining which data type to use for the MEMO column types 
 661         // (a type which allow to store large strings; like VARCHAR just with a bigger precision) 
 663     // NOTE: The first type in this enumeration that is determined to be supported by the 
 664     //       datasource/driver is the one that will be used. 
 665     SWORD PossibleSqlMemoTypes
[] = { 
 670     // Query the data source regarding data type information 
 673     // The way it was determined which SQL data types to use was by calling SQLGetInfo 
 674     // for all of the possible SQL data types to see which ones were supported.  If 
 675     // a type is not supported, the SQLFetch() that's called from getDataTypeInfo() 
 676     // fails with SQL_NO_DATA_FOUND.  This is ugly because I'm sure the three SQL data 
 677     // types I've selected below will not always be what we want.  These are just 
 678     // what happened to work against an Oracle 7/Intersolv combination.  The following is 
 679     // a complete list of the results I got back against the Oracle 7 database: 
 681     // SQL_BIGINT             SQL_NO_DATA_FOUND 
 682     // SQL_BINARY             SQL_NO_DATA_FOUND 
 683     // SQL_BIT                SQL_NO_DATA_FOUND 
 684     // SQL_CHAR               type name = 'CHAR', Precision = 255 
 685     // SQL_DATE               SQL_NO_DATA_FOUND 
 686     // SQL_DECIMAL            type name = 'NUMBER', Precision = 38 
 687     // SQL_DOUBLE             type name = 'NUMBER', Precision = 15 
 688     // SQL_FLOAT              SQL_NO_DATA_FOUND 
 689     // SQL_INTEGER            SQL_NO_DATA_FOUND 
 690     // SQL_LONGVARBINARY      type name = 'LONG RAW', Precision = 2 billion 
 691     // SQL_LONGVARCHAR        type name = 'LONG', Precision = 2 billion 
 692     // SQL_NUMERIC            SQL_NO_DATA_FOUND 
 693     // SQL_REAL               SQL_NO_DATA_FOUND 
 694     // SQL_SMALLINT           SQL_NO_DATA_FOUND 
 695     // SQL_TIME               SQL_NO_DATA_FOUND 
 696     // SQL_TIMESTAMP          type name = 'DATE', Precision = 19 
 697     // SQL_VARBINARY          type name = 'RAW', Precision = 255 
 698     // SQL_VARCHAR            type name = 'VARCHAR2', Precision = 2000 
 699     // ===================================================================== 
 700     // Results from a Microsoft Access 7.0 db, using a driver from Microsoft 
 702     // SQL_VARCHAR            type name = 'TEXT', Precision = 255 
 703     // SQL_TIMESTAMP          type name = 'DATETIME' 
 704     // SQL_DECIMAL            SQL_NO_DATA_FOUND 
 705     // SQL_NUMERIC            type name = 'CURRENCY', Precision = 19 
 706     // SQL_FLOAT              SQL_NO_DATA_FOUND 
 707     // SQL_REAL               type name = 'SINGLE', Precision = 7 
 708     // SQL_DOUBLE             type name = 'DOUBLE', Precision = 15 
 709     // SQL_INTEGER            type name = 'LONG', Precision = 10 
 711     // Query the data source for info about itself 
 712     if (!getDbInfo(failOnDataTypeUnsupported
)) 
 715     // --------------- Varchar - (Variable length character string) --------------- 
 716     for (iIndex 
= 0; iIndex 
< WXSIZEOF(PossibleSqlCharTypes
) && 
 717                      !getDataTypeInfo(PossibleSqlCharTypes
[iIndex
], typeInfVarchar
); ++iIndex
) 
 720     if (iIndex 
< WXSIZEOF(PossibleSqlCharTypes
)) 
 721         typeInfVarchar
.FsqlType 
= PossibleSqlCharTypes
[iIndex
]; 
 722     else if (failOnDataTypeUnsupported
) 
 725     // --------------- Float --------------- 
 726     for (iIndex 
= 0; iIndex 
< WXSIZEOF(PossibleSqlFloatTypes
) && 
 727                      !getDataTypeInfo(PossibleSqlFloatTypes
[iIndex
], typeInfFloat
); ++iIndex
) 
 730     if (iIndex 
< WXSIZEOF(PossibleSqlFloatTypes
)) 
 731         typeInfFloat
.FsqlType 
= PossibleSqlFloatTypes
[iIndex
]; 
 732     else if (failOnDataTypeUnsupported
) 
 735     // --------------- Integer ------------- 
 736     for (iIndex 
= 0; iIndex 
< WXSIZEOF(PossibleSqlIntegerTypes
) && 
 737                      !getDataTypeInfo(PossibleSqlIntegerTypes
[iIndex
], typeInfInteger
); ++iIndex
) 
 740     if (iIndex 
< WXSIZEOF(PossibleSqlIntegerTypes
)) 
 741         typeInfInteger
.FsqlType 
= PossibleSqlIntegerTypes
[iIndex
]; 
 742     else if (failOnDataTypeUnsupported
) 
 744         // If no non-floating point data types are supported, we'll 
 745         // use the type assigned for floats to store integers as well 
 746         if (!getDataTypeInfo(typeInfFloat
.FsqlType
, typeInfInteger
)) 
 748             if (failOnDataTypeUnsupported
) 
 752             typeInfInteger
.FsqlType 
= typeInfFloat
.FsqlType
; 
 755     // --------------- Date/Time --------------- 
 756     for (iIndex 
= 0; iIndex 
< WXSIZEOF(PossibleSqlDateTypes
) && 
 757                      !getDataTypeInfo(PossibleSqlDateTypes
[iIndex
], typeInfDate
); ++iIndex
) 
 760     if (iIndex 
< WXSIZEOF(PossibleSqlDateTypes
)) 
 761         typeInfDate
.FsqlType 
= PossibleSqlDateTypes
[iIndex
]; 
 762     else if (failOnDataTypeUnsupported
) 
 765     // --------------- BLOB --------------- 
 766     for (iIndex 
= 0; iIndex 
< WXSIZEOF(PossibleSqlBlobTypes
) && 
 767                      !getDataTypeInfo(PossibleSqlBlobTypes
[iIndex
], typeInfBlob
); ++iIndex
) 
 770     if (iIndex 
< WXSIZEOF(PossibleSqlBlobTypes
)) 
 771         typeInfBlob
.FsqlType 
= PossibleSqlBlobTypes
[iIndex
]; 
 772     else if (failOnDataTypeUnsupported
) 
 775     // --------------- MEMO --------------- 
 776     for (iIndex 
= 0; iIndex 
< WXSIZEOF(PossibleSqlMemoTypes
) && 
 777                      !getDataTypeInfo(PossibleSqlMemoTypes
[iIndex
], typeInfMemo
); ++iIndex
) 
 780     if (iIndex 
< WXSIZEOF(PossibleSqlMemoTypes
)) 
 781         typeInfMemo
.FsqlType 
= PossibleSqlMemoTypes
[iIndex
]; 
 782     else if (failOnDataTypeUnsupported
) 
 786 }  // wxDb::determineDataTypes 
 789 bool wxDb::open(bool failOnDataTypeUnsupported
) 
 792     If using Intersolv branded ODBC drivers, this is the place where you would substitute 
 793     your branded driver license information 
 795     SQLSetConnectOption(hdbc, 1041, (UDWORD) wxEmptyString); 
 796     SQLSetConnectOption(hdbc, 1042, (UDWORD) wxEmptyString); 
 799     // Mark database as open 
 802     // Allocate a statement handle for the database connection 
 803     if (SQLAllocStmt(hdbc
, &hstmt
) != SQL_SUCCESS
) 
 804         return(DispAllErrors(henv
, hdbc
)); 
 806     // Set Connection Options 
 807     if (!setConnectionOptions()) 
 810     if (!determineDataTypes(failOnDataTypeUnsupported
)) 
 813 #ifdef DBDEBUG_CONSOLE 
 814     cout 
<< wxT("VARCHAR DATA TYPE: ") << typeInfVarchar
.TypeName 
<< endl
; 
 815     cout 
<< wxT("INTEGER DATA TYPE: ") << typeInfInteger
.TypeName 
<< endl
; 
 816     cout 
<< wxT("FLOAT   DATA TYPE: ") << typeInfFloat
.TypeName 
<< endl
; 
 817     cout 
<< wxT("DATE    DATA TYPE: ") << typeInfDate
.TypeName 
<< endl
; 
 818     cout 
<< wxT("BLOB    DATA TYPE: ") << typeInfBlob
.TypeName 
<< endl
; 
 819     cout 
<< wxT("MEMO    DATA TYPE: ") << typeInfMemo
.TypeName 
<< endl
; 
 823     // Completed Successfully 
 827 bool wxDb::Open(const wxString
& inConnectStr
, bool failOnDataTypeUnsupported
) 
 829     wxASSERT(inConnectStr
.Length()); 
 830     return Open(inConnectStr
, NULL
, failOnDataTypeUnsupported
); 
 833 bool wxDb::Open(const wxString
& inConnectStr
, SQLHWND parentWnd
, bool failOnDataTypeUnsupported
) 
 841     if (!FwdOnlyCursors()) 
 843         // Specify that the ODBC cursor library be used, if needed.  This must be 
 844         // specified before the connection is made. 
 845         retcode 
= SQLSetConnectOption(hdbc
, SQL_ODBC_CURSORS
, SQL_CUR_USE_IF_NEEDED
); 
 847 #ifdef DBDEBUG_CONSOLE 
 848         if (retcode 
== SQL_SUCCESS
) 
 849             cout 
<< wxT("SQLSetConnectOption(CURSOR_LIB) successful") << endl
; 
 851             cout 
<< wxT("SQLSetConnectOption(CURSOR_LIB) failed") << endl
; 
 853         wxUnusedVar(retcode
); 
 857     // Connect to the data source 
 858     SQLTCHAR outConnectBuffer
[SQL_MAX_CONNECTSTR_LEN
+1];  // MS recommends at least 1k buffer 
 859     short outConnectBufferLen
; 
 861     inConnectionStr 
= inConnectStr
; 
 863     retcode 
= SQLDriverConnect(hdbc
, parentWnd
, (SQLTCHAR FAR 
*)inConnectionStr
.c_str(), 
 864                         (SWORD
)inConnectionStr
.Length(), (SQLTCHAR FAR 
*)outConnectBuffer
, 
 865                         sizeof(outConnectBuffer
), &outConnectBufferLen
, SQL_DRIVER_COMPLETE 
); 
 867     if ((retcode 
!= SQL_SUCCESS
) && 
 868         (retcode 
!= SQL_SUCCESS_WITH_INFO
)) 
 869         return(DispAllErrors(henv
, hdbc
)); 
 871     outConnectBuffer
[outConnectBufferLen
] = 0; 
 872     outConnectionStr 
= outConnectBuffer
; 
 873     dbOpenedWithConnectionString 
= true; 
 875     return open(failOnDataTypeUnsupported
); 
 878 /********** wxDb::Open() **********/ 
 879 bool wxDb::Open(const wxString 
&Dsn
, const wxString 
&Uid
, const wxString 
&AuthStr
, bool failOnDataTypeUnsupported
) 
 881     wxASSERT(Dsn
.Length()); 
 886     inConnectionStr 
= wxT(""); 
 887     outConnectionStr 
= wxT(""); 
 891     if (!FwdOnlyCursors()) 
 893         // Specify that the ODBC cursor library be used, if needed.  This must be 
 894         // specified before the connection is made. 
 895         retcode 
= SQLSetConnectOption(hdbc
, SQL_ODBC_CURSORS
, SQL_CUR_USE_IF_NEEDED
); 
 897 #ifdef DBDEBUG_CONSOLE 
 898         if (retcode 
== SQL_SUCCESS
) 
 899             cout 
<< wxT("SQLSetConnectOption(CURSOR_LIB) successful") << endl
; 
 901             cout 
<< wxT("SQLSetConnectOption(CURSOR_LIB) failed") << endl
; 
 903         wxUnusedVar( retcode 
); 
 907     // Connect to the data source 
 908     retcode 
= SQLConnect(hdbc
, (SQLTCHAR FAR 
*) dsn
.c_str(), SQL_NTS
, 
 909                          (SQLTCHAR FAR 
*) uid
.c_str(), SQL_NTS
, 
 910                          (SQLTCHAR FAR 
*) authStr
.c_str(), SQL_NTS
); 
 912     if ((retcode 
!= SQL_SUCCESS
) && 
 913         (retcode 
!= SQL_SUCCESS_WITH_INFO
)) 
 914         return(DispAllErrors(henv
, hdbc
)); 
 916     return open(failOnDataTypeUnsupported
); 
 921 bool wxDb::Open(wxDbConnectInf 
*dbConnectInf
, bool failOnDataTypeUnsupported
) 
 923     wxASSERT(dbConnectInf
); 
 925     // Use the connection string if one is present 
 926     if (dbConnectInf
->UseConnectionStr()) 
 927         return Open(GetConnectionInStr(), failOnDataTypeUnsupported
); 
 929         return Open(dbConnectInf
->GetDsn(), dbConnectInf
->GetUserID(), 
 930                     dbConnectInf
->GetPassword(), failOnDataTypeUnsupported
); 
 934 bool wxDb::Open(wxDb 
*copyDb
) 
 936     dsn              
= copyDb
->GetDatasourceName(); 
 937     uid              
= copyDb
->GetUsername(); 
 938     authStr          
= copyDb
->GetPassword(); 
 939     inConnectionStr  
= copyDb
->GetConnectionInStr(); 
 940     outConnectionStr 
= copyDb
->GetConnectionOutStr(); 
 944     if (!FwdOnlyCursors()) 
 946         // Specify that the ODBC cursor library be used, if needed.  This must be 
 947         // specified before the connection is made. 
 948         retcode 
= SQLSetConnectOption(hdbc
, SQL_ODBC_CURSORS
, SQL_CUR_USE_IF_NEEDED
); 
 950 #ifdef DBDEBUG_CONSOLE 
 951         if (retcode 
== SQL_SUCCESS
) 
 952             cout 
<< wxT("SQLSetConnectOption(CURSOR_LIB) successful") << endl
; 
 954             cout 
<< wxT("SQLSetConnectOption(CURSOR_LIB) failed") << endl
; 
 956         wxUnusedVar( retcode 
); 
 960     if (copyDb
->OpenedWithConnectionString()) 
 962         // Connect to the data source 
 963         SQLTCHAR outConnectBuffer
[SQL_MAX_CONNECTSTR_LEN
+1]; 
 964         short outConnectBufferLen
; 
 966         inConnectionStr 
= copyDb
->GetConnectionInStr(); 
 968         retcode 
= SQLDriverConnect(hdbc
, NULL
, (SQLTCHAR FAR 
*)inConnectionStr
.c_str(), 
 969                             (SWORD
)inConnectionStr
.Length(), (SQLTCHAR FAR 
*)outConnectBuffer
, 
 970                             sizeof(outConnectBuffer
), &outConnectBufferLen
, SQL_DRIVER_COMPLETE
); 
 972         if ((retcode 
!= SQL_SUCCESS
) && 
 973             (retcode 
!= SQL_SUCCESS_WITH_INFO
)) 
 974             return(DispAllErrors(henv
, hdbc
)); 
 976         outConnectBuffer
[outConnectBufferLen
] = 0; 
 977         outConnectionStr 
= outConnectBuffer
; 
 978         dbOpenedWithConnectionString 
= true; 
 982         // Connect to the data source 
 983         retcode 
= SQLConnect(hdbc
, (SQLTCHAR FAR 
*) dsn
.c_str(), SQL_NTS
, 
 984                              (SQLTCHAR FAR 
*) uid
.c_str(), SQL_NTS
, 
 985                              (SQLTCHAR FAR 
*) authStr
.c_str(), SQL_NTS
); 
 988     if ((retcode 
!= SQL_SUCCESS
) && 
 989         (retcode 
!= SQL_SUCCESS_WITH_INFO
)) 
 990         return(DispAllErrors(henv
, hdbc
)); 
 993     If using Intersolv branded ODBC drivers, this is the place where you would substitute 
 994     your branded driver license information 
 996     SQLSetConnectOption(hdbc, 1041, (UDWORD) wxEmptyString); 
 997     SQLSetConnectOption(hdbc, 1042, (UDWORD) wxEmptyString); 
1000     // Mark database as open 
1003     // Allocate a statement handle for the database connection 
1004     if (SQLAllocStmt(hdbc
, &hstmt
) != SQL_SUCCESS
) 
1005         return(DispAllErrors(henv
, hdbc
)); 
1007     // Set Connection Options 
1008     if (!setConnectionOptions()) 
1011     // Instead of Querying the data source for info about itself, it can just be copied 
1012     // from the wxDb instance that was passed in (copyDb). 
1013     wxStrcpy(dbInf
.serverName
,copyDb
->dbInf
.serverName
); 
1014     wxStrcpy(dbInf
.databaseName
,copyDb
->dbInf
.databaseName
); 
1015     wxStrcpy(dbInf
.dbmsName
,copyDb
->dbInf
.dbmsName
); 
1016     wxStrcpy(dbInf
.dbmsVer
,copyDb
->dbInf
.dbmsVer
); 
1017     dbInf
.maxConnections 
= copyDb
->dbInf
.maxConnections
; 
1018     dbInf
.maxStmts 
= copyDb
->dbInf
.maxStmts
; 
1019     wxStrcpy(dbInf
.driverName
,copyDb
->dbInf
.driverName
); 
1020     wxStrcpy(dbInf
.odbcVer
,copyDb
->dbInf
.odbcVer
); 
1021     wxStrcpy(dbInf
.drvMgrOdbcVer
,copyDb
->dbInf
.drvMgrOdbcVer
); 
1022     wxStrcpy(dbInf
.driverVer
,copyDb
->dbInf
.driverVer
); 
1023     dbInf
.apiConfLvl 
= copyDb
->dbInf
.apiConfLvl
; 
1024     dbInf
.cliConfLvl 
= copyDb
->dbInf
.cliConfLvl
; 
1025     dbInf
.sqlConfLvl 
= copyDb
->dbInf
.sqlConfLvl
; 
1026     wxStrcpy(dbInf
.outerJoins
,copyDb
->dbInf
.outerJoins
); 
1027     wxStrcpy(dbInf
.procedureSupport
,copyDb
->dbInf
.procedureSupport
); 
1028     wxStrcpy(dbInf
.accessibleTables
,copyDb
->dbInf
.accessibleTables
); 
1029     dbInf
.cursorCommitBehavior 
= copyDb
->dbInf
.cursorCommitBehavior
; 
1030     dbInf
.cursorRollbackBehavior 
= copyDb
->dbInf
.cursorRollbackBehavior
; 
1031     dbInf
.supportNotNullClause 
= copyDb
->dbInf
.supportNotNullClause
; 
1032     wxStrcpy(dbInf
.supportIEF
,copyDb
->dbInf
.supportIEF
); 
1033     dbInf
.txnIsolation 
= copyDb
->dbInf
.txnIsolation
; 
1034     dbInf
.txnIsolationOptions 
= copyDb
->dbInf
.txnIsolationOptions
; 
1035     dbInf
.fetchDirections 
= copyDb
->dbInf
.fetchDirections
; 
1036     dbInf
.lockTypes 
= copyDb
->dbInf
.lockTypes
; 
1037     dbInf
.posOperations 
= copyDb
->dbInf
.posOperations
; 
1038     dbInf
.posStmts 
= copyDb
->dbInf
.posStmts
; 
1039     dbInf
.scrollConcurrency 
= copyDb
->dbInf
.scrollConcurrency
; 
1040     dbInf
.scrollOptions 
= copyDb
->dbInf
.scrollOptions
; 
1041     dbInf
.staticSensitivity 
= copyDb
->dbInf
.staticSensitivity
; 
1042     dbInf
.txnCapable 
= copyDb
->dbInf
.txnCapable
; 
1043     dbInf
.loginTimeout 
= copyDb
->dbInf
.loginTimeout
; 
1045     // VARCHAR = Variable length character string 
1046     typeInfVarchar
.FsqlType         
= copyDb
->typeInfVarchar
.FsqlType
; 
1047     typeInfVarchar
.TypeName         
= copyDb
->typeInfVarchar
.TypeName
; 
1048     typeInfVarchar
.Precision        
= copyDb
->typeInfVarchar
.Precision
; 
1049     typeInfVarchar
.CaseSensitive    
= copyDb
->typeInfVarchar
.CaseSensitive
; 
1050     typeInfVarchar
.MaximumScale     
= copyDb
->typeInfVarchar
.MaximumScale
; 
1053     typeInfFloat
.FsqlType         
= copyDb
->typeInfFloat
.FsqlType
; 
1054     typeInfFloat
.TypeName         
= copyDb
->typeInfFloat
.TypeName
; 
1055     typeInfFloat
.Precision        
= copyDb
->typeInfFloat
.Precision
; 
1056     typeInfFloat
.CaseSensitive    
= copyDb
->typeInfFloat
.CaseSensitive
; 
1057     typeInfFloat
.MaximumScale     
= copyDb
->typeInfFloat
.MaximumScale
; 
1060     typeInfInteger
.FsqlType         
= copyDb
->typeInfInteger
.FsqlType
; 
1061     typeInfInteger
.TypeName         
= copyDb
->typeInfInteger
.TypeName
; 
1062     typeInfInteger
.Precision        
= copyDb
->typeInfInteger
.Precision
; 
1063     typeInfInteger
.CaseSensitive    
= copyDb
->typeInfInteger
.CaseSensitive
; 
1064     typeInfInteger
.MaximumScale     
= copyDb
->typeInfInteger
.MaximumScale
; 
1067     typeInfDate
.FsqlType         
= copyDb
->typeInfDate
.FsqlType
; 
1068     typeInfDate
.TypeName         
= copyDb
->typeInfDate
.TypeName
; 
1069     typeInfDate
.Precision        
= copyDb
->typeInfDate
.Precision
; 
1070     typeInfDate
.CaseSensitive    
= copyDb
->typeInfDate
.CaseSensitive
; 
1071     typeInfDate
.MaximumScale     
= copyDb
->typeInfDate
.MaximumScale
; 
1074     typeInfBlob
.FsqlType         
= copyDb
->typeInfBlob
.FsqlType
; 
1075     typeInfBlob
.TypeName         
= copyDb
->typeInfBlob
.TypeName
; 
1076     typeInfBlob
.Precision        
= copyDb
->typeInfBlob
.Precision
; 
1077     typeInfBlob
.CaseSensitive    
= copyDb
->typeInfBlob
.CaseSensitive
; 
1078     typeInfBlob
.MaximumScale     
= copyDb
->typeInfBlob
.MaximumScale
; 
1081     typeInfMemo
.FsqlType         
= copyDb
->typeInfMemo
.FsqlType
; 
1082     typeInfMemo
.TypeName         
= copyDb
->typeInfMemo
.TypeName
; 
1083     typeInfMemo
.Precision        
= copyDb
->typeInfMemo
.Precision
; 
1084     typeInfMemo
.CaseSensitive    
= copyDb
->typeInfMemo
.CaseSensitive
; 
1085     typeInfMemo
.MaximumScale     
= copyDb
->typeInfMemo
.MaximumScale
; 
1087 #ifdef DBDEBUG_CONSOLE 
1088     cout 
<< wxT("VARCHAR DATA TYPE: ") << typeInfVarchar
.TypeName 
<< endl
; 
1089     cout 
<< wxT("INTEGER DATA TYPE: ") << typeInfInteger
.TypeName 
<< endl
; 
1090     cout 
<< wxT("FLOAT   DATA TYPE: ") << typeInfFloat
.TypeName 
<< endl
; 
1091     cout 
<< wxT("DATE    DATA TYPE: ") << typeInfDate
.TypeName 
<< endl
; 
1092     cout 
<< wxT("BLOB    DATA TYPE: ") << typeInfBlob
.TypeName 
<< endl
; 
1093     cout 
<< wxT("MEMO    DATA TYPE: ") << typeInfMemo
.TypeName 
<< endl
; 
1097     // Completed Successfully 
1102 /********** wxDb::setConnectionOptions() **********/ 
1103 bool wxDb::setConnectionOptions(void) 
1105  * NOTE: The Intersolv/Oracle 7 driver was "Not Capable" of setting the login timeout. 
1110     // I need to get the DBMS name here, because some of the connection options 
1111     // are database specific and need to call the Dbms() function. 
1114     retcode 
= SQLGetInfo(hdbc
, SQL_DBMS_NAME
, (UCHAR 
*) dbInf
.dbmsName
, sizeof(dbInf
.dbmsName
), &cb
); 
1115     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO
) 
1116         return(DispAllErrors(henv
, hdbc
)); 
1118     /* retcode = */ SQLSetConnectOption(hdbc
, SQL_AUTOCOMMIT
, SQL_AUTOCOMMIT_OFF
); 
1119     /* retcode = */ SQLSetConnectOption(hdbc
, SQL_OPT_TRACE
, SQL_OPT_TRACE_OFF
); 
1120 //  SQLSetConnectOption(hdbc, SQL_TXN_ISOLATION, SQL_TXN_READ_COMMITTED);  // No dirty reads 
1122     // By default, MS Sql Server closes cursors on commit and rollback.  The following 
1123     // call to SQLSetConnectOption() is needed to force SQL Server to preserve cursors 
1124     // after a transaction.  This is a driver specific option and is not part of the 
1125     // ODBC standard.  Note: this behavior is specific to the ODBC interface to SQL Server. 
1126     // The database settings don't have any effect one way or the other. 
1127     if (Dbms() == dbmsMS_SQL_SERVER
) 
1129         const long SQL_PRESERVE_CURSORS 
= 1204L; 
1130         const long SQL_PC_ON 
= 1L; 
1131         /* retcode = */ SQLSetConnectOption(hdbc
, SQL_PRESERVE_CURSORS
, SQL_PC_ON
); 
1134     // Display the connection options to verify them 
1135 #ifdef DBDEBUG_CONSOLE 
1137     cout 
<< wxT("****** CONNECTION OPTIONS ******") << endl
; 
1139     retcode 
= SQLGetConnectOption(hdbc
, SQL_AUTOCOMMIT
, &l
); 
1140     if (retcode 
!= SQL_SUCCESS
) 
1141         return(DispAllErrors(henv
, hdbc
)); 
1142     cout 
<< wxT("AUTOCOMMIT: ") << (l 
== SQL_AUTOCOMMIT_OFF 
? "OFF" : "ON") << endl
; 
1144     retcode 
= SQLGetConnectOption(hdbc
, SQL_ODBC_CURSORS
, &l
); 
1145     if (retcode 
!= SQL_SUCCESS
) 
1146         return(DispAllErrors(henv
, hdbc
)); 
1147     cout 
<< wxT("ODBC CURSORS: "); 
1150         case(SQL_CUR_USE_IF_NEEDED
): 
1151             cout 
<< wxT("SQL_CUR_USE_IF_NEEDED"); 
1153         case(SQL_CUR_USE_ODBC
): 
1154             cout 
<< wxT("SQL_CUR_USE_ODBC"); 
1156         case(SQL_CUR_USE_DRIVER
): 
1157             cout 
<< wxT("SQL_CUR_USE_DRIVER"); 
1162     retcode 
= SQLGetConnectOption(hdbc
, SQL_OPT_TRACE
, &l
) 
1163     if (retcode 
!= SQL_SUCCESS
) 
1164         return(DispAllErrors(henv
, hdbc
)); 
1165     cout 
<< wxT("TRACING: ") << (l 
== SQL_OPT_TRACE_OFF 
? wxT("OFF") : wxT("ON")) << endl
; 
1170     // Completed Successfully 
1173 } // wxDb::setConnectionOptions() 
1176 /********** wxDb::getDbInfo() **********/ 
1177 bool wxDb::getDbInfo(bool failOnDataTypeUnsupported
) 
1182     retcode 
= SQLGetInfo(hdbc
, SQL_SERVER_NAME
, (UCHAR
*) dbInf
.serverName
, sizeof(dbInf
.serverName
), &cb
); 
1183     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1185         DispAllErrors(henv
, hdbc
); 
1186         if (failOnDataTypeUnsupported
) 
1190     retcode 
= SQLGetInfo(hdbc
, SQL_DATABASE_NAME
, (UCHAR
*) dbInf
.databaseName
, sizeof(dbInf
.databaseName
), &cb
); 
1191     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1193         DispAllErrors(henv
, hdbc
); 
1194         if (failOnDataTypeUnsupported
) 
1198     retcode 
= SQLGetInfo(hdbc
, SQL_DBMS_NAME
, (UCHAR
*) dbInf
.dbmsName
, sizeof(dbInf
.dbmsName
), &cb
); 
1199     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1201         DispAllErrors(henv
, hdbc
); 
1202         if (failOnDataTypeUnsupported
) 
1207     // After upgrading to MSVC6, the original 20 char buffer below was insufficient, 
1208     // causing database connectivity to fail in some cases. 
1209     retcode 
= SQLGetInfo(hdbc
, SQL_DBMS_VER
, (UCHAR
*) dbInf
.dbmsVer
, sizeof(dbInf
.dbmsVer
), &cb
); 
1210     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1212         DispAllErrors(henv
, hdbc
); 
1213         if (failOnDataTypeUnsupported
) 
1217     retcode 
= SQLGetInfo(hdbc
, SQL_ACTIVE_CONNECTIONS
, (UCHAR
*) &dbInf
.maxConnections
, sizeof(dbInf
.maxConnections
), &cb
); 
1218     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1220         DispAllErrors(henv
, hdbc
); 
1221         if (failOnDataTypeUnsupported
) 
1225     retcode 
= SQLGetInfo(hdbc
, SQL_ACTIVE_STATEMENTS
, (UCHAR
*) &dbInf
.maxStmts
, sizeof(dbInf
.maxStmts
), &cb
); 
1226     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1228         DispAllErrors(henv
, hdbc
); 
1229         if (failOnDataTypeUnsupported
) 
1233     retcode 
= SQLGetInfo(hdbc
, SQL_DRIVER_NAME
, (UCHAR
*) dbInf
.driverName
, sizeof(dbInf
.driverName
), &cb
); 
1234     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1236         DispAllErrors(henv
, hdbc
); 
1237         if (failOnDataTypeUnsupported
) 
1241     retcode 
= SQLGetInfo(hdbc
, SQL_DRIVER_ODBC_VER
, (UCHAR
*) dbInf
.odbcVer
, sizeof(dbInf
.odbcVer
), &cb
); 
1242     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1244         DispAllErrors(henv
, hdbc
); 
1245         if (failOnDataTypeUnsupported
) 
1249     retcode 
= SQLGetInfo(hdbc
, SQL_ODBC_VER
, (UCHAR
*) dbInf
.drvMgrOdbcVer
, sizeof(dbInf
.drvMgrOdbcVer
), &cb
); 
1250     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO
) 
1252         DispAllErrors(henv
, hdbc
); 
1253         if (failOnDataTypeUnsupported
) 
1257     retcode 
= SQLGetInfo(hdbc
, SQL_DRIVER_VER
, (UCHAR
*) dbInf
.driverVer
, sizeof(dbInf
.driverVer
), &cb
); 
1258     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1260         DispAllErrors(henv
, hdbc
); 
1261         if (failOnDataTypeUnsupported
) 
1265     retcode 
= SQLGetInfo(hdbc
, SQL_ODBC_API_CONFORMANCE
, (UCHAR
*) &dbInf
.apiConfLvl
, sizeof(dbInf
.apiConfLvl
), &cb
); 
1266     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1268         DispAllErrors(henv
, hdbc
); 
1269         if (failOnDataTypeUnsupported
) 
1273     retcode 
= SQLGetInfo(hdbc
, SQL_ODBC_SAG_CLI_CONFORMANCE
, (UCHAR
*) &dbInf
.cliConfLvl
, sizeof(dbInf
.cliConfLvl
), &cb
); 
1274     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1276         // Not all drivers support this call - Nick Gorham(unixODBC) 
1277         dbInf
.cliConfLvl 
= 0; 
1278         DispAllErrors(henv
, hdbc
); 
1279         if (failOnDataTypeUnsupported
) 
1283     retcode 
= SQLGetInfo(hdbc
, SQL_ODBC_SQL_CONFORMANCE
, (UCHAR
*) &dbInf
.sqlConfLvl
, sizeof(dbInf
.sqlConfLvl
), &cb
); 
1284     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1286         DispAllErrors(henv
, hdbc
); 
1287         if (failOnDataTypeUnsupported
) 
1291     retcode 
= SQLGetInfo(hdbc
, SQL_OUTER_JOINS
, (UCHAR
*) dbInf
.outerJoins
, sizeof(dbInf
.outerJoins
), &cb
); 
1292     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1294         DispAllErrors(henv
, hdbc
); 
1295         if (failOnDataTypeUnsupported
) 
1299     retcode 
= SQLGetInfo(hdbc
, SQL_PROCEDURES
, (UCHAR
*) dbInf
.procedureSupport
, sizeof(dbInf
.procedureSupport
), &cb
); 
1300     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1302         DispAllErrors(henv
, hdbc
); 
1303         if (failOnDataTypeUnsupported
) 
1307     retcode 
= SQLGetInfo(hdbc
, SQL_ACCESSIBLE_TABLES
, (UCHAR
*) dbInf
.accessibleTables
, sizeof(dbInf
.accessibleTables
), &cb
); 
1308     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1310         DispAllErrors(henv
, hdbc
); 
1311         if (failOnDataTypeUnsupported
) 
1315     retcode 
= SQLGetInfo(hdbc
, SQL_CURSOR_COMMIT_BEHAVIOR
, (UCHAR
*) &dbInf
.cursorCommitBehavior
, sizeof(dbInf
.cursorCommitBehavior
), &cb
); 
1316     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1318         DispAllErrors(henv
, hdbc
); 
1319         if (failOnDataTypeUnsupported
) 
1323     retcode 
= SQLGetInfo(hdbc
, SQL_CURSOR_ROLLBACK_BEHAVIOR
, (UCHAR
*) &dbInf
.cursorRollbackBehavior
, sizeof(dbInf
.cursorRollbackBehavior
), &cb
); 
1324     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1326         DispAllErrors(henv
, hdbc
); 
1327         if (failOnDataTypeUnsupported
) 
1331     retcode 
= SQLGetInfo(hdbc
, SQL_NON_NULLABLE_COLUMNS
, (UCHAR
*) &dbInf
.supportNotNullClause
, sizeof(dbInf
.supportNotNullClause
), &cb
); 
1332     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1334         DispAllErrors(henv
, hdbc
); 
1335         if (failOnDataTypeUnsupported
) 
1339     retcode 
= SQLGetInfo(hdbc
, SQL_ODBC_SQL_OPT_IEF
, (UCHAR
*) dbInf
.supportIEF
, sizeof(dbInf
.supportIEF
), &cb
); 
1340     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1342         DispAllErrors(henv
, hdbc
); 
1343         if (failOnDataTypeUnsupported
) 
1347     retcode 
= SQLGetInfo(hdbc
, SQL_DEFAULT_TXN_ISOLATION
, (UCHAR
*) &dbInf
.txnIsolation
, sizeof(dbInf
.txnIsolation
), &cb
); 
1348     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1350         DispAllErrors(henv
, hdbc
); 
1351         if (failOnDataTypeUnsupported
) 
1355     retcode 
= SQLGetInfo(hdbc
, SQL_TXN_ISOLATION_OPTION
, (UCHAR
*) &dbInf
.txnIsolationOptions
, sizeof(dbInf
.txnIsolationOptions
), &cb
); 
1356     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1358         DispAllErrors(henv
, hdbc
); 
1359         if (failOnDataTypeUnsupported
) 
1363     retcode 
= SQLGetInfo(hdbc
, SQL_FETCH_DIRECTION
, (UCHAR
*) &dbInf
.fetchDirections
, sizeof(dbInf
.fetchDirections
), &cb
); 
1364     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1366         DispAllErrors(henv
, hdbc
); 
1367         if (failOnDataTypeUnsupported
) 
1371     retcode 
= SQLGetInfo(hdbc
, SQL_LOCK_TYPES
, (UCHAR
*) &dbInf
.lockTypes
, sizeof(dbInf
.lockTypes
), &cb
); 
1372     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1374         DispAllErrors(henv
, hdbc
); 
1375         if (failOnDataTypeUnsupported
) 
1379     retcode 
= SQLGetInfo(hdbc
, SQL_POS_OPERATIONS
, (UCHAR
*) &dbInf
.posOperations
, sizeof(dbInf
.posOperations
), &cb
); 
1380     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1382         DispAllErrors(henv
, hdbc
); 
1383         if (failOnDataTypeUnsupported
) 
1387     retcode 
= SQLGetInfo(hdbc
, SQL_POSITIONED_STATEMENTS
, (UCHAR
*) &dbInf
.posStmts
, sizeof(dbInf
.posStmts
), &cb
); 
1388     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1390         DispAllErrors(henv
, hdbc
); 
1391         if (failOnDataTypeUnsupported
) 
1395     retcode 
= SQLGetInfo(hdbc
, SQL_SCROLL_CONCURRENCY
, (UCHAR
*) &dbInf
.scrollConcurrency
, sizeof(dbInf
.scrollConcurrency
), &cb
); 
1396     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1398         DispAllErrors(henv
, hdbc
); 
1399         if (failOnDataTypeUnsupported
) 
1403     retcode 
= SQLGetInfo(hdbc
, SQL_SCROLL_OPTIONS
, (UCHAR
*) &dbInf
.scrollOptions
, sizeof(dbInf
.scrollOptions
), &cb
); 
1404     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1406         DispAllErrors(henv
, hdbc
); 
1407         if (failOnDataTypeUnsupported
) 
1411     retcode 
= SQLGetInfo(hdbc
, SQL_STATIC_SENSITIVITY
, (UCHAR
*) &dbInf
.staticSensitivity
, sizeof(dbInf
.staticSensitivity
), &cb
); 
1412     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1414         DispAllErrors(henv
, hdbc
); 
1415         if (failOnDataTypeUnsupported
) 
1419     retcode 
= SQLGetInfo(hdbc
, SQL_TXN_CAPABLE
, (UCHAR
*) &dbInf
.txnCapable
, sizeof(dbInf
.txnCapable
), &cb
); 
1420     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1422         DispAllErrors(henv
, hdbc
); 
1423         if (failOnDataTypeUnsupported
) 
1427     retcode 
= SQLGetInfo(hdbc
, SQL_LOGIN_TIMEOUT
, (UCHAR
*) &dbInf
.loginTimeout
, sizeof(dbInf
.loginTimeout
), &cb
); 
1428     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO 
) 
1430         DispAllErrors(henv
, hdbc
); 
1431         if (failOnDataTypeUnsupported
) 
1435 #ifdef DBDEBUG_CONSOLE 
1436     cout 
<< wxT("***** DATA SOURCE INFORMATION *****") << endl
; 
1437     cout 
<< wxT(wxT("SERVER Name: ") << dbInf
.serverName 
<< endl
; 
1438     cout 
<< wxT("DBMS Name: ") << dbInf
.dbmsName 
<< wxT("; DBMS Version: ") << dbInf
.dbmsVer 
<< endl
; 
1439     cout 
<< wxT("ODBC Version: ") << dbInf
.odbcVer 
<< wxT("; Driver Version: ") << dbInf
.driverVer 
<< endl
; 
1441     cout 
<< wxT("API Conf. Level: "); 
1442     switch(dbInf
.apiConfLvl
) 
1444         case SQL_OAC_NONE
:      cout 
<< wxT("None");       break; 
1445         case SQL_OAC_LEVEL1
:    cout 
<< wxT("Level 1");    break; 
1446         case SQL_OAC_LEVEL2
:    cout 
<< wxT("Level 2");    break; 
1450     cout 
<< wxT("SAG CLI Conf. Level: "); 
1451     switch(dbInf
.cliConfLvl
) 
1453         case SQL_OSCC_NOT_COMPLIANT
:    cout 
<< wxT("Not Compliant");    break; 
1454         case SQL_OSCC_COMPLIANT
:        cout 
<< wxT("Compliant");        break; 
1458     cout 
<< wxT("SQL Conf. Level: "); 
1459     switch(dbInf
.sqlConfLvl
) 
1461         case SQL_OSC_MINIMUM
:     cout 
<< wxT("Minimum Grammar");     break; 
1462         case SQL_OSC_CORE
:        cout 
<< wxT("Core Grammar");        break; 
1463         case SQL_OSC_EXTENDED
:    cout 
<< wxT("Extended Grammar");    break; 
1467     cout 
<< wxT("Max. Connections: ")       << dbInf
.maxConnections   
<< endl
; 
1468     cout 
<< wxT("Outer Joins: ")            << dbInf
.outerJoins       
<< endl
; 
1469     cout 
<< wxT("Support for Procedures: ") << dbInf
.procedureSupport 
<< endl
; 
1470     cout 
<< wxT("All tables accessible : ") << dbInf
.accessibleTables 
<< endl
; 
1471     cout 
<< wxT("Cursor COMMIT Behavior: "); 
1472     switch(dbInf
.cursorCommitBehavior
) 
1474         case SQL_CB_DELETE
:        cout 
<< wxT("Delete cursors");      break; 
1475         case SQL_CB_CLOSE
:         cout 
<< wxT("Close cursors");       break; 
1476         case SQL_CB_PRESERVE
:      cout 
<< wxT("Preserve cursors");    break; 
1480     cout 
<< wxT("Cursor ROLLBACK Behavior: "); 
1481     switch(dbInf
.cursorRollbackBehavior
) 
1483         case SQL_CB_DELETE
:      cout 
<< wxT("Delete cursors");      break; 
1484         case SQL_CB_CLOSE
:       cout 
<< wxT("Close cursors");       break; 
1485         case SQL_CB_PRESERVE
:    cout 
<< wxT("Preserve cursors");    break; 
1489     cout 
<< wxT("Support NOT NULL clause: "); 
1490     switch(dbInf
.supportNotNullClause
) 
1492         case SQL_NNC_NULL
:        cout 
<< wxT("No");        break; 
1493         case SQL_NNC_NON_NULL
:    cout 
<< wxT("Yes");       break; 
1497     cout 
<< wxT("Support IEF (Ref. Integrity): ") << dbInf
.supportIEF   
<< endl
; 
1498     cout 
<< wxT("Login Timeout: ")                << dbInf
.loginTimeout 
<< endl
; 
1500     cout 
<< endl 
<< endl 
<< wxT("more ...") << endl
; 
1503     cout 
<< wxT("Default Transaction Isolation: "; 
1504     switch(dbInf
.txnIsolation
) 
1506         case SQL_TXN_READ_UNCOMMITTED
:  cout 
<< wxT("Read Uncommitted");    break; 
1507         case SQL_TXN_READ_COMMITTED
:    cout 
<< wxT("Read Committed");      break; 
1508         case SQL_TXN_REPEATABLE_READ
:   cout 
<< wxT("Repeatable Read");     break; 
1509         case SQL_TXN_SERIALIZABLE
:      cout 
<< wxT("Serializable");        break; 
1511         case SQL_TXN_VERSIONING
:        cout 
<< wxT("Versioning");          break; 
1516     cout 
<< wxT("Transaction Isolation Options: "); 
1517     if (dbInf
.txnIsolationOptions 
& SQL_TXN_READ_UNCOMMITTED
) 
1518         cout 
<< wxT("Read Uncommitted, "); 
1519     if (dbInf
.txnIsolationOptions 
& SQL_TXN_READ_COMMITTED
) 
1520         cout 
<< wxT("Read Committed, "); 
1521     if (dbInf
.txnIsolationOptions 
& SQL_TXN_REPEATABLE_READ
) 
1522         cout 
<< wxT("Repeatable Read, "); 
1523     if (dbInf
.txnIsolationOptions 
& SQL_TXN_SERIALIZABLE
) 
1524         cout 
<< wxT("Serializable, "); 
1526     if (dbInf
.txnIsolationOptions 
& SQL_TXN_VERSIONING
) 
1527         cout 
<< wxT("Versioning"); 
1531     cout 
<< wxT("Fetch Directions Supported:") << endl 
<< wxT("   "); 
1532     if (dbInf
.fetchDirections 
& SQL_FD_FETCH_NEXT
) 
1533         cout 
<< wxT("Next, "); 
1534     if (dbInf
.fetchDirections 
& SQL_FD_FETCH_PRIOR
) 
1535         cout 
<< wxT("Prev, "); 
1536     if (dbInf
.fetchDirections 
& SQL_FD_FETCH_FIRST
) 
1537         cout 
<< wxT("First, "); 
1538     if (dbInf
.fetchDirections 
& SQL_FD_FETCH_LAST
) 
1539         cout 
<< wxT("Last, "); 
1540     if (dbInf
.fetchDirections 
& SQL_FD_FETCH_ABSOLUTE
) 
1541         cout 
<< wxT("Absolute, "); 
1542     if (dbInf
.fetchDirections 
& SQL_FD_FETCH_RELATIVE
) 
1543         cout 
<< wxT("Relative, "); 
1545     if (dbInf
.fetchDirections 
& SQL_FD_FETCH_RESUME
) 
1546         cout 
<< wxT("Resume, "); 
1548     if (dbInf
.fetchDirections 
& SQL_FD_FETCH_BOOKMARK
) 
1549         cout 
<< wxT("Bookmark"); 
1552     cout 
<< wxT("Lock Types Supported (SQLSetPos): "); 
1553     if (dbInf
.lockTypes 
& SQL_LCK_NO_CHANGE
) 
1554         cout 
<< wxT("No Change, "); 
1555     if (dbInf
.lockTypes 
& SQL_LCK_EXCLUSIVE
) 
1556         cout 
<< wxT("Exclusive, "); 
1557     if (dbInf
.lockTypes 
& SQL_LCK_UNLOCK
) 
1558         cout 
<< wxT("UnLock"); 
1561     cout 
<< wxT("Position Operations Supported (SQLSetPos): "); 
1562     if (dbInf
.posOperations 
& SQL_POS_POSITION
) 
1563         cout 
<< wxT("Position, "); 
1564     if (dbInf
.posOperations 
& SQL_POS_REFRESH
) 
1565         cout 
<< wxT("Refresh, "); 
1566     if (dbInf
.posOperations 
& SQL_POS_UPDATE
) 
1567         cout 
<< wxT("Upd, ")); 
1568     if (dbInf
.posOperations 
& SQL_POS_DELETE
) 
1569         cout 
<< wxT("Del, "); 
1570     if (dbInf
.posOperations 
& SQL_POS_ADD
) 
1574     cout 
<< wxT("Positioned Statements Supported: "); 
1575     if (dbInf
.posStmts 
& SQL_PS_POSITIONED_DELETE
) 
1576         cout 
<< wxT("Pos delete, "); 
1577     if (dbInf
.posStmts 
& SQL_PS_POSITIONED_UPDATE
) 
1578         cout 
<< wxT("Pos update, "); 
1579     if (dbInf
.posStmts 
& SQL_PS_SELECT_FOR_UPDATE
) 
1580         cout 
<< wxT("Select for update"); 
1583     cout 
<< wxT("Scroll Concurrency: "); 
1584     if (dbInf
.scrollConcurrency 
& SQL_SCCO_READ_ONLY
) 
1585         cout 
<< wxT("Read Only, "); 
1586     if (dbInf
.scrollConcurrency 
& SQL_SCCO_LOCK
) 
1587         cout 
<< wxT("Lock, "); 
1588     if (dbInf
.scrollConcurrency 
& SQL_SCCO_OPT_ROWVER
) 
1589         cout 
<< wxT("Opt. Rowver, "); 
1590     if (dbInf
.scrollConcurrency 
& SQL_SCCO_OPT_VALUES
) 
1591         cout 
<< wxT("Opt. Values"); 
1594     cout 
<< wxT("Scroll Options: "); 
1595     if (dbInf
.scrollOptions 
& SQL_SO_FORWARD_ONLY
) 
1596         cout 
<< wxT("Fwd Only, "); 
1597     if (dbInf
.scrollOptions 
& SQL_SO_STATIC
) 
1598         cout 
<< wxT("Static, "); 
1599     if (dbInf
.scrollOptions 
& SQL_SO_KEYSET_DRIVEN
) 
1600         cout 
<< wxT("Keyset Driven, "); 
1601     if (dbInf
.scrollOptions 
& SQL_SO_DYNAMIC
) 
1602         cout 
<< wxT("Dynamic, "); 
1603     if (dbInf
.scrollOptions 
& SQL_SO_MIXED
) 
1604         cout 
<< wxT("Mixed"); 
1607     cout 
<< wxT("Static Sensitivity: "); 
1608     if (dbInf
.staticSensitivity 
& SQL_SS_ADDITIONS
) 
1609         cout 
<< wxT("Additions, "); 
1610     if (dbInf
.staticSensitivity 
& SQL_SS_DELETIONS
) 
1611         cout 
<< wxT("Deletions, "); 
1612     if (dbInf
.staticSensitivity 
& SQL_SS_UPDATES
) 
1613         cout 
<< wxT("Updates"); 
1616     cout 
<< wxT("Transaction Capable?: "); 
1617     switch(dbInf
.txnCapable
) 
1619         case SQL_TC_NONE
:          cout 
<< wxT("No");            break; 
1620         case SQL_TC_DML
:           cout 
<< wxT("DML Only");      break; 
1621         case SQL_TC_DDL_COMMIT
:    cout 
<< wxT("DDL Commit");    break; 
1622         case SQL_TC_DDL_IGNORE
:    cout 
<< wxT("DDL Ignore");    break; 
1623         case SQL_TC_ALL
:           cout 
<< wxT("DDL & DML");     break; 
1630     // Completed Successfully 
1633 } // wxDb::getDbInfo() 
1636 /********** wxDb::getDataTypeInfo() **********/ 
1637 bool wxDb::getDataTypeInfo(SWORD fSqlType
, wxDbSqlTypeInfo 
&structSQLTypeInfo
) 
1640  * fSqlType will be something like SQL_VARCHAR.  This parameter determines 
1641  * the data type inf. is gathered for. 
1643  * wxDbSqlTypeInfo is a structure that is filled in with data type information, 
1648     // Get information about the data type specified 
1649     if (SQLGetTypeInfo(hstmt
, fSqlType
) != SQL_SUCCESS
) 
1650         return(DispAllErrors(henv
, hdbc
, hstmt
)); 
1653     retcode 
= SQLFetch(hstmt
); 
1654     if (retcode 
!= SQL_SUCCESS
) 
1656 #ifdef DBDEBUG_CONSOLE 
1657         if (retcode 
== SQL_NO_DATA_FOUND
) 
1658             cout 
<< wxT("SQL_NO_DATA_FOUND fetching information about data type.") << endl
; 
1660         DispAllErrors(henv
, hdbc
, hstmt
); 
1661         SQLFreeStmt(hstmt
, SQL_CLOSE
); 
1665     wxChar typeName
[DB_TYPE_NAME_LEN
+1]; 
1667     // Obtain columns from the record 
1668     if (SQLGetData(hstmt
, 1, SQL_C_WXCHAR
, typeName
, sizeof(typeName
), &cbRet
) != SQL_SUCCESS
) 
1669         return(DispAllErrors(henv
, hdbc
, hstmt
)); 
1671     structSQLTypeInfo
.TypeName 
= typeName
; 
1673     // BJO 20000503: no more needed with new GetColumns... 
1676     if (Dbms() == dbmsMY_SQL
) 
1678         if (structSQLTypeInfo
.TypeName 
== wxT("middleint")) 
1679             structSQLTypeInfo
.TypeName 
= wxT("mediumint"); 
1680         else if (structSQLTypeInfo
.TypeName 
== wxT("middleint unsigned")) 
1681             structSQLTypeInfo
.TypeName 
= wxT("mediumint unsigned"); 
1682         else if (structSQLTypeInfo
.TypeName 
== wxT("integer")) 
1683             structSQLTypeInfo
.TypeName 
= wxT("int"); 
1684         else if (structSQLTypeInfo
.TypeName 
== wxT("integer unsigned")) 
1685             structSQLTypeInfo
.TypeName 
= wxT("int unsigned"); 
1686         else if (structSQLTypeInfo
.TypeName 
== wxT("middleint")) 
1687             structSQLTypeInfo
.TypeName 
= wxT("mediumint"); 
1688         else if (structSQLTypeInfo
.TypeName 
== wxT("varchar")) 
1689             structSQLTypeInfo
.TypeName 
= wxT("char"); 
1692     // BJO 20000427 : OpenLink driver 
1693     if (!wxStrncmp(dbInf
.driverName
, wxT("oplodbc"), 7) || 
1694         !wxStrncmp(dbInf
.driverName
, wxT("OLOD"), 4)) 
1696         if (structSQLTypeInfo
.TypeName 
== wxT("double precision")) 
1697             structSQLTypeInfo
.TypeName 
= wxT("real"); 
1701     if (SQLGetData(hstmt
, 3, SQL_C_LONG
, (UCHAR
*) &structSQLTypeInfo
.Precision
, 0, &cbRet
) != SQL_SUCCESS
) 
1702         return(DispAllErrors(henv
, hdbc
, hstmt
)); 
1703     if (SQLGetData(hstmt
, 8, SQL_C_SHORT
, (UCHAR
*) &structSQLTypeInfo
.CaseSensitive
, 0, &cbRet
) != SQL_SUCCESS
) 
1704         return(DispAllErrors(henv
, hdbc
, hstmt
)); 
1705 //    if (SQLGetData(hstmt, 14, SQL_C_SHORT, (UCHAR*) &structSQLTypeInfo.MinimumScale, 0, &cbRet) != SQL_SUCCESS) 
1706 //        return(DispAllErrors(henv, hdbc, hstmt)); 
1708     if (SQLGetData(hstmt
, 15, SQL_C_SHORT
,(UCHAR
*)  &structSQLTypeInfo
.MaximumScale
, 0, &cbRet
) != SQL_SUCCESS
) 
1709         return(DispAllErrors(henv
, hdbc
, hstmt
)); 
1711     if (structSQLTypeInfo
.MaximumScale 
< 0) 
1712         structSQLTypeInfo
.MaximumScale 
= 0; 
1714     // Close the statement handle which closes open cursors 
1715     if (SQLFreeStmt(hstmt
, SQL_CLOSE
) != SQL_SUCCESS
) 
1716         return(DispAllErrors(henv
, hdbc
, hstmt
)); 
1718     // Completed Successfully 
1721 } // wxDb::getDataTypeInfo() 
1724 /********** wxDb::Close() **********/ 
1725 void wxDb::Close(void) 
1727     // Close the Sql Log file 
1734     // Free statement handle 
1737         if (SQLFreeStmt(hstmt
, SQL_DROP
) != SQL_SUCCESS
) 
1738             DispAllErrors(henv
, hdbc
); 
1741     // Disconnect from the datasource 
1742     if (SQLDisconnect(hdbc
) != SQL_SUCCESS
) 
1743         DispAllErrors(henv
, hdbc
); 
1745     // Free the connection to the datasource 
1746     if (SQLFreeConnect(hdbc
) != SQL_SUCCESS
) 
1747         DispAllErrors(henv
, hdbc
); 
1749     // There should be zero Ctable objects still connected to this db object 
1750     wxASSERT(nTables 
== 0); 
1754     wxList::compatibility_iterator pNode
; 
1755     pNode 
= TablesInUse
.GetFirst(); 
1759         tiu 
= (wxTablesInUse 
*)pNode
->GetData(); 
1760         if (tiu
->pDb 
== this) 
1762             s
.Printf(wxT("(%-20s)     tableID:[%6lu]     pDb:[%p]"), tiu
->tableName
,tiu
->tableID
,tiu
->pDb
); 
1763             s2
.Printf(wxT("Orphaned table found using pDb:[%p]"),this); 
1764             wxLogDebug(s
.c_str(),s2
.c_str()); 
1766         pNode 
= pNode
->GetNext(); 
1770     // Copy the error messages to a global variable 
1772     for (i 
= 0; i 
< DB_MAX_ERROR_HISTORY
; i
++) 
1773         wxStrcpy(DBerrorList
[i
], errorList
[i
]); 
1775     dbmsType 
= dbmsUNIDENTIFIED
; 
1781 /********** wxDb::CommitTrans() **********/ 
1782 bool wxDb::CommitTrans(void) 
1786         // Commit the transaction 
1787         if (SQLTransact(henv
, hdbc
, SQL_COMMIT
) != SQL_SUCCESS
) 
1788             return(DispAllErrors(henv
, hdbc
)); 
1791     // Completed successfully 
1794 } // wxDb::CommitTrans() 
1797 /********** wxDb::RollbackTrans() **********/ 
1798 bool wxDb::RollbackTrans(void) 
1800     // Rollback the transaction 
1801     if (SQLTransact(henv
, hdbc
, SQL_ROLLBACK
) != SQL_SUCCESS
) 
1802         return(DispAllErrors(henv
, hdbc
)); 
1804     // Completed successfully 
1807 } // wxDb::RollbackTrans() 
1810 /********** wxDb::DispAllErrors() **********/ 
1811 bool wxDb::DispAllErrors(HENV aHenv
, HDBC aHdbc
, HSTMT aHstmt
) 
1813  * This function is called internally whenever an error condition prevents the user's 
1814  * request from being executed.  This function will query the datasource as to the 
1815  * actual error(s) that just occurred on the previous request of the datasource. 
1817  * The function will retrieve each error condition from the datasource and 
1818  * Printf the codes/text values into a string which it then logs via logError(). 
1819  * If in DBDEBUG_CONSOLE mode, the constructed string will be displayed in the console 
1820  * window and program execution will be paused until the user presses a key. 
1822  * This function always returns false, so that functions which call this function 
1823  * can have a line like "return (DispAllErrors(henv, hdbc));" to indicate the failure 
1824  * of the user's request, so that the calling code can then process the error message log. 
1827     wxString odbcErrMsg
; 
1829    while (SQLError(aHenv
, aHdbc
, aHstmt
, (SQLTCHAR FAR 
*) sqlState
, &nativeError
, (SQLTCHAR FAR 
*) errorMsg
, SQL_MAX_MESSAGE_LENGTH 
- 1, &cbErrorMsg
) == SQL_SUCCESS
) 
1831         odbcErrMsg
.Printf(wxT("SQL State = %s\nNative Error Code = %li\nError Message = %s\n"), sqlState
, nativeError
, errorMsg
); 
1832         logError(odbcErrMsg
, sqlState
); 
1835 #ifdef DBDEBUG_CONSOLE 
1836             // When run in console mode, use standard out to display errors. 
1837             cout 
<< odbcErrMsg
.c_str() << endl
; 
1838             cout 
<< wxT("Press any key to continue...") << endl
; 
1843             wxLogDebug(odbcErrMsg
,wxT("ODBC DEBUG MESSAGE from DispAllErrors()")); 
1848     return false;  // This function always returns false. 
1850 } // wxDb::DispAllErrors() 
1853 /********** wxDb::GetNextError() **********/ 
1854 bool wxDb::GetNextError(HENV aHenv
, HDBC aHdbc
, HSTMT aHstmt
) 
1856    if (SQLError(aHenv
, aHdbc
, aHstmt
, (SQLTCHAR FAR 
*) sqlState
, &nativeError
, (SQLTCHAR FAR 
*) errorMsg
, SQL_MAX_MESSAGE_LENGTH 
- 1, &cbErrorMsg
) == SQL_SUCCESS
) 
1861 } // wxDb::GetNextError() 
1864 /********** wxDb::DispNextError() **********/ 
1865 void wxDb::DispNextError(void) 
1867     wxString odbcErrMsg
; 
1869     odbcErrMsg
.Printf(wxT("SQL State = %s\nNative Error Code = %li\nError Message = %s\n"), sqlState
, nativeError
, errorMsg
); 
1870     logError(odbcErrMsg
, sqlState
); 
1875 #ifdef DBDEBUG_CONSOLE 
1876     // When run in console mode, use standard out to display errors. 
1877     cout 
<< odbcErrMsg
.c_str() << endl
; 
1878     cout 
<< wxT("Press any key to continue...")  << endl
; 
1883     wxLogDebug(odbcErrMsg
,wxT("ODBC DEBUG MESSAGE")); 
1884 #endif  // __WXDEBUG__ 
1886 } // wxDb::DispNextError() 
1889 /********** wxDb::logError() **********/ 
1890 void wxDb::logError(const wxString 
&errMsg
, const wxString 
&SQLState
) 
1892     wxASSERT(errMsg
.Length()); 
1894     static int pLast 
= -1; 
1897     if (++pLast 
== DB_MAX_ERROR_HISTORY
) 
1900         for (i 
= 0; i 
< DB_MAX_ERROR_HISTORY
-1; i
++) 
1901             wxStrcpy(errorList
[i
], errorList
[i
+1]); 
1905     wxStrncpy(errorList
[pLast
], errMsg
, DB_MAX_ERROR_MSG_LEN
); 
1906     errorList
[pLast
][DB_MAX_ERROR_MSG_LEN
] = 0; 
1908     if (SQLState
.Length()) 
1909         if ((dbStatus 
= TranslateSqlState(SQLState
)) != DB_ERR_FUNCTION_SEQUENCE_ERROR
) 
1910             DB_STATUS 
= dbStatus
; 
1912     // Add the errmsg to the sql log 
1913     WriteSqlLog(errMsg
); 
1915 }  // wxDb::logError() 
1918 /**********wxDb::TranslateSqlState()  **********/ 
1919 int wxDb::TranslateSqlState(const wxString 
&SQLState
) 
1921     if (!wxStrcmp(SQLState
, wxT("01000"))) 
1922         return(DB_ERR_GENERAL_WARNING
); 
1923     if (!wxStrcmp(SQLState
, wxT("01002"))) 
1924         return(DB_ERR_DISCONNECT_ERROR
); 
1925     if (!wxStrcmp(SQLState
, wxT("01004"))) 
1926         return(DB_ERR_DATA_TRUNCATED
); 
1927     if (!wxStrcmp(SQLState
, wxT("01006"))) 
1928         return(DB_ERR_PRIV_NOT_REVOKED
); 
1929     if (!wxStrcmp(SQLState
, wxT("01S00"))) 
1930         return(DB_ERR_INVALID_CONN_STR_ATTR
); 
1931     if (!wxStrcmp(SQLState
, wxT("01S01"))) 
1932         return(DB_ERR_ERROR_IN_ROW
); 
1933     if (!wxStrcmp(SQLState
, wxT("01S02"))) 
1934         return(DB_ERR_OPTION_VALUE_CHANGED
); 
1935     if (!wxStrcmp(SQLState
, wxT("01S03"))) 
1936         return(DB_ERR_NO_ROWS_UPD_OR_DEL
); 
1937     if (!wxStrcmp(SQLState
, wxT("01S04"))) 
1938         return(DB_ERR_MULTI_ROWS_UPD_OR_DEL
); 
1939     if (!wxStrcmp(SQLState
, wxT("07001"))) 
1940         return(DB_ERR_WRONG_NO_OF_PARAMS
); 
1941     if (!wxStrcmp(SQLState
, wxT("07006"))) 
1942         return(DB_ERR_DATA_TYPE_ATTR_VIOL
); 
1943     if (!wxStrcmp(SQLState
, wxT("08001"))) 
1944         return(DB_ERR_UNABLE_TO_CONNECT
); 
1945     if (!wxStrcmp(SQLState
, wxT("08002"))) 
1946         return(DB_ERR_CONNECTION_IN_USE
); 
1947     if (!wxStrcmp(SQLState
, wxT("08003"))) 
1948         return(DB_ERR_CONNECTION_NOT_OPEN
); 
1949     if (!wxStrcmp(SQLState
, wxT("08004"))) 
1950         return(DB_ERR_REJECTED_CONNECTION
); 
1951     if (!wxStrcmp(SQLState
, wxT("08007"))) 
1952         return(DB_ERR_CONN_FAIL_IN_TRANS
); 
1953     if (!wxStrcmp(SQLState
, wxT("08S01"))) 
1954         return(DB_ERR_COMM_LINK_FAILURE
); 
1955     if (!wxStrcmp(SQLState
, wxT("21S01"))) 
1956         return(DB_ERR_INSERT_VALUE_LIST_MISMATCH
); 
1957     if (!wxStrcmp(SQLState
, wxT("21S02"))) 
1958         return(DB_ERR_DERIVED_TABLE_MISMATCH
); 
1959     if (!wxStrcmp(SQLState
, wxT("22001"))) 
1960         return(DB_ERR_STRING_RIGHT_TRUNC
); 
1961     if (!wxStrcmp(SQLState
, wxT("22003"))) 
1962         return(DB_ERR_NUMERIC_VALUE_OUT_OF_RNG
); 
1963     if (!wxStrcmp(SQLState
, wxT("22005"))) 
1964         return(DB_ERR_ERROR_IN_ASSIGNMENT
); 
1965     if (!wxStrcmp(SQLState
, wxT("22008"))) 
1966         return(DB_ERR_DATETIME_FLD_OVERFLOW
); 
1967     if (!wxStrcmp(SQLState
, wxT("22012"))) 
1968         return(DB_ERR_DIVIDE_BY_ZERO
); 
1969     if (!wxStrcmp(SQLState
, wxT("22026"))) 
1970         return(DB_ERR_STR_DATA_LENGTH_MISMATCH
); 
1971     if (!wxStrcmp(SQLState
, wxT("23000"))) 
1972         return(DB_ERR_INTEGRITY_CONSTRAINT_VIOL
); 
1973     if (!wxStrcmp(SQLState
, wxT("24000"))) 
1974         return(DB_ERR_INVALID_CURSOR_STATE
); 
1975     if (!wxStrcmp(SQLState
, wxT("25000"))) 
1976         return(DB_ERR_INVALID_TRANS_STATE
); 
1977     if (!wxStrcmp(SQLState
, wxT("28000"))) 
1978         return(DB_ERR_INVALID_AUTH_SPEC
); 
1979     if (!wxStrcmp(SQLState
, wxT("34000"))) 
1980         return(DB_ERR_INVALID_CURSOR_NAME
); 
1981     if (!wxStrcmp(SQLState
, wxT("37000"))) 
1982         return(DB_ERR_SYNTAX_ERROR_OR_ACCESS_VIOL
); 
1983     if (!wxStrcmp(SQLState
, wxT("3C000"))) 
1984         return(DB_ERR_DUPLICATE_CURSOR_NAME
); 
1985     if (!wxStrcmp(SQLState
, wxT("40001"))) 
1986         return(DB_ERR_SERIALIZATION_FAILURE
); 
1987     if (!wxStrcmp(SQLState
, wxT("42000"))) 
1988         return(DB_ERR_SYNTAX_ERROR_OR_ACCESS_VIOL2
); 
1989     if (!wxStrcmp(SQLState
, wxT("70100"))) 
1990         return(DB_ERR_OPERATION_ABORTED
); 
1991     if (!wxStrcmp(SQLState
, wxT("IM001"))) 
1992         return(DB_ERR_UNSUPPORTED_FUNCTION
); 
1993     if (!wxStrcmp(SQLState
, wxT("IM002"))) 
1994         return(DB_ERR_NO_DATA_SOURCE
); 
1995     if (!wxStrcmp(SQLState
, wxT("IM003"))) 
1996         return(DB_ERR_DRIVER_LOAD_ERROR
); 
1997     if (!wxStrcmp(SQLState
, wxT("IM004"))) 
1998         return(DB_ERR_SQLALLOCENV_FAILED
); 
1999     if (!wxStrcmp(SQLState
, wxT("IM005"))) 
2000         return(DB_ERR_SQLALLOCCONNECT_FAILED
); 
2001     if (!wxStrcmp(SQLState
, wxT("IM006"))) 
2002         return(DB_ERR_SQLSETCONNECTOPTION_FAILED
); 
2003     if (!wxStrcmp(SQLState
, wxT("IM007"))) 
2004         return(DB_ERR_NO_DATA_SOURCE_DLG_PROHIB
); 
2005     if (!wxStrcmp(SQLState
, wxT("IM008"))) 
2006         return(DB_ERR_DIALOG_FAILED
); 
2007     if (!wxStrcmp(SQLState
, wxT("IM009"))) 
2008         return(DB_ERR_UNABLE_TO_LOAD_TRANSLATION_DLL
); 
2009     if (!wxStrcmp(SQLState
, wxT("IM010"))) 
2010         return(DB_ERR_DATA_SOURCE_NAME_TOO_LONG
); 
2011     if (!wxStrcmp(SQLState
, wxT("IM011"))) 
2012         return(DB_ERR_DRIVER_NAME_TOO_LONG
); 
2013     if (!wxStrcmp(SQLState
, wxT("IM012"))) 
2014         return(DB_ERR_DRIVER_KEYWORD_SYNTAX_ERROR
); 
2015     if (!wxStrcmp(SQLState
, wxT("IM013"))) 
2016         return(DB_ERR_TRACE_FILE_ERROR
); 
2017     if (!wxStrcmp(SQLState
, wxT("S0001"))) 
2018         return(DB_ERR_TABLE_OR_VIEW_ALREADY_EXISTS
); 
2019     if (!wxStrcmp(SQLState
, wxT("S0002"))) 
2020         return(DB_ERR_TABLE_NOT_FOUND
); 
2021     if (!wxStrcmp(SQLState
, wxT("S0011"))) 
2022         return(DB_ERR_INDEX_ALREADY_EXISTS
); 
2023     if (!wxStrcmp(SQLState
, wxT("S0012"))) 
2024         return(DB_ERR_INDEX_NOT_FOUND
); 
2025     if (!wxStrcmp(SQLState
, wxT("S0021"))) 
2026         return(DB_ERR_COLUMN_ALREADY_EXISTS
); 
2027     if (!wxStrcmp(SQLState
, wxT("S0022"))) 
2028         return(DB_ERR_COLUMN_NOT_FOUND
); 
2029     if (!wxStrcmp(SQLState
, wxT("S0023"))) 
2030         return(DB_ERR_NO_DEFAULT_FOR_COLUMN
); 
2031     if (!wxStrcmp(SQLState
, wxT("S1000"))) 
2032         return(DB_ERR_GENERAL_ERROR
); 
2033     if (!wxStrcmp(SQLState
, wxT("S1001"))) 
2034         return(DB_ERR_MEMORY_ALLOCATION_FAILURE
); 
2035     if (!wxStrcmp(SQLState
, wxT("S1002"))) 
2036         return(DB_ERR_INVALID_COLUMN_NUMBER
); 
2037     if (!wxStrcmp(SQLState
, wxT("S1003"))) 
2038         return(DB_ERR_PROGRAM_TYPE_OUT_OF_RANGE
); 
2039     if (!wxStrcmp(SQLState
, wxT("S1004"))) 
2040         return(DB_ERR_SQL_DATA_TYPE_OUT_OF_RANGE
); 
2041     if (!wxStrcmp(SQLState
, wxT("S1008"))) 
2042         return(DB_ERR_OPERATION_CANCELLED
); 
2043     if (!wxStrcmp(SQLState
, wxT("S1009"))) 
2044         return(DB_ERR_INVALID_ARGUMENT_VALUE
); 
2045     if (!wxStrcmp(SQLState
, wxT("S1010"))) 
2046         return(DB_ERR_FUNCTION_SEQUENCE_ERROR
); 
2047     if (!wxStrcmp(SQLState
, wxT("S1011"))) 
2048         return(DB_ERR_OPERATION_INVALID_AT_THIS_TIME
); 
2049     if (!wxStrcmp(SQLState
, wxT("S1012"))) 
2050         return(DB_ERR_INVALID_TRANS_OPERATION_CODE
); 
2051     if (!wxStrcmp(SQLState
, wxT("S1015"))) 
2052         return(DB_ERR_NO_CURSOR_NAME_AVAIL
); 
2053     if (!wxStrcmp(SQLState
, wxT("S1090"))) 
2054         return(DB_ERR_INVALID_STR_OR_BUF_LEN
); 
2055     if (!wxStrcmp(SQLState
, wxT("S1091"))) 
2056         return(DB_ERR_DESCRIPTOR_TYPE_OUT_OF_RANGE
); 
2057     if (!wxStrcmp(SQLState
, wxT("S1092"))) 
2058         return(DB_ERR_OPTION_TYPE_OUT_OF_RANGE
); 
2059     if (!wxStrcmp(SQLState
, wxT("S1093"))) 
2060         return(DB_ERR_INVALID_PARAM_NO
); 
2061     if (!wxStrcmp(SQLState
, wxT("S1094"))) 
2062         return(DB_ERR_INVALID_SCALE_VALUE
); 
2063     if (!wxStrcmp(SQLState
, wxT("S1095"))) 
2064         return(DB_ERR_FUNCTION_TYPE_OUT_OF_RANGE
); 
2065     if (!wxStrcmp(SQLState
, wxT("S1096"))) 
2066         return(DB_ERR_INF_TYPE_OUT_OF_RANGE
); 
2067     if (!wxStrcmp(SQLState
, wxT("S1097"))) 
2068         return(DB_ERR_COLUMN_TYPE_OUT_OF_RANGE
); 
2069     if (!wxStrcmp(SQLState
, wxT("S1098"))) 
2070         return(DB_ERR_SCOPE_TYPE_OUT_OF_RANGE
); 
2071     if (!wxStrcmp(SQLState
, wxT("S1099"))) 
2072         return(DB_ERR_NULLABLE_TYPE_OUT_OF_RANGE
); 
2073     if (!wxStrcmp(SQLState
, wxT("S1100"))) 
2074         return(DB_ERR_UNIQUENESS_OPTION_TYPE_OUT_OF_RANGE
); 
2075     if (!wxStrcmp(SQLState
, wxT("S1101"))) 
2076         return(DB_ERR_ACCURACY_OPTION_TYPE_OUT_OF_RANGE
); 
2077     if (!wxStrcmp(SQLState
, wxT("S1103"))) 
2078         return(DB_ERR_DIRECTION_OPTION_OUT_OF_RANGE
); 
2079     if (!wxStrcmp(SQLState
, wxT("S1104"))) 
2080         return(DB_ERR_INVALID_PRECISION_VALUE
); 
2081     if (!wxStrcmp(SQLState
, wxT("S1105"))) 
2082         return(DB_ERR_INVALID_PARAM_TYPE
); 
2083     if (!wxStrcmp(SQLState
, wxT("S1106"))) 
2084         return(DB_ERR_FETCH_TYPE_OUT_OF_RANGE
); 
2085     if (!wxStrcmp(SQLState
, wxT("S1107"))) 
2086         return(DB_ERR_ROW_VALUE_OUT_OF_RANGE
); 
2087     if (!wxStrcmp(SQLState
, wxT("S1108"))) 
2088         return(DB_ERR_CONCURRENCY_OPTION_OUT_OF_RANGE
); 
2089     if (!wxStrcmp(SQLState
, wxT("S1109"))) 
2090         return(DB_ERR_INVALID_CURSOR_POSITION
); 
2091     if (!wxStrcmp(SQLState
, wxT("S1110"))) 
2092         return(DB_ERR_INVALID_DRIVER_COMPLETION
); 
2093     if (!wxStrcmp(SQLState
, wxT("S1111"))) 
2094         return(DB_ERR_INVALID_BOOKMARK_VALUE
); 
2095     if (!wxStrcmp(SQLState
, wxT("S1C00"))) 
2096         return(DB_ERR_DRIVER_NOT_CAPABLE
); 
2097     if (!wxStrcmp(SQLState
, wxT("S1T00"))) 
2098         return(DB_ERR_TIMEOUT_EXPIRED
); 
2103 }  // wxDb::TranslateSqlState() 
2106 /**********  wxDb::Grant() **********/ 
2107 bool wxDb::Grant(int privileges
, const wxString 
&tableName
, const wxString 
&userList
) 
2111     // Build the grant statement 
2112     sqlStmt  
= wxT("GRANT "); 
2113     if (privileges 
== DB_GRANT_ALL
) 
2114         sqlStmt 
+= wxT("ALL"); 
2118         if (privileges 
& DB_GRANT_SELECT
) 
2120             sqlStmt 
+= wxT("SELECT"); 
2123         if (privileges 
& DB_GRANT_INSERT
) 
2126                 sqlStmt 
+= wxT(", "); 
2127             sqlStmt 
+= wxT("INSERT"); 
2129         if (privileges 
& DB_GRANT_UPDATE
) 
2132                 sqlStmt 
+= wxT(", "); 
2133             sqlStmt 
+= wxT("UPDATE"); 
2135         if (privileges 
& DB_GRANT_DELETE
) 
2138                 sqlStmt 
+= wxT(", "); 
2139             sqlStmt 
+= wxT("DELETE"); 
2143     sqlStmt 
+= wxT(" ON "); 
2144     sqlStmt 
+= SQLTableName(tableName
); 
2145     sqlStmt 
+= wxT(" TO "); 
2146     sqlStmt 
+= userList
; 
2148 #ifdef DBDEBUG_CONSOLE 
2149     cout 
<< endl 
<< sqlStmt
.c_str() << endl
; 
2152     WriteSqlLog(sqlStmt
); 
2154     return(ExecSql(sqlStmt
)); 
2159 /********** wxDb::CreateView() **********/ 
2160 bool wxDb::CreateView(const wxString 
&viewName
, const wxString 
&colList
, 
2161                       const wxString 
&pSqlStmt
, bool attemptDrop
) 
2165     // Drop the view first 
2166     if (attemptDrop 
&& !DropView(viewName
)) 
2169     // Build the create view statement 
2170     sqlStmt  
= wxT("CREATE VIEW "); 
2171     sqlStmt 
+= viewName
; 
2173     if (colList
.Length()) 
2175         sqlStmt 
+= wxT(" ("); 
2177         sqlStmt 
+= wxT(")"); 
2180     sqlStmt 
+= wxT(" AS "); 
2181     sqlStmt 
+= pSqlStmt
; 
2183     WriteSqlLog(sqlStmt
); 
2185 #ifdef DBDEBUG_CONSOLE 
2186     cout 
<< sqlStmt
.c_str() << endl
; 
2189     return(ExecSql(sqlStmt
)); 
2191 }  // wxDb::CreateView() 
2194 /********** wxDb::DropView()  **********/ 
2195 bool wxDb::DropView(const wxString 
&viewName
) 
2198  * NOTE: This function returns true if the View does not exist, but 
2199  *       only for identified databases.  Code will need to be added 
2200  *            below for any other databases when those databases are defined 
2201  *       to handle this situation consistently 
2205     sqlStmt
.Printf(wxT("DROP VIEW %s"), viewName
.c_str()); 
2207     WriteSqlLog(sqlStmt
); 
2209 #ifdef DBDEBUG_CONSOLE 
2210     cout 
<< endl 
<< sqlStmt
.c_str() << endl
; 
2213     if (SQLExecDirect(hstmt
, (SQLTCHAR FAR 
*) sqlStmt
.c_str(), SQL_NTS
) != SQL_SUCCESS
) 
2215         // Check for "Base table not found" error and ignore 
2216         GetNextError(henv
, hdbc
, hstmt
); 
2217         if (wxStrcmp(sqlState
,wxT("S0002")))  // "Base table not found" 
2219             // Check for product specific error codes 
2220             if (!((Dbms() == dbmsSYBASE_ASA    
&& !wxStrcmp(sqlState
,wxT("42000")))))  // 5.x (and lower?) 
2223                 DispAllErrors(henv
, hdbc
, hstmt
); 
2230     // Commit the transaction 
2236 }  // wxDb::DropView() 
2239 /********** wxDb::ExecSql()  **********/ 
2240 bool wxDb::ExecSql(const wxString 
&pSqlStmt
) 
2244     SQLFreeStmt(hstmt
, SQL_CLOSE
); 
2246     retcode 
= SQLExecDirect(hstmt
, (SQLTCHAR FAR 
*) pSqlStmt
.c_str(), SQL_NTS
); 
2247     if (retcode 
== SQL_SUCCESS 
|| 
2248         (Dbms() == dbmsDB2 
&& (retcode 
== SQL_SUCCESS_WITH_INFO 
|| retcode 
== SQL_NO_DATA_FOUND
))) 
2254         DispAllErrors(henv
, hdbc
, hstmt
); 
2258 }  // wxDb::ExecSql() 
2261 /********** wxDb::ExecSql() with column info **********/ 
2262 bool wxDb::ExecSql(const wxString 
&pSqlStmt
, wxDbColInf
** columns
, short& numcols
) 
2264     //execute the statement first 
2265     if (!ExecSql(pSqlStmt
)) 
2269     if (SQLNumResultCols(hstmt
, &noCols
) != SQL_SUCCESS
) 
2271         DispAllErrors(henv
, hdbc
, hstmt
); 
2280     //  Get column information 
2282     wxChar name
[DB_MAX_COLUMN_NAME_LEN
+1]; 
2285     wxDbColInf
* pColInf 
= new wxDbColInf
[noCols
]; 
2287     // Fill in column information (name, datatype) 
2288     for (colNum 
= 0; colNum 
< noCols
; colNum
++) 
2290         if (SQLColAttributes(hstmt
, (UWORD
)(colNum
+1), SQL_COLUMN_NAME
, 
2292             &Sword
, &Sqllen
) != SQL_SUCCESS
) 
2294             DispAllErrors(henv
, hdbc
, hstmt
); 
2299         wxStrncpy(pColInf
[colNum
].colName
, name
, DB_MAX_COLUMN_NAME_LEN
); 
2300         pColInf
[colNum
].colName
[DB_MAX_COLUMN_NAME_LEN
] = 0;  // Prevent buffer overrun 
2302         if (SQLColAttributes(hstmt
, (UWORD
)(colNum
+1), SQL_COLUMN_TYPE
, 
2303             NULL
, 0, &Sword
, &Sqllen
) != SQL_SUCCESS
) 
2305             DispAllErrors(henv
, hdbc
, hstmt
); 
2313     #if defined(SQL_WCHAR) 
2316     #if defined(SQL_WVARCHAR) 
2322                 pColInf
[colNum
].dbDataType 
= DB_DATA_TYPE_VARCHAR
; 
2324             case SQL_LONGVARCHAR
: 
2325                 pColInf
[colNum
].dbDataType 
= DB_DATA_TYPE_MEMO
; 
2331                 pColInf
[colNum
].dbDataType 
= DB_DATA_TYPE_INTEGER
; 
2338                 pColInf
[colNum
].dbDataType 
= DB_DATA_TYPE_FLOAT
; 
2342                 pColInf
[colNum
].dbDataType 
= DB_DATA_TYPE_DATE
; 
2345                 pColInf
[colNum
].dbDataType 
= DB_DATA_TYPE_BLOB
; 
2350                 errMsg
.Printf(wxT("SQL Data type %ld currently not supported by wxWidgets"), (long)Sqllen
); 
2351                 wxLogDebug(errMsg
,wxT("ODBC DEBUG MESSAGE")); 
2358 }  // wxDb::ExecSql() 
2360 /********** wxDb::GetNext()  **********/ 
2361 bool wxDb::GetNext(void) 
2363     if (SQLFetch(hstmt
) == SQL_SUCCESS
) 
2367         DispAllErrors(henv
, hdbc
, hstmt
); 
2371 }  // wxDb::GetNext() 
2374 /********** wxDb::GetData()  **********/ 
2375 bool wxDb::GetData(UWORD colNo
, SWORD cType
, PTR pData
, SDWORD maxLen
, SQLLEN FAR 
*cbReturned
) 
2378     wxASSERT(cbReturned
); 
2380     long bufferSize 
= maxLen
; 
2382     if (cType 
== SQL_C_WXCHAR
) 
2383         bufferSize 
= maxLen 
* sizeof(wxChar
); 
2385     if (SQLGetData(hstmt
, colNo
, cType
, pData
, bufferSize
, cbReturned
) == SQL_SUCCESS
) 
2389         DispAllErrors(henv
, hdbc
, hstmt
); 
2393 }  // wxDb::GetData() 
2396 /********** wxDb::GetKeyFields() **********/ 
2397 int wxDb::GetKeyFields(const wxString 
&tableName
, wxDbColInf
* colInf
, UWORD noCols
) 
2399     wxChar       szPkTable
[DB_MAX_TABLE_NAME_LEN
+1];  /* Primary key table name */ 
2400     wxChar       szFkTable
[DB_MAX_TABLE_NAME_LEN
+1];  /* Foreign key table name */ 
2402     wxChar       szPkCol
[DB_MAX_COLUMN_NAME_LEN
+1];   /* Primary key column     */ 
2403     wxChar       szFkCol
[DB_MAX_COLUMN_NAME_LEN
+1];   /* Foreign key column     */ 
2409      * ----------------------------------------------------------------------- 
2410      * -- 19991224 : mj10777 : Create                                   ------ 
2411      * --          : Three things are done and stored here :            ------ 
2412      * --          : 1) which Column(s) is/are Primary Key(s)           ------ 
2413      * --          : 2) which tables use this Key as a Foreign Key      ------ 
2414      * --          : 3) which columns are Foreign Key and the name      ------ 
2415      * --          :     of the Table where the Key is the Primary Key  ----- 
2416      * --          : Called from GetColumns(const wxString &tableName,  ------ 
2417      * --                           int *numCols,const wxChar *userID ) ------ 
2418      * ----------------------------------------------------------------------- 
2421     /*---------------------------------------------------------------------*/ 
2422     /* Get the names of the columns in the primary key.                    */ 
2423     /*---------------------------------------------------------------------*/ 
2424     retcode 
= SQLPrimaryKeys(hstmt
, 
2425                              NULL
, 0,                               /* Catalog name  */ 
2426                              NULL
, 0,                               /* Schema name   */ 
2427                              (SQLTCHAR FAR 
*) tableName
.c_str(), SQL_NTS
); /* Table name    */ 
2429     /*---------------------------------------------------------------------*/ 
2430     /* Fetch and display the result set. This will be a list of the        */ 
2431     /* columns in the primary key of the tableName table.                  */ 
2432     /*---------------------------------------------------------------------*/ 
2433     while ((retcode 
== SQL_SUCCESS
) || (retcode 
== SQL_SUCCESS_WITH_INFO
)) 
2435         retcode 
= SQLFetch(hstmt
); 
2436         if (retcode 
== SQL_SUCCESS 
|| retcode 
== SQL_SUCCESS_WITH_INFO
) 
2438             GetData( 4, SQL_C_WXCHAR
,  szPkCol
,    DB_MAX_COLUMN_NAME_LEN
+1, &cb
); 
2439             GetData( 5, SQL_C_SSHORT
, &iKeySeq
,    0,                        &cb
); 
2441             for (i
=0;i
<noCols
;i
++)                          // Find the Column name 
2442                 if (!wxStrcmp(colInf
[i
].colName
,szPkCol
))   // We have found the Column 
2443                     colInf
[i
].PkCol 
= iKeySeq
;              // Which Primary Key is this (first, second usw.) ? 
2446     SQLFreeStmt(hstmt
, SQL_CLOSE
);  /* Close the cursor (the hstmt is still allocated).      */ 
2448     /*---------------------------------------------------------------------*/ 
2449     /* Get all the foreign keys that refer to tableName primary key.       */ 
2450     /*---------------------------------------------------------------------*/ 
2451     retcode 
= SQLForeignKeys(hstmt
, 
2452                              NULL
, 0,                            /* Primary catalog */ 
2453                              NULL
, 0,                            /* Primary schema  */ 
2454                              (SQLTCHAR FAR 
*)tableName
.c_str(), SQL_NTS
,/* Primary table   */ 
2455                              NULL
, 0,                            /* Foreign catalog */ 
2456                              NULL
, 0,                            /* Foreign schema  */ 
2457                              NULL
, 0);                           /* Foreign table   */ 
2459     /*---------------------------------------------------------------------*/ 
2460     /* Fetch and display the result set. This will be all of the foreign   */ 
2461     /* keys in other tables that refer to the tableName  primary key.      */ 
2462     /*---------------------------------------------------------------------*/ 
2465     while ((retcode 
== SQL_SUCCESS
) || (retcode 
== SQL_SUCCESS_WITH_INFO
)) 
2467         retcode 
= SQLFetch(hstmt
); 
2468         if (retcode 
== SQL_SUCCESS 
|| retcode 
== SQL_SUCCESS_WITH_INFO
) 
2470             GetData( 3, SQL_C_WXCHAR
,  szPkTable
,   DB_MAX_TABLE_NAME_LEN
+1,   &cb
); 
2471             GetData( 4, SQL_C_WXCHAR
,  szPkCol
,     DB_MAX_COLUMN_NAME_LEN
+1,  &cb
); 
2472             GetData( 5, SQL_C_SSHORT
, &iKeySeq
,     0,                         &cb
); 
2473             GetData( 7, SQL_C_WXCHAR
,  szFkTable
,   DB_MAX_TABLE_NAME_LEN
+1,   &cb
); 
2474             GetData( 8, SQL_C_WXCHAR
,  szFkCol
,     DB_MAX_COLUMN_NAME_LEN
+1,  &cb
); 
2475             tempStr 
<< _T('[') << szFkTable 
<< _T(']');  // [ ] in case there is a blank in the Table name 
2479     tempStr
.Trim();     // Get rid of any unneeded blanks 
2480     if (!tempStr
.empty()) 
2482         for (i
=0; i
<noCols
; i
++) 
2483         {   // Find the Column name 
2484             if (!wxStrcmp(colInf
[i
].colName
, szPkCol
))           // We have found the Column, store the Information 
2486                 wxStrncpy(colInf
[i
].PkTableName
, tempStr
.c_str(), DB_MAX_TABLE_NAME_LEN
);  // Name of the Tables where this Primary Key is used as a Foreign Key 
2487                 colInf
[i
].PkTableName
[DB_MAX_TABLE_NAME_LEN
] = 0;  // Prevent buffer overrun 
2492     SQLFreeStmt(hstmt
, SQL_CLOSE
);  /* Close the cursor (the hstmt is still allocated). */ 
2494     /*---------------------------------------------------------------------*/ 
2495     /* Get all the foreign keys in the tablename table.                    */ 
2496     /*---------------------------------------------------------------------*/ 
2497     retcode 
= SQLForeignKeys(hstmt
, 
2498                              NULL
, 0,                             /* Primary catalog   */ 
2499                              NULL
, 0,                             /* Primary schema    */ 
2500                              NULL
, 0,                             /* Primary table     */ 
2501                              NULL
, 0,                             /* Foreign catalog   */ 
2502                              NULL
, 0,                             /* Foreign schema    */ 
2503                              (SQLTCHAR 
*)tableName
.c_str(), SQL_NTS
);/* Foreign table     */ 
2505     /*---------------------------------------------------------------------*/ 
2506     /*  Fetch and display the result set. This will be all of the          */ 
2507     /*  primary keys in other tables that are referred to by foreign       */ 
2508     /*  keys in the tableName table.                                       */ 
2509     /*---------------------------------------------------------------------*/ 
2510     while ((retcode 
== SQL_SUCCESS
) || (retcode 
== SQL_SUCCESS_WITH_INFO
)) 
2512         retcode 
= SQLFetch(hstmt
); 
2513         if (retcode 
== SQL_SUCCESS 
|| retcode 
== SQL_SUCCESS_WITH_INFO
) 
2515             GetData( 3, SQL_C_WXCHAR
,  szPkTable
,   DB_MAX_TABLE_NAME_LEN
+1,  &cb
); 
2516             GetData( 5, SQL_C_SSHORT
, &iKeySeq
,     0,                        &cb
); 
2517             GetData( 8, SQL_C_WXCHAR
,  szFkCol
,     DB_MAX_COLUMN_NAME_LEN
+1, &cb
); 
2519             for (i
=0; i
<noCols
; i
++)                            // Find the Column name 
2521                 if (!wxStrcmp(colInf
[i
].colName
,szFkCol
))       // We have found the (Foreign Key) Column 
2523                     colInf
[i
].FkCol 
= iKeySeq
;                  // Which Foreign Key is this (first, second usw.) ? 
2524                     wxStrncpy(colInf
[i
].FkTableName
, szFkTable
, DB_MAX_TABLE_NAME_LEN
);  // Name of the Table where this Foriegn is the Primary Key 
2525                     colInf
[i
].FkTableName
[DB_MAX_TABLE_NAME_LEN
] = 0;  // Prevent buffer overrun 
2530     SQLFreeStmt(hstmt
, SQL_CLOSE
);  /* Close the cursor (the hstmt is still allocated). */ 
2534 }  // wxDb::GetKeyFields() 
2538 /********** wxDb::GetColumns() **********/ 
2539 wxDbColInf 
*wxDb::GetColumns(wxChar 
*tableName
[], const wxChar 
*userID
) 
2541  *        1) The last array element of the tableName[] argument must be zero (null). 
2542  *            This is how the end of the array is detected. 
2543  *        2) This function returns an array of wxDbColInf structures.  If no columns 
2544  *            were found, or an error occurred, this pointer will be zero (null).  THE 
2545  *            CALLING FUNCTION IS RESPONSIBLE FOR DELETING THE MEMORY RETURNED WHEN IT 
2546  *            IS FINISHED WITH IT.  i.e. 
2548  *            wxDbColInf *colInf = pDb->GetColumns(tableList, userID); 
2551  *                // Use the column inf 
2553  *                // Destroy the memory 
2557  * userID is evaluated in the following manner: 
2558  *        userID == NULL  ... UserID is ignored 
2559  *        userID == ""    ... UserID set equal to 'this->uid' 
2560  *        userID != ""    ... UserID set equal to 'userID' 
2562  * NOTE: ALL column bindings associated with this wxDb instance are unbound 
2563  *       by this function.  This function should use its own wxDb instance 
2564  *       to avoid undesired unbinding of columns. 
2569     wxDbColInf 
*colInf 
= 0; 
2577     convertUserID(userID
,UserID
); 
2579     // Pass 1 - Determine how many columns there are. 
2580     // Pass 2 - Allocate the wxDbColInf array and fill in 
2581     //                the array with the column information. 
2583     for (pass 
= 1; pass 
<= 2; pass
++) 
2587             if (noCols 
== 0)  // Probably a bogus table name(s) 
2589             // Allocate n wxDbColInf objects to hold the column information 
2590             colInf 
= new wxDbColInf
[noCols
+1]; 
2593             // Mark the end of the array 
2594             wxStrcpy(colInf
[noCols
].tableName
, wxEmptyString
); 
2595             wxStrcpy(colInf
[noCols
].colName
, wxEmptyString
); 
2596             colInf
[noCols
].sqlDataType 
= 0; 
2598         // Loop through each table name 
2600         for (tbl 
= 0; tableName
[tbl
]; tbl
++) 
2602             TableName 
= tableName
[tbl
]; 
2603             // Oracle and Interbase table names are uppercase only, so force 
2604             // the name to uppercase just in case programmer forgot to do this 
2605             if ((Dbms() == dbmsORACLE
) || 
2606                 (Dbms() == dbmsFIREBIRD
) || 
2607                 (Dbms() == dbmsINTERBASE
)) 
2608                 TableName 
= TableName
.Upper(); 
2610             SQLFreeStmt(hstmt
, SQL_CLOSE
); 
2612             // MySQL, SQLServer, and Access cannot accept a user name when looking up column names, so we 
2613             // use the call below that leaves out the user name 
2614             if (!UserID
.empty() && 
2615                 Dbms() != dbmsMY_SQL 
&& 
2616                 Dbms() != dbmsACCESS 
&& 
2617                 Dbms() != dbmsMS_SQL_SERVER
) 
2619                 retcode 
= SQLColumns(hstmt
, 
2620                                      NULL
, 0,                                // All qualifiers 
2621                                      (SQLTCHAR 
*) UserID
.c_str(), SQL_NTS
,      // Owner 
2622                                      (SQLTCHAR 
*) TableName
.c_str(), SQL_NTS
, 
2623                                      NULL
, 0);                               // All columns 
2627                 retcode 
= SQLColumns(hstmt
, 
2628                                      NULL
, 0,                                // All qualifiers 
2630                                      (SQLTCHAR 
*) TableName
.c_str(), SQL_NTS
, 
2631                                      NULL
, 0);                               // All columns 
2633             if (retcode 
!= SQL_SUCCESS
) 
2634             {  // Error occurred, abort 
2635                 DispAllErrors(henv
, hdbc
, hstmt
); 
2638                 SQLFreeStmt(hstmt
, SQL_CLOSE
); 
2642             while ((retcode 
= SQLFetch(hstmt
)) == SQL_SUCCESS
) 
2644                 if (pass 
== 1)  // First pass, just add up the number of columns 
2646                 else  // Pass 2; Fill in the array of structures 
2648                     if (colNo 
< noCols
)  // Some extra error checking to prevent memory overwrites 
2650                         // NOTE: Only the ODBC 1.x fields are retrieved 
2651                         GetData( 1, SQL_C_WXCHAR
, (UCHAR
*)  colInf
[colNo
].catalog
,      128+1,                    &cb
); 
2652                         GetData( 2, SQL_C_WXCHAR
, (UCHAR
*)  colInf
[colNo
].schema
,       128+1,                    &cb
); 
2653                         GetData( 3, SQL_C_WXCHAR
, (UCHAR
*)  colInf
[colNo
].tableName
,    DB_MAX_TABLE_NAME_LEN
+1,  &cb
); 
2654                         GetData( 4, SQL_C_WXCHAR
, (UCHAR
*)  colInf
[colNo
].colName
,      DB_MAX_COLUMN_NAME_LEN
+1, &cb
); 
2655                         GetData( 5, SQL_C_SSHORT
, (UCHAR
*) &colInf
[colNo
].sqlDataType
,  0,                        &cb
); 
2656                         GetData( 6, SQL_C_WXCHAR
, (UCHAR
*)  colInf
[colNo
].typeName
,     128+1,                    &cb
); 
2657                         GetData( 7, SQL_C_SLONG
,  (UCHAR
*) &colInf
[colNo
].columnLength
, 0,                        &cb
); 
2658                         GetData( 8, SQL_C_SSHORT
, (UCHAR
*) &colInf
[colNo
].bufferSize
,   0,                        &cb
); 
2659                         GetData( 9, SQL_C_SSHORT
, (UCHAR
*) &colInf
[colNo
].decimalDigits
,0,                        &cb
); 
2660                         GetData(10, SQL_C_SSHORT
, (UCHAR
*) &colInf
[colNo
].numPrecRadix
, 0,                        &cb
); 
2661                         GetData(11, SQL_C_SSHORT
, (UCHAR
*) &colInf
[colNo
].nullable
,     0,                        &cb
); 
2662                         GetData(12, SQL_C_WXCHAR
, (UCHAR
*)  colInf
[colNo
].remarks
,      254+1,                    &cb
); 
2664                         // Determine the wxDb data type that is used to represent the native data type of this data source 
2665                         colInf
[colNo
].dbDataType 
= 0; 
2666                         if (!wxStricmp(typeInfVarchar
.TypeName
,colInf
[colNo
].typeName
)) 
2669                             // IODBC does not return a correct columnLength, so we set 
2670                             // columnLength = bufferSize if no column length was returned 
2671                             // IODBC returns the columnLength in bufferSize. (bug) 
2672                             if (colInf
[colNo
].columnLength 
< 1) 
2674                                colInf
[colNo
].columnLength 
= colInf
[colNo
].bufferSize
; 
2677                             colInf
[colNo
].dbDataType 
= DB_DATA_TYPE_VARCHAR
; 
2679                         else if (!wxStricmp(typeInfInteger
.TypeName
, colInf
[colNo
].typeName
)) 
2680                             colInf
[colNo
].dbDataType 
= DB_DATA_TYPE_INTEGER
; 
2681                         else if (!wxStricmp(typeInfFloat
.TypeName
, colInf
[colNo
].typeName
)) 
2682                             colInf
[colNo
].dbDataType 
= DB_DATA_TYPE_FLOAT
; 
2683                         else if (!wxStricmp(typeInfDate
.TypeName
, colInf
[colNo
].typeName
)) 
2684                             colInf
[colNo
].dbDataType 
= DB_DATA_TYPE_DATE
; 
2685                         else if (!wxStricmp(typeInfBlob
.TypeName
, colInf
[colNo
].typeName
)) 
2686                             colInf
[colNo
].dbDataType 
= DB_DATA_TYPE_BLOB
; 
2691             if (retcode 
!= SQL_NO_DATA_FOUND
) 
2692             {  // Error occurred, abort 
2693                 DispAllErrors(henv
, hdbc
, hstmt
); 
2696                 SQLFreeStmt(hstmt
, SQL_CLOSE
); 
2702     SQLFreeStmt(hstmt
, SQL_CLOSE
); 
2705 }  // wxDb::GetColumns() 
2708 /********** wxDb::GetColumns() **********/ 
2710 wxDbColInf 
*wxDb::GetColumns(const wxString 
&tableName
, UWORD 
*numCols
, const wxChar 
*userID
) 
2712 // Same as the above GetColumns() function except this one gets columns 
2713 // only for a single table, and if 'numCols' is not NULL, the number of 
2714 // columns stored in the returned wxDbColInf is set in '*numCols' 
2716 // userID is evaluated in the following manner: 
2717 //        userID == NULL  ... UserID is ignored 
2718 //        userID == ""    ... UserID set equal to 'this->uid' 
2719 //        userID != ""    ... UserID set equal to 'userID' 
2721 // NOTE: ALL column bindings associated with this wxDb instance are unbound 
2722 //       by this function.  This function should use its own wxDb instance 
2723 //       to avoid undesired unbinding of columns. 
2728     wxDbColInf 
*colInf 
= 0; 
2736     convertUserID(userID
,UserID
); 
2738     // Pass 1 - Determine how many columns there are. 
2739     // Pass 2 - Allocate the wxDbColInf array and fill in 
2740     //                the array with the column information. 
2742     for (pass 
= 1; pass 
<= 2; pass
++) 
2746             if (noCols 
== 0)  // Probably a bogus table name(s) 
2748             // Allocate n wxDbColInf objects to hold the column information 
2749             colInf 
= new wxDbColInf
[noCols
+1]; 
2752             // Mark the end of the array 
2753             wxStrcpy(colInf
[noCols
].tableName
, wxEmptyString
); 
2754             wxStrcpy(colInf
[noCols
].colName
, wxEmptyString
); 
2755             colInf
[noCols
].sqlDataType 
= 0; 
2758         TableName 
= tableName
; 
2759         // Oracle and Interbase table names are uppercase only, so force 
2760         // the name to uppercase just in case programmer forgot to do this 
2761         if ((Dbms() == dbmsORACLE
) || 
2762             (Dbms() == dbmsFIREBIRD
) || 
2763             (Dbms() == dbmsINTERBASE
)) 
2764             TableName 
= TableName
.Upper(); 
2766         SQLFreeStmt(hstmt
, SQL_CLOSE
); 
2768         // MySQL, SQLServer, and Access cannot accept a user name when looking up column names, so we 
2769         // use the call below that leaves out the user name 
2770         if (!UserID
.empty() && 
2771             Dbms() != dbmsMY_SQL 
&& 
2772             Dbms() != dbmsACCESS 
&& 
2773             Dbms() != dbmsMS_SQL_SERVER
) 
2775             retcode 
= SQLColumns(hstmt
, 
2776                                  NULL
, 0,                                // All qualifiers 
2777                                  (SQLTCHAR 
*) UserID
.c_str(), SQL_NTS
,    // Owner 
2778                                  (SQLTCHAR 
*) TableName
.c_str(), SQL_NTS
, 
2779                                  NULL
, 0);                               // All columns 
2783             retcode 
= SQLColumns(hstmt
, 
2784                                  NULL
, 0,                                 // All qualifiers 
2786                                  (SQLTCHAR 
*) TableName
.c_str(), SQL_NTS
, 
2787                                  NULL
, 0);                                // All columns 
2789         if (retcode 
!= SQL_SUCCESS
) 
2790         {  // Error occurred, abort 
2791             DispAllErrors(henv
, hdbc
, hstmt
); 
2794             SQLFreeStmt(hstmt
, SQL_CLOSE
); 
2800         while ((retcode 
= SQLFetch(hstmt
)) == SQL_SUCCESS
) 
2802             if (pass 
== 1)  // First pass, just add up the number of columns 
2804             else  // Pass 2; Fill in the array of structures 
2806                 if (colNo 
< noCols
)  // Some extra error checking to prevent memory overwrites 
2808                     // NOTE: Only the ODBC 1.x fields are retrieved 
2809                     GetData( 1, SQL_C_WXCHAR
, (UCHAR
*)  colInf
[colNo
].catalog
,      128+1,                     &cb
); 
2810                     GetData( 2, SQL_C_WXCHAR
, (UCHAR
*)  colInf
[colNo
].schema
,       128+1,                     &cb
); 
2811                     GetData( 3, SQL_C_WXCHAR
, (UCHAR
*)  colInf
[colNo
].tableName
,    DB_MAX_TABLE_NAME_LEN
+1,   &cb
); 
2812                     GetData( 4, SQL_C_WXCHAR
, (UCHAR
*)  colInf
[colNo
].colName
,      DB_MAX_COLUMN_NAME_LEN
+1,  &cb
); 
2813                     GetData( 5, SQL_C_SSHORT
, (UCHAR
*) &colInf
[colNo
].sqlDataType
,  0,                         &cb
); 
2814                     GetData( 6, SQL_C_WXCHAR
, (UCHAR
*)  colInf
[colNo
].typeName
,     128+1,                     &cb
); 
2815                     GetData( 7, SQL_C_SLONG
,  (UCHAR
*) &colInf
[colNo
].columnLength
, 0,                         &cb
); 
2816                     // BJO 991214 : SQL_C_SSHORT instead of SQL_C_SLONG, otherwise fails on Sparc (probably all 64 bit architectures) 
2817                     GetData( 8, SQL_C_SSHORT
, (UCHAR
*) &colInf
[colNo
].bufferSize
,   0,                         &cb
); 
2818                     GetData( 9, SQL_C_SSHORT
, (UCHAR
*) &colInf
[colNo
].decimalDigits
,0,                         &cb
); 
2819                     GetData(10, SQL_C_SSHORT
, (UCHAR
*) &colInf
[colNo
].numPrecRadix
, 0,                         &cb
); 
2820                     GetData(11, SQL_C_SSHORT
, (UCHAR
*) &colInf
[colNo
].nullable
,     0,                         &cb
); 
2821                     GetData(12, SQL_C_WXCHAR
, (UCHAR
*)  colInf
[colNo
].remarks
,      254+1,                     &cb
); 
2822                     // Start Values for Primary/Foriegn Key (=No) 
2823                     colInf
[colNo
].PkCol 
= 0;           // Primary key column   0=No; 1= First Key, 2 = Second Key etc. 
2824                     colInf
[colNo
].PkTableName
[0] = 0;  // Tablenames where Primary Key is used as a Foreign Key 
2825                     colInf
[colNo
].FkCol 
= 0;           // Foreign key column   0=No; 1= First Key, 2 = Second Key etc. 
2826                     colInf
[colNo
].FkTableName
[0] = 0;  // Foreign key table name 
2828                     // BJO 20000428 : Virtuoso returns type names with upper cases! 
2829                     if (Dbms() == dbmsVIRTUOSO
) 
2831                         wxString s 
= colInf
[colNo
].typeName
; 
2833                         wxStrcmp(colInf
[colNo
].typeName
, s
.c_str()); 
2836                     // Determine the wxDb data type that is used to represent the native data type of this data source 
2837                     colInf
[colNo
].dbDataType 
= 0; 
2838                     if (!wxStricmp(typeInfVarchar
.TypeName
, colInf
[colNo
].typeName
)) 
2841                         // IODBC does not return a correct columnLength, so we set 
2842                         // columnLength = bufferSize if no column length was returned 
2843                         // IODBC returns the columnLength in bufferSize. (bug) 
2844                         if (colInf
[colNo
].columnLength 
< 1) 
2846                            colInf
[colNo
].columnLength 
= colInf
[colNo
].bufferSize
; 
2850                         colInf
[colNo
].dbDataType 
= DB_DATA_TYPE_VARCHAR
; 
2852                     else if (!wxStricmp(typeInfInteger
.TypeName
, colInf
[colNo
].typeName
)) 
2853                         colInf
[colNo
].dbDataType 
= DB_DATA_TYPE_INTEGER
; 
2854                     else if (!wxStricmp(typeInfFloat
.TypeName
, colInf
[colNo
].typeName
)) 
2855                         colInf
[colNo
].dbDataType 
= DB_DATA_TYPE_FLOAT
; 
2856                     else if (!wxStricmp(typeInfDate
.TypeName
, colInf
[colNo
].typeName
)) 
2857                         colInf
[colNo
].dbDataType 
= DB_DATA_TYPE_DATE
; 
2858                     else if (!wxStricmp(typeInfBlob
.TypeName
, colInf
[colNo
].typeName
)) 
2859                         colInf
[colNo
].dbDataType 
= DB_DATA_TYPE_BLOB
; 
2865         if (retcode 
!= SQL_NO_DATA_FOUND
) 
2866         {  // Error occurred, abort 
2867             DispAllErrors(henv
, hdbc
, hstmt
); 
2870             SQLFreeStmt(hstmt
, SQL_CLOSE
); 
2877     SQLFreeStmt(hstmt
, SQL_CLOSE
); 
2879     // Store Primary and Foriegn Keys 
2880     GetKeyFields(tableName
,colInf
,noCols
); 
2886 }  // wxDb::GetColumns() 
2889 #else  // New GetColumns 
2894     These are tentative new GetColumns members which should be more database 
2895     independent and which always returns the columns in the order they were 
2898     - The first one (wxDbColInf *wxDb::GetColumns(wxChar *tableName[], const 
2899       wxChar* userID)) calls the second implementation for each separate table 
2900       before merging the results. This makes the code easier to maintain as 
2901       only one member (the second) makes the real work 
2902     - wxDbColInf *wxDb::GetColumns(const wxString &tableName, int *numCols, const 
2903       wxChar *userID) is a little bit improved 
2904     - It doesn't anymore rely on the type-name to find out which database-type 
2906     - It ends by sorting the columns, so that they are returned in the same 
2907       order they were created 
2917 wxDbColInf 
*wxDb::GetColumns(wxChar 
*tableName
[], const wxChar 
*userID
) 
2920     // The last array element of the tableName[] argument must be zero (null). 
2921     // This is how the end of the array is detected. 
2925     // How many tables ? 
2927     for (tbl 
= 0 ; tableName
[tbl
]; tbl
++); 
2929     // Create a table to maintain the columns for each separate table 
2930     _TableColumns 
*TableColumns 
= new _TableColumns
[tbl
]; 
2933     for (i 
= 0 ; i 
< tbl 
; i
++) 
2936         TableColumns
[i
].colInf 
= GetColumns(tableName
[i
], &TableColumns
[i
].noCols
, userID
); 
2937         if (TableColumns
[i
].colInf 
== NULL
) 
2939         noCols 
+= TableColumns
[i
].noCols
; 
2942     // Now merge all the separate table infos 
2943     wxDbColInf 
*colInf 
= new wxDbColInf
[noCols
+1]; 
2945     // Mark the end of the array 
2946     wxStrcpy(colInf
[noCols
].tableName
, wxEmptyString
); 
2947     wxStrcpy(colInf
[noCols
].colName
, wxEmptyString
); 
2948     colInf
[noCols
].sqlDataType 
= 0; 
2953     for (i 
= 0 ; i 
< tbl 
; i
++) 
2955         for (j 
= 0 ; j 
< TableColumns
[i
].noCols 
; j
++) 
2957             colInf
[offset
++] = TableColumns
[i
].colInf
[j
]; 
2961     delete [] TableColumns
; 
2964 }  // wxDb::GetColumns()  -- NEW 
2967 wxDbColInf 
*wxDb::GetColumns(const wxString 
&tableName
, int *numCols
, const wxChar 
*userID
) 
2969 // Same as the above GetColumns() function except this one gets columns 
2970 // only for a single table, and if 'numCols' is not NULL, the number of 
2971 // columns stored in the returned wxDbColInf is set in '*numCols' 
2973 // userID is evaluated in the following manner: 
2974 //        userID == NULL  ... UserID is ignored 
2975 //        userID == ""    ... UserID set equal to 'this->uid' 
2976 //        userID != ""    ... UserID set equal to 'userID' 
2978 // NOTE: ALL column bindings associated with this wxDb instance are unbound 
2979 //       by this function.  This function should use its own wxDb instance 
2980 //       to avoid undesired unbinding of columns. 
2984     wxDbColInf 
*colInf 
= 0; 
2992     convertUserID(userID
,UserID
); 
2994     // Pass 1 - Determine how many columns there are. 
2995     // Pass 2 - Allocate the wxDbColInf array and fill in 
2996     //                the array with the column information. 
2998     for (pass 
= 1; pass 
<= 2; pass
++) 
3002             if (noCols 
== 0)  // Probably a bogus table name(s) 
3004             // Allocate n wxDbColInf objects to hold the column information 
3005             colInf 
= new wxDbColInf
[noCols
+1]; 
3008             // Mark the end of the array 
3009             wxStrcpy(colInf
[noCols
].tableName
, wxEmptyString
); 
3010             wxStrcpy(colInf
[noCols
].colName
, wxEmptyString
); 
3011             colInf
[noCols
].sqlDataType 
= 0; 
3014         TableName 
= tableName
; 
3015         // Oracle and Interbase table names are uppercase only, so force 
3016         // the name to uppercase just in case programmer forgot to do this 
3017         if ((Dbms() == dbmsORACLE
) || 
3018             (Dbms() == dbmsFIREBIRD
) || 
3019             (Dbms() == dbmsINTERBASE
)) 
3020             TableName 
= TableName
.Upper(); 
3022         SQLFreeStmt(hstmt
, SQL_CLOSE
); 
3024         // MySQL, SQLServer, and Access cannot accept a user name when looking up column names, so we 
3025         // use the call below that leaves out the user name 
3026         if (!UserID
.empty() && 
3027             Dbms() != dbmsMY_SQL 
&& 
3028             Dbms() != dbmsACCESS 
&& 
3029             Dbms() != dbmsMS_SQL_SERVER
) 
3031             retcode 
= SQLColumns(hstmt
, 
3032                                  NULL
, 0,                              // All qualifiers 
3033                                  (UCHAR 
*) UserID
.c_str(), SQL_NTS
,    // Owner 
3034                                  (UCHAR 
*) TableName
.c_str(), SQL_NTS
, 
3035                                  NULL
, 0);                             // All columns 
3039             retcode 
= SQLColumns(hstmt
, 
3040                                  NULL
, 0,                              // All qualifiers 
3042                                  (UCHAR 
*) TableName
.c_str(), SQL_NTS
, 
3043                                  NULL
, 0);                             // All columns 
3045         if (retcode 
!= SQL_SUCCESS
) 
3046         {  // Error occurred, abort 
3047             DispAllErrors(henv
, hdbc
, hstmt
); 
3050             SQLFreeStmt(hstmt
, SQL_CLOSE
); 
3056         while ((retcode 
= SQLFetch(hstmt
)) == SQL_SUCCESS
) 
3058             if (pass 
== 1)  // First pass, just add up the number of columns 
3060             else  // Pass 2; Fill in the array of structures 
3062                 if (colNo 
< noCols
)  // Some extra error checking to prevent memory overwrites 
3064                     // NOTE: Only the ODBC 1.x fields are retrieved 
3065                     GetData( 1, SQL_C_WXCHAR
, (UCHAR
*)  colInf
[colNo
].catalog
,      128+1,                     &cb
); 
3066                     GetData( 2, SQL_C_WXCHAR
, (UCHAR
*)  colInf
[colNo
].schema
,       128+1,                     &cb
); 
3067                     GetData( 3, SQL_C_WXCHAR
, (UCHAR
*)  colInf
[colNo
].tableName
,    DB_MAX_TABLE_NAME_LEN
+1,   &cb
); 
3068                     GetData( 4, SQL_C_WXCHAR
, (UCHAR
*)  colInf
[colNo
].colName
,      DB_MAX_COLUMN_NAME_LEN
+1,  &cb
); 
3069                     GetData( 5, SQL_C_SSHORT
, (UCHAR
*) &colInf
[colNo
].sqlDataType
,  0,                         &cb
); 
3070                     GetData( 6, SQL_C_WXCHAR
, (UCHAR
*)  colInf
[colNo
].typeName
,     128+1,                     &cb
); 
3071                     GetData( 7, SQL_C_SLONG
,  (UCHAR
*) &colInf
[colNo
].columnLength
, 0,                         &cb
); 
3072                     GetData( 8, SQL_C_SSHORT
, (UCHAR
*) &colInf
[colNo
].bufferSize
,   0,                         &cb
); 
3073                     GetData( 9, SQL_C_SSHORT
, (UCHAR
*) &colInf
[colNo
].decimalDigits
,0,                         &cb
); 
3074                     GetData(10, SQL_C_SSHORT
, (UCHAR
*) &colInf
[colNo
].numPrecRadix
, 0,                         &cb
); 
3075                     GetData(11, SQL_C_SSHORT
, (UCHAR
*) &colInf
[colNo
].nullable
,     0,                         &cb
); 
3076                     GetData(12, SQL_C_WXCHAR
, (UCHAR
*)  colInf
[colNo
].remarks
,      254+1,                     &cb
); 
3077                     // Start Values for Primary/Foriegn Key (=No) 
3078                     colInf
[colNo
].PkCol 
= 0;           // Primary key column   0=No; 1= First Key, 2 = Second Key etc. 
3079                     colInf
[colNo
].PkTableName
[0] = 0;  // Tablenames where Primary Key is used as a Foreign Key 
3080                     colInf
[colNo
].FkCol 
= 0;           // Foreign key column   0=No; 1= First Key, 2 = Second Key etc. 
3081                     colInf
[colNo
].FkTableName
[0] = 0;  // Foreign key table name 
3084                     // IODBC does not return a correct columnLength, so we set 
3085                     // columnLength = bufferSize if no column length was returned 
3086                     // IODBC returns the columnLength in bufferSize. (bug) 
3087                     if (colInf
[colNo
].columnLength 
< 1) 
3089                        colInf
[colNo
].columnLength 
= colInf
[colNo
].bufferSize
; 
3093                     // Determine the wxDb data type that is used to represent the native data type of this data source 
3094                     colInf
[colNo
].dbDataType 
= 0; 
3095                     // Get the intern datatype 
3096                     switch (colInf
[colNo
].sqlDataType
) 
3099     #if defined(SQL_WCHAR) 
3102     #if defined(SQL_WVARCHAR) 
3108                             colInf
[colNo
].dbDataType 
= DB_DATA_TYPE_VARCHAR
; 
3110                         case SQL_LONGVARCHAR
:                         
3111                             colInf
[colNo
].dbDataType 
= DB_DATA_TYPE_MEMO
; 
3117                             colInf
[colNo
].dbDataType 
= DB_DATA_TYPE_INTEGER
; 
3124                             colInf
[colNo
].dbDataType 
= DB_DATA_TYPE_FLOAT
; 
3128                             colInf
[colNo
].dbDataType 
= DB_DATA_TYPE_DATE
; 
3131                             colInf
[colNo
].dbDataType 
= DB_DATA_TYPE_BLOB
; 
3136                             errMsg
.Printf(wxT("SQL Data type %d currently not supported by wxWidgets"), colInf
[colNo
].sqlDataType
); 
3137                             wxLogDebug(errMsg
,wxT("ODBC DEBUG MESSAGE")); 
3144         if (retcode 
!= SQL_NO_DATA_FOUND
) 
3145         {  // Error occurred, abort 
3146             DispAllErrors(henv
, hdbc
, hstmt
); 
3149             SQLFreeStmt(hstmt
, SQL_CLOSE
); 
3156     SQLFreeStmt(hstmt
, SQL_CLOSE
); 
3158     // Store Primary and Foreign Keys 
3159     GetKeyFields(tableName
,colInf
,noCols
); 
3161     /////////////////////////////////////////////////////////////////////////// 
3162     // Now sort the the columns in order to make them appear in the right order 
3163     /////////////////////////////////////////////////////////////////////////// 
3165     // Build a generic SELECT statement which returns 0 rows 
3168     Stmt
.Printf(wxT("select * from \"%s\" where 0=1"), tableName
); 
3171     if (SQLExecDirect(hstmt
, (UCHAR FAR 
*) Stmt
.c_str(), SQL_NTS
) != SQL_SUCCESS
) 
3173         DispAllErrors(henv
, hdbc
, hstmt
); 
3177     // Get the number of result columns 
3178     if (SQLNumResultCols (hstmt
, &noCols
) != SQL_SUCCESS
) 
3180         DispAllErrors(henv
, hdbc
, hstmt
); 
3184     if (noCols 
== 0) // Probably a bogus table name 
3193     for (colNum 
= 0; colNum 
< noCols
; colNum
++) 
3195         if (SQLColAttributes(hstmt
,colNum
+1, SQL_COLUMN_NAME
, 
3197             &Sword
, &Sdword
) != SQL_SUCCESS
) 
3199             DispAllErrors(henv
, hdbc
, hstmt
); 
3203         wxString Name1 
= name
; 
3204         Name1 
= Name1
.Upper(); 
3206         // Where is this name in the array ? 
3207         for (i 
= colNum 
; i 
< noCols 
; i
++) 
3209             wxString Name2 
=  colInf
[i
].colName
; 
3210             Name2 
= Name2
.Upper(); 
3213                 if (colNum 
!= i
) // swap to sort 
3215                     wxDbColInf tmpColInf 
= colInf
[colNum
]; 
3216                     colInf
[colNum
] =  colInf
[i
]; 
3217                     colInf
[i
] = tmpColInf
; 
3223     SQLFreeStmt(hstmt
, SQL_CLOSE
); 
3225     /////////////////////////////////////////////////////////////////////////// 
3227     /////////////////////////////////////////////////////////////////////////// 
3233 }  // wxDb::GetColumns() 
3236 #endif  // #else OLD_GETCOLUMNS 
3239 /********** wxDb::GetColumnCount() **********/ 
3240 int wxDb::GetColumnCount(const wxString 
&tableName
, const wxChar 
*userID
) 
3242  * Returns a count of how many columns are in a table. 
3243  * If an error occurs in computing the number of columns 
3244  * this function will return a -1 for the count 
3246  * userID is evaluated in the following manner: 
3247  *        userID == NULL  ... UserID is ignored 
3248  *        userID == ""    ... UserID set equal to 'this->uid' 
3249  *        userID != ""    ... UserID set equal to 'userID' 
3251  * NOTE: ALL column bindings associated with this wxDb instance are unbound 
3252  *       by this function.  This function should use its own wxDb instance 
3253  *       to avoid undesired unbinding of columns. 
3263     convertUserID(userID
,UserID
); 
3265     TableName 
= tableName
; 
3266     // Oracle and Interbase table names are uppercase only, so force 
3267     // the name to uppercase just in case programmer forgot to do this 
3268     if ((Dbms() == dbmsORACLE
) || 
3269         (Dbms() == dbmsFIREBIRD
) || 
3270         (Dbms() == dbmsINTERBASE
)) 
3271         TableName 
= TableName
.Upper(); 
3273     SQLFreeStmt(hstmt
, SQL_CLOSE
); 
3275     // MySQL, SQLServer, and Access cannot accept a user name when looking up column names, so we 
3276     // use the call below that leaves out the user name 
3277     if (!UserID
.empty() && 
3278         Dbms() != dbmsMY_SQL 
&& 
3279         Dbms() != dbmsACCESS 
&& 
3280         Dbms() != dbmsMS_SQL_SERVER
) 
3282         retcode 
= SQLColumns(hstmt
, 
3283                              NULL
, 0,                                // All qualifiers 
3284                              (SQLTCHAR 
*) UserID
.c_str(), SQL_NTS
,      // Owner 
3285                              (SQLTCHAR 
*) TableName
.c_str(), SQL_NTS
, 
3286                              NULL
, 0);                               // All columns 
3290         retcode 
= SQLColumns(hstmt
, 
3291                              NULL
, 0,                                // All qualifiers 
3293                              (SQLTCHAR 
*) TableName
.c_str(), SQL_NTS
, 
3294                              NULL
, 0);                               // All columns 
3296     if (retcode 
!= SQL_SUCCESS
) 
3297     {  // Error occurred, abort 
3298         DispAllErrors(henv
, hdbc
, hstmt
); 
3299         SQLFreeStmt(hstmt
, SQL_CLOSE
); 
3303     // Count the columns 
3304     while ((retcode 
= SQLFetch(hstmt
)) == SQL_SUCCESS
) 
3307     if (retcode 
!= SQL_NO_DATA_FOUND
) 
3308     {  // Error occurred, abort 
3309         DispAllErrors(henv
, hdbc
, hstmt
); 
3310         SQLFreeStmt(hstmt
, SQL_CLOSE
); 
3314     SQLFreeStmt(hstmt
, SQL_CLOSE
); 
3317 }  // wxDb::GetColumnCount() 
3320 /********** wxDb::GetCatalog() *******/ 
3321 wxDbInf 
*wxDb::GetCatalog(const wxChar 
*userID
) 
3323  * --------------------------------------------------------------------- 
3324  * -- 19991203 : mj10777 : Create                                 ------ 
3325  * --          : Creates a wxDbInf with Tables / Cols Array       ------ 
3326  * --          : uses SQLTables and fills pTableInf;              ------ 
3327  * --          : pColInf is set to NULL and numCols to 0;         ------ 
3328  * --          : returns pDbInf (wxDbInf)                         ------ 
3329  * --            - if unsuccessful (pDbInf == NULL)               ------ 
3330  * --          : pColInf can be filled with GetColumns(..);       ------ 
3331  * --          : numCols   can be filled with GetColumnCount(..); ------ 
3332  * --------------------------------------------------------------------- 
3334  * userID is evaluated in the following manner: 
3335  *        userID == NULL  ... UserID is ignored 
3336  *        userID == ""    ... UserID set equal to 'this->uid' 
3337  *        userID != ""    ... UserID set equal to 'userID' 
3339  * NOTE: ALL column bindings associated with this wxDb instance are unbound 
3340  *       by this function.  This function should use its own wxDb instance 
3341  *       to avoid undesired unbinding of columns. 
3344     int      noTab 
= 0;     // Counter while filling table entries 
3348     wxString tblNameSave
; 
3351     convertUserID(userID
,UserID
); 
3353     //------------------------------------------------------------- 
3354     // Create the Database Array of catalog entries 
3356     wxDbInf 
*pDbInf 
= new wxDbInf
; 
3358     //------------------------------------------------------------- 
3359     // Table Information 
3360     // Pass 1 - Determine how many Tables there are. 
3361     // Pass 2 - Create the Table array and fill it 
3362     //        - Create the Cols array = NULL 
3363     //------------------------------------------------------------- 
3365     for (pass 
= 1; pass 
<= 2; pass
++) 
3367         SQLFreeStmt(hstmt
, SQL_CLOSE
);   // Close if Open 
3368         tblNameSave
.Empty(); 
3370         if (!UserID
.empty() && 
3371             Dbms() != dbmsMY_SQL 
&& 
3372             Dbms() != dbmsACCESS 
&& 
3373             Dbms() != dbmsMS_SQL_SERVER
) 
3375             retcode 
= SQLTables(hstmt
, 
3376                                 NULL
, 0,                             // All qualifiers 
3377                                 (SQLTCHAR 
*) UserID
.c_str(), SQL_NTS
,   // User specified 
3378                                 NULL
, 0,                             // All tables 
3379                                 NULL
, 0);                            // All columns 
3383             retcode 
= SQLTables(hstmt
, 
3384                                 NULL
, 0,           // All qualifiers 
3385                                 NULL
, 0,           // User specified 
3386                                 NULL
, 0,           // All tables 
3387                                 NULL
, 0);          // All columns 
3390         if (retcode 
!= SQL_SUCCESS
) 
3392             DispAllErrors(henv
, hdbc
, hstmt
); 
3394             SQLFreeStmt(hstmt
, SQL_CLOSE
); 
3398         while ((retcode 
= SQLFetch(hstmt
)) == SQL_SUCCESS
)   // Table Information 
3400             if (pass 
== 1)  // First pass, just count the Tables 
3402                 if (pDbInf
->numTables 
== 0) 
3404                     GetData( 1, SQL_C_WXCHAR
,   (UCHAR
*)  pDbInf
->catalog
,  128+1, &cb
); 
3405                     GetData( 2, SQL_C_WXCHAR
,   (UCHAR
*)  pDbInf
->schema
,   128+1, &cb
); 
3407                  pDbInf
->numTables
++;      // Counter for Tables 
3409             if (pass 
== 2) // Create and fill the Table entries 
3411                 if (pDbInf
->pTableInf 
== NULL
)   // Has the Table Array been created 
3412                 {  // no, then create the Array 
3413                     pDbInf
->pTableInf 
= new wxDbTableInf
[pDbInf
->numTables
]; 
3415                 } // if (pDbInf->pTableInf == NULL)   // Has the Table Array been created 
3417                 GetData( 3, SQL_C_WXCHAR
,   (UCHAR
*)  (pDbInf
->pTableInf
+noTab
)->tableName
,    DB_MAX_TABLE_NAME_LEN
+1, &cb
); 
3418                 GetData( 4, SQL_C_WXCHAR
,   (UCHAR
*)  (pDbInf
->pTableInf
+noTab
)->tableType
,    30+1,                    &cb
); 
3419                 GetData( 5, SQL_C_WXCHAR
,   (UCHAR
*)  (pDbInf
->pTableInf
+noTab
)->tableRemarks
, 254+1,                   &cb
); 
3425     SQLFreeStmt(hstmt
, SQL_CLOSE
); 
3427     // Query how many columns are in each table 
3428     for (noTab
=0;noTab
<pDbInf
->numTables
;noTab
++) 
3430         (pDbInf
->pTableInf
+noTab
)->numCols 
= (UWORD
)GetColumnCount((pDbInf
->pTableInf
+noTab
)->tableName
,UserID
); 
3435 }  // wxDb::GetCatalog() 
3438 /********** wxDb::Catalog() **********/ 
3439 bool wxDb::Catalog(const wxChar 
*userID
, const wxString 
&fileName
) 
3441  * Creates the text file specified in 'filename' which will contain 
3442  * a minimal data dictionary of all tables accessible by the user specified 
3445  * userID is evaluated in the following manner: 
3446  *        userID == NULL  ... UserID is ignored 
3447  *        userID == ""    ... UserID set equal to 'this->uid' 
3448  *        userID != ""    ... UserID set equal to 'userID' 
3450  * NOTE: ALL column bindings associated with this wxDb instance are unbound 
3451  *       by this function.  This function should use its own wxDb instance 
3452  *       to avoid undesired unbinding of columns. 
3455     wxASSERT(fileName
.Length()); 
3459     wxChar    tblName
[DB_MAX_TABLE_NAME_LEN
+1]; 
3460     wxString  tblNameSave
; 
3461     wxChar    colName
[DB_MAX_COLUMN_NAME_LEN
+1]; 
3463     wxChar    typeName
[30+1]; 
3464     SDWORD    precision
, length
; 
3466     FILE *fp 
= wxFopen(fileName
.c_str(),wxT("wt")); 
3470     SQLFreeStmt(hstmt
, SQL_CLOSE
); 
3473     convertUserID(userID
,UserID
); 
3475     if (!UserID
.empty() && 
3476         Dbms() != dbmsMY_SQL 
&& 
3477         Dbms() != dbmsACCESS 
&& 
3478         Dbms() != dbmsFIREBIRD 
&& 
3479         Dbms() != dbmsINTERBASE 
&& 
3480         Dbms() != dbmsMS_SQL_SERVER
) 
3482         retcode 
= SQLColumns(hstmt
, 
3483                              NULL
, 0,                                // All qualifiers 
3484                              (SQLTCHAR 
*) UserID
.c_str(), SQL_NTS
,      // User specified 
3485                              NULL
, 0,                                // All tables 
3486                              NULL
, 0);                               // All columns 
3490         retcode 
= SQLColumns(hstmt
, 
3491                              NULL
, 0,    // All qualifiers 
3492                              NULL
, 0,    // User specified 
3493                              NULL
, 0,    // All tables 
3494                              NULL
, 0);   // All columns 
3496     if (retcode 
!= SQL_SUCCESS
) 
3498         DispAllErrors(henv
, hdbc
, hstmt
); 
3504     tblNameSave
.Empty(); 
3509         retcode 
= SQLFetch(hstmt
); 
3510         if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO
) 
3513         GetData(3,SQL_C_WXCHAR
,  (UCHAR 
*) tblName
,     DB_MAX_TABLE_NAME_LEN
+1, &cb
); 
3514         GetData(4,SQL_C_WXCHAR
,  (UCHAR 
*) colName
,     DB_MAX_COLUMN_NAME_LEN
+1,&cb
); 
3515         GetData(5,SQL_C_SSHORT
,  (UCHAR 
*)&sqlDataType
, 0,                       &cb
); 
3516         GetData(6,SQL_C_WXCHAR
,  (UCHAR 
*) typeName
,    sizeof(typeName
),        &cb
); 
3517         GetData(7,SQL_C_SLONG
,   (UCHAR 
*)&precision
,   0,                       &cb
); 
3518         GetData(8,SQL_C_SLONG
,   (UCHAR 
*)&length
,      0,                       &cb
); 
3520         if (wxStrcmp(tblName
, tblNameSave
.c_str())) 
3523                 wxFputs(wxT("\n"), fp
); 
3524             wxFputs(wxT("================================ "), fp
); 
3525             wxFputs(wxT("================================ "), fp
); 
3526             wxFputs(wxT("===================== "), fp
); 
3527             wxFputs(wxT("========= "), fp
); 
3528             wxFputs(wxT("=========\n"), fp
); 
3529             outStr
.Printf(wxT("%-32s %-32s %-21s %9s %9s\n"), 
3530                 wxT("TABLE NAME"), wxT("COLUMN NAME"), wxT("DATA TYPE"), wxT("PRECISION"), wxT("LENGTH")); 
3531             wxFputs(outStr
.c_str(), fp
); 
3532             wxFputs(wxT("================================ "), fp
); 
3533             wxFputs(wxT("================================ "), fp
); 
3534             wxFputs(wxT("===================== "), fp
); 
3535             wxFputs(wxT("========= "), fp
); 
3536             wxFputs(wxT("=========\n"), fp
); 
3537             tblNameSave 
= tblName
; 
3540         outStr
.Printf(wxT("%-32s %-32s (%04d)%-15s %9ld %9ld\n"), 
3541             tblName
, colName
, sqlDataType
, typeName
, precision
, length
); 
3542         if (wxFputs(outStr
.c_str(), fp
) == EOF
) 
3544             SQLFreeStmt(hstmt
, SQL_CLOSE
); 
3551     if (retcode 
!= SQL_NO_DATA_FOUND
) 
3552         DispAllErrors(henv
, hdbc
, hstmt
); 
3554     SQLFreeStmt(hstmt
, SQL_CLOSE
); 
3557     return(retcode 
== SQL_NO_DATA_FOUND
); 
3559 }  // wxDb::Catalog() 
3562 bool wxDb::TableExists(const wxString 
&tableName
, const wxChar 
*userID
, const wxString 
&tablePath
) 
3564  * Table name can refer to a table, view, alias or synonym.  Returns true 
3565  * if the object exists in the database.  This function does not indicate 
3566  * whether or not the user has privleges to query or perform other functions 
3569  * userID is evaluated in the following manner: 
3570  *        userID == NULL  ... UserID is ignored 
3571  *        userID == ""    ... UserID set equal to 'this->uid' 
3572  *        userID != ""    ... UserID set equal to 'userID' 
3575     wxASSERT(tableName
.Length()); 
3579     if (Dbms() == dbmsDBASE
) 
3582         if (tablePath
.Length()) 
3583             dbName
.Printf(wxT("%s/%s.dbf"), tablePath
.c_str(), tableName
.c_str()); 
3585             dbName
.Printf(wxT("%s.dbf"), tableName
.c_str()); 
3588         exists 
= wxFileExists(dbName
); 
3593     convertUserID(userID
,UserID
); 
3595     TableName 
= tableName
; 
3596     // Oracle and Interbase table names are uppercase only, so force 
3597     // the name to uppercase just in case programmer forgot to do this 
3598     if ((Dbms() == dbmsORACLE
) || 
3599         (Dbms() == dbmsFIREBIRD
) || 
3600         (Dbms() == dbmsINTERBASE
)) 
3601         TableName 
= TableName
.Upper(); 
3603     SQLFreeStmt(hstmt
, SQL_CLOSE
); 
3606     // Some databases cannot accept a user name when looking up table names, 
3607     // so we use the call below that leaves out the user name 
3608     if (!UserID
.empty() && 
3609         Dbms() != dbmsMY_SQL 
&& 
3610         Dbms() != dbmsACCESS 
&& 
3611         Dbms() != dbmsMS_SQL_SERVER 
&& 
3612         Dbms() != dbmsDB2 
&& 
3613         Dbms() != dbmsFIREBIRD 
&& 
3614         Dbms() != dbmsINTERBASE 
&& 
3615         Dbms() != dbmsPERVASIVE_SQL
) 
3617         retcode 
= SQLTables(hstmt
, 
3618                             NULL
, 0,                                  // All qualifiers 
3619                             (SQLTCHAR 
*) UserID
.c_str(), SQL_NTS
,        // Only tables owned by this user 
3620                             (SQLTCHAR FAR 
*)TableName
.c_str(), SQL_NTS
, 
3621                             NULL
, 0);                                 // All table types 
3625         retcode 
= SQLTables(hstmt
, 
3626                             NULL
, 0,                                  // All qualifiers 
3627                             NULL
, 0,                                  // All owners 
3628                             (SQLTCHAR FAR 
*)TableName
.c_str(), SQL_NTS
, 
3629                             NULL
, 0);                                 // All table types 
3631     if (retcode 
!= SQL_SUCCESS
) 
3632         return(DispAllErrors(henv
, hdbc
, hstmt
)); 
3634     retcode 
= SQLFetch(hstmt
); 
3635     if (retcode 
!= SQL_SUCCESS 
&& retcode 
!= SQL_SUCCESS_WITH_INFO
) 
3637         SQLFreeStmt(hstmt
, SQL_CLOSE
); 
3638         return(DispAllErrors(henv
, hdbc
, hstmt
)); 
3641     SQLFreeStmt(hstmt
, SQL_CLOSE
); 
3645 }  // wxDb::TableExists() 
3648 /********** wxDb::TablePrivileges() **********/ 
3649 bool wxDb::TablePrivileges(const wxString 
&tableName
, const wxString 
&priv
, const wxChar 
*userID
, 
3650                             const wxChar 
*schema
, const wxString 
&WXUNUSED(tablePath
)) 
3652     wxASSERT(tableName
.Length()); 
3654     wxDbTablePrivilegeInfo  result
; 
3658     // We probably need to be able to dynamically set this based on 
3659     // the driver type, and state. 
3660     wxChar curRole
[]=wxT("public"); 
3664     wxString UserID
,Schema
; 
3665     convertUserID(userID
,UserID
); 
3666     convertUserID(schema
,Schema
); 
3668     TableName 
= tableName
; 
3669     // Oracle and Interbase table names are uppercase only, so force 
3670     // the name to uppercase just in case programmer forgot to do this 
3671     if ((Dbms() == dbmsORACLE
) || 
3672         (Dbms() == dbmsFIREBIRD
) || 
3673         (Dbms() == dbmsINTERBASE
)) 
3674         TableName 
= TableName
.Upper(); 
3676     SQLFreeStmt(hstmt
, SQL_CLOSE
); 
3678     // Some databases cannot accept a user name when looking up table names, 
3679     // so we use the call below that leaves out the user name 
3680     if (!Schema
.empty() && 
3681         Dbms() != dbmsMY_SQL 
&& 
3682         Dbms() != dbmsACCESS 
&& 
3683         Dbms() != dbmsMS_SQL_SERVER
) 
3685         retcode 
= SQLTablePrivileges(hstmt
, 
3687                                      (SQLTCHAR FAR 
*)Schema
.c_str(), SQL_NTS
,               // Schema 
3688                                      (SQLTCHAR FAR 
*)TableName
.c_str(), SQL_NTS
); 
3692         retcode 
= SQLTablePrivileges(hstmt
, 
3695                                      (SQLTCHAR FAR 
*)TableName
.c_str(), SQL_NTS
); 
3698 #ifdef DBDEBUG_CONSOLE 
3699     wxFprintf(stderr 
,wxT("SQLTablePrivileges() returned %i \n"),retcode
); 
3702     if ((retcode 
!= SQL_SUCCESS
) && (retcode 
!= SQL_SUCCESS_WITH_INFO
)) 
3703         return (DispAllErrors(henv
, hdbc
, hstmt
)); 
3705     bool failed 
= false; 
3706     retcode 
= SQLFetch(hstmt
); 
3707     while (retcode 
== SQL_SUCCESS 
|| retcode 
== SQL_SUCCESS_WITH_INFO
) 
3709         if (SQLGetData(hstmt
, 1, SQL_C_WXCHAR
, (UCHAR
*) result
.tableQual
, sizeof(result
.tableQual
), &cbRetVal
) != SQL_SUCCESS
) 
3712         if (!failed 
&& SQLGetData(hstmt
, 2, SQL_C_WXCHAR
, (UCHAR
*) result
.tableOwner
, sizeof(result
.tableOwner
), &cbRetVal
) != SQL_SUCCESS
) 
3715         if (!failed 
&& SQLGetData(hstmt
, 3, SQL_C_WXCHAR
, (UCHAR
*) result
.tableName
, sizeof(result
.tableName
), &cbRetVal
) != SQL_SUCCESS
) 
3718         if (!failed 
&& SQLGetData(hstmt
, 4, SQL_C_WXCHAR
, (UCHAR
*) result
.grantor
, sizeof(result
.grantor
), &cbRetVal
) != SQL_SUCCESS
) 
3721         if (!failed 
&& SQLGetData(hstmt
, 5, SQL_C_WXCHAR
, (UCHAR
*) result
.grantee
, sizeof(result
.grantee
), &cbRetVal
) != SQL_SUCCESS
) 
3724         if (!failed 
&& SQLGetData(hstmt
, 6, SQL_C_WXCHAR
, (UCHAR
*) result
.privilege
, sizeof(result
.privilege
), &cbRetVal
) != SQL_SUCCESS
) 
3727         if (!failed 
&& SQLGetData(hstmt
, 7, SQL_C_WXCHAR
, (UCHAR
*) result
.grantable
, sizeof(result
.grantable
), &cbRetVal
) != SQL_SUCCESS
) 
3732             return(DispAllErrors(henv
, hdbc
, hstmt
)); 
3734 #ifdef DBDEBUG_CONSOLE 
3735         wxFprintf(stderr
,wxT("Scanning %s privilege on table %s.%s granted by %s to %s\n"), 
3736                 result
.privilege
,result
.tableOwner
,result
.tableName
, 
3737                 result
.grantor
, result
.grantee
); 
3740         if (UserID
.IsSameAs(result
.tableOwner
,false)) 
3742             SQLFreeStmt(hstmt
, SQL_CLOSE
); 
3746         if (UserID
.IsSameAs(result
.grantee
,false) && 
3747             !wxStrcmp(result
.privilege
,priv
)) 
3749             SQLFreeStmt(hstmt
, SQL_CLOSE
); 
3753         if (!wxStrcmp(result
.grantee
,curRole
) && 
3754             !wxStrcmp(result
.privilege
,priv
)) 
3756             SQLFreeStmt(hstmt
, SQL_CLOSE
); 
3760         retcode 
= SQLFetch(hstmt
); 
3763     SQLFreeStmt(hstmt
, SQL_CLOSE
); 
3766 }  // wxDb::TablePrivileges 
3769 const wxString 
wxDb::SQLTableName(const wxChar 
*tableName
) 
3773     if (Dbms() == dbmsACCESS
) 
3774         TableName 
= _T("\""); 
3775     TableName 
+= tableName
; 
3776     if (Dbms() == dbmsACCESS
) 
3777         TableName 
+= _T("\""); 
3780 }  // wxDb::SQLTableName() 
3783 const wxString 
wxDb::SQLColumnName(const wxChar 
*colName
) 
3787     if (Dbms() == dbmsACCESS
) 
3790     if (Dbms() == dbmsACCESS
) 
3791         ColName 
+= _T("\""); 
3794 }  // wxDb::SQLColumnName() 
3797 /********** wxDb::SetSqlLogging() **********/ 
3798 bool wxDb::SetSqlLogging(wxDbSqlLogState state
, const wxString 
&filename
, bool append
) 
3800     wxASSERT(state 
== sqlLogON  
|| state 
== sqlLogOFF
); 
3801     wxASSERT(state 
== sqlLogOFF 
|| filename
.Length()); 
3803     if (state 
== sqlLogON
) 
3807             fpSqlLog 
= wxFopen(filename
.c_str(), (append 
? wxT("at") : wxT("wt"))); 
3808             if (fpSqlLog 
== NULL
) 
3816             if (fclose(fpSqlLog
)) 
3822     sqlLogState 
= state
; 
3825 }  // wxDb::SetSqlLogging() 
3828 /********** wxDb::WriteSqlLog() **********/ 
3829 bool wxDb::WriteSqlLog(const wxString 
&logMsg
) 
3831     wxASSERT(logMsg
.Length()); 
3833     if (fpSqlLog 
== 0 || sqlLogState 
== sqlLogOFF
) 
3836     if (wxFputs(wxT("\n"),   fpSqlLog
) == EOF
) 
3838     if (wxFputs(logMsg
, fpSqlLog
) == EOF
) 
3840     if (wxFputs(wxT("\n"),   fpSqlLog
) == EOF
) 
3845 }  // wxDb::WriteSqlLog() 
3848 /********** wxDb::Dbms() **********/ 
3849 wxDBMS 
wxDb::Dbms(void) 
3851  * Be aware that not all database engines use the exact same syntax, and not 
3852  * every ODBC compliant database is compliant to the same level of compliancy. 
3853  * Some manufacturers support the minimum Level 1 compliancy, and others up 
3854  * through Level 3.  Others support subsets of features for levels above 1. 
3856  * If you find an inconsistency between the wxDb class and a specific database 
3857  * engine, and an identifier to this section, and special handle the database in 
3858  * the area where behavior is non-conforming with the other databases. 
3861  * NOTES ABOUT ISSUES SPECIFIC TO EACH DATABASE ENGINE 
3862  * --------------------------------------------------- 
3865  *        - Currently the only database supported by the class to support VIEWS 
3868  *        - Does not support the SQL_TIMESTAMP structure 
3869  *        - Supports only one cursor and one connect (apparently? with Microsoft driver only?) 
3870  *        - Does not automatically create the primary index if the 'keyField' param of SetColDef 
3871  *            is true.  The user must create ALL indexes from their program. 
3872  *        - Table names can only be 8 characters long 
3873  *        - Column names can only be 10 characters long 
3876  *        - To lock a record during QUERY functions, the reserved word 'HOLDLOCK' must be added 
3877  *            after every table name involved in the query/join if that tables matching record(s) 
3879  *        - Ignores the keywords 'FOR UPDATE'.  Use the HOLDLOCK functionality described above 
3881  * SYBASE (Enterprise) 
3882  *        - If a column is part of the Primary Key, the column cannot be NULL 
3883  *        - Maximum row size is somewhere in the neighborhood of 1920 bytes 
3886  *        - If a column is part of the Primary Key, the column cannot be NULL 
3887  *        - Cannot support selecting for update [::CanSelectForUpdate()].  Always returns FALSE 
3888  *        - Columns that are part of primary or secondary keys must be defined as being NOT NULL 
3889  *            when they are created.  Some code is added in ::CreateIndex to try to adjust the 
3890  *            column definition if it is not defined correctly, but it is experimental 
3891  *        - Does not support sub-queries in SQL statements 
3894  *        - Does not support the keywords 'ASC' or 'DESC' as of release v6.5.0 
3895  *        - Does not support sub-queries in SQL statements 
3898  *        - Primary keys must be declared as NOT NULL 
3899  *        - Table and index names must not be longer than 13 characters in length (technically 
3900  *          table names can be up to 18 characters, but the primary index is created using the 
3901  *          base table name plus "_PIDX", so the limit if the table has a primary index is 13. 
3906  *        - Columns that are part of primary keys must be defined as being NOT NULL 
3907  *          when they are created.  Some code is added in ::CreateIndex to try to adjust the 
3908  *          column definition if it is not defined correctly, but it is experimental 
3911     // Should only need to do this once for each new database connection 
3912     // so return the value we already determined it to be to save time 
3913     // and lots of string comparisons 
3914     if (dbmsType 
!= dbmsUNIDENTIFIED
) 
3917 #ifdef DBDEBUG_CONSOLE 
3918                // When run in console mode, use standard out to display errors. 
3919                cout 
<< "Database connecting to: " << dbInf
.dbmsName 
<< endl
; 
3920 #endif  // DBDEBUG_CONSOLE 
3922     wxLogDebug(wxT("Database connecting to: ")); 
3923     wxLogDebug(dbInf
.dbmsName
); 
3925     wxChar baseName
[25+1]; 
3926     wxStrncpy(baseName
, dbInf
.dbmsName
, 25); 
3929     // RGG 20001025 : add support for Interbase 
3930     // GT : Integrated to base classes on 20001121 
3931     if (!wxStricmp(dbInf
.dbmsName
,wxT("Interbase"))) 
3932         return((wxDBMS
)(dbmsType 
= dbmsINTERBASE
)); 
3934     // BJO 20000428 : add support for Virtuoso 
3935     if (!wxStricmp(dbInf
.dbmsName
,wxT("OpenLink Virtuoso VDBMS"))) 
3936       return((wxDBMS
)(dbmsType 
= dbmsVIRTUOSO
)); 
3938     if (!wxStricmp(dbInf
.dbmsName
,wxT("Adaptive Server Anywhere"))) 
3939         return((wxDBMS
)(dbmsType 
= dbmsSYBASE_ASA
)); 
3941     // BJO 20000427 : The "SQL Server" string is also returned by SQLServer when 
3942     // connected through an OpenLink driver. 
3943     // Is it also returned by Sybase Adapatitve server? 
3944     // OpenLink driver name is OLOD3032.DLL for msw and oplodbc.so for unix 
3945     if (!wxStricmp(dbInf
.dbmsName
,wxT("SQL Server"))) 
3947       if (!wxStrncmp(dbInf
.driverName
, wxT("oplodbc"), 7) || 
3948           !wxStrncmp(dbInf
.driverName
, wxT("OLOD"), 4)) 
3949             return ((wxDBMS
)(dbmsMS_SQL_SERVER
)); 
3951             return ((wxDBMS
)(dbmsType 
= dbmsSYBASE_ASE
)); 
3954     if (!wxStricmp(dbInf
.dbmsName
,wxT("Microsoft SQL Server"))) 
3955         return((wxDBMS
)(dbmsType 
= dbmsMS_SQL_SERVER
)); 
3958     if (!wxStricmp(baseName
,wxT("PostgreSQL")))  // v6.5.0 
3959         return((wxDBMS
)(dbmsType 
= dbmsPOSTGRES
)); 
3962     if (!wxStricmp(baseName
,wxT("Pervasive"))) 
3963         return((wxDBMS
)(dbmsType 
= dbmsPERVASIVE_SQL
)); 
3966     if (!wxStricmp(baseName
,wxT("Informix"))) 
3967         return((wxDBMS
)(dbmsType 
= dbmsINFORMIX
)); 
3969     if (!wxStricmp(baseName
,wxT("Firebird"))) 
3970         return((wxDBMS
)(dbmsType 
= dbmsFIREBIRD
)); 
3973     if (!wxStricmp(baseName
,wxT("Oracle"))) 
3974         return((wxDBMS
)(dbmsType 
= dbmsORACLE
)); 
3975     if (!wxStricmp(baseName
,wxT("ACCESS"))) 
3976         return((wxDBMS
)(dbmsType 
= dbmsACCESS
)); 
3977     if (!wxStricmp(baseName
,wxT("Sybase"))) 
3978       return((wxDBMS
)(dbmsType 
= dbmsSYBASE_ASE
)); 
3981     if (!wxStricmp(baseName
,wxT("DBASE"))) 
3982         return((wxDBMS
)(dbmsType 
= dbmsDBASE
)); 
3983     if (!wxStricmp(baseName
,wxT("xBase"))) 
3984         return((wxDBMS
)(dbmsType 
= dbmsXBASE_SEQUITER
)); 
3985     if (!wxStricmp(baseName
,wxT("MySQL"))) 
3986         return((wxDBMS
)(dbmsType 
= dbmsMY_SQL
)); 
3987     if (!wxStricmp(baseName
,wxT("MaxDB"))) 
3988         return((wxDBMS
)(dbmsType 
= dbmsMAXDB
)); 
3991     if (!wxStricmp(baseName
,wxT("DB2"))) 
3992         return((wxDBMS
)(dbmsType 
= dbmsDB2
)); 
3994     return((wxDBMS
)(dbmsType 
= dbmsUNIDENTIFIED
)); 
3999 bool wxDb::ModifyColumn(const wxString 
&tableName
, const wxString 
&columnName
, 
4000                         int dataType
, ULONG columnLength
, 
4001                         const wxString 
&optionalParam
) 
4003     wxASSERT(tableName
.Length()); 
4004     wxASSERT(columnName
.Length()); 
4005     wxASSERT((dataType 
== DB_DATA_TYPE_VARCHAR 
&& columnLength 
> 0) || 
4006              dataType 
!= DB_DATA_TYPE_VARCHAR
); 
4008     // Must specify a columnLength if modifying a VARCHAR type column 
4009     if (dataType 
== DB_DATA_TYPE_VARCHAR 
&& !columnLength
) 
4012     wxString dataTypeName
; 
4014     wxString alterSlashModify
; 
4018         case DB_DATA_TYPE_VARCHAR 
: 
4019             dataTypeName 
= typeInfVarchar
.TypeName
; 
4021         case DB_DATA_TYPE_INTEGER 
: 
4022             dataTypeName 
= typeInfInteger
.TypeName
; 
4024         case DB_DATA_TYPE_FLOAT 
: 
4025             dataTypeName 
= typeInfFloat
.TypeName
; 
4027         case DB_DATA_TYPE_DATE 
: 
4028             dataTypeName 
= typeInfDate
.TypeName
; 
4030         case DB_DATA_TYPE_BLOB 
: 
4031             dataTypeName 
= typeInfBlob
.TypeName
; 
4037     // Set the modify or alter syntax depending on the type of database connected to 
4041             alterSlashModify 
= _T("MODIFY"); 
4043         case dbmsMS_SQL_SERVER 
: 
4044             alterSlashModify 
= _T("ALTER COLUMN"); 
4046         case dbmsUNIDENTIFIED 
: 
4048         case dbmsSYBASE_ASA 
: 
4049         case dbmsSYBASE_ASE 
: 
4054         case dbmsXBASE_SEQUITER 
: 
4056             alterSlashModify 
= _T("MODIFY"); 
4060     // create the SQL statement 
4061     if ( Dbms() == dbmsMY_SQL 
) 
4063         sqlStmt
.Printf(wxT("ALTER TABLE %s %s %s %s"), tableName
.c_str(), alterSlashModify
.c_str(), 
4064               columnName
.c_str(), dataTypeName
.c_str()); 
4068         sqlStmt
.Printf(wxT("ALTER TABLE \"%s\" \"%s\" \"%s\" %s"), tableName
.c_str(), alterSlashModify
.c_str(), 
4069               columnName
.c_str(), dataTypeName
.c_str()); 
4072     // For varchars only, append the size of the column 
4073     if (dataType 
== DB_DATA_TYPE_VARCHAR 
&& 
4074         (Dbms() != dbmsMY_SQL 
|| dataTypeName 
!= _T("text"))) 
4077         s
.Printf(wxT("(%lu)"), columnLength
); 
4081     // for passing things like "NOT NULL" 
4082     if (optionalParam
.Length()) 
4084         sqlStmt 
+= wxT(" "); 
4085         sqlStmt 
+= optionalParam
; 
4088     return ExecSql(sqlStmt
); 
4090 } // wxDb::ModifyColumn() 
4093 /********** wxDbGetConnection() **********/ 
4094 wxDb WXDLLIMPEXP_ODBC 
*wxDbGetConnection(wxDbConnectInf 
*pDbConfig
, bool FwdOnlyCursors
) 
4098     // Used to keep a pointer to a DB connection that matches the requested 
4099     // DSN and FwdOnlyCursors settings, even if it is not FREE, so that the 
4100     // data types can be copied from it (using the wxDb::Open(wxDb *) function) 
4101     // rather than having to re-query the datasource to get all the values 
4102     // using the wxDb::Open(Dsn,Uid,AuthStr) function 
4103     wxDb 
*matchingDbConnection 
= NULL
; 
4105     // Scan the linked list searching for an available database connection 
4106     // that's already been opened but is currently not in use. 
4107     for (pList 
= PtrBegDbList
; pList
; pList 
= pList
->PtrNext
) 
4109         // The database connection must be for the same datasource 
4110         // name and must currently not be in use. 
4112             (pList
->PtrDb
->FwdOnlyCursors() == FwdOnlyCursors
)) 
4114             if (pDbConfig
->UseConnectionStr()) 
4116                 if (pList
->PtrDb
->OpenedWithConnectionString() && 
4117                      (!wxStrcmp(pDbConfig
->GetConnectionStr(), pList
->ConnectionStr
))) 
4119                     // Found a free connection 
4120                     pList
->Free 
= false; 
4121                     return(pList
->PtrDb
); 
4126                 if (!pList
->PtrDb
->OpenedWithConnectionString() && 
4127                      (!wxStrcmp(pDbConfig
->GetDsn(), pList
->Dsn
))) 
4129                     // Found a free connection 
4130                     pList
->Free 
= false; 
4131                     return(pList
->PtrDb
); 
4136         if (pDbConfig
->UseConnectionStr()) 
4138             if (!wxStrcmp(pDbConfig
->GetConnectionStr(), pList
->ConnectionStr
)) 
4139                 matchingDbConnection 
= pList
->PtrDb
; 
4143             if (!wxStrcmp(pDbConfig
->GetDsn(), pList
->Dsn
) && 
4144                 !wxStrcmp(pDbConfig
->GetUserID(), pList
->Uid
) && 
4145                 !wxStrcmp(pDbConfig
->GetPassword(), pList
->AuthStr
)) 
4146                 matchingDbConnection 
= pList
->PtrDb
; 
4150     // No available connections.  A new connection must be made and 
4151     // appended to the end of the linked list. 
4154         // Find the end of the list 
4155         for (pList 
= PtrBegDbList
; pList
->PtrNext
; pList 
= pList
->PtrNext
); 
4156         // Append a new list item 
4157         pList
->PtrNext 
= new wxDbList
; 
4158         pList
->PtrNext
->PtrPrev 
= pList
; 
4159         pList 
= pList
->PtrNext
; 
4163         // Create the first node on the list 
4164         pList 
= PtrBegDbList 
= new wxDbList
; 
4168     // Initialize new node in the linked list 
4170     pList
->Free             
= false; 
4171     pList
->Dsn              
= pDbConfig
->GetDsn(); 
4172     pList
->Uid              
= pDbConfig
->GetUserID(); 
4173     pList
->AuthStr          
= pDbConfig
->GetPassword(); 
4174     pList
->ConnectionStr    
= pDbConfig
->GetConnectionStr(); 
4176     pList
->PtrDb 
= new wxDb(pDbConfig
->GetHenv(), FwdOnlyCursors
); 
4180     if (!matchingDbConnection
) 
4182         if (pDbConfig
->UseConnectionStr()) 
4184             opened 
= pList
->PtrDb
->Open(pDbConfig
->GetConnectionStr()); 
4188             opened 
= pList
->PtrDb
->Open(pDbConfig
->GetDsn(), pDbConfig
->GetUserID(), pDbConfig
->GetPassword()); 
4192         opened 
= pList
->PtrDb
->Open(matchingDbConnection
); 
4194     // Connect to the datasource 
4197         pList
->PtrDb
->setCached(true);  // Prevent a user from deleting a cached connection 
4198         pList
->PtrDb
->SetSqlLogging(SQLLOGstate
, SQLLOGfn
, true); 
4199         return(pList
->PtrDb
); 
4201     else  // Unable to connect, destroy list item 
4204             pList
->PtrPrev
->PtrNext 
= 0; 
4206             PtrBegDbList 
= 0;        // Empty list again 
4208         pList
->PtrDb
->CommitTrans(); // Commit any open transactions on wxDb object 
4209         pList
->PtrDb
->Close();       // Close the wxDb object 
4210         delete pList
->PtrDb
;         // Deletes the wxDb object 
4211         delete pList
;                // Deletes the linked list object 
4215 }  // wxDbGetConnection() 
4218 /********** wxDbFreeConnection() **********/ 
4219 bool WXDLLIMPEXP_ODBC 
wxDbFreeConnection(wxDb 
*pDb
) 
4223     // Scan the linked list searching for the database connection 
4224     for (pList 
= PtrBegDbList
; pList
; pList 
= pList
->PtrNext
) 
4226         if (pList
->PtrDb 
== pDb
)  // Found it, now free it!!! 
4227             return (pList
->Free 
= true); 
4230     // Never found the database object, return failure 
4233 }  // wxDbFreeConnection() 
4236 /********** wxDbCloseConnections() **********/ 
4237 void WXDLLIMPEXP_ODBC 
wxDbCloseConnections(void) 
4239     wxDbList 
*pList
, *pNext
; 
4241     // Traverse the linked list closing database connections and freeing memory as I go. 
4242     for (pList 
= PtrBegDbList
; pList
; pList 
= pNext
) 
4244         pNext 
= pList
->PtrNext
;       // Save the pointer to next 
4245         pList
->PtrDb
->CommitTrans();  // Commit any open transactions on wxDb object 
4246         pList
->PtrDb
->Close();        // Close the wxDb object 
4247         pList
->PtrDb
->setCached(false);  // Allows deletion of the wxDb instance 
4248         delete pList
->PtrDb
;          // Deletes the wxDb object 
4249         delete pList
;                 // Deletes the linked list object 
4252     // Mark the list as empty 
4255 }  // wxDbCloseConnections() 
4258 /********** wxDbConnectionsInUse() **********/ 
4259 int WXDLLIMPEXP_ODBC 
wxDbConnectionsInUse(void) 
4264     // Scan the linked list counting db connections that are currently in use 
4265     for (pList 
= PtrBegDbList
; pList
; pList 
= pList
->PtrNext
) 
4267         if (pList
->Free 
== false) 
4273 }  // wxDbConnectionsInUse() 
4277 /********** wxDbLogExtendedErrorMsg() **********/ 
4278 // DEBUG ONLY function 
4279 const wxChar WXDLLIMPEXP_ODBC 
*wxDbLogExtendedErrorMsg(const wxChar 
*userText
, 
4281                                                   const wxChar 
*ErrFile
, 
4284     static wxString msg
; 
4289     if (ErrFile 
|| ErrLine
) 
4291         msg 
+= wxT("File: "); 
4293         msg 
+= wxT("   Line: "); 
4294         tStr
.Printf(wxT("%d"),ErrLine
); 
4295         msg 
+= tStr
.c_str(); 
4299     msg
.Append (wxT("\nODBC errors:\n")); 
4302     // Display errors for this connection 
4304     for (i 
= 0; i 
< DB_MAX_ERROR_HISTORY
; i
++) 
4306         if (pDb
->errorList
[i
]) 
4308             msg
.Append(pDb
->errorList
[i
]); 
4309             if (wxStrcmp(pDb
->errorList
[i
], wxEmptyString
) != 0) 
4310                 msg
.Append(wxT("\n")); 
4311             // Clear the errmsg buffer so the next error will not 
4312             // end up showing the previous error that have occurred 
4313             wxStrcpy(pDb
->errorList
[i
], wxEmptyString
); 
4318     wxLogDebug(msg
.c_str()); 
4321 }  // wxDbLogExtendedErrorMsg() 
4324 /********** wxDbSqlLog() **********/ 
4325 bool wxDbSqlLog(wxDbSqlLogState state
, const wxChar 
*filename
) 
4327     bool append 
= false; 
4330     for (pList 
= PtrBegDbList
; pList
; pList 
= pList
->PtrNext
) 
4332         if (!pList
->PtrDb
->SetSqlLogging(state
,filename
,append
)) 
4337     SQLLOGstate 
= state
; 
4338     SQLLOGfn 
= filename
; 
4346 /********** wxDbCreateDataSource() **********/ 
4347 int wxDbCreateDataSource(const wxString 
&driverName
, const wxString 
&dsn
, const wxString 
&description
, 
4348                          bool sysDSN
, const wxString 
&defDir
, wxWindow 
*parent
) 
4350  * !!!! ONLY FUNCTIONAL UNDER MSW with VC6 !!!! 
4351  * Very rudimentary creation of an ODBC data source. 
4353  * ODBC driver must be ODBC 3.0 compliant to use this function 
4358 //!!!! ONLY FUNCTIONAL UNDER MSW with VC6 !!!! 
4364         dsnLocation 
= ODBC_ADD_SYS_DSN
; 
4366         dsnLocation 
= ODBC_ADD_DSN
; 
4368     // NOTE: The decimal 2 is an invalid character in all keyword pairs 
4369     // so that is why I used it, as wxString does not deal well with 
4370     // embedded nulls in strings 
4371     setupStr
.Printf(wxT("DSN=%s%cDescription=%s%cDefaultDir=%s%c"),dsn
,2,description
,2,defDir
,2); 
4373     // Replace the separator from above with the '\0' separator needed 
4374     // by the SQLConfigDataSource() function 
4378         k 
= setupStr
.Find((wxChar
)2,true); 
4379         if (k 
!= wxNOT_FOUND
) 
4380             setupStr
[(UINT
)k
] = wxT('\0'); 
4382     while (k 
!= wxNOT_FOUND
); 
4384     result 
= SQLConfigDataSource((HWND
)parent
->GetHWND(), dsnLocation
, 
4385                                  driverName
, setupStr
.c_str()); 
4387     if ((result 
!= SQL_SUCCESS
) && (result 
!= SQL_SUCCESS_WITH_INFO
)) 
4389         // check for errors caused by ConfigDSN based functions 
4392         wxChar errMsg
[SQL_MAX_MESSAGE_LENGTH
]; 
4393         errMsg
[0] = wxT('\0'); 
4395         // This function is only supported in ODBC drivers v3.0 compliant and above 
4396         SQLInstallerError(1,&retcode
,errMsg
,SQL_MAX_MESSAGE_LENGTH
-1,&cb
); 
4399 #ifdef DBDEBUG_CONSOLE 
4400                // When run in console mode, use standard out to display errors. 
4401                cout 
<< errMsg 
<< endl
; 
4402                cout 
<< wxT("Press any key to continue...") << endl
; 
4404 #endif  // DBDEBUG_CONSOLE 
4407                wxLogDebug(errMsg
,wxT("ODBC DEBUG MESSAGE")); 
4408 #endif  // __WXDEBUG__ 
4414     // Using iODBC/unixODBC or some other compiler which does not support the APIs 
4415     // necessary to use this function, so this function is not supported 
4417     wxLogDebug(wxT("wxDbCreateDataSource() not available except under VC++/MSW"),wxT("ODBC DEBUG MESSAGE")); 
4420 #endif  // __VISUALC__ 
4424 }  // wxDbCreateDataSource() 
4428 /********** wxDbGetDataSource() **********/ 
4429 bool wxDbGetDataSource(HENV henv
, wxChar 
*Dsn
, SWORD DsnMaxLength
, wxChar 
*DsDesc
, 
4430                        SWORD DsDescMaxLength
, UWORD direction
) 
4432  * Dsn and DsDesc will contain the data source name and data source 
4433  * description upon return 
4437     SWORD lengthDsn 
= (SWORD
)(DsnMaxLength
*sizeof(wxChar
)); 
4438     SWORD lengthDsDesc 
= (SWORD
)(DsDescMaxLength
*sizeof(wxChar
)); 
4440     if (SQLDataSources(henv
, direction
, (SQLTCHAR FAR 
*) Dsn
, lengthDsn
, &cb1
, 
4441                        (SQLTCHAR FAR 
*) DsDesc
, lengthDsDesc
, &cb2
) == SQL_SUCCESS
) 
4446 }  // wxDbGetDataSource() 
4449 // Change this to 0 to remove use of all deprecated functions 
4450 #if wxODBC_BACKWARD_COMPATABILITY 
4451 /******************************************************************** 
4452  ******************************************************************** 
4454  * The following functions are all DEPRECATED and are included for 
4455  * backward compatibility reasons only 
4457  ******************************************************************** 
4458  ********************************************************************/ 
4459 bool SqlLog(sqlLog state
, const wxChar 
*filename
) 
4461     return wxDbSqlLog((enum wxDbSqlLogState
)state
, filename
); 
4463 /***** DEPRECATED: use wxGetDataSource() *****/ 
4464 bool GetDataSource(HENV henv
, char *Dsn
, SWORD DsnMax
, char *DsDesc
, SWORD DsDescMax
, 
4467     return wxDbGetDataSource(henv
, Dsn
, DsnMax
, DsDesc
, DsDescMax
, direction
); 
4469 /***** DEPRECATED: use wxDbGetConnection() *****/ 
4470 wxDb WXDLLIMPEXP_ODBC 
*GetDbConnection(DbStuff 
*pDbStuff
, bool FwdOnlyCursors
) 
4472     return wxDbGetConnection((wxDbConnectInf 
*)pDbStuff
, FwdOnlyCursors
); 
4474 /***** DEPRECATED: use wxDbFreeConnection() *****/ 
4475 bool WXDLLIMPEXP_ODBC 
FreeDbConnection(wxDb 
*pDb
) 
4477     return wxDbFreeConnection(pDb
); 
4479 /***** DEPRECATED: use wxDbCloseConnections() *****/ 
4480 void WXDLLIMPEXP_ODBC 
CloseDbConnections(void) 
4482     wxDbCloseConnections(); 
4484 /***** DEPRECATED: use wxDbConnectionsInUse() *****/ 
4485 int WXDLLIMPEXP_ODBC 
NumberDbConnectionsInUse(void) 
4487     return wxDbConnectionsInUse();