gambas-source-code/gb.db.sqlite3/src/sqlitedataset.cpp
Benoît Minisini a8915f2700 [GB.DB.SQLITE3]
* BUG: Handle SQLITE_BUSY error for all requests, even SELECT queries.


git-svn-id: svn://localhost/gambas/trunk@5544 867c0c6c-44f3-4631-809d-bfa615b0a4ec
2013-02-12 01:33:38 +00:00

1084 lines
24 KiB
C++

/***************************************************************************
sqlitedataset.cpp
(c) 2000-2012 Benoît Minisini <gambas@users.sourceforge.net>
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
the Free Software Foundation; either version 2, or (at your option)
any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
MA 02110-1301, USA.
***************************************************************************/
/**********************************************************************
* Copyright (c) 2002, Leo Seib, Hannover
*
* Project:SQLiteDataset C++ Dynamic Library
* Module: SQLiteDataset class realisation file
* Author: Leo Seib E-Mail: lev@almaty.pointstrike.net
* Begin: 5/04/2002
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*
**********************************************************************/
#include <stdio.h>
#include <ctype.h>
#include <stdarg.h>
#include <time.h>
#include <iostream>
#include <string.h>
#include <math.h>
#include "sqlitedataset.h"
#include <unistd.h>
#include "gambas.h"
/**************************************************************/
// BM: my own sqlite3_exec(), that pass the statement pointer to the callback
typedef
int (*my_sqlite3_callback)(void *, int, char **, char **, sqlite3_stmt *);
static int my_sqlite3_exec(
sqlite3 *db, /* The database on which the SQL executes */
const char *zSql, /* The SQL to be executed */
my_sqlite3_callback xCallback, /* Invoke this callback routine */
void *pArg, /* First argument to xCallback() */
char **pzErrMsg /* Write error messages here */
)
{
int rc = SQLITE_OK;
const char *zLeftover;
sqlite3_stmt *pStmt = 0;
char **azCols = 0;
int nRetry = 0;
//int nChange = 0;
int nCallback;
if( zSql==0 ) return SQLITE_OK;
while( (rc==SQLITE_OK || (rc==SQLITE_SCHEMA && (++nRetry)<2)) && zSql[0] ){
int nCol;
char **azVals = 0;
pStmt = 0;
rc = sqlite3_prepare(db, zSql, -1, &pStmt, &zLeftover);
if( rc!=SQLITE_OK ){
if( pStmt ) sqlite3_finalize(pStmt);
continue;
}
if( !pStmt ){
/* this happens for a comment or white-space */
zSql = zLeftover;
continue;
}
//db->nChange += nChange;
nCallback = 0;
nCol = sqlite3_column_count(pStmt);
if (nCol == 0)
goto exec_out;
//azCols = sqliteMalloc(2*nCol*sizeof(const char *));
GB.Alloc(POINTER(&azCols), 2*nCol*sizeof(const char *));
if( nCol && !azCols ){
rc = SQLITE_NOMEM;
goto exec_out;
}
while( 1 ){
int i;
rc = sqlite3_step(pStmt);
/* Invoke the callback function if required */
if( xCallback && (SQLITE_ROW==rc ||
//(SQLITE_DONE==rc && !nCallback && db->flags&SQLITE_NullCallback)) ){
(SQLITE_DONE==rc && !nCallback && 1)) ){
if( 0==nCallback ){
for(i=0; i<nCol; i++){
azCols[i] = (char *)sqlite3_column_name(pStmt, i);
}
nCallback++;
}
if( rc==SQLITE_ROW ){
azVals = &azCols[nCol];
for(i=0; i<nCol; i++){
if (sqlite3_column_type(pStmt, i) == SQLITE_BLOB)
azVals[i] = (char *)sqlite3_column_blob(pStmt, i);
else
azVals[i] = (char *)sqlite3_column_text(pStmt, i);
}
}
if( xCallback(pArg, nCol, azVals, azCols, pStmt) ){
rc = SQLITE_ABORT;
goto exec_out;
}
}
if( rc!=SQLITE_ROW ){
rc = sqlite3_finalize(pStmt);
pStmt = 0;
//if( db->pVdbe==0 ){
// nChange = db->nChange;
//}
if( rc!=SQLITE_SCHEMA ){
nRetry = 0;
zSql = zLeftover;
while( isspace((unsigned char)zSql[0]) ) zSql++;
}
break;
}
}
GB.Free(POINTER(&azCols));
azCols = 0;
}
exec_out:
if( pStmt ) sqlite3_finalize(pStmt);
if( azCols )
GB.Free(POINTER(&azCols));
//if( sqlite3_malloc_failed ){
// rc = SQLITE_NOMEM;
//}
// GB.Free(POINTER(pzErrMsg));
//
// if( rc!=SQLITE_OK && rc==sqlite3_errcode(db) && pzErrMsg ){
// //*pzErrMsg = malloc(1+strlen(sqlite3_errmsg(db)));
// GB.Alloc(POINTER(pzErrMsg), 1+strlen(sqlite3_errmsg(db)));
// if( *pzErrMsg ){
// strcpy(*pzErrMsg, sqlite3_errmsg(db));
// }
// }
return rc;
}
//************* Callback function ***************************
static int callback(void *res_ptr, int ncol, char **reslt, char **cols, sqlite3_stmt *stmt)
{
/* NG: Type definition */
// typedef vector<string> Tables;
Tables tables;
Tables::iterator it;
char *item;
char *table;
result_set *r = (result_set *) res_ptr; //dynamic_cast<result_set*>(res_ptr);
int sz = r->records.size();
//if (reslt == NULL ) cout << "EMPTY!!!\n";
if (!r->record_header.size())
{
/*IF*/ for (int i = 0; i < ncol; i++)
{
item = strchr(cols[i], (int) '.');
if (!item)
{ /* Field does not include table info */
item = cols[i];
r->record_header[i].name = item; //NG
table = NULL;
r->record_header[i].field_table = "";
r->record_header[i].type = ft_String; //default type to string
}
else
{
//table = strndup(cols[i], strchr(cols[i], (int)'.') - cols[i]);
table = GB.NewString(cols[i], strchr(cols[i], (int) '.') - cols[i]);
// table = strdup(reslt[0]);
r->record_header[i].name = item + 1;
r->record_header[i].field_table = table;
r->record_header[i].type = ft_String; //default type to string
}
if (!table)
{
/* Field does not contain table info,
* so let's default to string. This
* has probably happened because aliases
* are being used */
}
else
{
/* Check Table Name and add to list */
bool TableRegistered = false;
for (it = tables.begin(); it != tables.end(); it++)
{
if (strcmp((*it).data(), table) == 0)
TableRegistered = true;
}
if (TableRegistered == false)
{
tables.push_back(table);
}
}
GB.FreeString(&table); //from strdup
}
SetFieldType(r, tables); // Set all the field types
for (int i = 0; i < ncol; i++)
{
/* Should table name be included in field name */
if (tables.size() > 1)
{
r->record_header[i].name = cols[i];
}
/* if (tables.size() < 1){
r->record_header[i].type = ft_String;//Where type cannot be found
//default to string
}*/
}
} /* IF close */
//sql_record rec;
if (reslt != NULL)
{
for (int i = 0; i < ncol; i++)
{
//fprintf(stderr, "callback: [%d] %s: %s\n", i, cols[i], reslt[i]);
if (reslt[i] == NULL)
{
r->records[sz][i].set_isNull(r->record_header[i].type);
}
else
{
switch (r->record_header[i].type)
{
case ft_Blob:
if (stmt)
r->records[sz][i].set_asBlob(reslt[i], sqlite3_column_bytes(stmt, i));
break;
default:
r->records[sz][i].set_asString(reslt[i], r->record_header[i].type);
}
}
}
}
//printf("Fsz: [%i]\n",r->record_header.size());
//printf("Recs: [%i]\n",r->records.size());
//cout << " val |" <<reslt<<"|"<<cols<<"|"<<"\n\n";
return 0;
}
static int old_callback(void *res_ptr, int ncol, char **reslt, char **cols)
{
return callback(res_ptr, ncol, reslt, cols, NULL);
}
//************* SqliteDatabase implementation ***************
SqliteDatabase::SqliteDatabase()
{
active = false;
_in_transaction = false; // for transaction
error = "Unknown database error"; //S_NO_CONNECTION;
host = "";
port = "";
db = ":memory:";
login = "";
passwd = "";
}
SqliteDatabase::~SqliteDatabase()
{
disconnect();
}
Dataset *SqliteDatabase::CreateDataset() const
{
return new SqliteDataset((SqliteDatabase *) this);
}
int SqliteDatabase::status(void)
{
if (active == false)
return DB_CONNECTION_NONE;
return DB_CONNECTION_OK;
}
int SqliteDatabase::setErr(int err_code)
{
last_err = err_code;
switch (err_code)
{
case SQLITE_OK:
error = "Successful result";
break;
case SQLITE_ERROR:
error = "SQL error or missing database";
break;
case SQLITE_INTERNAL:
error = "Internal logic error - Report this error on the mailing-list at sqlite.org";
break;
case SQLITE_PERM:
error = "Access permission denied";
break;
case SQLITE_ABORT:
error = "Callback routine requested an abort";
break;
case SQLITE_BUSY:
error = "The database file is locked";
break;
case SQLITE_LOCKED:
error = "A table in the database is locked";
break;
case SQLITE_NOMEM:
error = "Out of memory";
break;
case SQLITE_READONLY:
error = "Attempt to write a readonly database";
break;
case SQLITE_INTERRUPT:
error = "Operation terminated by sqlite_interrupt()";
break;
case SQLITE_IOERR:
error = "Some kind of disk I/O error occurred";
break;
case SQLITE_CORRUPT:
error = "The database disk image is malformed";
break;
case SQLITE_NOTFOUND:
error = "(Internal Only) Table or record not found";
break;
case SQLITE_FULL:
error = "Insertion failed because database is full";
break;
case SQLITE_CANTOPEN:
error = "Unable to open the database file";
break;
case SQLITE_PROTOCOL:
error = "Database lock protocol error";
break;
case SQLITE_EMPTY:
error = "(Internal Only) Database table is empty";
break;
case SQLITE_SCHEMA:
error = "The database schema changed";
break;
case SQLITE_TOOBIG:
error = "Too much data for one row of a table";
break;
case SQLITE_CONSTRAINT:
error = "Abort due to constraint violation";
break;
case SQLITE_MISMATCH:
error = "Data type mismatch";
break;
default:
error = "Undefined SQLite error";
}
return err_code;
}
const char *SqliteDatabase::getErrorMsg()
{
return error.c_str();
}
int SqliteDatabase::connect()
{
disconnect();
if (sqlite3_open(db.c_str(), &conn) == SQLITE_OK)
{
//cout << "Connected!\n";
//char *err = NULL;
if (setErr
(sqlite3_exec
(getHandle(), "PRAGMA empty_result_callbacks=ON", NULL, NULL,
NULL)) != SQLITE_OK)
{
GB.Error(getErrorMsg());
}
active = true;
/* NG 29/12/2005 - 3.2.1 introduced a problem with columns names
* which is resolved by setting short columns off first */
if (setErr
(sqlite3_exec
(getHandle(), "PRAGMA short_column_names=OFF", NULL, NULL,
NULL)) != SQLITE_OK)
{ //NG
GB.Error(getErrorMsg()); //NG
}
if (setErr
(sqlite3_exec
(getHandle(), "PRAGMA full_column_names=ON", NULL, NULL,
NULL)) != SQLITE_OK)
{ //NG
GB.Error(getErrorMsg()); //NG
}
return DB_CONNECTION_OK;
sqlite3_close(conn);
}
return DB_CONNECTION_NONE;
};
void SqliteDatabase::disconnect(void)
{
if (active == false)
return;
sqlite3_close(conn);
active = false;
};
int SqliteDatabase::create()
{
return connect();
};
int SqliteDatabase::drop()
{
if (active == false)
return DB_ERROR;
disconnect();
if (!unlink(db.c_str()))
return DB_ERROR;
return DB_COMMAND_OK;
};
long SqliteDatabase::nextid(const char *sname)
{
if (!active)
return DB_UNEXPECTED_RESULT;
int id;
result_set res;
char sqlcmd[512];
sprintf(sqlcmd, "select nextid from %s where seq_name = '%s'",
sequence_table.c_str(), sname);
res.conn = getHandle(); //NG
if ((last_err =
my_sqlite3_exec(getHandle(), sqlcmd, &callback, &res, NULL)) != SQLITE_OK)
{
return DB_UNEXPECTED_RESULT;
}
if (res.records.size() == 0)
{
id = 1;
sprintf(sqlcmd, "insert into %s (nextid,seq_name) values (%d,'%s')",
sequence_table.c_str(), id, sname);
if ((last_err =
sqlite3_exec(conn, sqlcmd, NULL, NULL, NULL)) != SQLITE_OK)
return DB_UNEXPECTED_RESULT;
return id;
}
else
{
id = res.records[0][0].get_asInteger() + 1;
sprintf(sqlcmd, "update %s set nextid=%d where seq_name = '%s'",
sequence_table.c_str(), id, sname);
if ((last_err =
sqlite3_exec(conn, sqlcmd, NULL, NULL, NULL)) != SQLITE_OK)
return DB_UNEXPECTED_RESULT;
return id;
}
}
// methods for transactions
// ---------------------------------------------
void SqliteDatabase::start_transaction()
{
if (active)
{
last_err = sqlite3_exec(conn, "begin", NULL, NULL, NULL);
_in_transaction = true;
}
}
void SqliteDatabase::commit_transaction()
{
if (active)
{
last_err = sqlite3_exec(conn, "commit", NULL, NULL, NULL);
_in_transaction = false;
}
}
void SqliteDatabase::rollback_transaction()
{
if (active)
{
last_err = sqlite3_exec(conn, "rollback", NULL, NULL, NULL);
_in_transaction = false;
}
}
//************* SqliteDataset implementation ***************
SqliteDataset::SqliteDataset():Dataset()
{
haveError = false;
db = NULL;
// errmsg = NULL;
}
SqliteDataset::SqliteDataset(SqliteDatabase * newDb):Dataset(newDb)
{
haveError = false;
db = newDb;
// errmsg = NULL;
}
SqliteDataset::~SqliteDataset()
{
/* if (errmsg)
sqlite3_free_table(&errmsg);*/
}
//--------- protected functions implementation -----------------//
sqlite3 *SqliteDataset::handle()
{
if (db != NULL)
{
return dynamic_cast < SqliteDatabase * >(db)->getHandle();
}
else
return NULL;
}
void SqliteDataset::make_query(StringList & _sql)
{
string query;
try
{
if (autocommit)
db->start_transaction();
if (db == NULL)
GB.Error("No database connection");
//close();
for (list < string >::iterator i = _sql.begin(); i != _sql.end(); i++)
{
query = *i;
Dataset::parse_sql(query);
//cout << "Executing: "<<query<<"\n\n";
if (db->
setErr(sqlite3_exec
(this->handle(), query.c_str(), NULL, NULL,
NULL)) != SQLITE_OK)
{
GB.Error(db->getErrorMsg());
}
} // end of for
if (db->in_transaction() && autocommit)
db->commit_transaction();
active = true;
ds_state = dsSelect;
refresh();
} // end of try
catch(...)
{
if (db->in_transaction())
db->rollback_transaction();
}
}
void SqliteDataset::make_insert()
{
make_query(insert_sql);
last();
}
void SqliteDataset::make_edit()
{
make_query(update_sql);
}
void SqliteDataset::make_deletion()
{
make_query(delete_sql);
}
// BM: One copy of field_value less by commenting 'edit_object'
void SqliteDataset::fill_fields()
{
//cout <<"rr "<<result.records.size()<<"|" << frecno <<"\n";
if ((db == NULL) || (result.record_header.size() == 0)
|| (result.records.size() < (uint) frecno))
return;
if (fields_object->size() == 0) // Filling columns name
for (uint i = 0; i < result.record_header.size(); i++)
{
(*fields_object)[i].props = result.record_header[i];
//(*edit_object)[i].props = result.record_header[i];
}
//Filling result
if (result.records.size() != 0)
{
for (uint i = 0; i < result.records[frecno].size(); i++)
{
(*fields_object)[i].val = result.records[frecno][i];
//(*edit_object)[i].val = result.records[frecno][i];
}
}
else
{
field_value tmp;
for (uint i = 0; i < result.record_header.size(); i++)
{
(*fields_object)[i].val = tmp;
//(*edit_object)[i].val = "";
}
}
}
//------------- public functions implementation -----------------//
// BM: should retry if error = SQLITE_SCHEMA
int SqliteDataset::exec(const string & sql)
{
int res;
int retry;
if (!handle())
GB.Error("No database connection");
exec_res.record_header.clear();
exec_res.records.clear();
exec_res.conn = handle(); //NG
//if ((strncmp("select",sql.c_str(),6) == 0) || (strncmp("SELECT",sql.c_str(),6) == 0))
for (retry = 1; retry <= 2; retry++)
{
res = sqlite3_exec(handle(), sql.c_str(), old_callback, &exec_res, NULL);
if (res != SQLITE_SCHEMA)
break;
}
db->setErr(res);
//if (res != SQLITE_OK)
// GB.Error(db->getErrorMsg());
return (res == SQLITE_OK);
}
int SqliteDataset::exec()
{
return exec(sql);
}
const void *SqliteDataset::getExecRes()
{
return &exec_res;
}
bool SqliteDataset::query(const char *query)
{
int res;
int retry;
//try{
if (db == NULL)
GB.Error("Database is not defined");
if (dynamic_cast < SqliteDatabase * >(db)->getHandle() == NULL)
GB.Error("No database connection");
if ((strncasecmp("select", query, 6) != 0) /*&&
(strncasecmp("PRAGMA table",query,12) !=0) &&
(strncasecmp("PRAGMA index",query,12) !=0) */
)
GB.Error("Syntax error in request: SELECT expected.");
//close();
//cout << "Curr size "<<num_rows()<<"\n\n";
result.conn = handle(); //NG
for (retry = 1; retry <= 2; retry++)
{
res = my_sqlite3_exec(handle(), query, &callback, &result, NULL);
if (res != SQLITE_SCHEMA)
break;
}
db->setErr(res);
if (res == SQLITE_OK)
{
active = true;
ds_state = dsSelect;
//cout << "Curr size2 "<<num_rows()<<"\n";
this->first();
//cout << "Curr size3 "<<num_rows()<<"\n";
//cout << "Curr fcount "<<field_count()<<"\n\n";
}
return (res == SQLITE_OK);
}
bool SqliteDataset::query(const string & q)
{
return query(q.c_str());
}
void SqliteDataset::open(const string & sql)
{
set_select_sql(sql);
open();
}
void SqliteDataset::open()
{
if (select_sql.size())
{
query(select_sql.c_str());
}
else
{
ds_state = dsInactive;
}
}
void SqliteDataset::close()
{
Dataset::close();
result.record_header.clear();
result.records.clear();
edit_object->clear();
fields_object->clear();
ds_state = dsInactive;
active = false;
delete this;
}
void SqliteDataset::cancel()
{
if ((ds_state == dsInsert) || (ds_state == dsEdit))
{
if (result.record_header.size())
ds_state = dsSelect;
else
ds_state = dsInactive;
}
}
int SqliteDataset::num_rows()
{
return result.records.size();
}
bool SqliteDataset::eof()
{
return feof;
}
bool SqliteDataset::bof()
{
return fbof;
}
void SqliteDataset::first()
{
Dataset::first();
this->fill_fields();
//cout << "In first "<< fields_object->size()<<"\n";
}
void SqliteDataset::last()
{
Dataset::last();
fill_fields();
}
void SqliteDataset::prev(void)
{
Dataset::prev();
fill_fields();
}
void SqliteDataset::next(void)
{
Dataset::next();
if (!eof())
fill_fields();
}
//bool SqliteDataset::seek(int pos=0) {
bool SqliteDataset::seek(int pos)
{
if (ds_state == dsSelect)
{
Dataset::seek(pos);
fill_fields();
return true;
}
return false;
}
long SqliteDataset::nextid(const char *seq_name)
{
if (handle())
return db->nextid(seq_name);
else
return DB_UNEXPECTED_RESULT;
}
/* Helper function */
void SetFieldType(result_set * r, Tables tables)
{
Tables::iterator it;
sqlite3_stmt *vm;
const char *tail;
unsigned int len;
// result_set res;
char sqlcmd[512];
for (it = tables.begin(); it != tables.end(); it++)
{
sprintf(sqlcmd, "PRAGMA table_info('%s')", (*it).data());
if (sqlite3_prepare(r->conn, sqlcmd, -1, &vm, &tail) != SQLITE_OK)
{
return;
}
while (sqlite3_step(vm) == SQLITE_ROW)
{
/* Type is in 2 */
/* field name is 1 */
/* not null indicator is 3 */
/* dflt value is 4 */
for (uint e = 0; e < r->record_header.size(); e++)
{
if (r->record_header[e].name ==
(char *) sqlite3_column_text(vm, 1) &&
r->record_header[e].field_table == (*it).data())
{
r->record_header[e].type = GetFieldType((char *)
sqlite3_column_text(vm, 2),
&len);
r->record_header[e].field_len = len;
/* Is not null */
r->record_header[e].notnull = sqlite3_column_text(vm, 3)[0];
}
} /* For end */
} /* while end */
sqlite3_finalize(vm);
}
}
/* Return fType and length from String field*/
fType GetFieldType(const char *Type, unsigned int *length)
{
char *upper;
char *_left, *_right;
fType rType; /* For return */
unsigned int rTypeLen = 0;
int i;
upper = GB.NewZeroString(Type);
for (i = 0; i < GB.StringLength(upper); i++)
upper[i] = toupper(upper[i]);
Type = upper;
if (!Type) Type = "";
if (strstr(Type, "CHAR(") /* note the opening bracket */
|| strstr(Type, "CLOB") || strstr(Type, "TEXT") /* also catches TINYTEXT */
|| strstr(Type, "VARCHAR")
|| strstr(Type, "ENUM") || strstr(Type, "SET") || strstr(Type, "YEAR"))
{ /* MySQL 2 or 4 digit year (string) */
rType = ft_String;
}
else if (strstr(Type, "CHAR") /* this is a 1-byte value */
|| strstr(Type, "TINYINT")
|| strstr(Type, "INT1") || strstr(Type, "BOOL"))
{
rType = ft_Boolean;
/* Length is for when value is used as a string */
rTypeLen = ft_Boolean_Length;
}
else if (strstr(Type, "SMALLINT") || strstr(Type, "INT2"))
{
rType = ft_Short;
/* Length is for when value is used as a string */
rTypeLen = ft_Short_Length;
}
else if (strstr(Type, "MEDIUMINT"))
{
rType = ft_Short;
/* Length is for when value is used as a string */
rTypeLen = ft_Short_Length;
}
else if (strstr(Type, "BIGINT") || strstr(Type, "INT8"))
{
rType = ft_LongDouble;
/* Length is for when value is used as a string */
rTypeLen = ft_LongDouble_Length;
}
else if (strstr(Type, "INTEGER")
|| strstr(Type, "INT") || strstr(Type, "INT4"))
{
rType = ft_Long;
/* Length is for when value is used as a string */
rTypeLen = ft_Long_Length;
}
else if (strstr(Type, "DECIMAL") || strstr(Type, "NUMERIC"))
{
rType = ft_Float;
/* Length is for when value is used as a string */
rTypeLen = ft_Float_Length;
}
else if (strstr(Type, "TIMESTAMP") || strstr(Type, "DATETIME"))
{
rType = ft_Date;
rTypeLen = ft_Date_Length;
}
else if (strstr(Type, "DATE"))
{
rType = ft_Date;
/* Length is for when value is used as a string */
rTypeLen = ft_Date_Length;
}
else if (strstr(Type, "TIME"))
{
rType = ft_Date;
/* Length is for when value is used as a string */
rTypeLen = ft_Date_Length;
}
else if (strstr(Type, "DOUBLE") || strstr(Type, "FLOAT8"))
{
rType = ft_Double;
/* Length is for when value is used as a string */
rTypeLen = ft_Double_Length;
}
else if (strstr(Type, "REAL") /* this is PostgreSQL "real", not
MySQL "real" which is a
synonym of "double" */
|| strstr(Type, "FLOAT")
|| strstr(Type, "FLOAT4"))
{
rType = ft_Float;
/* Length is for when value is used as a string */
rTypeLen = ft_Float_Length;
}
else if (strstr(Type, "BLOB")) // BM
{
rType = ft_Blob;
}
else
{
rType = ft_String;
/* most reasonable default */
}
if (rType == ft_String)
{
/* if a length has been defined it will be between () */
_right = (char *)rindex(Type, ')');
_left = (char *)index(Type, '(');
if (_right)
{
_right = '\0';
rTypeLen = atoi(_left + 1);
}
else
{
/* set a default length */
rTypeLen = DEFAULT_STRING_LENGTH;
}
}
if (length != NULL)
*length = rTypeLen;
GB.FreeString(&upper);
return rType;
}