move all the sql query in an own thread

This commit is contained in:
Marco Martin 2011-04-28 21:31:11 +02:00
parent 084b7a1226
commit f358f3de6d
4 changed files with 229 additions and 180 deletions

View File

@ -96,17 +96,8 @@ StorageJob::StorageJob(const QString& destination,
m_rdb->ref(); m_rdb->ref();
if (!m_rdb->database()->open()) {
kWarning() << "Unable to open the plasma storage cache database: " << m_rdb->database()->lastError();
} else if (!m_rdb->database()->tables().contains(m_clientName)) {
QSqlQuery query(*m_rdb->database());
query.prepare(QString("create table ") + m_clientName + " (valueGroup varchar(256), id varchar(256), txt TEXT, int INTEGER, float REAL, binary BLOB, creationTime datetime, accessTime datetime, primary key (valueGroup, id))");
if (!query.exec()) {
kWarning() << "Unable to create table for" << m_clientName;
m_rdb->database()->close();
}
}
Plasma::StorageThread::self()->start(); Plasma::StorageThread::self()->start();
connect(Plasma::StorageThread::self(), SIGNAL(newResult(StorageJob *, const QVariant &)), this, SLOT(resultSlot(StorageJob *, const QVariant &)));
qRegisterMetaType<StorageJob *>(); qRegisterMetaType<StorageJob *>();
} }
@ -127,6 +118,13 @@ QVariantHash StorageJob::data() const
return m_data; return m_data;
} }
QString StorageJob::clientName() const
{
return m_clientName;
}
void StorageJob::start() void StorageJob::start()
{ {
if (!m_rdb->database()->isOpen()) { if (!m_rdb->database()->isOpen()) {
@ -144,178 +142,28 @@ void StorageJob::start()
//kDebug() << operationName(); //kDebug() << operationName();
m_rdb->database()->transaction(); m_rdb->database()->transaction();
if (operationName() == "save") { if (operationName() == "save") {
QSqlQuery query(*m_rdb->database()); QMetaObject::invokeMethod(Plasma::StorageThread::self(), "save", Qt::QueuedConnection, Q_ARG(StorageJob *, this), Q_ARG(const QVariantMap&, params));
if (params.value("key").toString().isNull()) {
m_data.insert(params.value("key").toString(), params.value("data"));
}
QHashIterator<QString, QVariant> it(m_data);
QString ids;
while (it.hasNext()) {
it.next();
QSqlField field(":id", QVariant::String);
field.setValue(it.key());
if (!ids.isEmpty()) {
ids.append(", ");
}
ids.append(m_rdb->database()->driver()->formatValue(field));
}
query.prepare("delete from " + m_clientName + " where valueGroup = :valueGroup and id in (" + ids + ");");
query.bindValue(":valueGroup", valueGroup);
if (!query.exec()) {
m_rdb->database()->commit();
setResult(false);
return;
}
query.prepare("insert into " + m_clientName + " values(:valueGroup, :id, :txt, :int, :float, :binary, date('now'), date('now'))");
query.bindValue(":valueGroup", valueGroup);
query.bindValue(":txt", QVariant());
query.bindValue(":int", QVariant());
query.bindValue(":float", QVariant());
query.bindValue(":binary", QVariant());
const QString key = params.value("key").toString();
if (!key.isEmpty()) {
m_data.insert(key, params["data"]);
}
it.toFront();
while (it.hasNext()) {
it.next();
//kDebug() << "going to insert" << valueGroup << it.key();
query.bindValue(":id", it.key());
QString field;
switch (QMetaType::Type(it.value().type())) {
case QVariant::String:
field = ":txt";
break;
case QVariant::Int:
field = ":int";
break;
case QVariant::Double:
case QMetaType::Float:
field = ":float";
break;
case QVariant::ByteArray:
field = ":binary";
break;
default:
continue;
break;
}
query.bindValue(field, it.value());
if (!query.exec()) {
//kDebug() << "query failed:" << query.lastQuery() << query.lastError().text();
m_rdb->database()->commit();
setResult(false);
return;
}
query.bindValue(field, QVariant());
}
setResult(true);
} else if (operationName() == "retrieve") { } else if (operationName() == "retrieve") {
QSqlQuery query(*m_rdb->database());
QMetaObject::invokeMethod(Plasma::StorageThread::self(), "retrieve", Qt::QueuedConnection, Q_ARG(StorageJob *, this), Q_ARG(const QVariantMap&, params)); QMetaObject::invokeMethod(Plasma::StorageThread::self(), "retrieve", Qt::QueuedConnection, Q_ARG(StorageJob *, this), Q_ARG(const QVariantMap&, params));
//a bit redundant but should be the faster way with less string concatenation as possible
if (params["key"].toString().isEmpty()) {
//update modification time
query.prepare("update " + m_clientName + " set accessTime=date('now') where valueGroup=:valueGroup");
query.bindValue(":valueGroup", valueGroup);
query.exec();
query.prepare("select * from " + m_clientName + " where valueGroup=:valueGroup");
query.bindValue(":valueGroup", valueGroup);
} else {
//update modification time
query.prepare("update " + m_clientName + " set accessTime=date('now') where valueGroup=:valueGroup and id=:key");
query.bindValue(":valueGroup", valueGroup);
query.bindValue(":key", params["key"].toString());
query.exec();
query.prepare("select * from " + m_clientName + " where valueGroup=:valueGroup and id=:key");
query.bindValue(":valueGroup", valueGroup);
query.bindValue(":key", params["key"].toString());
}
const bool success = query.exec();
m_data.clear();
if (success) {
QSqlRecord rec = query.record();
const int keyColumn = rec.indexOf("id");
const int textColumn = rec.indexOf("txt");
const int intColumn = rec.indexOf("int");
const int floatColumn = rec.indexOf("float");
const int binaryColumn = rec.indexOf("binary");
while (query.next()) {
const QString key = query.value(keyColumn).toString();
if (!query.value(textColumn).isNull()) {
m_data.insert(key, query.value(textColumn));
} else if (!query.value(intColumn).isNull()) {
m_data.insert(key, query.value(intColumn));
} else if (!query.value(floatColumn).isNull()) {
m_data.insert(key, query.value(floatColumn));
} else if (!query.value(binaryColumn).isNull()) {
m_data.insert(key, query.value(binaryColumn));
}
}
setResult(m_data);
} else {
setResult(false);
}
} else if (operationName() == "delete") { } else if (operationName() == "delete") {
QSqlQuery query(*m_rdb->database()); QMetaObject::invokeMethod(Plasma::StorageThread::self(), "delete", Qt::QueuedConnection, Q_ARG(StorageJob *, this), Q_ARG(const QVariantMap&, params));
if (params["key"].toString().isEmpty()) {
query.prepare("delete from "+m_clientName+" where valueGroup=:valueGroup");
query.bindValue(":valueGroup", valueGroup);
} else {
query.prepare("delete from "+m_clientName+" where valueGroup=:valueGroup and id=:key");
query.bindValue(":valueGroup", valueGroup);
query.bindValue(":key", params["key"].toString());
}
const bool success = query.exec();
setResult(success);
} else if (operationName() == "expire") { } else if (operationName() == "expire") {
QSqlQuery query(*m_rdb->database()); QMetaObject::invokeMethod(Plasma::StorageThread::self(), "expire", Qt::QueuedConnection, Q_ARG(StorageJob *, this), Q_ARG(const QVariantMap&, params));
if (valueGroup.isEmpty()) {
query.prepare("delete from "+m_clientName+" where accessTime < :date");
QDateTime time(QDateTime::currentDateTime());
time.addSecs(-params["age"].toUInt());
query.bindValue(":date", time.toTime_t());
} else {
query.prepare("delete from "+m_clientName+" where valueGroup=:valueGroup and accessTime < :date");
query.bindValue(":valueGroup", valueGroup);
QDateTime time(QDateTime::currentDateTime());
time.addSecs(-params["age"].toUInt());
query.bindValue(":date", time.toTime_t());
}
const bool success = query.exec();
setResult(success);
} else { } else {
setError(true); setError(true);
} }
m_rdb->database()->commit(); m_rdb->database()->commit();
} }
void StorageJob::resultSlot(StorageJob *job, const QVariant &result)
{
if (job == this) {
setResult(result);
}
}
Plasma::ServiceJob* Storage::createJob(const QString &operation, QMap<QString, QVariant> &parameters) Plasma::ServiceJob* Storage::createJob(const QString &operation, QMap<QString, QVariant> &parameters)
{ {
if (m_clientName.isEmpty()) { if (m_clientName.isEmpty()) {

View File

@ -43,6 +43,10 @@ public:
void setData(const QVariantHash &data); void setData(const QVariantHash &data);
QVariantHash data() const; QVariantHash data() const;
void start(); void start();
QString clientName() const;
protected Q_SLOTS:
void resultSlot(StorageJob *job, const QVariant &result);
private: private:
RefCountedDatabase *m_rdb; RefCountedDatabase *m_rdb;

View File

@ -19,6 +19,12 @@
#include "storagethread_p.h" #include "storagethread_p.h"
#include <QSqlError>
#include <QSqlQuery>
#include <QSqlField>
#include <QSqlDriver>
#include <QSqlRecord>
#include "kdebug.h" #include "kdebug.h"
namespace Plasma namespace Plasma
@ -40,6 +46,7 @@ K_GLOBAL_STATIC(StorageThreadSingleton, privateStorageThreadSelf)
StorageThread::StorageThread(QObject *parent) StorageThread::StorageThread(QObject *parent)
: QThread(parent) : QThread(parent)
{ {
} }
StorageThread::~StorageThread() StorageThread::~StorageThread()
@ -53,31 +60,220 @@ Plasma::StorageThread *StorageThread::self()
return &privateStorageThreadSelf->self; return &privateStorageThreadSelf->self;
} }
void StorageThread::initializeDb(StorageJob* caller)
{
if (!m_db.open()) {
kWarning() << "Unable to open the plasma storage cache database: " << m_db.lastError();
} else if (!m_db.tables().contains(caller->clientName())) {
QSqlQuery query(m_db);
query.prepare(QString("create table ") + caller->clientName() + " (valueGroup varchar(256), id varchar(256), txt TEXT, int INTEGER, float REAL, binary BLOB, creationTime datetime, accessTime datetime, primary key (valueGroup, id))");
if (!query.exec()) {
kWarning() << "Unable to create table for" << caller->clientName();
m_db.close();
}
}
}
void StorageThread::save(StorageJob* caller, const QVariantMap &params) void StorageThread::save(StorageJob* caller, const QVariantMap &params)
{ {
bool success = true; initializeDb(caller);
QString valueGroup = params["group"].toString();
if (valueGroup.isEmpty()) {
valueGroup = "default";
}
QSqlQuery query(m_db);
if (params.value("key").toString().isNull()) {
caller->data().insert(params.value("key").toString(), params.value("data"));
}
emit newResult(caller, success); QHashIterator<QString, QVariant> it(caller->data());
QString ids;
while (it.hasNext()) {
it.next();
QSqlField field(":id", QVariant::String);
field.setValue(it.key());
if (!ids.isEmpty()) {
ids.append(", ");
}
ids.append(m_db.driver()->formatValue(field));
}
query.prepare("delete from " + caller->clientName() + " where valueGroup = :valueGroup and id in (" + ids + ");");
query.bindValue(":valueGroup", valueGroup);
if (!query.exec()) {
m_db.commit();
emit newResult(caller, false);
return;
}
query.prepare("insert into " + caller->clientName() + " values(:valueGroup, :id, :txt, :int, :float, :binary, date('now'), date('now'))");
query.bindValue(":valueGroup", valueGroup);
query.bindValue(":txt", QVariant());
query.bindValue(":int", QVariant());
query.bindValue(":float", QVariant());
query.bindValue(":binary", QVariant());
const QString key = params.value("key").toString();
if (!key.isEmpty()) {
caller->data().insert(key, params["data"]);
}
it.toFront();
while (it.hasNext()) {
it.next();
//kDebug() << "going to insert" << valueGroup << it.key();
query.bindValue(":id", it.key());
QString field;
switch (QMetaType::Type(it.value().type())) {
case QVariant::String:
field = ":txt";
break;
case QVariant::Int:
field = ":int";
break;
case QVariant::Double:
case QMetaType::Float:
field = ":float";
break;
case QVariant::ByteArray:
field = ":binary";
break;
default:
continue;
break;
}
query.bindValue(field, it.value());
if (!query.exec()) {
//kDebug() << "query failed:" << query.lastQuery() << query.lastError().text();
m_db.commit();
emit newResult(caller, false);
return;
}
query.bindValue(field, QVariant());
}
emit newResult(caller, true);
} }
void StorageThread::retrieve(StorageJob* caller, const QVariantMap &params) void StorageThread::retrieve(StorageJob* caller, const QVariantMap &params)
{ {
kWarning()<<"RETRIEVE"; initializeDb(caller);
bool success = true; QString valueGroup = params["group"].toString();
emit newResult(caller, success); if (valueGroup.isEmpty()) {
valueGroup = "default";
}
QSqlQuery query(m_db);
//a bit redundant but should be the faster way with less string concatenation as possible
if (params["key"].toString().isEmpty()) {
//update modification time
query.prepare("update " + caller->clientName() + " set accessTime=date('now') where valueGroup=:valueGroup");
query.bindValue(":valueGroup", valueGroup);
query.exec();
query.prepare("select * from " + caller->clientName() + " where valueGroup=:valueGroup");
query.bindValue(":valueGroup", valueGroup);
} else {
//update modification time
query.prepare("update " + caller->clientName() + " set accessTime=date('now') where valueGroup=:valueGroup and id=:key");
query.bindValue(":valueGroup", valueGroup);
query.bindValue(":key", params["key"].toString());
query.exec();
query.prepare("select * from " + caller->clientName() + " where valueGroup=:valueGroup and id=:key");
query.bindValue(":valueGroup", valueGroup);
query.bindValue(":key", params["key"].toString());
}
const bool success = query.exec();
QVariant result;
caller->data().clear();
if (success) {
QSqlRecord rec = query.record();
const int keyColumn = rec.indexOf("id");
const int textColumn = rec.indexOf("txt");
const int intColumn = rec.indexOf("int");
const int floatColumn = rec.indexOf("float");
const int binaryColumn = rec.indexOf("binary");
while (query.next()) {
const QString key = query.value(keyColumn).toString();
if (!query.value(textColumn).isNull()) {
caller->data().insert(key, query.value(textColumn));
} else if (!query.value(intColumn).isNull()) {
caller->data().insert(key, query.value(intColumn));
} else if (!query.value(floatColumn).isNull()) {
caller->data().insert(key, query.value(floatColumn));
} else if (!query.value(binaryColumn).isNull()) {
caller->data().insert(key, query.value(binaryColumn));
}
}
result = caller->data();
} else {
result = false;
}
emit newResult(caller, result);
} }
void StorageThread::deleteEntry(StorageJob* caller, const QVariantMap &params) void StorageThread::deleteEntry(StorageJob* caller, const QVariantMap &params)
{ {
initializeDb(caller);
QString valueGroup = params["group"].toString();
if (valueGroup.isEmpty()) {
valueGroup = "default";
}
QSqlQuery query(m_db);
if (params["key"].toString().isEmpty()) {
query.prepare("delete from "+caller->clientName()+" where valueGroup=:valueGroup");
query.bindValue(":valueGroup", valueGroup);
} else {
query.prepare("delete from "+caller->clientName()+" where valueGroup=:valueGroup and id=:key");
query.bindValue(":valueGroup", valueGroup);
query.bindValue(":key", params["key"].toString());
}
const bool success = query.exec();
const bool success = true;
emit newResult(caller, success); emit newResult(caller, success);
} }
void StorageThread::expire(StorageJob* caller, const QVariantMap &params) void StorageThread::expire(StorageJob* caller, const QVariantMap &params)
{ {
initializeDb(caller);
QString valueGroup = params["group"].toString();
if (valueGroup.isEmpty()) {
valueGroup = "default";
}
QSqlQuery query(m_db);
if (valueGroup.isEmpty()) {
query.prepare("delete from "+caller->clientName()+" where accessTime < :date");
QDateTime time(QDateTime::currentDateTime());
time.addSecs(-params["age"].toUInt());
query.bindValue(":date", time.toTime_t());
} else {
query.prepare("delete from "+caller->clientName()+" where valueGroup=:valueGroup and accessTime < :date");
query.bindValue(":valueGroup", valueGroup);
QDateTime time(QDateTime::currentDateTime());
time.addSecs(-params["age"].toUInt());
query.bindValue(":date", time.toTime_t());
}
const bool success = query.exec();
bool success = true;
emit newResult(caller, success); emit newResult(caller, success);
} }

View File

@ -39,6 +39,7 @@ public:
void run(); void run();
static Plasma::StorageThread *self(); static Plasma::StorageThread *self();
void initializeDb(StorageJob* caller);
public Q_SLOTS: public Q_SLOTS:
void save(StorageJob* caller, const QVariantMap &parameters); void save(StorageJob* caller, const QVariantMap &parameters);
@ -47,7 +48,7 @@ public Q_SLOTS:
void expire(StorageJob* caller, const QVariantMap &parameters); void expire(StorageJob* caller, const QVariantMap &parameters);
Q_SIGNALS: Q_SIGNALS:
void newResult(StorageJob* caller, bool result); void newResult(StorageJob* caller, const QVariant &result);
private: private:
QSqlDatabase m_db; QSqlDatabase m_db;