ctkPluginDatabase.cpp 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804
  1. /*=============================================================================
  2. Library: CTK
  3. Copyright (c) 2010 German Cancer Research Center,
  4. Division of Medical and Biological Informatics
  5. Licensed under the Apache License, Version 2.0 (the "License");
  6. you may not use this file except in compliance with the License.
  7. You may obtain a copy of the License at
  8. http://www.apache.org/licenses/LICENSE-2.0
  9. Unless required by applicable law or agreed to in writing, software
  10. distributed under the License is distributed on an "AS IS" BASIS,
  11. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. See the License for the specific language governing permissions and
  13. limitations under the License.
  14. =============================================================================*/
  15. #include "ctkPluginDatabase_p.h"
  16. #include "ctkPluginDatabaseException.h"
  17. #include "ctkPlugin.h"
  18. #include "ctkPluginConstants.h"
  19. #include "ctkPluginException.h"
  20. #include "ctkPluginArchive_p.h"
  21. #include "ctkPluginStorage_p.h"
  22. #include "ctkServiceException.h"
  23. #include <QApplication>
  24. #include <QFileInfo>
  25. #include <QUrl>
  26. #include <QServiceManager>
  27. #include <QDebug>
  28. //database table names
  29. #define PLUGINS_TABLE "Plugins"
  30. #define PLUGIN_RESOURCES_TABLE "PluginResources"
  31. enum TBindIndexes
  32. {
  33. EBindIndex=0,
  34. EBindIndex1,
  35. EBindIndex2,
  36. EBindIndex3,
  37. EBindIndex4,
  38. EBindIndex5,
  39. EBindIndex6,
  40. EBindIndex7
  41. };
  42. ctkPluginDatabase::ctkPluginDatabase(ctkPluginStorage* storage)
  43. :m_isDatabaseOpen(false), m_inTransaction(false), m_PluginStorage(storage)
  44. {
  45. }
  46. ctkPluginDatabase::~ctkPluginDatabase()
  47. {
  48. close();
  49. }
  50. void ctkPluginDatabase::open()
  51. {
  52. if (m_isDatabaseOpen)
  53. return;
  54. QString path;
  55. //Create full path to database
  56. if(m_databasePath.isEmpty ())
  57. m_databasePath = getDatabasePath();
  58. path = m_databasePath;
  59. QFileInfo dbFileInfo(path);
  60. if (!dbFileInfo.dir().exists())
  61. {
  62. if(!QDir::root().mkpath(dbFileInfo.path()))
  63. {
  64. close();
  65. QString errorText("Could not create database directory: %1");
  66. throw ctkPluginDatabaseException(errorText.arg(dbFileInfo.path()), ctkPluginDatabaseException::DB_CREATE_DIR_ERROR);
  67. }
  68. }
  69. m_connectionName = dbFileInfo.completeBaseName();
  70. QSqlDatabase database;
  71. if (QSqlDatabase::contains(m_connectionName))
  72. {
  73. database = QSqlDatabase::database(m_connectionName);
  74. }
  75. else
  76. {
  77. database = QSqlDatabase::addDatabase("QSQLITE", m_connectionName);
  78. database.setDatabaseName(path);
  79. }
  80. if (!database.isValid())
  81. {
  82. close();
  83. throw ctkPluginDatabaseException(QString("Invalid database connection: %1").arg(m_connectionName),
  84. ctkPluginDatabaseException::DB_CONNECTION_INVALID);
  85. }
  86. //Create or open database
  87. if (!database.isOpen())
  88. {
  89. if (!database.open())
  90. {
  91. close();
  92. throw ctkPluginDatabaseException(QString("Could not open database. ") + database.lastError().text(),
  93. ctkPluginDatabaseException::DB_SQL_ERROR);
  94. }
  95. }
  96. m_isDatabaseOpen = true;
  97. //Check if the sqlite version supports foreign key constraints
  98. QSqlQuery query(database);
  99. if (!query.exec("PRAGMA foreign_keys"))
  100. {
  101. close();
  102. throw ctkPluginDatabaseException(QString("Check for foreign key support failed."),
  103. ctkPluginDatabaseException::DB_SQL_ERROR);
  104. }
  105. if (!query.next())
  106. {
  107. close();
  108. throw ctkPluginDatabaseException(QString("SQLite db does not support foreign keys. It is either older than 3.6.19 or was compiled with SQLITE_OMIT_FOREIGN_KEY or SQLITE_OMIT_TRIGGER"),
  109. ctkPluginDatabaseException::DB_SQL_ERROR);
  110. }
  111. query.finish();
  112. query.clear();
  113. //Enable foreign key support
  114. if (!query.exec("PRAGMA foreign_keys = ON"))
  115. {
  116. close();
  117. throw ctkPluginDatabaseException(QString("Enabling foreign key support failed."),
  118. ctkPluginDatabaseException::DB_SQL_ERROR);
  119. }
  120. query.finish();
  121. //Check database structure (tables) and recreate tables if neccessary
  122. //If one of the tables is missing remove all tables and recreate them
  123. //This operation is required in order to avoid data coruption
  124. if (!checkTables())
  125. {
  126. if (dropTables())
  127. {
  128. createTables();
  129. }
  130. else
  131. {
  132. //dropTables() should've handled error message
  133. //and warning
  134. close();
  135. }
  136. }
  137. //Update database based on the recorded timestamps
  138. updateDB();
  139. }
  140. void ctkPluginDatabase::updateDB()
  141. {
  142. checkConnection();
  143. QSqlDatabase database = QSqlDatabase::database(m_connectionName);
  144. QSqlQuery query(database);
  145. beginTransaction(&query, Write);
  146. QString statement = "SELECT ID, Location, LocalPath, Timestamp, SymbolicName, Version FROM Plugins WHERE State != ?";
  147. QList<QVariant> bindValues;
  148. bindValues.append(ctkPlugin::UNINSTALLED);
  149. QList<qlonglong> outdatedIds;
  150. QList<QPair<QString,QString> > outdatedPlugins;
  151. QStringList outdatedServiceNames;
  152. try
  153. {
  154. executeQuery(&query, statement, bindValues);
  155. while (query.next())
  156. {
  157. QFileInfo pluginInfo(query.value(EBindIndex2).toString());
  158. if (pluginInfo.lastModified() > QDateTime::fromString(query.value(EBindIndex3).toString(), Qt::ISODate))
  159. {
  160. outdatedIds.append(query.value(EBindIndex).toLongLong());
  161. outdatedPlugins.append(qMakePair(query.value(EBindIndex1).toString(), query.value(EBindIndex2).toString()));
  162. outdatedServiceNames.append(query.value(EBindIndex4).toString() + "_" + query.value(EBindIndex5).toString());
  163. }
  164. }
  165. }
  166. catch (...)
  167. {
  168. rollbackTransaction(&query);
  169. throw;
  170. }
  171. query.finish();
  172. query.clear();
  173. try
  174. {
  175. statement = "DELETE FROM Plugins WHERE ID=?";
  176. QListIterator<qlonglong> idIter(outdatedIds);
  177. while (idIter.hasNext())
  178. {
  179. bindValues.clear();
  180. bindValues.append(idIter.next());
  181. executeQuery(&query, statement, bindValues);
  182. }
  183. }
  184. catch (...)
  185. {
  186. rollbackTransaction(&query);
  187. throw;
  188. }
  189. try
  190. {
  191. QtMobility::QServiceManager serviceManager;
  192. QStringListIterator serviceNameIter(outdatedServiceNames);
  193. while (serviceNameIter.hasNext())
  194. {
  195. QString serviceName = serviceNameIter.next();
  196. serviceManager.removeService(serviceName);
  197. QtMobility::QServiceManager::Error error = serviceManager.error();
  198. if (!(error == QtMobility::QServiceManager::NoError ||
  199. error == QtMobility::QServiceManager::ComponentNotFound))
  200. {
  201. throw ctkServiceException(QString("Removing service named ") + serviceName +
  202. " failed: " + QString::number(static_cast<unsigned int>(error)));
  203. }
  204. }
  205. }
  206. catch (...)
  207. {
  208. rollbackTransaction(&query);
  209. throw;
  210. }
  211. commitTransaction(&query);
  212. QListIterator<QPair<QString,QString> > locationIter(outdatedPlugins);
  213. while (locationIter.hasNext())
  214. {
  215. const QPair<QString,QString>& locations = locationIter.next();
  216. insertPlugin(QUrl(locations.first), locations.second, false);
  217. }
  218. }
  219. ctkPluginArchive* ctkPluginDatabase::insertPlugin(const QUrl& location, const QString& localPath, bool createArchive)
  220. {
  221. checkConnection();
  222. // Assemble the data for the sql record
  223. QFileInfo fileInfo(localPath);
  224. const QString lastModified = fileInfo.lastModified().toString(Qt::ISODate);
  225. QString resourcePrefix = fileInfo.baseName();
  226. if (resourcePrefix.startsWith("lib"))
  227. {
  228. resourcePrefix = resourcePrefix.mid(3);
  229. }
  230. resourcePrefix.replace("_", ".");
  231. resourcePrefix = QString(":/") + resourcePrefix + "/";
  232. QSqlDatabase database = QSqlDatabase::database(m_connectionName);
  233. QSqlQuery query(database);
  234. beginTransaction(&query, Write);
  235. QString statement = "INSERT INTO Plugins(Location,LocalPath,SymbolicName,Version,State,Timestamp) VALUES(?,?,?,?,?,?)";
  236. QList<QVariant> bindValues;
  237. bindValues.append(location.toString());
  238. bindValues.append(localPath);
  239. bindValues.append(QString("na"));
  240. bindValues.append(QString("na"));
  241. bindValues.append(ctkPlugin::INSTALLED);
  242. bindValues.append(lastModified);
  243. qlonglong pluginId = -1;
  244. try
  245. {
  246. executeQuery(&query, statement, bindValues);
  247. QVariant lastId = query.lastInsertId();
  248. if (lastId.isValid())
  249. {
  250. pluginId = lastId.toLongLong();
  251. }
  252. }
  253. catch (...)
  254. {
  255. rollbackTransaction(&query);
  256. throw;
  257. }
  258. // Load the plugin and cache the resources
  259. QPluginLoader pluginLoader;
  260. pluginLoader.setFileName(localPath);
  261. if (!pluginLoader.load())
  262. {
  263. rollbackTransaction(&query);
  264. throw ctkPluginException(QString("The plugin could not be loaded: %1").arg(localPath));
  265. }
  266. QDirIterator dirIter(resourcePrefix, QDirIterator::Subdirectories);
  267. while (dirIter.hasNext())
  268. {
  269. QString resourcePath = dirIter.next();
  270. if (QFileInfo(resourcePath).isDir()) continue;
  271. QFile resourceFile(resourcePath);
  272. resourceFile.open(QIODevice::ReadOnly);
  273. QByteArray resourceData = resourceFile.readAll();
  274. resourceFile.close();
  275. statement = "INSERT INTO PluginResources(PluginID, ResourcePath, Resource) VALUES(?,?,?)";
  276. bindValues.clear();
  277. bindValues.append(QVariant::fromValue<qlonglong>(pluginId));
  278. bindValues.append(resourcePath.mid(resourcePrefix.size()-1));
  279. bindValues.append(resourceData);
  280. try
  281. {
  282. executeQuery(&query, statement, bindValues);
  283. }
  284. catch (...)
  285. {
  286. rollbackTransaction(&query);
  287. throw;
  288. }
  289. }
  290. pluginLoader.unload();
  291. try
  292. {
  293. ctkPluginArchive* archive = new ctkPluginArchive(m_PluginStorage, location, localPath,
  294. pluginId);;
  295. statement = "UPDATE Plugins SET SymbolicName=?,Version=? WHERE ID=?";
  296. QString versionString = archive->getAttribute(ctkPluginConstants::PLUGIN_VERSION);
  297. bindValues.clear();
  298. bindValues.append(archive->getAttribute(ctkPluginConstants::PLUGIN_SYMBOLICNAME));
  299. bindValues.append(versionString.isEmpty() ? "0.0.0" : versionString);
  300. bindValues.append(pluginId);
  301. if (!createArchive)
  302. {
  303. delete archive;
  304. archive = 0;
  305. }
  306. executeQuery(&query, statement, bindValues);
  307. commitTransaction(&query);
  308. return archive;
  309. }
  310. catch (...)
  311. {
  312. rollbackTransaction(&query);
  313. throw;
  314. }
  315. }
  316. QStringList ctkPluginDatabase::findResourcesPath(long pluginId, const QString& path) const
  317. {
  318. checkConnection();
  319. QString statement = "SELECT SUBSTR(ResourcePath,?) FROM PluginResources WHERE PluginID=? AND SUBSTR(ResourcePath,1,?)=?";
  320. QString resourcePath = path.startsWith('/') ? path : QString("/") + path;
  321. if (!resourcePath.endsWith('/'))
  322. resourcePath += "/";
  323. QList<QVariant> bindValues;
  324. bindValues.append(resourcePath.size()+1);
  325. bindValues.append(qlonglong(pluginId));
  326. bindValues.append(resourcePath.size());
  327. bindValues.append(resourcePath);
  328. QSqlDatabase database = QSqlDatabase::database(m_connectionName);
  329. QSqlQuery query(database);
  330. executeQuery(&query, statement, bindValues);
  331. QStringList paths;
  332. while (query.next())
  333. {
  334. QString currPath = query.value(EBindIndex).toString();
  335. int slashIndex = currPath.indexOf('/');
  336. if (slashIndex > 0)
  337. {
  338. currPath = currPath.left(slashIndex+1);
  339. }
  340. paths << currPath;
  341. }
  342. return paths;
  343. }
  344. void ctkPluginDatabase::removeArchive(const ctkPluginArchive *pa)
  345. {
  346. checkConnection();
  347. QSqlDatabase database = QSqlDatabase::database(m_connectionName);
  348. QSqlQuery query(database);
  349. QString statement = "DELETE FROM Plugins WHERE ID=?";
  350. QList<QVariant> bindValues;
  351. bindValues.append(pa->getPluginId());
  352. executeQuery(&query, statement, bindValues);
  353. }
  354. void ctkPluginDatabase::executeQuery(QSqlQuery *query, const QString &statement, const QList<QVariant> &bindValues) const
  355. {
  356. Q_ASSERT(query != 0);
  357. bool success = false;
  358. enum {Prepare =0 , Execute=1};
  359. for (int stage=Prepare; stage <= Execute; ++stage)
  360. {
  361. if ( stage == Prepare)
  362. success = query->prepare(statement);
  363. else // stage == Execute
  364. success = query->exec();
  365. if (!success)
  366. {
  367. QString errorText = "Problem: Could not %1 statement: %2\n"
  368. "Reason: %3\n"
  369. "Parameters: %4\n";
  370. QString parameters;
  371. if (bindValues.count() > 0)
  372. {
  373. for (int i = 0; i < bindValues.count(); ++i)
  374. {
  375. parameters.append(QString("\n\t[") + QString::number(i) + "]: " + bindValues.at(i).toString());
  376. }
  377. }
  378. else
  379. {
  380. parameters = "None";
  381. }
  382. ctkPluginDatabaseException::Type errorType;
  383. int result = query->lastError().number();
  384. if (result == 26 || result == 11) //SQLILTE_NOTADB || SQLITE_CORRUPT
  385. {
  386. qWarning() << "ctkPluginFramework:- Database file is corrupt or invalid:" << getDatabasePath();
  387. errorType = ctkPluginDatabaseException::DB_FILE_INVALID;
  388. }
  389. else if (result == 8) //SQLITE_READONLY
  390. errorType = ctkPluginDatabaseException::DB_WRITE_ERROR;
  391. else
  392. errorType = ctkPluginDatabaseException::DB_SQL_ERROR;
  393. query->finish();
  394. query->clear();
  395. throw ctkPluginDatabaseException(errorText.arg(stage == Prepare ? "prepare":"execute")
  396. .arg(statement).arg(query->lastError().text()).arg(parameters), errorType);
  397. }
  398. if (stage == Prepare)
  399. {
  400. foreach(const QVariant &bindValue, bindValues)
  401. query->addBindValue(bindValue);
  402. }
  403. }
  404. }
  405. void ctkPluginDatabase::close()
  406. {
  407. if (m_isDatabaseOpen)
  408. {
  409. QSqlDatabase database = QSqlDatabase::database(m_connectionName, false);
  410. if (database.isValid())
  411. {
  412. if(database.isOpen())
  413. {
  414. database.close();
  415. m_isDatabaseOpen = false;
  416. return;
  417. }
  418. }
  419. else
  420. {
  421. throw ctkPluginDatabaseException(QString("Problem closing database: Invalid connection %1").arg(m_connectionName));
  422. }
  423. }
  424. }
  425. void ctkPluginDatabase::setDatabasePath(const QString &databasePath)
  426. {
  427. m_databasePath = QDir::toNativeSeparators(databasePath);
  428. }
  429. QString ctkPluginDatabase::getDatabasePath() const
  430. {
  431. QString path;
  432. if(m_databasePath.isEmpty())
  433. {
  434. QSettings settings;
  435. path = settings.value("PluginDB/Path").toString();
  436. if (path.isEmpty())
  437. {
  438. path = QApplication::applicationDirPath();
  439. if (path.lastIndexOf("/") != path.length() -1)
  440. {
  441. path.append("/");
  442. }
  443. QString appName = QApplication::applicationName();
  444. appName.replace(" ", "");
  445. if (!appName.isEmpty())
  446. {
  447. path.append(appName + "_plugins.db");
  448. }
  449. else
  450. {
  451. path.append("pluginfw.db");
  452. qWarning() << "Warning: Using generic plugin database name. You should "
  453. "set an application name via QCoreApplication::setApplicationName()";
  454. }
  455. }
  456. path = QDir::toNativeSeparators(path);
  457. }
  458. else
  459. {
  460. path = m_databasePath;
  461. }
  462. qDebug() << "Using database:" << path;
  463. return path;
  464. }
  465. QByteArray ctkPluginDatabase::getPluginResource(long pluginId, const QString& res) const
  466. {
  467. checkConnection();
  468. QSqlDatabase database = QSqlDatabase::database(m_connectionName);
  469. QSqlQuery query(database);
  470. QString statement = "SELECT Resource FROM PluginResources WHERE PluginID=? AND ResourcePath=?";
  471. QString resourcePath = res.startsWith('/') ? res : QString("/") + res;
  472. QList<QVariant> bindValues;
  473. bindValues.append(qlonglong(pluginId));
  474. bindValues.append(resourcePath);
  475. executeQuery(&query, statement, bindValues);
  476. if (query.next())
  477. {
  478. return query.value(EBindIndex).toByteArray();
  479. }
  480. return QByteArray();
  481. }
  482. void ctkPluginDatabase::createTables()
  483. {
  484. QSqlDatabase database = QSqlDatabase::database(m_connectionName);
  485. QSqlQuery query(database);
  486. //Begin Transaction
  487. beginTransaction(&query, Write);
  488. QString statement("CREATE TABLE Plugins("
  489. "ID INTEGER PRIMARY KEY,"
  490. "Location TEXT NOT NULL UNIQUE,"
  491. "LocalPath TEXT NOT NULL UNIQUE,"
  492. "SymbolicName TEXT NOT NULL,"
  493. "Version TEXT NOT NULL,"
  494. "State INTEGER NOT NULL,"
  495. "Timestamp TEXT NOT NULL)");
  496. try
  497. {
  498. executeQuery(&query, statement);
  499. }
  500. catch (...)
  501. {
  502. rollbackTransaction(&query);
  503. throw;
  504. }
  505. statement = "CREATE TABLE PluginResources("
  506. "PluginID INTEGER NOT NULL,"
  507. "ResourcePath TEXT NOT NULL, "
  508. "Resource BLOB NOT NULL,"
  509. "FOREIGN KEY(PluginID) REFERENCES Plugins(ID) ON DELETE CASCADE)";
  510. try
  511. {
  512. executeQuery(&query, statement);
  513. }
  514. catch (...)
  515. {
  516. rollbackTransaction(&query);
  517. throw;
  518. }
  519. try
  520. {
  521. commitTransaction(&query);
  522. }
  523. catch (...)
  524. {
  525. rollbackTransaction(&query);
  526. throw;
  527. }
  528. }
  529. bool ctkPluginDatabase::checkTables() const
  530. {
  531. bool bTables(false);
  532. QStringList tables = QSqlDatabase::database(m_connectionName).tables();
  533. if (tables.contains(PLUGINS_TABLE)
  534. && tables.contains(PLUGIN_RESOURCES_TABLE))
  535. {
  536. bTables = true;
  537. }
  538. return bTables;
  539. }
  540. bool ctkPluginDatabase::dropTables()
  541. {
  542. //Execute transaction for deleting the database tables
  543. QSqlDatabase database = QSqlDatabase::database(m_connectionName);
  544. QSqlQuery query(database);
  545. QStringList expectedTables;
  546. expectedTables << PLUGINS_TABLE << PLUGIN_RESOURCES_TABLE;
  547. if (database.tables().count() > 0)
  548. {
  549. beginTransaction(&query, Write);
  550. QStringList actualTables = database.tables();
  551. foreach(const QString expectedTable, expectedTables)
  552. {
  553. if (actualTables.contains(expectedTable))
  554. {
  555. try
  556. {
  557. executeQuery(&query, QString("DROP TABLE ") + expectedTable);
  558. }
  559. catch (...)
  560. {
  561. rollbackTransaction(&query);
  562. throw;
  563. }
  564. }
  565. try
  566. {
  567. commitTransaction(&query);
  568. }
  569. catch (...)
  570. {
  571. rollbackTransaction(&query);
  572. throw;
  573. }
  574. }
  575. }
  576. return true;
  577. }
  578. bool ctkPluginDatabase::isOpen() const
  579. {
  580. return m_isDatabaseOpen;
  581. }
  582. void ctkPluginDatabase::checkConnection() const
  583. {
  584. if(!m_isDatabaseOpen)
  585. {
  586. throw ctkPluginDatabaseException("Database not open.", ctkPluginDatabaseException::DB_NOT_OPEN_ERROR);
  587. }
  588. if (!QSqlDatabase::database(m_connectionName).isValid())
  589. {
  590. throw ctkPluginDatabaseException(QString("Database connection invalid: %1").arg(m_connectionName),
  591. ctkPluginDatabaseException::DB_CONNECTION_INVALID);
  592. }
  593. }
  594. void ctkPluginDatabase::beginTransaction(QSqlQuery *query, TransactionType type)
  595. {
  596. bool success;
  597. if (type == Read)
  598. success = query->exec(QLatin1String("BEGIN"));
  599. else
  600. success = query->exec(QLatin1String("BEGIN IMMEDIATE"));
  601. if (!success) {
  602. int result = query->lastError().number();
  603. if (result == 26 || result == 11) //SQLITE_NOTADB || SQLITE_CORRUPT
  604. {
  605. throw ctkPluginDatabaseException(QString("ctkPluginFramework: Database file is corrupt or invalid: %1").arg(getDatabasePath()),
  606. ctkPluginDatabaseException::DB_FILE_INVALID);
  607. }
  608. else if (result == 8) //SQLITE_READONLY
  609. {
  610. throw ctkPluginDatabaseException(QString("ctkPluginFramework: Insufficient permissions to write to database: %1").arg(getDatabasePath()),
  611. ctkPluginDatabaseException::DB_WRITE_ERROR);
  612. }
  613. else
  614. throw ctkPluginDatabaseException(QString("ctkPluginFramework: ") + query->lastError().text(),
  615. ctkPluginDatabaseException::DB_SQL_ERROR);
  616. }
  617. }
  618. void ctkPluginDatabase::commitTransaction(QSqlQuery *query)
  619. {
  620. Q_ASSERT(query != 0);
  621. query->finish();
  622. query->clear();
  623. if (!query->exec(QLatin1String("COMMIT")))
  624. {
  625. throw ctkPluginDatabaseException(QString("ctkPluginFramework: ") + query->lastError().text(),
  626. ctkPluginDatabaseException::DB_SQL_ERROR);
  627. }
  628. }
  629. void ctkPluginDatabase::rollbackTransaction(QSqlQuery *query)
  630. {
  631. Q_ASSERT(query !=0);
  632. query->finish();
  633. query->clear();
  634. if (!query->exec(QLatin1String("ROLLBACK")))
  635. {
  636. throw ctkPluginDatabaseException(QString("ctkPluginFramework: ") + query->lastError().text(),
  637. ctkPluginDatabaseException::DB_SQL_ERROR);
  638. }
  639. }
  640. QList<ctkPluginArchive*> ctkPluginDatabase::getPluginArchives() const
  641. {
  642. checkConnection();
  643. QSqlQuery query(QSqlDatabase::database(m_connectionName));
  644. QString statement("SELECT ID, Location, LocalPath FROM Plugins WHERE State != ?");
  645. QList<QVariant> bindValues;
  646. bindValues.append(ctkPlugin::UNINSTALLED);
  647. executeQuery(&query, statement, bindValues);
  648. QList<ctkPluginArchive*> archives;
  649. while (query.next())
  650. {
  651. const long id = query.value(EBindIndex).toLongLong();
  652. const QUrl location(query.value(EBindIndex1).toString());
  653. const QString localPath(query.value(EBindIndex2).toString());
  654. if (id <= 0 || location.isEmpty() || localPath.isEmpty())
  655. {
  656. throw ctkPluginDatabaseException(QString("Database integrity corrupted, row %1 contains empty values.").arg(id),
  657. ctkPluginDatabaseException::DB_FILE_INVALID);
  658. }
  659. try
  660. {
  661. ctkPluginArchive* pa = new ctkPluginArchive(m_PluginStorage, location, localPath, id);
  662. archives.append(pa);
  663. }
  664. catch (const ctkPluginException& exc)
  665. {
  666. qWarning() << exc;
  667. }
  668. }
  669. return archives;
  670. }