adatabase.cpp 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528
  1. /*
  2. *openPilot Log - A FOSS Pilot Logbook Application
  3. *Copyright (C) 2020 Felix Turowsky
  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 3 of the License, or
  8. *(at your option) any later version.
  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, see <https://www.gnu.org/licenses/>.
  17. */
  18. #include "adatabase.h"
  19. namespace experimental {
  20. ADataBase* ADataBase::instance = nullptr;
  21. ADataBase* ADataBase::getInstance()
  22. {
  23. if(!instance)
  24. instance = new ADataBase();
  25. return instance;
  26. }
  27. bool ADataBase::connect()
  28. {
  29. const QString driver("QSQLITE");
  30. if (!QSqlDatabase::isDriverAvailable(driver))
  31. return false;
  32. QDir directory("data");
  33. QString databaseLocation = directory.filePath("logbook.db");
  34. QSqlDatabase db = QSqlDatabase::addDatabase(driver);
  35. db.setDatabaseName(databaseLocation);
  36. if (!db.open())
  37. return false;
  38. DEB("Database connection established.");
  39. // Enable foreign key restrictions
  40. QSqlQuery query("PRAGMA foreign_keys = ON;");
  41. tableNames = db.tables();
  42. QStringList columnNames;
  43. for (const auto &table : tableNames) {
  44. columnNames.clear();
  45. QSqlRecord fields = db.record(table);
  46. for (int i = 0; i < fields.count(); i++) {
  47. columnNames.append(fields.field(i).name());
  48. tableColumns.insert(table, columnNames);
  49. }
  50. }
  51. DEB("Database Tables: " << tableNames);
  52. return true;
  53. }
  54. void ADataBase::disconnect()
  55. {
  56. auto db = ADataBase::database();
  57. db.close();
  58. db.removeDatabase(db.connectionName());
  59. DEB("Database connection closed.");
  60. }
  61. QSqlDatabase ADataBase::database()
  62. {
  63. return QSqlDatabase::database("qt_sql_default_connection");
  64. }
  65. bool ADataBase::commit(AEntry entry)
  66. {
  67. if (exists(entry)) {
  68. return update(entry);
  69. } else {
  70. return insert(entry);
  71. }
  72. }
  73. bool ADataBase::remove(AEntry entry)
  74. {
  75. if (!exists(entry)) {
  76. DEB("Error: Database entry not found.");
  77. lastError = "Database Error: Database entry not found.";
  78. return false;
  79. }
  80. QString statement = "DELETE FROM " + entry.getPosition().tableName +
  81. " WHERE ROWID=" + QString::number(entry.getPosition().rowId);
  82. QSqlQuery query(statement);
  83. if (query.lastError().type() == QSqlError::NoError)
  84. {
  85. DEB("Entry " << entry.getPosition().tableName << entry.getPosition().rowId << " removed.");
  86. emit dataBaseUpdated();
  87. lastError = QString();
  88. return true;
  89. } else {
  90. DEB("Unable to delete.");
  91. DEB("Query: " << statement);
  92. DEB("Query Error: " << query.lastError().text());
  93. lastError = query.lastError().text();
  94. return false;
  95. }
  96. }
  97. bool ADataBase::removeMany(QList<DataPosition> data_position_list)
  98. {
  99. int errorCount = 0;
  100. QSqlQuery query;
  101. query.prepare("BEGIN EXCLUSIVE TRANSACTION");
  102. query.exec();
  103. for (const auto data_position : data_position_list) {
  104. if (!exists(data_position)) {
  105. lastError = "Database Error: Database entry not found.";
  106. errorCount++;
  107. }
  108. QString statement = "DELETE FROM " + data_position.first +
  109. " WHERE ROWID=" + QString::number(data_position.second);
  110. query.prepare(statement);
  111. query.exec();
  112. if (!(query.lastError().type() == QSqlError::NoError))
  113. errorCount++;
  114. }
  115. if (errorCount == 0) {
  116. query.prepare("COMMIT");
  117. query.exec();
  118. if(query.lastError().type() == QSqlError::NoError) {
  119. emit dataBaseUpdated();
  120. lastError = QString();
  121. return true;
  122. } else {
  123. lastError = "Transaction unsuccessful (Interrupted). Error count: " + QString::number(errorCount);
  124. return false;
  125. }
  126. } else {
  127. query.prepare("ROLLBACK");
  128. query.exec();
  129. lastError = "Transaction unsuccessful (no changes have been made). Error count: " + QString::number(errorCount);
  130. return false;
  131. }
  132. }
  133. bool ADataBase::exists(AEntry entry)
  134. {
  135. if(entry.getPosition().second == 0)
  136. return false;
  137. //Check database for row id
  138. QString statement = "SELECT COUNT(*) FROM " + entry.getPosition().tableName +
  139. " WHERE ROWID=" + QString::number(entry.getPosition().rowId);
  140. QSqlQuery query;
  141. query.prepare(statement);
  142. query.setForwardOnly(true);
  143. query.exec();
  144. //this returns either 1 or 0 since row ids are unique
  145. if (!query.isActive()) {
  146. lastError = query.lastError().text();
  147. DEB("Query Error: " << query.lastError().text() << statement);
  148. return false;
  149. }
  150. query.next();
  151. int rowId = query.value(0).toInt();
  152. if (rowId) {
  153. DEB("Entry " << entry.getPosition() << " exists.");
  154. return true;
  155. } else {
  156. DEB("Database entry not found.");
  157. lastError = "Database Error: Database entry not found.";
  158. return false;
  159. }
  160. }
  161. bool ADataBase::exists(DataPosition data_position)
  162. {
  163. if(data_position.second == 0)
  164. return false;
  165. //Check database for row id
  166. QString statement = "SELECT COUNT(*) FROM " + data_position.first +
  167. " WHERE ROWID=" + QString::number(data_position.second);
  168. QSqlQuery query;
  169. query.prepare(statement);
  170. query.setForwardOnly(true);
  171. query.exec();
  172. //this returns either 1 or 0 since row ids are unique
  173. if (!query.isActive()) {
  174. lastError = query.lastError().text();
  175. DEB("Query Error: " << query.lastError().text() << statement);
  176. }
  177. query.next();
  178. int rowId = query.value(0).toInt();
  179. if (rowId) {
  180. DEB("Entry exists at DataPosition: " << data_position);
  181. return true;
  182. } else {
  183. DEB("No entry exists at DataPosition: " << data_position);
  184. lastError = "Database Error: Database entry not found.";
  185. return false;
  186. }
  187. }
  188. bool ADataBase::update(AEntry updated_entry)
  189. {
  190. auto data = updated_entry.getData();
  191. QString statement = "UPDATE " + updated_entry.getPosition().tableName + " SET ";
  192. for (auto i = data.constBegin(); i != data.constEnd(); ++i) {
  193. if (i.value() == QString()) {
  194. statement += i.key() + QLatin1String("=NULL") + QLatin1String(", ");
  195. } else {
  196. statement += i.key() + QLatin1String("=\"") + i.value() + QLatin1String("\", ");
  197. }
  198. }
  199. statement.chop(2); // Remove last comma
  200. statement.append(QLatin1String(" WHERE ROWID=") + QString::number(updated_entry.getPosition().rowId));
  201. DEB("UPDATE QUERY: " << statement);
  202. QSqlQuery query(statement);
  203. if (query.lastError().type() == QSqlError::NoError)
  204. {
  205. DEB("Entry successfully committed.");
  206. emit dataBaseUpdated();
  207. lastError = QString();
  208. return true;
  209. } else {
  210. DEB("Unable to commit.");
  211. DEB("Query: " << statement);
  212. DEB("Query Error: " << query.lastError().text());
  213. lastError = query.lastError().text();
  214. return false;
  215. }
  216. }
  217. bool ADataBase::insert(AEntry new_entry)
  218. {
  219. auto data = new_entry.getData();
  220. DEB("Inserting...");
  221. QString statement = "INSERT INTO " + new_entry.getPosition().tableName + QLatin1String(" (");
  222. QMap<QString, QString>::iterator i;
  223. for (i = data.begin(); i != data.end(); ++i) {
  224. statement += i.key() + QLatin1String(", ");
  225. }
  226. statement.chop(2);
  227. statement += QLatin1String(") VALUES (");
  228. for (i = data.begin(); i != data.end(); ++i) {
  229. if (i.value() == "") {
  230. statement += QLatin1String("NULL, ");
  231. } else {
  232. statement += QLatin1String("\"") + i.value() + QLatin1String("\", ");
  233. }
  234. }
  235. statement.chop(2);
  236. statement += QLatin1String(")");
  237. DEB(statement);
  238. QSqlQuery query(statement);
  239. //check result.
  240. if (query.lastError().type() == QSqlError::NoError)
  241. {
  242. DEB("Entry successfully committed.");
  243. emit dataBaseUpdated();
  244. lastError = QString();
  245. return true;
  246. } else {
  247. DEB("Unable to commit.");
  248. DEB("Query: " << statement);
  249. DEB("Query Error: " << query.lastError().text());
  250. lastError = query.lastError().text();
  251. return false;
  252. }
  253. }
  254. TableData ADataBase::getEntryData(DataPosition data_position)
  255. {
  256. // check table exists
  257. if (!tableNames.contains(data_position.first)) {
  258. DEB(data_position.first << " not a table in the database. Unable to retreive Entry data.");
  259. return TableData();
  260. }
  261. //Check Database for rowId
  262. QString statement = "SELECT COUNT(*) FROM " + data_position.first
  263. + " WHERE ROWID=" + QString::number(data_position.second);
  264. QSqlQuery check_query;
  265. check_query.prepare(statement);
  266. check_query.setForwardOnly(true);
  267. check_query.exec();
  268. if (check_query.lastError().type() != QSqlError::NoError) {
  269. DEB("SQL error: " << check_query.lastError().text());
  270. DEB("Statement: " << statement);
  271. lastError = check_query.lastError().text();
  272. return TableData();
  273. }
  274. check_query.next();
  275. if (check_query.value(0).toInt() == 0) {
  276. DEB("No Entry found for row id: " << data_position.second );
  277. lastError = "Database Error: Database entry not found.";
  278. return TableData();
  279. }
  280. // Retreive TableData
  281. DEB("Retreiving data for row id: " << data_position.second);
  282. statement = "SELECT * FROM " + data_position.first
  283. + " WHERE ROWID=" + QString::number(data_position.second);
  284. QSqlQuery select_query;
  285. select_query.prepare(statement);
  286. select_query.setForwardOnly(true);
  287. select_query.exec();
  288. if (select_query.lastError().type() != QSqlError::NoError) {
  289. DEB("SQL error: " << select_query.lastError().text());
  290. DEB("Statement: " << statement);
  291. lastError = select_query.lastError().text();
  292. return TableData();
  293. }
  294. select_query.next();
  295. TableData entry_data;
  296. for (const auto &column : tableColumns.value(data_position.first)) {
  297. entry_data.insert(column, select_query.value(column).toString());
  298. }
  299. return entry_data;
  300. }
  301. AEntry ADataBase::getEntry(DataPosition data_position)
  302. {
  303. AEntry entry(data_position);
  304. entry.setData(getEntryData(data_position));
  305. return entry;
  306. }
  307. APilotEntry ADataBase::getPilotEntry(RowId row_id)
  308. {
  309. APilotEntry pilot_entry(row_id);
  310. pilot_entry.setData(getEntryData(pilot_entry.getPosition()));
  311. return pilot_entry;
  312. }
  313. ATailEntry ADataBase::getTailEntry(RowId row_id)
  314. {
  315. ATailEntry tail_entry(row_id);
  316. tail_entry.setData(getEntryData(tail_entry.getPosition()));
  317. return tail_entry;
  318. }
  319. AAircraftEntry ADataBase::getAircraftEntry(RowId row_id)
  320. {
  321. AAircraftEntry aircraft_entry(row_id);
  322. aircraft_entry.setData(getEntryData(aircraft_entry.getPosition()));
  323. return aircraft_entry;
  324. }
  325. AFlightEntry ADataBase::getFlightEntry(RowId row_id)
  326. {
  327. AFlightEntry flight_entry(row_id);
  328. flight_entry.setData(getEntryData(flight_entry.getPosition()));
  329. return flight_entry;
  330. }
  331. const QStringList ADataBase::getCompletionList(ADataBase::DatabaseTarget target)
  332. {
  333. QString statement;
  334. switch (target) {
  335. case pilots:
  336. statement.append("SELECT piclastname||\", \"||picfirstname FROM pilots");
  337. break;
  338. case aircraft:
  339. statement.append("SELECT make||\" \"||model FROM aircraft WHERE model IS NOT NULL "
  340. "UNION "
  341. "SELECT make||\" \"||model||\"-\"||variant FROM aircraft WHERE variant IS NOT NULL");
  342. break;
  343. case airport_identifier_all:
  344. statement.append("SELECT icao FROM airports UNION SELECT iata FROM airports");
  345. break;
  346. case registrations:
  347. statement.append("SELECT registration FROM tails");
  348. break;
  349. case companies:
  350. statement.append("SELECT company FROM pilots");
  351. break;
  352. default:
  353. DEB("Not a valid completer target for this function.");
  354. return QStringList();
  355. }
  356. QSqlQuery query;
  357. query.prepare(statement);
  358. query.setForwardOnly(true);
  359. query.exec();
  360. if(!query.isActive()) {
  361. lastError = query.lastError().text();
  362. return QStringList();
  363. }
  364. QStringList completer_list;
  365. while (query.next())
  366. completer_list.append(query.value(0).toString());
  367. completer_list.sort();
  368. completer_list.removeAll(QString(""));
  369. completer_list.removeDuplicates();
  370. return completer_list;
  371. }
  372. const QMap<QString, int> ADataBase::getIdMap(ADataBase::DatabaseTarget target)
  373. {
  374. QString statement;
  375. switch (target) {
  376. case pilots:
  377. statement.append("SELECT ROWID, piclastname||\", \"||picfirstname FROM pilots");
  378. break;
  379. case aircraft:
  380. statement.append("SELECT ROWID, make||\" \"||model FROM aircraft WHERE model IS NOT NULL "
  381. "UNION "
  382. "SELECT ROWID, make||\" \"||model||\"-\"||variant FROM aircraft WHERE variant IS NOT NULL");
  383. break;
  384. case airport_identifier_icao:
  385. statement.append("SELECT ROWID, icao FROM airports");
  386. break;
  387. case airport_identifier_iata:
  388. statement.append("SELECT ROWID, iata FROM airports WHERE iata NOT NULL");
  389. break;
  390. case airport_names:
  391. statement.append("SELECT ROWID, name FROM airports");
  392. break;
  393. case tails:
  394. statement.append("SELECT ROWID, registration FROM tails");
  395. break;
  396. default:
  397. DEB("Not a valid completer target for this function.");
  398. return QMap<QString, int>();
  399. }
  400. auto id_map = QMap<QString, int>();
  401. auto query = QSqlQuery(statement);
  402. if (!query.isActive()) {
  403. DEB("No result found. Check Query and Error.");
  404. DEB("Query: " << statement);
  405. DEB("Error: " << query.lastError().text());
  406. lastError = query.lastError().text();
  407. return QMap<QString, int>();
  408. } else {
  409. QVector<QString> query_result;
  410. while (query.next()) {
  411. id_map.insert(query.value(1).toString(), query.value(0).toInt());
  412. }
  413. return id_map;
  414. }
  415. }
  416. int ADataBase::getLastEntry(ADataBase::DatabaseTarget target)
  417. {
  418. QString statement = "SELECT MAX(ROWID) FROM ";
  419. switch (target) {
  420. case pilots:
  421. statement.append("pilots");
  422. break;
  423. case aircraft:
  424. statement.append("aircraft");
  425. break;
  426. case tails:
  427. statement.append("tails");
  428. break;
  429. default:
  430. DEB("Not a valid completer target for this function.");
  431. return 0;
  432. }
  433. auto query = QSqlQuery(statement);
  434. if (query.first()) {
  435. return query.value(0).toInt();
  436. } else {
  437. lastError = "Database Error: Database entry not found.";
  438. DEB("No entry found.");
  439. return 0;
  440. }
  441. }
  442. QVector<QString> ADataBase::customQuery(QString statement, int return_values)
  443. {
  444. QSqlQuery query(statement);
  445. query.exec();
  446. if (!query.first()) {
  447. DEB("No result found. Check Query and Error.");
  448. DEB("Error: " << query.lastError().text());
  449. DEB("Statement: " << statement);
  450. lastError = query.lastError().text();
  451. return QVector<QString>();
  452. } else {
  453. query.first();
  454. query.previous();
  455. QVector<QString> result;
  456. while (query.next()) {
  457. for (int i = 0; i < return_values ; i++) {
  458. result.append(query.value(i).toString());
  459. }
  460. }
  461. emit dataBaseUpdated();
  462. lastError = QString();
  463. return result;
  464. }
  465. }
  466. ADataBase* aDB() { return ADataBase::getInstance(); }
  467. }