/* * Bittorrent Client using Qt and libtorrent. * Copyright (C) 2021 Vladimir Golovnev * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * * In addition, as a special exception, the copyright holders give permission to * link this program with the OpenSSL project's "OpenSSL" library (or with * modified versions of it that use the same license as the "OpenSSL" library), * and distribute the linked executables. You must obey the GNU General Public * License in all respects for all of the code used other than "OpenSSL". If you * modify file(s), you may extend this exception to your version of the file(s), * but you are not obligated to do so. If you do not wish to do so, delete this * exception statement from your version. */ #include "dbresumedatastorage.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "base/exceptions.h" #include "base/global.h" #include "base/logger.h" #include "base/profile.h" #include "base/utils/fs.h" #include "base/utils/string.h" #include "infohash.h" #include "loadtorrentparams.h" #include "torrentinfo.h" namespace { const char DB_CONNECTION_NAME[] = "ResumeDataStorage"; const int DB_VERSION = 1; const char DB_TABLE_META[] = "meta"; const char DB_TABLE_TORRENTS[] = "torrents"; struct Column { QString name; QString placeholder; }; Column makeColumn(const char *columnName) { return {QLatin1String(columnName), (QLatin1Char(':') + QLatin1String(columnName))}; } const Column DB_COLUMN_ID = makeColumn("id"); const Column DB_COLUMN_TORRENT_ID = makeColumn("torrent_id"); const Column DB_COLUMN_QUEUE_POSITION = makeColumn("queue_position"); const Column DB_COLUMN_NAME = makeColumn("name"); const Column DB_COLUMN_CATEGORY = makeColumn("category"); const Column DB_COLUMN_TAGS = makeColumn("tags"); const Column DB_COLUMN_TARGET_SAVE_PATH = makeColumn("target_save_path"); const Column DB_COLUMN_CONTENT_LAYOUT = makeColumn("content_layout"); const Column DB_COLUMN_RATIO_LIMIT = makeColumn("ratio_limit"); const Column DB_COLUMN_SEEDING_TIME_LIMIT = makeColumn("seeding_time_limit"); const Column DB_COLUMN_HAS_OUTER_PIECES_PRIORITY = makeColumn("has_outer_pieces_priority"); const Column DB_COLUMN_HAS_SEED_STATUS = makeColumn("has_seed_status"); const Column DB_COLUMN_OPERATING_MODE = makeColumn("operating_mode"); const Column DB_COLUMN_STOPPED = makeColumn("stopped"); const Column DB_COLUMN_RESUMEDATA = makeColumn("libtorrent_resume_data"); const Column DB_COLUMN_METADATA = makeColumn("metadata"); const Column DB_COLUMN_VALUE = makeColumn("value"); template QString fromLTString(const LTStr &str) { return QString::fromUtf8(str.data(), static_cast(str.size())); } QString quoted(const QString &name) { const QLatin1Char quote {'`'}; return (quote + name + quote); } QString makeCreateTableStatement(const QString &tableName, const QStringList &items) { return QString::fromLatin1("CREATE TABLE %1 (%2)").arg(quoted(tableName), items.join(QLatin1Char(','))); } QString makeInsertStatement(const QString &tableName, const QVector &columns) { QStringList names; names.reserve(columns.size()); QStringList values; values.reserve(columns.size()); for (const Column &column : columns) { names.append(quoted(column.name)); values.append(column.placeholder); } const QString jointNames = names.join(QLatin1Char(',')); const QString jointValues = values.join(QLatin1Char(',')); return QString::fromLatin1("INSERT INTO %1 (%2) VALUES (%3)") .arg(quoted(tableName), jointNames, jointValues); } QString makeOnConflictUpdateStatement(const Column &constraint, const QVector &columns) { QStringList names; names.reserve(columns.size()); QStringList values; values.reserve(columns.size()); for (const Column &column : columns) { names.append(quoted(column.name)); values.append(column.placeholder); } const QString jointNames = names.join(QLatin1Char(',')); const QString jointValues = values.join(QLatin1Char(',')); return QString::fromLatin1(" ON CONFLICT (%1) DO UPDATE SET (%2) = (%3)") .arg(quoted(constraint.name), jointNames, jointValues); } QString makeColumnDefinition(const Column &column, const char *definition) { return QString::fromLatin1("%1 %2").arg(quoted(column.name), QLatin1String(definition)); } } namespace BitTorrent { class DBResumeDataStorage::Worker final : public QObject { Q_DISABLE_COPY(Worker) public: Worker(const QString &dbPath, const QString &dbConnectionName); void openDatabase() const; void closeDatabase() const; void store(const TorrentID &id, const LoadTorrentParams &resumeData) const; void remove(const TorrentID &id) const; void storeQueue(const QVector &queue) const; private: const QString m_path; const QString m_connectionName; }; } BitTorrent::DBResumeDataStorage::DBResumeDataStorage(const QString &dbPath, QObject *parent) : ResumeDataStorage {parent} , m_ioThread {new QThread(this)} { const bool needCreateDB = !QFile::exists(dbPath); auto db = QSqlDatabase::addDatabase(QLatin1String("QSQLITE"), DB_CONNECTION_NAME); db.setDatabaseName(dbPath); if (!db.open()) throw RuntimeError(db.lastError().text()); if (needCreateDB) createDB(); m_asyncWorker = new Worker(dbPath, QLatin1String("ResumeDataStorageWorker")); m_asyncWorker->moveToThread(m_ioThread); connect(m_ioThread, &QThread::finished, m_asyncWorker, &QObject::deleteLater); m_ioThread->start(); RuntimeError *errPtr = nullptr; QMetaObject::invokeMethod(m_asyncWorker, [this, &errPtr]() { try { m_asyncWorker->openDatabase(); } catch (const RuntimeError &err) { errPtr = new RuntimeError(err); } }, Qt::BlockingQueuedConnection); if (errPtr) throw *errPtr; } BitTorrent::DBResumeDataStorage::~DBResumeDataStorage() { QMetaObject::invokeMethod(m_asyncWorker, &Worker::closeDatabase); QSqlDatabase::removeDatabase(DB_CONNECTION_NAME); m_ioThread->quit(); m_ioThread->wait(); } QVector BitTorrent::DBResumeDataStorage::registeredTorrents() const { const auto selectTorrentIDStatement = QString::fromLatin1("SELECT %1 FROM %2 ORDER BY %3;") .arg(quoted(DB_COLUMN_TORRENT_ID.name), quoted(DB_TABLE_TORRENTS), quoted(DB_COLUMN_QUEUE_POSITION.name)); auto db = QSqlDatabase::database(DB_CONNECTION_NAME); QSqlQuery query {db}; if (!query.exec(selectTorrentIDStatement)) throw RuntimeError(query.lastError().text()); QVector registeredTorrents; registeredTorrents.reserve(query.size()); while (query.next()) registeredTorrents.append(BitTorrent::TorrentID::fromString(query.value(0).toString())); return registeredTorrents; } std::optional BitTorrent::DBResumeDataStorage::load(const TorrentID &id) const { const QString selectTorrentStatement = QString(QLatin1String("SELECT * FROM %1 WHERE %2 = %3;")) .arg(quoted(DB_TABLE_TORRENTS), quoted(DB_COLUMN_TORRENT_ID.name), DB_COLUMN_TORRENT_ID.placeholder); auto db = QSqlDatabase::database(DB_CONNECTION_NAME); QSqlQuery query {db}; try { if (!query.prepare(selectTorrentStatement)) throw RuntimeError(query.lastError().text()); query.bindValue(DB_COLUMN_TORRENT_ID.placeholder, id.toString()); if (!query.exec()) throw RuntimeError(query.lastError().text()); if (!query.next()) throw RuntimeError(tr("Not found.")); } catch (const RuntimeError &err) { LogMsg(tr("Couldn't load resume data of torrent '%1'. Error: %2") .arg(id.toString(), err.message()), Log::CRITICAL); return std::nullopt; } LoadTorrentParams resumeData; resumeData.restored = true; resumeData.name = query.value(DB_COLUMN_NAME.name).toString(); resumeData.category = query.value(DB_COLUMN_CATEGORY.name).toString(); const QString tagsData = query.value(DB_COLUMN_TAGS.name).toString(); if (!tagsData.isEmpty()) { const QStringList tagList = tagsData.split(QLatin1Char(',')); resumeData.tags.insert(tagList.cbegin(), tagList.cend()); } resumeData.savePath = Profile::instance()->fromPortablePath( Utils::Fs::toUniformPath(query.value(DB_COLUMN_TARGET_SAVE_PATH.name).toString())); resumeData.hasSeedStatus = query.value(DB_COLUMN_HAS_SEED_STATUS.name).toBool(); resumeData.firstLastPiecePriority = query.value(DB_COLUMN_HAS_OUTER_PIECES_PRIORITY.name).toBool(); resumeData.ratioLimit = query.value(DB_COLUMN_RATIO_LIMIT.name).toInt() / 1000.0; resumeData.seedingTimeLimit = query.value(DB_COLUMN_SEEDING_TIME_LIMIT.name).toInt(); resumeData.contentLayout = Utils::String::toEnum( query.value(DB_COLUMN_CONTENT_LAYOUT.name).toString(), TorrentContentLayout::Original); resumeData.operatingMode = Utils::String::toEnum( query.value(DB_COLUMN_OPERATING_MODE.name).toString(), TorrentOperatingMode::AutoManaged); resumeData.stopped = query.value(DB_COLUMN_STOPPED.name).toBool(); const QByteArray bencodedResumeData = query.value(DB_COLUMN_RESUMEDATA.name).toByteArray(); lt::error_code ec; const lt::bdecode_node root = lt::bdecode(bencodedResumeData, ec); lt::add_torrent_params &p = resumeData.ltAddTorrentParams; p = lt::read_resume_data(root, ec); p.save_path = Profile::instance()->fromPortablePath(fromLTString(p.save_path)).toStdString(); const QByteArray bencodedMetadata = query.value(DB_COLUMN_METADATA.name).toByteArray(); auto metadata = TorrentInfo::load(bencodedMetadata); if (metadata.isValid()) p.ti = metadata.nativeInfo(); return resumeData; } void BitTorrent::DBResumeDataStorage::store(const TorrentID &id, const LoadTorrentParams &resumeData) const { QMetaObject::invokeMethod(m_asyncWorker, [this, id, resumeData]() { m_asyncWorker->store(id, resumeData); }); } void BitTorrent::DBResumeDataStorage::remove(const BitTorrent::TorrentID &id) const { QMetaObject::invokeMethod(m_asyncWorker, [this, id]() { m_asyncWorker->remove(id); }); } void BitTorrent::DBResumeDataStorage::storeQueue(const QVector &queue) const { QMetaObject::invokeMethod(m_asyncWorker, [this, queue]() { m_asyncWorker->storeQueue(queue); }); } void BitTorrent::DBResumeDataStorage::createDB() const { auto db = QSqlDatabase::database(DB_CONNECTION_NAME); if (!db.transaction()) throw RuntimeError(db.lastError().text()); QSqlQuery query {db}; try { const QStringList tableMetaItems = { makeColumnDefinition(DB_COLUMN_ID, "INTEGER PRIMARY KEY"), makeColumnDefinition(DB_COLUMN_NAME, "TEXT NOT NULL UNIQUE"), makeColumnDefinition(DB_COLUMN_VALUE, "BLOB") }; const QString createTableMetaQuery = makeCreateTableStatement(DB_TABLE_META, tableMetaItems); if (!query.exec(createTableMetaQuery)) throw RuntimeError(query.lastError().text()); const QString insertMetaVersionQuery = makeInsertStatement(DB_TABLE_META, {DB_COLUMN_NAME, DB_COLUMN_VALUE}); if (!query.prepare(insertMetaVersionQuery)) throw RuntimeError(query.lastError().text()); query.bindValue(DB_COLUMN_NAME.placeholder, QString::fromLatin1("version")); query.bindValue(DB_COLUMN_VALUE.placeholder, DB_VERSION); if (!query.exec()) throw RuntimeError(query.lastError().text()); const QStringList tableTorrentsItems = { makeColumnDefinition(DB_COLUMN_ID, "INTEGER PRIMARY KEY"), makeColumnDefinition(DB_COLUMN_TORRENT_ID, "BLOB NOT NULL UNIQUE"), makeColumnDefinition(DB_COLUMN_QUEUE_POSITION, "INTEGER NOT NULL DEFAULT -1"), makeColumnDefinition(DB_COLUMN_NAME, "TEXT"), makeColumnDefinition(DB_COLUMN_CATEGORY, "TEXT"), makeColumnDefinition(DB_COLUMN_TAGS, "TEXT"), makeColumnDefinition(DB_COLUMN_TARGET_SAVE_PATH, "TEXT"), makeColumnDefinition(DB_COLUMN_CONTENT_LAYOUT, "TEXT NOT NULL"), makeColumnDefinition(DB_COLUMN_RATIO_LIMIT, "INTEGER NOT NULL"), makeColumnDefinition(DB_COLUMN_SEEDING_TIME_LIMIT, "INTEGER NOT NULL"), makeColumnDefinition(DB_COLUMN_HAS_OUTER_PIECES_PRIORITY, "INTEGER NOT NULL"), makeColumnDefinition(DB_COLUMN_HAS_SEED_STATUS, "INTEGER NOT NULL"), makeColumnDefinition(DB_COLUMN_OPERATING_MODE, "TEXT NOT NULL"), makeColumnDefinition(DB_COLUMN_STOPPED, "INTEGER NOT NULL"), makeColumnDefinition(DB_COLUMN_RESUMEDATA, "BLOB NOT NULL"), makeColumnDefinition(DB_COLUMN_METADATA, "BLOB") }; const QString createTableTorrentsQuery = makeCreateTableStatement(DB_TABLE_TORRENTS, tableTorrentsItems); if (!query.exec(createTableTorrentsQuery)) throw RuntimeError(query.lastError().text()); if (!db.commit()) throw RuntimeError(db.lastError().text()); } catch (const RuntimeError &err) { db.rollback(); throw err; } } BitTorrent::DBResumeDataStorage::Worker::Worker(const QString &dbPath, const QString &dbConnectionName) : m_path {dbPath} , m_connectionName {dbConnectionName} { } void BitTorrent::DBResumeDataStorage::Worker::openDatabase() const { auto db = QSqlDatabase::addDatabase(QLatin1String("QSQLITE"), m_connectionName); db.setDatabaseName(m_path); if (!db.open()) throw RuntimeError(db.lastError().text()); } void BitTorrent::DBResumeDataStorage::Worker::closeDatabase() const { QSqlDatabase::removeDatabase(m_connectionName); } void BitTorrent::DBResumeDataStorage::Worker::store(const TorrentID &id, const LoadTorrentParams &resumeData) const { // We need to adjust native libtorrent resume data lt::add_torrent_params p = resumeData.ltAddTorrentParams; p.save_path = Profile::instance()->toPortablePath(QString::fromStdString(p.save_path)).toStdString(); if (resumeData.stopped) { p.flags |= lt::torrent_flags::paused; p.flags &= ~lt::torrent_flags::auto_managed; } else { // Torrent can be actually "running" but temporarily "paused" to perform some // service jobs behind the scenes so we need to restore it as "running" if (resumeData.operatingMode == BitTorrent::TorrentOperatingMode::AutoManaged) { p.flags |= lt::torrent_flags::auto_managed; } else { p.flags &= ~lt::torrent_flags::paused; p.flags &= ~lt::torrent_flags::auto_managed; } } QVector columns { DB_COLUMN_TORRENT_ID, DB_COLUMN_NAME, DB_COLUMN_CATEGORY, DB_COLUMN_TAGS, DB_COLUMN_TARGET_SAVE_PATH, DB_COLUMN_CONTENT_LAYOUT, DB_COLUMN_RATIO_LIMIT, DB_COLUMN_SEEDING_TIME_LIMIT, DB_COLUMN_HAS_OUTER_PIECES_PRIORITY, DB_COLUMN_HAS_SEED_STATUS, DB_COLUMN_OPERATING_MODE, DB_COLUMN_STOPPED, DB_COLUMN_RESUMEDATA }; // metadata is stored in separate column QByteArray bencodedMetadata; bencodedMetadata.reserve(512 * 1024); const std::shared_ptr torrentInfo = std::move(p.ti); if (torrentInfo) { const lt::create_torrent torrentCreator = lt::create_torrent(*torrentInfo); const lt::entry metadata = torrentCreator.generate(); lt::bencode(std::back_inserter(bencodedMetadata), metadata); columns.append(DB_COLUMN_METADATA); } QByteArray bencodedResumeData; bencodedResumeData.reserve(256 * 1024); lt::bencode(std::back_inserter(bencodedResumeData), lt::write_resume_data(p)); const QString insertTorrentStatement = makeInsertStatement(DB_TABLE_TORRENTS, columns) + makeOnConflictUpdateStatement(DB_COLUMN_TORRENT_ID, columns); auto db = QSqlDatabase::database(m_connectionName); QSqlQuery query {db}; try { if (!query.prepare(insertTorrentStatement)) throw RuntimeError(query.lastError().text()); query.bindValue(DB_COLUMN_TORRENT_ID.placeholder, id.toString()); query.bindValue(DB_COLUMN_NAME.placeholder, resumeData.name); query.bindValue(DB_COLUMN_CATEGORY.placeholder, resumeData.category); query.bindValue(DB_COLUMN_TAGS.placeholder, (resumeData.tags.isEmpty() ? QVariant(QVariant::String) : resumeData.tags.join(QLatin1String(",")))); query.bindValue(DB_COLUMN_TARGET_SAVE_PATH.placeholder, Profile::instance()->toPortablePath(resumeData.savePath)); query.bindValue(DB_COLUMN_CONTENT_LAYOUT.placeholder, Utils::String::fromEnum(resumeData.contentLayout)); query.bindValue(DB_COLUMN_RATIO_LIMIT.placeholder, static_cast(resumeData.ratioLimit * 1000)); query.bindValue(DB_COLUMN_SEEDING_TIME_LIMIT.placeholder, resumeData.seedingTimeLimit); query.bindValue(DB_COLUMN_HAS_OUTER_PIECES_PRIORITY.placeholder, resumeData.firstLastPiecePriority); query.bindValue(DB_COLUMN_HAS_SEED_STATUS.placeholder, resumeData.hasSeedStatus); query.bindValue(DB_COLUMN_OPERATING_MODE.placeholder, Utils::String::fromEnum(resumeData.operatingMode)); query.bindValue(DB_COLUMN_STOPPED.placeholder, resumeData.stopped); query.bindValue(DB_COLUMN_RESUMEDATA.placeholder, bencodedResumeData); if (torrentInfo) query.bindValue(DB_COLUMN_METADATA.placeholder, bencodedMetadata); if (!query.exec()) throw RuntimeError(query.lastError().text()); } catch (const RuntimeError &err) { LogMsg(tr("Couldn't store resume data for torrent '%1'. Error: %2") .arg(id.toString(), err.message()), Log::CRITICAL); } } void BitTorrent::DBResumeDataStorage::Worker::remove(const TorrentID &id) const { const auto deleteTorrentStatement = QString::fromLatin1("DELETE FROM %1 WHERE %2 = %3;") .arg(quoted(DB_TABLE_TORRENTS), quoted(DB_COLUMN_TORRENT_ID.name), DB_COLUMN_TORRENT_ID.placeholder); auto db = QSqlDatabase::database(m_connectionName); QSqlQuery query {db}; try { if (!query.prepare(deleteTorrentStatement)) throw RuntimeError(query.lastError().text()); query.bindValue(DB_COLUMN_TORRENT_ID.placeholder, id.toString()); if (!query.exec()) throw RuntimeError(query.lastError().text()); } catch (const RuntimeError &err) { LogMsg(tr("Couldn't delete resume data of torrent '%1'. Error: %2") .arg(id.toString(), err.message()), Log::CRITICAL); } } void BitTorrent::DBResumeDataStorage::Worker::storeQueue(const QVector &queue) const { const auto updateQueuePosStatement = QString::fromLatin1("UPDATE %1 SET %2 = %3 WHERE %4 = %5;") .arg(quoted(DB_TABLE_TORRENTS), quoted(DB_COLUMN_QUEUE_POSITION.name), DB_COLUMN_QUEUE_POSITION.placeholder , quoted(DB_COLUMN_TORRENT_ID.name), DB_COLUMN_TORRENT_ID.placeholder); auto db = QSqlDatabase::database(m_connectionName); try { if (!db.transaction()) throw RuntimeError(db.lastError().text()); QSqlQuery query {db}; try { if (!query.prepare(updateQueuePosStatement)) throw RuntimeError(query.lastError().text()); int pos = 0; for (const TorrentID &torrentID : queue) { query.bindValue(DB_COLUMN_TORRENT_ID.placeholder, torrentID.toString()); query.bindValue(DB_COLUMN_QUEUE_POSITION.placeholder, pos++); if (!query.exec()) throw RuntimeError(query.lastError().text()); } if (!db.commit()) throw RuntimeError(db.lastError().text()); } catch (const RuntimeError &err) { db.rollback(); throw err; } } catch (const RuntimeError &err) { LogMsg(tr("Couldn't store torrents queue positions. Error: %1") .arg(err.message()), Log::CRITICAL); } }