/* Copyright (C) 2003-2015 LiveCode Ltd. This file is part of LiveCode. LiveCode is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License v3 as published by the Free Software Foundation. LiveCode 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 LiveCode. If not see . */ /*DBCURSOR_SQLITE - CUSROR OBJECT FOR MYSQL DATABASES CHILD OF DBCURSOR Attributes: mysql_res - result set structure for MYSQL database isBOF - True if at beginning of resultset (inherited) isEOF - True if at end of resultset (inherited) recordNum - Number of current record -1 (inherited) recordCount - Number of records in resultset (inherited) fieldCount - Number of columns (inherited) fields - Array of fields (inherited) connection - Pointer to dbconnection_mysql object that created cursor Methods: Open - opens cursor and retrieves resultset from connection Close - closes cursor and frees any resources used by cursor getFieldsInformation - get column names, types, and info getRowData - Retrieve the data from the current row. first - move to first row of resultset next - move to next row of resultset prev - move to previous row of resultset last - move to last row of resultset */ #include "dbsqlite.h" #include #define ENTER DBCursor_SQLITE::DBCursor_SQLITE(SqliteDatabase &db, bool enable_binary) : mDB(db) { ENTER; mDataset = mDB.CreateDataset(); m_enable_binary = enable_binary; } DBCursor_SQLITE::~DBCursor_SQLITE() { ENTER; close(); delete mDataset; } /*Open - opens cursor and retrieves resultset from connection Output: False on error*/ Bool DBCursor_SQLITE::open(DBConnection *newconnection) { ENTER; if (!newconnection->getIsConnected()) return False; connection = newconnection; try { recordCount = mDataset -> num_rows(); fieldCount = mDataset -> field_count(); if(!getFieldsInformation()) return False; if(recordCount != 0) first(); } catch(DbErrors &e) { MDEBUG("***** Caught error [%s]\n", e.getMsg()); return False; } MDEBUG0("Cursor::open OK\n"); return True; } //Close - close cursor and free resources used by cursor void DBCursor_SQLITE::close() { ENTER; MDEBUG0("\n--\n"); FreeFields(); isBOF = False; isEOF = True; recordNum = recordCount = fieldCount = 0; MDEBUG0("\nBye cursor\n"); } /*first - move to first row of resultset Output - False on error*/ Bool DBCursor_SQLITE::first() { ENTER; try { //exit if no more records if (recordCount == 0) return False; recordNum = 0; mDataset->first(); if(!getRowData()) return False; isBOF = True; isEOF = False; } catch(DbErrors &e) { MDEBUG("***** Caught error [%s]\n", e.getMsg()); return False; } return True; } /*last - move to last row of resultset Output - False on error*/ Bool DBCursor_SQLITE::last() { ENTER; try { if (recordCount == 0) return False; recordNum = recordCount-1; mDataset->last(); if (!getRowData()) return False; isBOF = False; isEOF = True; } catch(DbErrors &e) { MDEBUG("***** Caught error [%s]\n", e.getMsg()); return False; } return True; } /*next - move to next row of resultset Output - False on error*/ Bool DBCursor_SQLITE::next() { ENTER; try { //exit if at last record or there are no records if (recordCount == 0 || isEOF == True) return False; isBOF = False; recordNum++; if (recordNum == recordCount) { isEOF = true; recordNum = recordCount-1; return False; } mDataset->next(); getRowData(); } catch(DbErrors &e) { MDEBUG("***** Caught error [%s]\n", e.getMsg()); return False; } return True; } /*prev - move to first row of resultset Output - False on error*/ Bool DBCursor_SQLITE::prev() { ENTER; try { if (recordCount == 0) return False; if (isBOF) return False; isEOF = false; recordNum--; if (recordNum == -1) { isBOF =True; recordNum = 0; return False; } mDataset->prev(); getRowData(); } catch(DbErrors &e) { MDEBUG("***** Caught error [%s]\n", e.getMsg()); return False; } return True; } Bool DBCursor_SQLITE::move(int p_record_index) { if (recordCount == 0) return False; if (p_record_index < 0) return False; else if (p_record_index > recordCount - 1) return False; isBOF = False; isEOF = False; if (!mDataset -> seek(p_record_index)) return False; recordNum = p_record_index; getRowData(); return True; } /*getFieldsInformation - get column names, types, and info Output: False on error*/ Bool DBCursor_SQLITE::getFieldsInformation() { ENTER; //create an array of columns fields = new (nothrow) DBField *[fieldCount]; Fields *t_fields_object; t_fields_object = mDataset -> get_fields_object(); for(int i = 0 ; i < fieldCount; i++) { DBField *tfield = new (nothrow) DBField(); fields[i] = tfield; const char *name = mDataset->fieldName(i); if(name) { MDEBUG("Field name=[%s]\n", name); const field_value fv = mDataset->get_field_value_by_index(i); // OK-2008-10-29 : [[Bug 6588]] - Get the field type from the props structure instead of from the field value, // because we are returning the declared column type rather than the actual type of the data in the column. field_prop t_props; t_props = (*t_fields_object)[i].props; fType type; type = t_props . type; /////////////////// // TYPE if (strlen(name) > F_NAMESIZE -6) strncpy(tfield->fieldName, name, F_NAMESIZE-6); else strcpy(tfield->fieldName, name); switch(type) { case ft_Short : case ft_UShort : MDEBUG0("Type: smallint\n"); tfield->fieldType = FT_SMALLINT; break; case ft_Long : case ft_ULong : MDEBUG0("Type: integer\n"); tfield->fieldType = FT_INTEGER; break; case ft_Float: MDEBUG0("Type: float\n"); tfield->fieldType = FT_FLOAT; break; case ft_Double: case ft_LongDouble: MDEBUG0("Type: double\n"); tfield->fieldType = FT_DOUBLE; break; case ft_Object : if (m_enable_binary) tfield -> fieldType = FT_BLOB; else tfield -> fieldType = FT_STRING; break; case ft_String : default: MDEBUG0("Type: string\n"); tfield->fieldType = FT_STRING; break; } //get field number tfield->fieldNum = i+1; //get max length tfield->maxlength = MAX_BYTES_PER_ROW; //TODO: get column characteristics tfield->isAutoIncrement = 0; tfield->isPrimaryKey = 0; tfield->isUnique = 0; tfield->isNotNull = 0; // OK-2009-05-26: [[Bug 8065]] - Field data must start out as NULL so we can tell // whether or not it needs to be freed. tfield -> data = NULL; } } return True; } /*getRowData - Retrieve the data from the current row. Output: False on error*/ Bool DBCursor_SQLITE::getRowData() { ENTER; Bool ret = true; try { for(int i = 0; i < fieldCount; i++) { const field_value fv = mDataset->get_field_value_by_index(i); // OK-2009-05-26: [[Bug 8065]] - Clear out any data from previous records, otherwise we no longer // have a pointer to it and the memory is lost. if (fields[i] -> data != NULL) { delete fields[i] -> data; fields[i] -> data = NULL; } //distinguish between empty column and null column fields[i]->isNull = fv.get_isNull(); if(!fields[i]->isNull) { std::string tmp = fv.get_asString(); // MW-2014-01-29: [[ Sqlite382 ]] If the field type is BLOB and we aren't in binary mode, // we must decode the data; otherwise we just use the contents directly. if (fv.get_fType() == ft_Object && !m_enable_binary) { // This code has moved from libsqlite/sqlitedataset.cpp:callback(). int bufsize; bufsize = tmp.size(); char *mybuff; mybuff = (char *)malloc(bufsize); memset(mybuff,0,bufsize); bufsize = sqlite_decode_binary((const unsigned char *)tmp.c_str(), bufsize, ( unsigned char *)mybuff, bufsize); if (bufsize == -1) { fields[i] -> data = NULL; fields[i] -> dataSize = 0; fields[i] -> isNull = True; free(mybuff); } else { fields[i] -> data = mybuff; fields[i] -> dataSize = bufsize; } } else { int bufsize = tmp.size(); fields[i]->data = new (nothrow) char[tmp.size() + 1]; memset(fields[i]->data,0,bufsize+1); memcpy(fields[i]->data, tmp.data(), tmp.size()); fields[i]->dataSize = tmp.size(); } // OK-2009-05-26: [[Bug 8065]] - Because we copied the data in the line above, // We have to set FreeBuffer otherwise it will not be freed when the cursor is closed, // leading to memory leaks. fields[i] -> freeBuffer = True; } MDEBUG("getRowData() -- fv=[%s]\n", fields[i]->data); } } catch(DbErrors &e) { ret = False; } return ret; }