47442705a00a1c90801c66c41820aa7ef842ce9d
[quassel.git] / src / core / abstractsqlstorage.cpp
1 /***************************************************************************
2  *   Copyright (C) 2005-07 by the Quassel IRC Team                         *
3  *   devel@quassel-irc.org                                                 *
4  *                                                                         *
5  *   This program is free software; you can redistribute it and/or modify  *
6  *   it under the terms of the GNU General Public License as published by  *
7  *   the Free Software Foundation; either version 2 of the License, or     *
8  *   (at your option) version 3.                                           *
9  *                                                                         *
10  *   This program is distributed in the hope that it will be useful,       *
11  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        *
12  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         *
13  *   GNU General Public License for more details.                          *
14  *                                                                         *
15  *   You should have received a copy of the GNU General Public License     *
16  *   along with this program; if not, write to the                         *
17  *   Free Software Foundation, Inc.,                                       *
18  *   59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.             *
19  ***************************************************************************/
20
21 #include "abstractsqlstorage.h"
22
23 #include <QSqlError>
24 #include <QSqlQuery>
25
26 AbstractSqlStorage::AbstractSqlStorage(QObject *parent)
27   : Storage(parent),
28     _schemaVersion(0)
29 {
30 }
31
32 AbstractSqlStorage::~AbstractSqlStorage() {
33   QHash<QPair<QString, int>, QSqlQuery *>::iterator iter = _queryCache.begin();
34   while(iter != _queryCache.end()) {
35     delete *iter;
36     iter = _queryCache.erase(iter);
37   }
38   
39   {
40     QSqlDatabase db = QSqlDatabase::database("quassel_connection");
41     db.commit();
42     db.close();
43   }
44   QSqlDatabase::removeDatabase("quassel_connection");  
45 }
46
47 QSqlDatabase AbstractSqlStorage::logDb() {
48   QSqlDatabase db = QSqlDatabase::database("quassel_connection");
49   if(db.isValid() && db.isOpen())
50     return db;
51
52   if(!openDb()) {
53     qWarning() << "Unable to Open Database" << engineName();
54     qWarning() << " -" << db.lastError().text();
55   }
56
57   return QSqlDatabase::database("quassel_connection");
58 }
59
60 bool AbstractSqlStorage::openDb() {
61   QSqlDatabase db = QSqlDatabase::database("quassel_connection");
62   if(db.isValid() && !db.isOpen())
63     return db.open();
64
65   db = QSqlDatabase::addDatabase(driverName(), "quassel_connection");
66   db.setDatabaseName(databaseName());
67
68   if(!hostName().isEmpty())
69     db.setHostName(hostName());
70
71   if(!userName().isEmpty()) {
72     db.setUserName(userName());
73     db.setPassword(password());
74   }
75
76   return db.open();
77 }
78
79 bool AbstractSqlStorage::init(const QVariantMap &settings) {
80   Q_UNUSED(settings)
81   QSqlDatabase db = logDb();
82   if(!db.isValid() || !db.isOpen())
83     return false;
84
85   if(installedSchemaVersion() == -1) {
86     qDebug() << "Storage Schema is missing!";
87     return false;
88   }
89
90   if(installedSchemaVersion() > schemaVersion()) {
91     qWarning() << "Installed Schema is newer then any known Version.";
92     return false;
93   }
94   
95   if(installedSchemaVersion() < schemaVersion()) {
96     qWarning() << "Installed Schema is not up to date. Upgrading...";
97     if(!upgradeDb())
98       return false;
99   }
100   
101   qDebug() << "Storage Backend is ready. Quassel Schema Version:" << installedSchemaVersion();
102   return true;
103 }
104
105 QString AbstractSqlStorage::queryString(const QString &queryName, int version) {
106   if(version == 0)
107     version = schemaVersion();
108     
109   QFileInfo queryInfo(QString(":/SQL/%1/%2/%3.sql").arg(engineName()).arg(version).arg(queryName));
110   if(!queryInfo.exists() || !queryInfo.isFile() || !queryInfo.isReadable()) {
111     qWarning() << "Unable to read SQL-Query" << queryName << "for Engine" << engineName();
112     return QString();
113   }
114
115   QFile queryFile(queryInfo.filePath());
116   if(!queryFile.open(QIODevice::ReadOnly | QIODevice::Text))
117     return QString();
118   QString query = QTextStream(&queryFile).readAll();
119   queryFile.close();
120   
121   return query.trimmed();
122 }
123
124 QString AbstractSqlStorage::queryString(const QString &queryName) {
125   return queryString(queryName, 0);
126 }
127
128 QSqlQuery *AbstractSqlStorage::cachedQuery(const QString &queryName, int version) {
129   QPair<QString, int> queryId = qMakePair(queryName, version);
130   if(!_queryCache.contains(queryId)) {
131     QSqlQuery *query = new QSqlQuery(logDb());
132     query->prepare(queryString(queryName, version));
133     _queryCache[queryId] = query;
134   }
135   return _queryCache[queryId];
136 }
137
138 QSqlQuery *AbstractSqlStorage::cachedQuery(const QString &queryName) {
139   return cachedQuery(queryName, 0);
140 }
141
142 QStringList AbstractSqlStorage::setupQueries() {
143   QStringList queries;
144   QDir dir = QDir(QString(":/SQL/%1/%2/").arg(engineName()).arg(schemaVersion()));
145   foreach(QFileInfo fileInfo, dir.entryInfoList(QStringList() << "setup*", QDir::NoFilter, QDir::Name)) {
146     queries << queryString(fileInfo.baseName());
147   }
148   return queries;
149 }
150
151 bool AbstractSqlStorage::setup(const QVariantMap &settings) {
152   Q_UNUSED(settings)
153   QSqlDatabase db = logDb();
154   if(!db.isOpen()) {
155     qWarning() << "Unable to setup Logging Backend!";
156     return false;
157   }
158
159   foreach(QString queryString, setupQueries()) {
160     QSqlQuery query = db.exec(queryString);
161     if(!watchQuery(&query)) {
162       qWarning() << "Unable to setup Logging Backend!";
163       return false;
164     }
165   }
166   return true;
167 }
168
169 QStringList AbstractSqlStorage::upgradeQueries(int version) {
170   QStringList queries;
171   QDir dir = QDir(QString(":/SQL/%1/%2/").arg(engineName()).arg(version));
172   foreach(QFileInfo fileInfo, dir.entryInfoList(QStringList() << "upgrade*", QDir::NoFilter, QDir::Name)) {
173     qDebug() << queryString(fileInfo.baseName());
174     queries << queryString(fileInfo.baseName());
175   }
176   return queries;
177 }
178
179 bool AbstractSqlStorage::upgradeDb() {
180   if(schemaVersion() <= installedSchemaVersion())
181     return true;
182
183   QSqlDatabase db = logDb();
184
185   for(int ver = installedSchemaVersion() + 1; ver <= schemaVersion(); ver++) {
186     foreach(QString queryString, upgradeQueries(ver)) {
187       QSqlQuery query = db.exec(queryString);
188       if(!watchQuery(&query)) {
189         qWarning() << "Unable to upgrade Logging Backend!";
190         return false;
191       }
192     }
193   }
194   return true;
195 }
196
197
198 int AbstractSqlStorage::schemaVersion() {
199   // returns the newest Schema Version!
200   // not the currently used one! (though it can be the same)
201   if(_schemaVersion > 0)
202     return _schemaVersion;
203
204   int version;
205   bool ok;
206   QDir dir = QDir(":/SQL/" + engineName());
207   foreach(QFileInfo fileInfo, dir.entryInfoList()) {
208     if(!fileInfo.isDir())
209       continue;
210
211     version = fileInfo.fileName().toInt(&ok);
212     if(!ok)
213       continue;
214
215     if(version > _schemaVersion)
216       _schemaVersion = version;
217   }
218   return _schemaVersion;
219 }
220
221 bool AbstractSqlStorage::watchQuery(QSqlQuery *query) {
222   if(query->lastError().isValid()) {
223     qWarning() << "unhandled Error in QSqlQuery!";
224     qWarning() << "                  last Query:" << query->lastQuery();
225     qWarning() << "              executed Query:" << query->executedQuery();
226     qWarning() << "                bound Values:" << query->boundValues();
227     qWarning() << "                Error Number:" << query->lastError().number();
228     qWarning() << "               Error Message:" << query->lastError().text();
229     qWarning() << "              Driver Message:" << query->lastError().driverText();
230     qWarning() << "                  DB Message:" << query->lastError().databaseText();
231     
232     return false;
233   }
234   return true;
235 }