[ODBC main.c]
* Updated so non-data-producing queries don't fail with error. * Added routine to throw detailed ODBC errors as Gambas errors. * Cosmetic changes to the C code.
This commit is contained in:
parent
b99546d40c
commit
9f8ab729fe
1 changed files with 372 additions and 229 deletions
|
@ -4,7 +4,7 @@
|
||||||
|
|
||||||
(c) 2004-2007 Andrea Bortolan <andrea_bortolan@yahoo.it>
|
(c) 2004-2007 Andrea Bortolan <andrea_bortolan@yahoo.it>
|
||||||
(c) 2000-2017 Benoît Minisini <gambas@users.sourceforge.net>
|
(c) 2000-2017 Benoît Minisini <gambas@users.sourceforge.net>
|
||||||
(c) 2015 zxMarce <d4t4full@gmail.com>
|
(c) 2015-2017 zxMarce <d4t4full@gmail.com>
|
||||||
|
|
||||||
This program is free software; you can redistribute it and/or modify
|
This program is free software; you can redistribute it and/or modify
|
||||||
it under the terms of the GNU General Public License as published by
|
it under the terms of the GNU General Public License as published by
|
||||||
|
@ -76,13 +76,13 @@ static DB_DRIVER _driver;
|
||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
// SQLHENV odbcEnvHandle;
|
// SQLHENV odbcEnvHandle;
|
||||||
// SQLHDBC odbcHandle;
|
// SQLHDBC odbcHandle;
|
||||||
SQLHANDLE odbcEnvHandle;
|
SQLHANDLE odbcEnvHandle; //ODBC environment handle
|
||||||
SQLHANDLE odbcHandle;
|
SQLHANDLE odbcHandle; //ODBC connection handle
|
||||||
SQLUSMALLINT FetchScroll_exist;
|
SQLUSMALLINT FetchScroll_exist; //Flag
|
||||||
char *dsn_name;
|
char *dsn_name; //DSN name
|
||||||
char *user_name;
|
char *user_name; //Logged-in user name
|
||||||
}
|
}
|
||||||
ODBC_CONN;
|
ODBC_CONN;
|
||||||
|
|
||||||
|
@ -120,19 +120,19 @@ ODBC_TABLES;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* zxMarce: Routine to print to console the errors from the ODBC subsystem.
|
* zxMarce: Routine to print to console the errors from the ODBC subsystem.
|
||||||
* Adapted to obey Gambas' DB.IsDebug() and use SQLGetDiagRecW() instead of SQLGetDiagRec().
|
* Adapted to obey Gambas' DB.IsDebug().
|
||||||
* Mostly from http://www.easysoft.com/developer/interfaces/odbc/diagnostics_error_status_codes.html
|
* Mostly from http://www.easysoft.com/developer/interfaces/odbc/diagnostics_error_status_codes.html
|
||||||
*/
|
*/
|
||||||
void reportODBCError(char *fn,
|
void reportODBCError(const char *fn,
|
||||||
SQLHANDLE handle,
|
SQLHANDLE handle,
|
||||||
SQLSMALLINT type
|
SQLSMALLINT type
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
|
|
||||||
SQLINTEGER i = 0;
|
SQLINTEGER i = 0;
|
||||||
SQLINTEGER native;
|
SQLINTEGER native;
|
||||||
SQLWCHAR state[7];
|
SQLTCHAR state[7];
|
||||||
SQLWCHAR text[256];
|
SQLTCHAR text[256];
|
||||||
SQLSMALLINT len;
|
SQLSMALLINT len;
|
||||||
SQLRETURN ret;
|
SQLRETURN ret;
|
||||||
|
|
||||||
|
@ -141,134 +141,203 @@ void reportODBCError(char *fn,
|
||||||
fprintf(stderr, "gb.db.odbc: %s\n", fn);
|
fprintf(stderr, "gb.db.odbc: %s\n", fn);
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
ret = SQLGetDiagRecW(type, handle, ++i, state, &native, text, sizeof(text), &len);
|
ret = SQLGetDiagRec(type, handle, ++i, state, &native, text, sizeof(text), &len);
|
||||||
|
|
||||||
if (SQL_SUCCEEDED(ret))
|
if (SQL_SUCCEEDED(ret))
|
||||||
fprintf(stderr, "gb.db.odbc: %s:%d:%d:%s\n", (char *)state, (int)i, (int)native, (char *)text);
|
fprintf(stderr, "gb.db.odbc: %d:%s:%d:%s\n", (int)i, (char *)state, (int)native, (char *)text);
|
||||||
}
|
}
|
||||||
while (ret == SQL_SUCCESS);
|
while (ret == SQL_SUCCESS);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* 20170623 zxMarce: Routine to publish underlying ODBC error(s) as a normal
|
||||||
|
* Gambas error. Based on reportODBCError above, but does NOT obey DB.IsDebug(),
|
||||||
|
* as this is intended as a component-centric "official" error report routine.
|
||||||
|
* It is the intention to replace most GB.Error() calls with this routine.
|
||||||
|
* Thanks to Tobias Boege for native GB string handling tips!
|
||||||
|
*/
|
||||||
|
void throwODBCError(const char *failedODBCFunctionName,
|
||||||
|
SQLHANDLE handle,
|
||||||
|
SQLSMALLINT handleType
|
||||||
|
)
|
||||||
|
{
|
||||||
|
|
||||||
|
SQLINTEGER i = 0;
|
||||||
|
SQLINTEGER native;
|
||||||
|
SQLTCHAR state[7];
|
||||||
|
SQLTCHAR text[512];
|
||||||
|
char *errorText = GB.NewString("gb.db.odbc: ", 12);
|
||||||
|
SQLSMALLINT len;
|
||||||
|
SQLRETURN ret;
|
||||||
|
|
||||||
|
errorText = GB.AddString(errorText, (char *)failedODBCFunctionName, 0),
|
||||||
|
errorText = GB.AddString(errorText, " failed:", 0);
|
||||||
|
|
||||||
|
do
|
||||||
|
{
|
||||||
|
ret = SQLGetDiagRec(handleType, handle, ++i, state, &native, text, sizeof(text), &len);
|
||||||
|
if (SQL_SUCCEEDED(ret))
|
||||||
|
{
|
||||||
|
errorText = GB.AddString(errorText, "\n", 1);
|
||||||
|
errorText = GB.AddString(errorText, (char *)state, 0);
|
||||||
|
errorText = GB.AddString(errorText, (char *)text, len);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
while (ret == SQL_SUCCESS);
|
||||||
|
|
||||||
|
GB.Error(errorText);
|
||||||
|
GB.FreeString(&errorText);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/* zxMarce: This is one way -hope there's an easier one- to retrieve a rowset
|
/* zxMarce: This is one way -hope there's an easier one- to retrieve a rowset
|
||||||
* count for SELECT statements. Four steps (must have an scrollable cursor!):
|
* count for SELECT statements. Four steps (must have an scrollable cursor!):
|
||||||
* 1- Remember the current row.
|
* 1- Remember the current row.
|
||||||
* 2- Seek down to the last row in the rowset
|
* 2- Seek down to the last row in the rowset
|
||||||
* 3- Get the last row's index (recno)
|
* 3- Get the last row's index (recno)
|
||||||
* 4- Seek back to wherever we were at in step 1
|
* 4- Seek back to wherever we were at in step 1
|
||||||
|
* 20161110 zxMarce: Ok, it did not work that OK for Firebird; it looks like
|
||||||
|
* the FB driver returns one-less than the record count (record count seems to
|
||||||
|
* be zero-based), so we will instead do as follows, if we have a scrollable
|
||||||
|
* recordset:
|
||||||
|
* 1- Remember the current row.
|
||||||
|
* 2- Seek up to the first row in the rowset
|
||||||
|
* 3- Get the first row's index (firstRecNo)
|
||||||
|
* 4- Seek down to the last row in the rowset
|
||||||
|
* 5- Get the last row's index (lastRecNo)
|
||||||
|
* 6- Seek back to wherever we were at in step 1
|
||||||
|
* 7- Return (lastRecNo - firstRecNo + 1).
|
||||||
*/
|
*/
|
||||||
int GetRecordCount(SQLHANDLE stmtHandle, SQLINTEGER cursorScrollable)
|
int GetRecordCount(SQLHANDLE stmtHandle, SQLINTEGER cursorScrollable)
|
||||||
{
|
{
|
||||||
SQLRETURN retcode; //ODBC call return values
|
SQLRETURN retcode; //ODBC call return values
|
||||||
int formerRecIdx = 0; //Where we were when this all started.
|
int formerRecIdx = 0; //Where we were when this all started.
|
||||||
SQLINTEGER myRecCnt = -1; //Default for when there's no cursor.
|
SQLINTEGER myRecCnt = -1; //Default for when there's no cursor.
|
||||||
|
SQLINTEGER firstRecNo = 0; //20161111 holder for 1st recno.
|
||||||
|
SQLINTEGER lastRecNo = 0; //20161111 holder for last recno.
|
||||||
|
char mssg[128]; //Error reporting text.
|
||||||
|
|
||||||
//Make sure the statement has a cursor
|
//Make sure the statement has a cursor
|
||||||
if (stmtHandle && (cursorScrollable == SQL_TRUE))
|
if (!(stmtHandle && (cursorScrollable == SQL_TRUE)))
|
||||||
{
|
{
|
||||||
|
if (DB.IsDebug())
|
||||||
|
{
|
||||||
|
fprintf(stderr, "gb.db.odbc: Cannot do GetRecordCount()!\n");
|
||||||
|
}
|
||||||
|
return ((int) myRecCnt);
|
||||||
|
}
|
||||||
|
|
||||||
//Tell ODBC we won't be actually reading data (speeds process up).
|
//Tell ODBC we won't be actually reading data (speeds process up).
|
||||||
//SQL_ATTR_RETRIEVE_DATA = [SQL_RD_ON] | SQL_RD_OFF
|
//SQL_ATTR_RETRIEVE_DATA = [SQL_RD_ON] | SQL_RD_OFF
|
||||||
retcode = SQLSetStmtAttr(stmtHandle, SQL_ATTR_RETRIEVE_DATA, (SQLPOINTER) SQL_RD_OFF, 0);
|
retcode = SQLSetStmtAttr(stmtHandle, SQL_ATTR_RETRIEVE_DATA, (SQLPOINTER) SQL_RD_OFF, 0);
|
||||||
if (!SQL_SUCCEEDED(retcode))
|
if (!SQL_SUCCEEDED(retcode))
|
||||||
{
|
{
|
||||||
reportODBCError("SQLSetStmtAttr SQL_ATTR_RETRIEVE_DATA",
|
reportODBCError("SQLSetStmtAttr SQL_ATTR_RETRIEVE_DATA",
|
||||||
stmtHandle,
|
stmtHandle,
|
||||||
SQL_HANDLE_STMT
|
SQL_HANDLE_STMT);
|
||||||
);
|
}
|
||||||
}
|
|
||||||
|
|
||||||
//Fetch current row's index so we can return to it when done.
|
//Fetch current row's index so we can return to it when done.
|
||||||
retcode = SQLGetStmtAttr(stmtHandle, SQL_ATTR_ROW_NUMBER, &formerRecIdx, 0, 0);
|
retcode = SQLGetStmtAttr(stmtHandle, SQL_ATTR_ROW_NUMBER, &formerRecIdx, 0, 0);
|
||||||
if (!SQL_SUCCEEDED(retcode))
|
if (!SQL_SUCCEEDED(retcode))
|
||||||
{
|
{
|
||||||
reportODBCError("SQLGetStmtAttr SQL_ATTR_ROW_NUMBER",
|
reportODBCError("SQLGetStmtAttr SQL_ATTR_ROW_NUMBER",
|
||||||
stmtHandle,
|
stmtHandle,
|
||||||
SQL_HANDLE_STMT
|
SQL_HANDLE_STMT);
|
||||||
);
|
}
|
||||||
}
|
|
||||||
|
|
||||||
//Advance the cursor to the last record.
|
//Make sure the statement has a cursor
|
||||||
retcode = SQLFetchScroll(stmtHandle, SQL_FETCH_LAST, (SQLINTEGER) 0);
|
if (formerRecIdx < 0)
|
||||||
if (SQL_SUCCEEDED(retcode))
|
{
|
||||||
{
|
if (DB.IsDebug())
|
||||||
|
{
|
||||||
//Fetch the last record's index
|
fprintf(stderr, "gb.db.odbc.GetRecordCount: Current record returned %d, returning -1 as count.\n", formerRecIdx);
|
||||||
retcode = SQLGetStmtAttr(stmtHandle, SQL_ATTR_ROW_NUMBER, &myRecCnt, 0, 0);
|
}
|
||||||
if (SQL_SUCCEEDED(retcode))
|
return ((int) myRecCnt);
|
||||||
{
|
}
|
||||||
|
|
||||||
//Set ret value
|
|
||||||
if (DB.IsDebug())
|
|
||||||
{
|
|
||||||
fprintf(stderr, "gb.db.odbc.GetRecordCount: Success, count=%d\n", (int) myRecCnt);
|
|
||||||
}
|
|
||||||
|
|
||||||
} else {
|
|
||||||
|
|
||||||
reportODBCError("SQLGetStmtAttr SQL_ATTR_ROW_NUMBER",
|
|
||||||
stmtHandle,
|
|
||||||
SQL_HANDLE_STMT
|
|
||||||
);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
//Return cursor to original row.
|
|
||||||
retcode = SQLFetchScroll(stmtHandle, SQL_FETCH_ABSOLUTE, (SQLINTEGER) formerRecIdx);
|
|
||||||
//Since we have set the "do not read data" statement attribute, this call (may) return
|
|
||||||
//code 100 (SQL_NO_DATA) but that's OK for our purposes of just counting rows.
|
|
||||||
if (!SQL_SUCCEEDED(retcode) && (retcode != SQL_NO_DATA))
|
|
||||||
{
|
|
||||||
char mssg[128];
|
|
||||||
snprintf(mssg,
|
|
||||||
sizeof(mssg),
|
|
||||||
"SQLFetchScroll SQL_FETCH_ABSOLUTE (code %d) (rec %d)",
|
|
||||||
(int)retcode,
|
|
||||||
formerRecIdx
|
|
||||||
);
|
|
||||||
|
|
||||||
reportODBCError(mssg,
|
|
||||||
stmtHandle,
|
|
||||||
SQL_HANDLE_STMT
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
} else {
|
|
||||||
|
|
||||||
reportODBCError("SQLFetchScroll SQL_FETCH_LAST",
|
|
||||||
stmtHandle,
|
|
||||||
SQL_HANDLE_STMT
|
|
||||||
);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
//Tell ODBC we will be reading data now.
|
|
||||||
//SQL_ATTR_RETRIEVE_DATA = [SQL_RD_ON] | SQL_RD_OFF
|
|
||||||
retcode = SQLSetStmtAttr(stmtHandle, SQL_ATTR_RETRIEVE_DATA, (SQLPOINTER) SQL_RD_ON, 0);
|
|
||||||
if (!SQL_SUCCEEDED(retcode))
|
|
||||||
{
|
|
||||||
reportODBCError("SQLSetStmtAttr SQL_ATTR_RETRIEVE_DATA",
|
|
||||||
stmtHandle,
|
|
||||||
SQL_HANDLE_STMT
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
//Try to get (back?) to the first record, abort if not possible.
|
||||||
|
retcode = SQLFetchScroll(stmtHandle, SQL_FETCH_FIRST, (SQLINTEGER) 0);
|
||||||
|
if (!SQL_SUCCEEDED(retcode))
|
||||||
|
{
|
||||||
|
reportODBCError("SQLFetchScroll SQL_FETCH_FIRST", stmtHandle, SQL_HANDLE_STMT);
|
||||||
|
retcode = SQLSetStmtAttr(stmtHandle, SQL_ATTR_RETRIEVE_DATA, (SQLPOINTER) SQL_RD_ON, 0);
|
||||||
|
return ((int) myRecCnt);
|
||||||
} else {
|
} else {
|
||||||
|
//Fetch the first record's index
|
||||||
|
retcode = SQLGetStmtAttr(stmtHandle, SQL_ATTR_ROW_NUMBER, &firstRecNo, 0, 0);
|
||||||
|
if (SQL_SUCCEEDED(retcode))
|
||||||
|
{
|
||||||
|
//Inform first recno if in Debug mode and carry on
|
||||||
if (DB.IsDebug())
|
if (DB.IsDebug())
|
||||||
{
|
{
|
||||||
fprintf(stderr, "gb.db.odbc: Cannot do GetRecordCount()!\n");
|
fprintf(stderr, "gb.db.odbc.GetRecordCount: First recno=%d\n", (int) firstRecNo);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
//Could not fetch the first recno: Abort!
|
||||||
|
reportODBCError("SQLFetchScroll SQL_ATTR_ROW_NUMBER (first recno)", stmtHandle, SQL_HANDLE_STMT);
|
||||||
|
retcode = SQLSetStmtAttr(stmtHandle, SQL_ATTR_RETRIEVE_DATA, (SQLPOINTER) SQL_RD_ON, 0);
|
||||||
|
return ((int) myRecCnt);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//Advance the cursor to the last record.
|
||||||
|
retcode = SQLFetchScroll(stmtHandle, SQL_FETCH_LAST, (SQLINTEGER) 0);
|
||||||
|
if (SQL_SUCCEEDED(retcode))
|
||||||
|
{
|
||||||
|
|
||||||
|
//Fetch the last record's index
|
||||||
|
retcode = SQLGetStmtAttr(stmtHandle, SQL_ATTR_ROW_NUMBER, &lastRecNo, 0, 0);
|
||||||
|
if (SQL_SUCCEEDED(retcode))
|
||||||
|
{
|
||||||
|
//Set ret value
|
||||||
|
if (DB.IsDebug())
|
||||||
|
{
|
||||||
|
fprintf(stderr, "gb.db.odbc.GetRecordCount: Last recno=%d\n", (int) lastRecNo);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
} else {
|
||||||
|
reportODBCError("SQLGetStmtAttr SQL_ATTR_ROW_NUMBER (last recno)", stmtHandle, SQL_HANDLE_STMT);
|
||||||
|
}
|
||||||
|
|
||||||
|
//Return cursor to original row.
|
||||||
|
retcode = SQLFetchScroll(stmtHandle, SQL_FETCH_ABSOLUTE, (SQLINTEGER) formerRecIdx);
|
||||||
|
//Since we have set the "do not read data" statement attribute, this call (may) return
|
||||||
|
//code 100 (SQL_NO_DATA) but that's OK for our purposes of just counting rows.
|
||||||
|
if (!SQL_SUCCEEDED(retcode) && (retcode != SQL_NO_DATA))
|
||||||
|
{
|
||||||
|
snprintf(mssg, sizeof(mssg), "SQLFetchScroll SQL_FETCH_ABSOLUTE (code %d) (rec %d)",
|
||||||
|
(int)retcode, formerRecIdx);
|
||||||
|
reportODBCError(mssg, stmtHandle, SQL_HANDLE_STMT);
|
||||||
|
}
|
||||||
|
|
||||||
|
} else {
|
||||||
|
reportODBCError("SQLFetchScroll SQL_FETCH_LAST", stmtHandle, SQL_HANDLE_STMT);
|
||||||
|
}
|
||||||
|
|
||||||
|
//Tell ODBC we will be reading data now.
|
||||||
|
//SQL_ATTR_RETRIEVE_DATA = [SQL_RD_ON] | SQL_RD_OFF
|
||||||
|
retcode = SQLSetStmtAttr(stmtHandle, SQL_ATTR_RETRIEVE_DATA, (SQLPOINTER) SQL_RD_ON, 0);
|
||||||
|
if (!SQL_SUCCEEDED(retcode))
|
||||||
|
{
|
||||||
|
reportODBCError("SQLSetStmtAttr SQL_ATTR_RETRIEVE_DATA", stmtHandle, SQL_HANDLE_STMT);
|
||||||
|
}
|
||||||
|
|
||||||
|
myRecCnt = (lastRecNo - firstRecNo + 1);
|
||||||
|
if (DB.IsDebug())
|
||||||
|
{
|
||||||
|
fprintf(stderr, "gb.db.odbc.GetRecordCount: Record count=%d\n", (int) myRecCnt);
|
||||||
}
|
}
|
||||||
|
|
||||||
return ((int) myRecCnt);
|
return ((int) myRecCnt);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/* BM: Replaces malloc() and free() by GB.Alloc() and GB.Free() */
|
/* BM: Replaces malloc() and free() by GB.Alloc() and GB.Free() */
|
||||||
|
|
||||||
static void *my_malloc(size_t size)
|
static void *my_malloc(size_t size)
|
||||||
{
|
{
|
||||||
void *ptr;
|
void *ptr;
|
||||||
|
@ -461,7 +530,7 @@ fflush(stderr);
|
||||||
{
|
{
|
||||||
case SQL_BINARY:
|
case SQL_BINARY:
|
||||||
return GB_T_BOOLEAN;
|
return GB_T_BOOLEAN;
|
||||||
/*case INT8OID: */
|
/*case INT8OID: */
|
||||||
case SQL_NUMERIC:
|
case SQL_NUMERIC:
|
||||||
return GB_T_FLOAT;
|
return GB_T_FLOAT;
|
||||||
case SQL_DECIMAL:
|
case SQL_DECIMAL:
|
||||||
|
@ -471,7 +540,7 @@ fflush(stderr);
|
||||||
case SQL_SMALLINT:
|
case SQL_SMALLINT:
|
||||||
return GB_T_INTEGER;
|
return GB_T_INTEGER;
|
||||||
case SQL_BIGINT:
|
case SQL_BIGINT:
|
||||||
// New datatype bigint 64 bits
|
// New datatype bigint 64 bits
|
||||||
return GB_T_LONG;
|
return GB_T_LONG;
|
||||||
case SQL_FLOAT:
|
case SQL_FLOAT:
|
||||||
return GB_T_FLOAT;
|
return GB_T_FLOAT;
|
||||||
|
@ -486,7 +555,8 @@ fflush(stderr);
|
||||||
return GB_T_DATE;
|
return GB_T_DATE;
|
||||||
case SQL_LONGVARCHAR:
|
case SQL_LONGVARCHAR:
|
||||||
case SQL_VARBINARY:
|
case SQL_VARBINARY:
|
||||||
case SQL_LONGVARBINARY: // Data type for BLOB
|
case SQL_LONGVARBINARY:
|
||||||
|
// Data type for BLOB
|
||||||
return DB_T_BLOB;
|
return DB_T_BLOB;
|
||||||
case SQL_CHAR:
|
case SQL_CHAR:
|
||||||
default:
|
default:
|
||||||
|
@ -593,21 +663,65 @@ static const char *get_quote(void)
|
||||||
}
|
}
|
||||||
|
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
|
zxMarce: Added 20160928 to retrieve the connected Database (or 'Catalog')
|
||||||
|
name directly from unixODBC/driver.
|
||||||
|
Actually, this routine *should* be called once after every operation,
|
||||||
|
because SPs or even plaintext queries could change the Catalog (for example
|
||||||
|
by using the "USE <newcatalog>" command), but we don't have access to a
|
||||||
|
DB_DESC structure from the query-running code.
|
||||||
|
*****************************************************************************/
|
||||||
|
void GetConnectedDBName(DB_DESC *desc, ODBC_CONN *odbc)
|
||||||
|
{
|
||||||
|
|
||||||
|
SQLRETURN retcode;
|
||||||
|
SQLINTEGER charsNeeded = 0;
|
||||||
|
SQLTCHAR *dbName;
|
||||||
|
|
||||||
|
/*zxMarce: Attribute to fetch is SQL_ATTR_CURRENT_CATALOG
|
||||||
|
We call the function first with a NULL buffer pointer so as to
|
||||||
|
retrieve the necessary buffer size.
|
||||||
|
*/
|
||||||
|
retcode = SQLGetConnectAttrA(odbc->odbcHandle, SQL_ATTR_CURRENT_CATALOG,
|
||||||
|
NULL, (SQLINTEGER) 0,
|
||||||
|
(SQLINTEGER *) &charsNeeded
|
||||||
|
);
|
||||||
|
|
||||||
|
if (SQL_SUCCEEDED(retcode))
|
||||||
|
{
|
||||||
|
dbName = malloc(sizeof(SQLTCHAR) * charsNeeded++);
|
||||||
|
dbName[sizeof(SQLTCHAR) * charsNeeded++] = 0;
|
||||||
|
/*zxMarce: We call the function again, this time specifying a
|
||||||
|
hopefully big enough buffer for storing the catalog name.
|
||||||
|
*/
|
||||||
|
retcode = SQLGetConnectAttr(odbc->odbcHandle, SQL_ATTR_CURRENT_CATALOG,
|
||||||
|
dbName, charsNeeded,
|
||||||
|
&charsNeeded
|
||||||
|
);
|
||||||
|
GB.FreeString(&desc->name);
|
||||||
|
desc->name = GB.NewZeroString((char *)dbName);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (DB.IsDebug())
|
||||||
|
{
|
||||||
|
if (desc->name)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "gb.db.odbc.GetConnectedDBName: desc->name (%d chars):'%s'.\n", (int)charsNeeded, desc->name);
|
||||||
|
} else {
|
||||||
|
fprintf(stderr, "gb.db.odbc.GetConnectedDBName: desc->name: NULL.\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/*****************************************************************************
|
||||||
open_database()
|
open_database()
|
||||||
|
|
||||||
Connect to a database.
|
Connect to a database.
|
||||||
|
|
||||||
<desc> points at a structure describing each connection parameter.
|
<desc> points at a structure describing each connection parameter.
|
||||||
|
|
||||||
This function must return a database handle, or NULL if the connection
|
This function must return a database handle, or NULL if the connection
|
||||||
has failed.
|
has failed.
|
||||||
|
|
||||||
The name of the database can be NULL, meaning a default database.
|
The name of the database can be NULL, meaning a default database.
|
||||||
|
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
static int open_database(DB_DESC *desc, DB_DATABASE *db)
|
||||||
static int open_database(DB_DESC * desc, DB_DATABASE *db)
|
|
||||||
{
|
{
|
||||||
|
|
||||||
#ifdef ODBC_DEBUG_HEADER
|
#ifdef ODBC_DEBUG_HEADER
|
||||||
|
@ -617,11 +731,11 @@ fflush(stderr);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
//int V_OD_erg;
|
//int V_OD_erg;
|
||||||
SQLRETURN retcode;
|
SQLRETURN retcode;
|
||||||
ODBC_CONN *odbc;
|
ODBC_CONN *odbc;
|
||||||
bool hostIsAConnString;
|
bool hostIsAConnString;
|
||||||
char *host;
|
char *host;
|
||||||
char *user;
|
char *user;
|
||||||
|
|
||||||
host = desc->host;
|
host = desc->host;
|
||||||
if (!host)
|
if (!host)
|
||||||
|
@ -637,60 +751,68 @@ fflush(stderr);
|
||||||
odbc = (ODBC_CONN *)malloc(sizeof(ODBC_CONN));
|
odbc = (ODBC_CONN *)malloc(sizeof(ODBC_CONN));
|
||||||
odbc->odbcHandle = NULL;
|
odbc->odbcHandle = NULL;
|
||||||
odbc->odbcEnvHandle = NULL;
|
odbc->odbcEnvHandle = NULL;
|
||||||
|
odbc->dsn_name = NULL;
|
||||||
|
|
||||||
/* Allocate the Environment handle */
|
/* Allocate the Environment handle */
|
||||||
retcode = SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &odbc->odbcEnvHandle);
|
retcode = SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &odbc->odbcEnvHandle);
|
||||||
|
|
||||||
if ((retcode != SQL_SUCCESS) && (retcode != SQL_SUCCESS_WITH_INFO))
|
if (!SQL_SUCCEEDED(retcode))
|
||||||
{
|
{
|
||||||
free(odbc);
|
free(odbc);
|
||||||
GB.Error("Unable to allocate the environment handle");
|
GB.Error("Unable to allocate ODBC environment handle");
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Set the Envoronment attributes */
|
/* Set the Envoronment attributes */
|
||||||
retcode =SQLSetEnvAttr(odbc->odbcEnvHandle, SQL_ATTR_ODBC_VERSION,(void *) SQL_OV_ODBC3, 0);
|
retcode = SQLSetEnvAttr(odbc->odbcEnvHandle, SQL_ATTR_ODBC_VERSION,(void *) SQL_OV_ODBC3, 0);
|
||||||
|
|
||||||
if ((retcode != SQL_SUCCESS) && (retcode != SQL_SUCCESS_WITH_INFO))
|
if (!SQL_SUCCEEDED(retcode))
|
||||||
{
|
{
|
||||||
SQLFreeHandle(SQL_HANDLE_ENV, odbc->odbcEnvHandle);
|
SQLFreeHandle(SQL_HANDLE_ENV, odbc->odbcEnvHandle);
|
||||||
free(odbc);
|
free(odbc);
|
||||||
GB.Error("Unable to set the environment attributes");
|
GB.Error("Unable to set ODBC environment attributes");
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Allocate the Database Connection handle */
|
/* Allocate the Database Connection handle */
|
||||||
retcode = SQLAllocHandle(SQL_HANDLE_DBC, odbc->odbcEnvHandle, &odbc->odbcHandle);
|
retcode = SQLAllocHandle(SQL_HANDLE_DBC, odbc->odbcEnvHandle, &odbc->odbcHandle);
|
||||||
|
|
||||||
if ((retcode != SQL_SUCCESS) && (retcode != SQL_SUCCESS_WITH_INFO))
|
if (!SQL_SUCCEEDED(retcode))
|
||||||
{
|
{
|
||||||
SQLFreeHandle(SQL_HANDLE_ENV, odbc->odbcEnvHandle);
|
SQLFreeHandle(SQL_HANDLE_ENV, odbc->odbcEnvHandle);
|
||||||
free(odbc);
|
free(odbc);
|
||||||
GB.Error("Unable to allocate the ODBC handle");
|
GB.Error("Unable to allocate ODBC database handle");
|
||||||
return TRUE;
|
return TRUE;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
SQLSetConnectAttr(odbc->odbcHandle, SQL_ATTR_LOGIN_TIMEOUT, (SQLPOINTER)(intptr_t)db->timeout, 0);
|
/* 20170818 zxMarce: The following timeout was incorrectly set. The right timeout is the CONNECT
|
||||||
|
* timeout. The LOGIN timeout is actually used once the connection is established. Also took the
|
||||||
|
* opportunity to make sure ODBC uses cursors either from the Driver or the Driver Manager, thus
|
||||||
|
* almost ensuring cursors are available.
|
||||||
|
*/
|
||||||
|
//SQLSetConnectAttr(odbc->odbcHandle, SQL_ATTR_LOGIN_TIMEOUT, (SQLPOINTER)(intptr_t)db->timeout, 0);
|
||||||
|
SQLSetConnectAttr(odbc->odbcHandle, SQL_ATTR_CONNECTION_TIMEOUT, (SQLPOINTER)(intptr_t)db->timeout, 0);
|
||||||
|
SQLSetConnectAttr(odbc->odbcHandle, SQL_ATTR_ODBC_CURSORS, (SQLPOINTER)SQL_CUR_USE_IF_NEEDED, 0);
|
||||||
|
|
||||||
if (hostIsAConnString)
|
if (hostIsAConnString)
|
||||||
{
|
{
|
||||||
/* zxMarce: Connect to Database (desc->host is an ODBC Connection String) */
|
/* zxMarce: Connect to Database (desc->host is an ODBC Connection String) */
|
||||||
retcode = SQLDriverConnect(odbc->odbcHandle, 0, (SQLCHAR *)host, SQL_NTS, 0, 0, 0, SQL_DRIVER_NOPROMPT);
|
retcode = SQLDriverConnect(odbc->odbcHandle, 0, (SQLCHAR *)host, SQL_NTS, 0, 0, 0, SQL_DRIVER_NOPROMPT);
|
||||||
/* The last three zero params in the call above can be used to retrieve the actual connstring used,
|
/* The last three zero params in the call above can be used to retrieve the actual connstring used,
|
||||||
should unixODBC "complete" the passed ConnString with data from a matching defined DSN. Not
|
should unixODBC "complete" the passed ConnString with data from a matching defined DSN. Not
|
||||||
doing it here, but maybe useful to fill in the other Gambas Connection object properties (user,
|
doing it here, but maybe useful to fill in the other Gambas Connection object properties (user,
|
||||||
pass, etc) after parsing it. Also note that the ConnString MAY refer to a DSN, and include
|
pass, etc) after parsing it. Also note that the ConnString MAY refer to a DSN, and include
|
||||||
user/pass, if desired.
|
user/pass, if desired.
|
||||||
Example - ODBC-ConnString, all one line (must assign this to the Connection.Host property in
|
Example - ODBC-ConnString for FreeTDS, all one line (must assign this to the Connection.Host
|
||||||
Gambas code and then call Connection.Open):
|
property in Gambas code and then call Connection.Open):
|
||||||
"Driver=<driverSectionNameInODBCInst.Ini>;
|
"Driver=FreeTDS;
|
||||||
Server=<serverNameOrIP>;
|
TDS_Version=<useNormally'7.2'>;
|
||||||
Port=<serverTcpPort>;
|
Server=<serverNameOrIP>;
|
||||||
Database=<defaultDatabase>;
|
Port=<serverTcpPort>;
|
||||||
UId=<userName>;
|
Database=<defaultDatabase>"
|
||||||
Pwd=<password>;
|
UId=<userName>;
|
||||||
TDS_Version=<useNormally'7.2'>"
|
Pwd=<password>;
|
||||||
*/
|
*/
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
|
@ -698,10 +820,28 @@ fflush(stderr);
|
||||||
retcode = SQLConnect(odbc->odbcHandle, (SQLCHAR *)host, SQL_NTS, (SQLCHAR *)user, SQL_NTS, (SQLCHAR *) desc->password, SQL_NTS);
|
retcode = SQLConnect(odbc->odbcHandle, (SQLCHAR *)host, SQL_NTS, (SQLCHAR *)user, SQL_NTS, (SQLCHAR *) desc->password, SQL_NTS);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//zxMarce: Must bail out NOW if failed to connect, or nonsense errors will appear.
|
||||||
|
if (!SQL_SUCCEEDED(retcode))
|
||||||
|
{
|
||||||
|
throwODBCError((hostIsAConnString ? "SQLDriverConnect" : "SQLConnect"),
|
||||||
|
odbc->odbcHandle,
|
||||||
|
SQL_HANDLE_DBC
|
||||||
|
);
|
||||||
|
//GB.Error("Error connecting to database");
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
retcode = SQLSetConnectAttr(odbc->odbcHandle, SQL_ATTR_AUTOCOMMIT, (void *) SQL_AUTOCOMMIT_ON, SQL_NTS);
|
retcode = SQLSetConnectAttr(odbc->odbcHandle, SQL_ATTR_AUTOCOMMIT, (void *) SQL_AUTOCOMMIT_ON, SQL_NTS);
|
||||||
|
|
||||||
odbc->dsn_name = malloc(sizeof(char) * strlen(host));
|
/* desc->name is a pointer intended to point to the database name ('catalog', in
|
||||||
strcpy(odbc->dsn_name, host);
|
* MSSQL parlance) to which we are connected. But that is NOT the actual database
|
||||||
|
* name when we use a connstring. When we use a connstring, the '.Database' connection
|
||||||
|
* property would then have the whole connection string, so we tweak it here to make it
|
||||||
|
* point to the actual database (catalog?) name.
|
||||||
|
*/
|
||||||
|
//odbc->dsn_name = malloc(sizeof(char) * strlen(host));
|
||||||
|
//strcpy(odbc->dsn_name, host);
|
||||||
|
GetConnectedDBName(desc, odbc);
|
||||||
|
|
||||||
odbc->user_name = malloc(sizeof(char) * strlen(user));
|
odbc->user_name = malloc(sizeof(char) * strlen(user));
|
||||||
strcpy(odbc->user_name, user);
|
strcpy(odbc->user_name, user);
|
||||||
|
@ -709,11 +849,14 @@ fflush(stderr);
|
||||||
db->version = 3;
|
db->version = 3;
|
||||||
|
|
||||||
retcode = SQLGetFunctions(odbc->odbcHandle, SQL_API_SQLFETCHSCROLL, &odbc->FetchScroll_exist);
|
retcode = SQLGetFunctions(odbc->odbcHandle, SQL_API_SQLFETCHSCROLL, &odbc->FetchScroll_exist);
|
||||||
if ((retcode != SQL_SUCCESS) && (retcode != SQL_SUCCESS_WITH_INFO))
|
if (!SQL_SUCCEEDED(retcode))
|
||||||
{
|
{
|
||||||
GB.Error("Error calling the ODBC SQLGetFunction API");
|
throwODBCError("SQLGetFunctions SQL_API_SQLFETCHSCROLL",
|
||||||
|
odbc->odbcHandle,
|
||||||
|
SQL_HANDLE_DBC
|
||||||
|
);
|
||||||
|
//GB.Error("Error calling the ODBC SQLGetFunctions API");
|
||||||
return TRUE;
|
return TRUE;
|
||||||
//printf("ERROR calling the SQLGetFunction API\n");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* flags */
|
/* flags */
|
||||||
|
@ -748,11 +891,12 @@ fflush(stderr);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
ODBC_CONN *conn = (ODBC_CONN *)db->handle;
|
ODBC_CONN *conn = (ODBC_CONN *)db->handle;
|
||||||
|
//SQLRETURN retcode = 0;
|
||||||
|
|
||||||
if (conn->odbcHandle)
|
if (conn->odbcHandle)
|
||||||
SQLDisconnect(conn->odbcHandle);
|
SQLDisconnect(conn->odbcHandle);
|
||||||
else
|
else
|
||||||
GB.Error("ODBC module internal error");
|
GB.Error("ODBC module internal error disconnecting hDBC");
|
||||||
|
|
||||||
if (conn->odbcHandle)
|
if (conn->odbcHandle)
|
||||||
{
|
{
|
||||||
|
@ -760,7 +904,7 @@ fflush(stderr);
|
||||||
conn->odbcHandle = NULL;
|
conn->odbcHandle = NULL;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
GB.Error("ODBC module internal error");
|
GB.Error("ODBC module internal error freeing hDBC");
|
||||||
|
|
||||||
if (conn->odbcEnvHandle)
|
if (conn->odbcEnvHandle)
|
||||||
{
|
{
|
||||||
|
@ -768,19 +912,26 @@ fflush(stderr);
|
||||||
conn->odbcEnvHandle = NULL;
|
conn->odbcEnvHandle = NULL;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
GB.Error("ODBC module internal error");
|
GB.Error("ODBC module internal error freeing hENV");
|
||||||
|
|
||||||
if (conn->dsn_name)
|
if (conn->dsn_name)
|
||||||
|
{
|
||||||
free(conn->dsn_name);
|
free(conn->dsn_name);
|
||||||
|
conn->dsn_name = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
if (conn->user_name)
|
if (conn->user_name)
|
||||||
|
{
|
||||||
free(conn->user_name);
|
free(conn->user_name);
|
||||||
|
conn->user_name = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
if (conn)
|
if (conn)
|
||||||
{
|
{
|
||||||
free(conn);
|
free(conn);
|
||||||
db->handle = NULL;
|
db->handle = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -912,7 +1063,7 @@ static void query_get_param(int index, char **str, int *len, char quote)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Internal function to implement the query execution */
|
/* Internal function to implement the query execution */
|
||||||
static int do_query(DB_DATABASE *db, const char *error, ODBC_RESULT ** res, const char *qtemp, int nsubst, ...)
|
static int do_query(DB_DATABASE *db, const char *error, ODBC_RESULT **res, const char *qtemp, int nsubst, ...)
|
||||||
{
|
{
|
||||||
|
|
||||||
#ifdef ODBC_DEBUG_HEADER
|
#ifdef ODBC_DEBUG_HEADER
|
||||||
|
@ -923,8 +1074,8 @@ fflush(stderr);
|
||||||
|
|
||||||
va_list args;
|
va_list args;
|
||||||
ODBC_CONN *handle = (ODBC_CONN *)db->handle;
|
ODBC_CONN *handle = (ODBC_CONN *)db->handle;
|
||||||
SQLRETURN retcode= SQL_SUCCESS;
|
SQLRETURN retcode = SQL_SUCCESS;
|
||||||
ODBC_RESULT * odbcres;
|
ODBC_RESULT *odbcres;
|
||||||
const char *query;
|
const char *query;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
|
@ -942,63 +1093,67 @@ fflush(stderr);
|
||||||
query = qtemp;
|
query = qtemp;
|
||||||
|
|
||||||
if (DB.IsDebug())
|
if (DB.IsDebug())
|
||||||
fprintf(stderr, "gb.db.odbc: %p: %s\n", handle, query);
|
fprintf(stderr, "gb.db.odbc.do_query: res %p, dbc handle %p, query '%s'\n", res, handle, query);
|
||||||
|
|
||||||
GB.AllocZero(POINTER(&odbcres), sizeof(ODBC_RESULT));
|
GB.AllocZero(POINTER(&odbcres), sizeof(ODBC_RESULT));
|
||||||
|
|
||||||
/* Allocate the space for the result structure */
|
/* Allocate the space for the result structure */
|
||||||
|
|
||||||
retcode = SQLAllocHandle(SQL_HANDLE_STMT, handle->odbcHandle, &odbcres->odbcStatHandle);
|
retcode = SQLAllocHandle(SQL_HANDLE_STMT, handle->odbcHandle, &odbcres->odbcStatHandle);
|
||||||
if ((retcode != SQL_SUCCESS) && (retcode != SQL_SUCCESS_WITH_INFO))
|
if ((retcode != SQL_SUCCESS) && (retcode != SQL_SUCCESS_WITH_INFO))
|
||||||
{
|
{
|
||||||
GB.Error("Cannot allocate statement handle");
|
//GB.Error("Cannot allocate statement handle");
|
||||||
|
throwODBCError("SQLAllocHandle", handle->odbcHandle, SQL_HANDLE_DBC);
|
||||||
return retcode;
|
return retcode;
|
||||||
}
|
}
|
||||||
|
|
||||||
retcode = SQLSetStmtAttr(odbcres->odbcStatHandle, SQL_ATTR_CURSOR_SCROLLABLE, (SQLPOINTER) SQL_SCROLLABLE, 0);
|
retcode = SQLSetStmtAttr(odbcres->odbcStatHandle, SQL_ATTR_CURSOR_SCROLLABLE, (SQLPOINTER) SQL_SCROLLABLE, 0);
|
||||||
if ((retcode != SQL_SUCCESS) && (retcode != SQL_SUCCESS_WITH_INFO))
|
//if ((retcode != SQL_SUCCESS) && (retcode != SQL_SUCCESS_WITH_INFO))
|
||||||
{
|
//{
|
||||||
odbcres->Cursor_Scrollable = SQL_FALSE;
|
// odbcres->Cursor_Scrollable = SQL_FALSE;
|
||||||
}
|
//}
|
||||||
else odbcres->Cursor_Scrollable = SQL_TRUE;
|
//else odbcres->Cursor_Scrollable = SQL_TRUE;
|
||||||
|
odbcres->Cursor_Scrollable = ((retcode != SQL_SUCCESS) && (retcode != SQL_SUCCESS_WITH_INFO)) ? SQL_FALSE : SQL_TRUE;
|
||||||
odbcres->Function_exist = handle->FetchScroll_exist;
|
odbcres->Function_exist = handle->FetchScroll_exist;
|
||||||
|
|
||||||
/* Execute the query */
|
/* Execute the query */
|
||||||
retcode = SQLExecDirect(odbcres->odbcStatHandle, (SQLCHAR *) query, SQL_NTS);
|
retcode = SQLExecDirect(odbcres->odbcStatHandle, (SQLCHAR *) query, SQL_NTS);
|
||||||
if ((retcode != SQL_SUCCESS) && (retcode != SQL_SUCCESS_WITH_INFO))
|
if ((retcode != SQL_SUCCESS) && (retcode != SQL_SUCCESS_WITH_INFO) && (retcode != SQL_NO_DATA))
|
||||||
{
|
{
|
||||||
|
if (DB.IsDebug())
|
||||||
|
fprintf(stderr, "gb.db.odbc.do_query: SQLExecDirect() returned code %d\n", (int)retcode);
|
||||||
|
throwODBCError("SQLExecDirect", odbcres->odbcStatHandle, SQL_HANDLE_STMT);
|
||||||
SQLFreeHandle(SQL_HANDLE_STMT, odbcres->odbcStatHandle);
|
SQLFreeHandle(SQL_HANDLE_STMT, odbcres->odbcStatHandle);
|
||||||
GB.Error("Error while executing the statement");
|
//GB.Error("Error while executing the statement");
|
||||||
return retcode;
|
return retcode;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (res)
|
if (res)
|
||||||
{
|
{
|
||||||
/*retcode = SQLRowCount(odbcres->odbcStatHandle, &odbcres->count);
|
|
||||||
if (retcode != SQL_SUCCESS && retcode != SQL_SUCCESS_WITH_INFO)
|
if (retcode == SQL_NO_DATA)
|
||||||
|
{
|
||||||
|
odbcres->count = 0;
|
||||||
|
retcode = SQL_SUCCESS;
|
||||||
|
}
|
||||||
|
else
|
||||||
{
|
{
|
||||||
SQLFreeHandle(SQL_HANDLE_STMT, odbcres->odbcStatHandle);
|
odbcres->count = GetRecordCount(odbcres->odbcStatHandle, odbcres->Cursor_Scrollable);
|
||||||
GB.Error("Unable to retrieve row count");
|
}
|
||||||
return retcode;
|
|
||||||
}*/
|
|
||||||
odbcres->count = GetRecordCount(odbcres->odbcStatHandle, odbcres->Cursor_Scrollable);
|
|
||||||
if (DB.IsDebug())
|
|
||||||
fprintf(stderr, "gb.db.odbc: -> %d rows\n", (int)odbcres->count);
|
|
||||||
*res = odbcres;
|
*res = odbcres;
|
||||||
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
|
||||||
SQLFreeHandle(SQL_HANDLE_STMT, odbcres->odbcStatHandle);
|
SQLFreeHandle(SQL_HANDLE_STMT, odbcres->odbcStatHandle);
|
||||||
GB.Free(POINTER(&odbcres));
|
GB.Free(POINTER(&odbcres));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return retcode;
|
return retcode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/* Internal function - free the result structure create to allocate the result row */
|
/* Internal function - free the result structure create to allocate the result row */
|
||||||
static void query_free_result(ODBC_RESULT * result)
|
static void query_free_result(ODBC_RESULT * result)
|
||||||
{
|
{
|
||||||
|
@ -1013,14 +1168,14 @@ fflush(stderr);
|
||||||
current = (ODBC_FIELDS *) result->fields;
|
current = (ODBC_FIELDS *) result->fields;
|
||||||
next = (ODBC_FIELDS *) result->fields;
|
next = (ODBC_FIELDS *) result->fields;
|
||||||
|
|
||||||
while(current!=NULL)
|
while(current != NULL)
|
||||||
{
|
{
|
||||||
next = (ODBC_FIELDS *) current->next;
|
next = (ODBC_FIELDS *) current->next;
|
||||||
|
|
||||||
if (current->fieldata != NULL)
|
if (current->fieldata != NULL)
|
||||||
{
|
{
|
||||||
free(current->fieldata); //091107
|
free(current->fieldata); //091107
|
||||||
current->fieldata = NULL; //091107
|
current->fieldata = NULL; //091107
|
||||||
}
|
}
|
||||||
|
|
||||||
if(current != NULL)
|
if(current != NULL)
|
||||||
|
@ -1033,10 +1188,11 @@ fflush(stderr);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if(result != NULL){
|
if(result != NULL)
|
||||||
free(result);
|
{
|
||||||
result = NULL; //091107
|
free(result);
|
||||||
}
|
result = NULL; //091107
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1055,8 +1211,6 @@ fflush(stderr);
|
||||||
<result> can be NULL, when we don't care getting the result.
|
<result> can be NULL, when we don't care getting the result.
|
||||||
|
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
|
|
||||||
static int exec_query(DB_DATABASE *db, const char *query, DB_RESULT * result, const char *err)
|
static int exec_query(DB_DATABASE *db, const char *query, DB_RESULT * result, const char *err)
|
||||||
{
|
{
|
||||||
#ifdef ODBC_DEBUG_HEADER
|
#ifdef ODBC_DEBUG_HEADER
|
||||||
|
@ -1064,7 +1218,7 @@ fprintf(stderr,"[ODBC][%s][%d]\n",__FILE__,__LINE__);
|
||||||
fprintf(stderr,"\texec_query\n");
|
fprintf(stderr,"\texec_query\n");
|
||||||
fflush(stderr);
|
fflush(stderr);
|
||||||
#endif
|
#endif
|
||||||
return do_query(db, err, (ODBC_RESULT **) result, query, 0);
|
return do_query(db, err, (ODBC_RESULT **) result, query, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int get_num_columns(ODBC_RESULT *result)
|
static int get_num_columns(ODBC_RESULT *result)
|
||||||
|
@ -1230,13 +1384,14 @@ fflush(stderr);
|
||||||
|
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
static void query_release(DB_RESULT result, DB_INFO * info)
|
static void query_release(DB_RESULT result, DB_INFO *info)
|
||||||
{
|
{
|
||||||
#ifdef ODBC_DEBUG_HEADER
|
#ifdef ODBC_DEBUG_HEADER
|
||||||
fprintf(stderr,"[ODBC][%s][%d]\n",__FILE__,__LINE__);
|
fprintf(stderr,"[ODBC][%s][%d]\n",__FILE__,__LINE__);
|
||||||
fprintf(stderr,"\tquery_release\n");
|
fprintf(stderr,"\tquery_release\n");
|
||||||
fflush(stderr);
|
fflush(stderr);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
ODBC_RESULT *res = (ODBC_RESULT *) result;
|
ODBC_RESULT *res = (ODBC_RESULT *) result;
|
||||||
|
|
||||||
/*if (res != NULL)*/ //query_free_result(res);
|
/*if (res != NULL)*/ //query_free_result(res);
|
||||||
|
@ -1273,7 +1428,7 @@ fflush(stderr);
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
|
|
||||||
static int query_fill(DB_DATABASE *db, DB_RESULT result, int pos, GB_VARIANT_VALUE * buffer, int next)
|
static int query_fill(DB_DATABASE *db, DB_RESULT result, int pos, GB_VARIANT_VALUE * buffer, int next)
|
||||||
{
|
{
|
||||||
ODBC_RESULT *res = (ODBC_RESULT *) result;
|
ODBC_RESULT *res = (ODBC_RESULT *) result;
|
||||||
GB_VARIANT value;
|
GB_VARIANT value;
|
||||||
|
@ -1458,17 +1613,17 @@ fflush(stderr);
|
||||||
|
|
||||||
while (i < field )
|
while (i < field )
|
||||||
{
|
{
|
||||||
if (cfield->next== NULL)
|
if (cfield->next == NULL)
|
||||||
{
|
{
|
||||||
GB.Error("ODBC module :Internal error1");
|
GB.Error("ODBC module: Internal error 1");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
cfield=(ODBC_FIELDS *) cfield->next;
|
cfield=(ODBC_FIELDS *) cfield->next;
|
||||||
|
|
||||||
if (cfield== NULL)
|
if (cfield == NULL)
|
||||||
{
|
{
|
||||||
GB.Error("ODBC module :Internal error2");
|
GB.Error("ODBC module: Internal error 2");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1477,7 +1632,7 @@ fflush(stderr);
|
||||||
|
|
||||||
if (i > field)
|
if (i > field)
|
||||||
{
|
{
|
||||||
GB.Error("ODBC module : Internal error");
|
GB.Error("ODBC module: Internal error");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1489,9 +1644,9 @@ fflush(stderr);
|
||||||
|
|
||||||
DB.Query.Init();
|
DB.Query.Init();
|
||||||
|
|
||||||
retcode = SQLGetData(res->odbcStatHandle,field+1,SQL_C_BINARY , blob->data,blob->length, &strlen);
|
retcode = SQLGetData(res->odbcStatHandle, field+1, SQL_C_BINARY, blob->data, blob->length, &strlen);
|
||||||
|
|
||||||
if (retcode != SQL_SUCCESS && retcode != SQL_SUCCESS_WITH_INFO)
|
if ((retcode != SQL_SUCCESS) && (retcode != SQL_SUCCESS_WITH_INFO))
|
||||||
{
|
{
|
||||||
GB.Error("Unable to retrieve blob data");
|
GB.Error("Unable to retrieve blob data");
|
||||||
free(blob->data);
|
free(blob->data);
|
||||||
|
@ -1499,9 +1654,7 @@ fflush(stderr);
|
||||||
blob->data = NULL;
|
blob->data = NULL;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
blob->data = NULL; //
|
blob->data = NULL; //
|
||||||
blob->length = 0;
|
blob->length = 0;
|
||||||
return;
|
return;
|
||||||
|
@ -1550,8 +1703,7 @@ fflush(stderr);
|
||||||
|
|
||||||
ODBC_RESULT *res = (ODBC_RESULT *) result;
|
ODBC_RESULT *res = (ODBC_RESULT *) result;
|
||||||
|
|
||||||
SQLDescribeCol(res->odbcStatHandle, field + 1, colname, sizeof(colname),
|
SQLDescribeCol(res->odbcStatHandle, field + 1, colname, sizeof(colname), &colnamelen, &coltype, &precision, &scale, NULL);
|
||||||
&colnamelen, &coltype, &precision, &scale, NULL);
|
|
||||||
//colnamer = malloc(sizeof(char) * strlen((char *) colname) + 1);
|
//colnamer = malloc(sizeof(char) * strlen((char *) colname) + 1);
|
||||||
strcpy(_buffer, (char *) colname);
|
strcpy(_buffer, (char *) colname);
|
||||||
return _buffer;
|
return _buffer;
|
||||||
|
@ -1925,7 +2077,7 @@ fflush(stderr);
|
||||||
|
|
||||||
current = fieldstr;
|
current = fieldstr;
|
||||||
|
|
||||||
retcode=SQLNumResultCols(statHandle2, &colsNum);
|
retcode=SQLNumResultCols(statHandle2, &colsNum);
|
||||||
|
|
||||||
SQLFreeHandle(SQL_HANDLE_STMT, statHandle2);
|
SQLFreeHandle(SQL_HANDLE_STMT, statHandle2);
|
||||||
|
|
||||||
|
@ -1934,14 +2086,12 @@ retcode=SQLNumResultCols(statHandle2, &colsNum);
|
||||||
|
|
||||||
retcode = SQLAllocHandle(SQL_HANDLE_STMT, han->odbcHandle, &statHandle);
|
retcode = SQLAllocHandle(SQL_HANDLE_STMT, han->odbcHandle, &statHandle);
|
||||||
|
|
||||||
|
|
||||||
if ((retcode != SQL_SUCCESS) && (retcode != SQL_SUCCESS_WITH_INFO))
|
if ((retcode != SQL_SUCCESS) && (retcode != SQL_SUCCESS_WITH_INFO))
|
||||||
{
|
{
|
||||||
return retcode;
|
return retcode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
if (!SQL_SUCCEEDED(nReturn = SQLPrimaryKeys(statHandle, 0, 0, 0, SQL_NTS, (SQLCHAR *)table, SQL_NTS)))
|
if (!SQL_SUCCEEDED(nReturn = SQLPrimaryKeys(statHandle, 0, 0, 0, SQL_NTS, (SQLCHAR *)table, SQL_NTS)))
|
||||||
{
|
{
|
||||||
|
|
||||||
|
@ -1949,9 +2099,6 @@ retcode=SQLNumResultCols(statHandle2, &colsNum);
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
retcode=SQLNumResultCols(statHandle, &colsNum);
|
retcode=SQLNumResultCols(statHandle, &colsNum);
|
||||||
|
|
||||||
i = 0;
|
i = 0;
|
||||||
|
@ -1959,8 +2106,6 @@ retcode=SQLNumResultCols(statHandle2, &colsNum);
|
||||||
while (SQL_SUCCEEDED(SQLFetch(statHandle)))
|
while (SQL_SUCCEEDED(SQLFetch(statHandle)))
|
||||||
{
|
{
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
if (!SQL_SUCCEEDED(SQLGetData (statHandle, 4, SQL_C_CHAR, szColumnName, sizeof(szColumnName), 0)))
|
if (!SQL_SUCCEEDED(SQLGetData (statHandle, 4, SQL_C_CHAR, szColumnName, sizeof(szColumnName), 0)))
|
||||||
strcpy((char *) szColumnName, "Unknown");
|
strcpy((char *) szColumnName, "Unknown");
|
||||||
|
|
||||||
|
@ -1987,31 +2132,29 @@ retcode=SQLNumResultCols(statHandle2, &colsNum);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
GB.Alloc(POINTER(&info->index), sizeof(int) * i);
|
GB.Alloc(POINTER(&info->index), sizeof(int) * i);
|
||||||
info->nindex = i;
|
info->nindex = i;
|
||||||
|
|
||||||
|
|
||||||
SQLFreeHandle(SQL_HANDLE_STMT, statHandle);
|
SQLFreeHandle(SQL_HANDLE_STMT, statHandle);
|
||||||
|
|
||||||
for (n = 0; n < i; n++)
|
for (n = 0; n < i; n++)
|
||||||
info->index[n] = inx[n];
|
info->index[n] = inx[n];
|
||||||
|
|
||||||
|
|
||||||
free(res);
|
free(res);
|
||||||
|
|
||||||
while(current != NULL){
|
while(current != NULL){
|
||||||
if (current->next != NULL){
|
if (current->next != NULL)
|
||||||
fieldstr = (ODBC_FIELDS *)current->next ;
|
{
|
||||||
free (current);
|
fieldstr = (ODBC_FIELDS *)current->next ;
|
||||||
|
free (current);
|
||||||
current=fieldstr;
|
current=fieldstr;
|
||||||
}else {
|
}
|
||||||
free (current);
|
else
|
||||||
current =NULL;
|
{
|
||||||
|
free (current);
|
||||||
|
current =NULL;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
@ -2399,7 +2542,7 @@ static int table_delete(DB_DATABASE *db, const char *table)
|
||||||
|
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
static int table_create(DB_DATABASE *db, const char *table, DB_FIELD * fields, char **primary, const char *type_not_used)
|
static int table_create(DB_DATABASE *db, const char *table, DB_FIELD *fields, char **primary, const char *type_not_used)
|
||||||
{
|
{
|
||||||
#ifdef ODBC_DEBUG_HEADER
|
#ifdef ODBC_DEBUG_HEADER
|
||||||
fprintf(stderr,"[ODBC][%s][%d]\n",__FILE__,__LINE__);
|
fprintf(stderr,"[ODBC][%s][%d]\n",__FILE__,__LINE__);
|
||||||
|
@ -2426,7 +2569,7 @@ fflush(stderr);
|
||||||
else
|
else
|
||||||
comma = TRUE;
|
comma = TRUE;
|
||||||
|
|
||||||
DB.Query.Add(fp->name);
|
DB.Query.Add(fp->name);
|
||||||
|
|
||||||
//AB autoincrement field is mapped to Integer because this is Database dependent
|
//AB autoincrement field is mapped to Integer because this is Database dependent
|
||||||
if (fp->type == DB_T_SERIAL)
|
if (fp->type == DB_T_SERIAL)
|
||||||
|
@ -3125,7 +3268,7 @@ static int user_list(DB_DATABASE *db, char ***users)
|
||||||
|
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
static int user_info(DB_DATABASE *db, const char *name, DB_USER * info)
|
static int user_info(DB_DATABASE *db, const char *name, DB_USER *info)
|
||||||
{
|
{
|
||||||
//GB.Error("ODBC does not implement this function");
|
//GB.Error("ODBC does not implement this function");
|
||||||
return TRUE;
|
return TRUE;
|
||||||
|
|
Loading…
Reference in a new issue