mirror of
https://github.com/ultimatepp/ultimatepp.git
synced 2026-05-16 06:05:58 -06:00
440 lines
11 KiB
C++
440 lines
11 KiB
C++
#include <Core/Core.h>
|
|
#include <Sql/Sql.h>
|
|
#include "lib/sqlite3.h"
|
|
#include "Sqlite3.h"
|
|
|
|
NAMESPACE_UPP
|
|
|
|
#define LLOG(x) // LOG(x)
|
|
|
|
class Sqlite3Connection : public SqlConnection {
|
|
protected:
|
|
virtual void SetParam(int i, const Value& r);
|
|
virtual bool Execute();
|
|
virtual int GetRowsProcessed() const;
|
|
virtual Value GetInsertedId() const;
|
|
virtual bool Fetch();
|
|
virtual void GetColumn(int i, Ref f) const;
|
|
virtual void Cancel();
|
|
virtual SqlSession& GetSession() const;
|
|
virtual String ToString() const;
|
|
|
|
typedef Sqlite3Session::sqlite3 sqlite3;
|
|
typedef Sqlite3Session::sqlite3_stmt sqlite3_stmt;
|
|
|
|
private:
|
|
Sqlite3Session& session;
|
|
sqlite3* db;
|
|
Vector<Value> param;
|
|
|
|
sqlite3_stmt* current_stmt;
|
|
String current_stmt_string;
|
|
bool got_first_row;
|
|
bool got_row_data;
|
|
|
|
friend class Sqlite3Session;
|
|
void BindParam(int i, const Value& r);
|
|
|
|
public:
|
|
Sqlite3Connection(Sqlite3Session& the_session, sqlite3 *the_db);
|
|
virtual ~Sqlite3Connection();
|
|
};
|
|
|
|
Sqlite3Connection::Sqlite3Connection(Sqlite3Session& the_session, sqlite3 *the_db)
|
|
: session(the_session), db(the_db), current_stmt(NULL), got_first_row(false), got_row_data(false)
|
|
{}
|
|
|
|
Sqlite3Connection::~Sqlite3Connection()
|
|
{
|
|
Cancel();
|
|
}
|
|
|
|
void Sqlite3Connection::Cancel()
|
|
{
|
|
if (current_stmt) {
|
|
if (sqlite3_reset(current_stmt) != SQLITE_OK)
|
|
session.SetError(sqlite3_errmsg(db), "Resetting statement: " + current_stmt_string);
|
|
if (sqlite3_finalize(current_stmt) != SQLITE_OK)
|
|
session.SetError(sqlite3_errmsg(db), "Finalizing statement: "+ current_stmt_string);
|
|
current_stmt = NULL;
|
|
current_stmt_string.Clear();
|
|
}
|
|
}
|
|
|
|
void Sqlite3Connection::SetParam(int i, const Value& r)
|
|
{
|
|
LLOG(Format("SetParam(%d,%s)",i,r.ToString()));
|
|
param.At(i) = r;
|
|
}
|
|
|
|
void Sqlite3Connection::BindParam(int i, const Value& r) {
|
|
if (IsNull(r))
|
|
sqlite3_bind_null(current_stmt,i);
|
|
else switch (r.GetType()) {
|
|
case STRING_V:
|
|
case WSTRING_V: {
|
|
WString p = r;
|
|
sqlite3_bind_text16(current_stmt,i,p,2*p.GetLength(),SQLITE_TRANSIENT);
|
|
break;
|
|
}
|
|
case BOOL_V:
|
|
case INT_V:
|
|
sqlite3_bind_int(current_stmt, i, int(r));
|
|
break;
|
|
case INT64_V:
|
|
sqlite3_bind_int64(current_stmt, i, int64(r));
|
|
break;
|
|
case DOUBLE_V:
|
|
sqlite3_bind_double(current_stmt, i, double(r));
|
|
break;
|
|
case DATE_V: {
|
|
Date d = r;
|
|
String p = Format("%04d-%02d-%02d", d.year, d.month, d.day);
|
|
sqlite3_bind_text(current_stmt,i,p,p.GetLength(),SQLITE_TRANSIENT);
|
|
}
|
|
break;
|
|
case TIME_V: {
|
|
Time t = r;
|
|
String p = Format("%04d-%02d-%02d %02d:%02d:%02d",
|
|
t.year, t.month, t.day, t.hour, t.minute, t.second);
|
|
sqlite3_bind_text(current_stmt,i,p,p.GetLength(),SQLITE_TRANSIENT);
|
|
}
|
|
break;
|
|
default:
|
|
NEVER();
|
|
}
|
|
}
|
|
|
|
int ParseForArgs(const char* sqlcmd)
|
|
{
|
|
int numargs = 0;
|
|
const char* ptr = sqlcmd;
|
|
while (*ptr)
|
|
if(*ptr == '\'')
|
|
while(*++ptr && (*ptr != '\'' || *++ptr && *ptr == '\''))
|
|
;
|
|
else if(*ptr++ == '?')
|
|
++numargs;
|
|
return numargs;
|
|
}
|
|
|
|
bool Sqlite3Connection::Execute() {
|
|
Cancel();
|
|
if(statement.GetLength() == 0) {
|
|
session.SetError("Empty statement", String("Preparing: ") + statement);
|
|
return false;
|
|
}
|
|
if (SQLITE_OK != sqlite3_prepare(db,statement,statement.GetLength(),¤t_stmt,NULL)) {
|
|
LLOG("Sqlite3Connection::Compile(" << statement << ") -> error");
|
|
session.SetError(sqlite3_errmsg(db), String("Preparing: ") + statement);
|
|
return false;
|
|
}
|
|
current_stmt_string = statement;
|
|
int nparams = ParseForArgs(current_stmt_string);
|
|
ASSERT(nparams == param.GetCount());
|
|
for (int i = 0; i < nparams; ++i)
|
|
BindParam(i+1,param[i]);
|
|
param.Clear();
|
|
// Make sure that compiling the statement never fails.
|
|
ASSERT(NULL != current_stmt);
|
|
int retcode = sqlite3_step(current_stmt);
|
|
if ((retcode != SQLITE_DONE) && (retcode != SQLITE_ROW)) {
|
|
session.SetError(sqlite3_errmsg(db), current_stmt_string);
|
|
return false;
|
|
}
|
|
got_first_row = got_row_data = (retcode==SQLITE_ROW);
|
|
if (got_row_data) {
|
|
int numfields = sqlite3_column_count(current_stmt);
|
|
info.SetCount(numfields);
|
|
for (int i = 0; i < numfields; ++i) {
|
|
SqlColumnInfo& field = info[i];
|
|
field.name = sqlite3_column_name(current_stmt,i);
|
|
String coltype = sqlite3_column_decltype(current_stmt,i);
|
|
switch (sqlite3_column_type(current_stmt,i)) {
|
|
case SQLITE_INTEGER:
|
|
field.type = INT_V;
|
|
break;
|
|
case SQLITE_FLOAT:
|
|
field.type = DOUBLE_V;
|
|
break;
|
|
case SQLITE_TEXT:
|
|
if(coltype == "date")
|
|
field.type = DATE_V;
|
|
else
|
|
if(coltype == "datetime")
|
|
field.type = TIME_V;
|
|
else
|
|
field.type = WSTRING_V;
|
|
break;
|
|
case SQLITE_NULL:
|
|
field.type = VOID_V;
|
|
break;
|
|
case SQLITE_BLOB:
|
|
field.type = STRING_V;
|
|
break;
|
|
default:
|
|
NEVER();
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
int Sqlite3Connection::GetRowsProcessed() const
|
|
{
|
|
LLOG("GetRowsProcessed()");
|
|
return sqlite3_changes(db);
|
|
}
|
|
|
|
Value Sqlite3Connection::GetInsertedId() const
|
|
{
|
|
LLOG("GetInsertedId()");
|
|
return sqlite3_last_insert_rowid(db);
|
|
}
|
|
|
|
bool Sqlite3Connection::Fetch() {
|
|
ASSERT(NULL != current_stmt);
|
|
if (!got_row_data)
|
|
return false;
|
|
if (got_first_row) {
|
|
got_first_row = false;
|
|
return true;
|
|
}
|
|
ASSERT(got_row_data);
|
|
int retcode = sqlite3_step(current_stmt);
|
|
if ((retcode != SQLITE_DONE) && (retcode != SQLITE_ROW))
|
|
session.SetError(sqlite3_errmsg(db), String("Fetching prepared statement: ")+current_stmt_string );
|
|
got_row_data = (retcode==SQLITE_ROW);
|
|
return got_row_data;
|
|
}
|
|
|
|
void Sqlite3Connection::GetColumn(int i, Ref f) const {
|
|
ASSERT(NULL != current_stmt);
|
|
if(i == -1) {
|
|
f = Value(sqlite3_last_insert_rowid(db));
|
|
return;
|
|
}
|
|
|
|
ASSERT(got_row_data);
|
|
String coltype = sqlite3_column_decltype(current_stmt,i);
|
|
switch (sqlite3_column_type(current_stmt,i)) {
|
|
case SQLITE_INTEGER:
|
|
f = sqlite3_column_int64(current_stmt,i);
|
|
break;
|
|
case SQLITE_FLOAT:
|
|
f = sqlite3_column_double(current_stmt,i);
|
|
break;
|
|
case SQLITE_TEXT:
|
|
if(coltype == "date"){
|
|
const char *s = (const char *)sqlite3_column_text(current_stmt, i);
|
|
if(strlen(s) >= 10)
|
|
f = Value(Date(atoi(s), atoi(s + 5), atoi(s + 8)));
|
|
else
|
|
f = Null;
|
|
}
|
|
else
|
|
if(coltype == "datetime") {
|
|
const char *s = (const char *)sqlite3_column_text(current_stmt, i);
|
|
if(strlen(s) >= 19)
|
|
f = Value(Time(atoi(s), atoi(s + 5), atoi(s + 8), atoi(s + 11), atoi(s + 14), atoi(s + 17)));
|
|
else
|
|
f = Null;
|
|
}
|
|
else
|
|
f = Value(WString((const wchar*)sqlite3_column_text16(current_stmt,i)));
|
|
break;
|
|
case SQLITE_NULL:
|
|
f = Null;
|
|
break;
|
|
case SQLITE_BLOB:
|
|
f = Value(String( (const byte*)sqlite3_column_blob(current_stmt,i),
|
|
sqlite3_column_bytes(current_stmt,i) ));
|
|
break;
|
|
default:
|
|
NEVER();
|
|
break;
|
|
}
|
|
return;
|
|
}
|
|
SqlSession& Sqlite3Connection::GetSession() const { return session; }
|
|
String Sqlite3Connection::ToString() const {
|
|
return statement;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
bool Sqlite3Session::Open(const char* filename) {
|
|
// Only open db once.
|
|
ASSERT(NULL == db);
|
|
current_filename = filename;
|
|
// By default, sqlite3 associates the opened db with "main.*"
|
|
// However, using the ATTACH sql command, it can connect to more databases.
|
|
// I don't know how to get the list of attached databases from the API
|
|
current_dbname = "main";
|
|
bool opened = (SQLITE_OK == sqlite3_open(filename, &db));
|
|
return opened;
|
|
}
|
|
void Sqlite3Session::Close() {
|
|
if (NULL != db) {
|
|
int retval;
|
|
#ifndef flagNOAPPSQL
|
|
if(SQL.IsOpen() && &SQL.GetSession() == this)
|
|
SQL.Cancel();
|
|
#endif
|
|
retval = sqlite3_close(db);
|
|
// If this function fails, that means that some of the
|
|
// prepared statements have not been finalized.
|
|
// See lib/sqlite3.h:91
|
|
ASSERT(SQLITE_OK == retval);
|
|
db = NULL;
|
|
}
|
|
}
|
|
SqlConnection *Sqlite3Session::CreateConnection() {
|
|
return new Sqlite3Connection(*this, db);
|
|
}
|
|
void Sqlite3Session::Begin() {
|
|
static const char begin[] = "BEGIN;";
|
|
if(trace)
|
|
*trace << begin << "\n";
|
|
if(SQLITE_OK != sqlite3_exec(db,begin,NULL,NULL,NULL))
|
|
SetError(sqlite3_errmsg(db), begin);
|
|
}
|
|
void Sqlite3Session::Commit() {
|
|
static const char commit[] = "COMMIT;";
|
|
if(trace)
|
|
*trace << commit << "\n";
|
|
if(SQLITE_OK != sqlite3_exec(db,commit,NULL,NULL,NULL))
|
|
SetError(sqlite3_errmsg(db), commit);
|
|
}
|
|
void Sqlite3Session::Rollback() {
|
|
static const char rollback[] = "ROLLBACK;";
|
|
if(trace)
|
|
*trace << rollback << "\n";
|
|
if(SQLITE_OK != sqlite3_exec(db,rollback,NULL,NULL,NULL))
|
|
SetError(sqlite3_errmsg(db), rollback);
|
|
}
|
|
Vector<String> Sqlite3Session::EnumDatabases() {
|
|
Vector<String> out;
|
|
Sql sql(*this);
|
|
sql.Execute("PRAGMA database_list;");
|
|
while (sql.Fetch())
|
|
out.Add(sql[1]); // sql[1] is database name, sql[2] is filename
|
|
return out;
|
|
}
|
|
|
|
Vector<String> Sqlite3Session::EnumTables(String database) {
|
|
Vector<String> out;
|
|
String dbn=database;
|
|
if(dbn.IsEmpty()) dbn=current_dbname; // for backward compatibility
|
|
Sql sql(*this);
|
|
sql.Execute("SELECT name FROM "+dbn+".sqlite_master WHERE type='table' AND name NOT LIKE 'sqlite_%' ORDER BY 1;");
|
|
while (sql.Fetch())
|
|
out.Add(sql[0]);
|
|
return out;
|
|
}
|
|
|
|
Vector<String> Sqlite3Session::EnumViews(String database) {
|
|
Vector<String> out;
|
|
String dbn=database;
|
|
if(dbn.IsEmpty()) dbn=current_dbname;
|
|
Sql sql(*this);
|
|
sql.Execute("SELECT name FROM "+dbn+".sqlite_master WHERE type='view' AND name NOT LIKE 'sqlite_%' ORDER BY 1;");
|
|
while (sql.Fetch())
|
|
out.Add(sql[0]);
|
|
return out;
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
Vector<SqlColumnInfo> Sqlite3Session::EnumColumns(String database, String table) {
|
|
Vector<SqlColumnInfo> out;
|
|
SqlColumnInfo info;
|
|
String ColType;
|
|
Sql sql(*this);
|
|
sql.Execute("PRAGMA table_info("+table+");");
|
|
while (sql.Fetch()) {
|
|
info.width = info.scale = info.precision = 0;
|
|
info.name = sql[1];
|
|
ColType = sql[2];
|
|
if(ColType =="integer")
|
|
info.type = INT_V;
|
|
else
|
|
if(ColType =="real")
|
|
info.type = DOUBLE_V;
|
|
else
|
|
if (ColType =="date")
|
|
info.type = DATE_V;
|
|
else
|
|
if (ColType == "datetime")
|
|
info.type = TIME_V;
|
|
else
|
|
info.type = STRING_V;
|
|
out.Add(info);
|
|
}
|
|
return out;
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
const char *Sqlite3ReadString(const char *s, String& stmt) {
|
|
stmt.Cat(*s);
|
|
int c = *s++;
|
|
for(;;) {
|
|
if(*s == '\0') break;
|
|
else
|
|
if(*s == '\'' && s[1] == '\'') {
|
|
stmt.Cat('\'');
|
|
s += 2;
|
|
}
|
|
else
|
|
if(*s == c) {
|
|
stmt.Cat(c);
|
|
s++;
|
|
break;
|
|
}
|
|
else
|
|
if(*s == '\\') {
|
|
stmt.Cat('\\');
|
|
if(*++s)
|
|
stmt.Cat(*s++);
|
|
}
|
|
else
|
|
stmt.Cat(*s++);
|
|
}
|
|
return s;
|
|
}
|
|
|
|
bool Sqlite3PerformScript(const String& txt, StatementExecutor& se, Gate2<int, int> progress_canceled) {
|
|
const char *text = txt;
|
|
for(;;) {
|
|
String stmt;
|
|
while(*text <= 32 && *text > 0) text++;
|
|
if(*text == '\0') break;
|
|
for(;;) {
|
|
if(*text == '\0')
|
|
break;
|
|
if(*text == ';')
|
|
break;
|
|
else
|
|
if(*text == '\'')
|
|
text = Sqlite3ReadString(text, stmt);
|
|
else
|
|
if(*text == '\"')
|
|
text = Sqlite3ReadString(text, stmt);
|
|
else
|
|
stmt.Cat(*text++);
|
|
}
|
|
if(progress_canceled(text - txt.Begin(), txt.GetLength()))
|
|
return false;
|
|
if(!se.Execute(stmt))
|
|
return false;
|
|
if(*text) text++;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
END_UPP_NAMESPACE
|