/* * Bittorrent Client using Qt and libtorrent. * Copyright (C) 2015 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. */ #ifndef UPGRADE_H #define UPGRADE_H #include #if LIBTORRENT_VERSION_NUM >= 10100 #include #endif #include #include #if LIBTORRENT_VERSION_NUM < 10100 #include #endif #include #include #ifndef DISABLE_GUI #include #endif #include #include #include "base/logger.h" #include "base/utils/fs.h" #include "base/utils/misc.h" #include "base/utils/string.h" #include "base/preferences.h" #include "base/qinisettings.h" bool userAcceptsUpgrade() { #ifdef DISABLE_GUI std::cout << std::endl << "*** " << qPrintable(QObject::tr("Upgrade")) << " ***" << std::endl; char ret = '\0'; do { std::cout << qPrintable(QObject::tr("You updated from an older version that saved things differently. You must migrate to the new saving system. You will not be able to use an older version than v3.3.0 again. Continue? [y/n]")) << std::endl; ret = getchar(); // Read pressed key } while ((ret != 'y') && (ret != 'Y') && (ret != 'n') && (ret != 'N')); if ((ret == 'y') || (ret == 'Y')) return true; #else QMessageBox msgBox; msgBox.setText(QObject::tr("You updated from an older version that saved things differently. You must migrate to the new saving system. If you continue, you will not be able to use an older version than v3.3.0 again.")); msgBox.setWindowTitle(QObject::tr("Upgrade")); msgBox.addButton(QMessageBox::Abort); msgBox.addButton(QMessageBox::Ok); msgBox.setDefaultButton(QMessageBox::Abort); msgBox.show(); // Need to be shown or to moveToCenter does not work msgBox.move(Utils::Misc::screenCenter(&msgBox)); if (msgBox.exec() == QMessageBox::Ok) return true; #endif return false; } bool upgradeResumeFile(const QString &filepath, const QVariantHash &oldTorrent = QVariantHash()) { QFile file1(filepath); if (!file1.open(QIODevice::ReadOnly)) return false; QByteArray data = file1.readAll(); file1.close(); libtorrent::error_code ec; #if LIBTORRENT_VERSION_NUM < 10100 libtorrent::lazy_entry fastOld; libtorrent::lazy_bdecode(data.constData(), data.constData() + data.size(), fastOld, ec); if (ec || (fastOld.type() != libtorrent::lazy_entry::dict_t)) return false; #else libtorrent::bdecode_node fastOld; libtorrent::bdecode(data.constData(), data.constData() + data.size(), fastOld, ec); if (ec || (fastOld.type() != libtorrent::bdecode_node::dict_t)) return false; #endif libtorrent::entry fastNew; fastNew = fastOld; bool v3_3 = false; int queuePosition = 0; QString outFilePath = filepath; QRegExp rx(QLatin1String("([A-Fa-f0-9]{40})\\.fastresume\\.(\\d+)$")); if (rx.indexIn(filepath) != -1) { // old v3.3.x format queuePosition = rx.cap(2).toInt(); v3_3 = true; outFilePath.replace(QRegExp("\\.\\d+$"), ""); } else { queuePosition = fastOld.dict_find_int_value("qBt-queuePosition", 0); fastNew["qBt-name"] = Utils::String::toStdString(oldTorrent.value("name").toString()); fastNew["qBt-tempPathDisabled"] = false; } // in versions < 3.3 we have -1 for seeding torrents, so we convert it to 0 fastNew["qBt-queuePosition"] = (queuePosition >= 0 ? queuePosition : 0); QFile file2(outFilePath); QVector out; libtorrent::bencode(std::back_inserter(out), fastNew); if (file2.open(QIODevice::WriteOnly)) { if (file2.write(&out[0], out.size()) == out.size()) { if (v3_3) Utils::Fs::forceRemove(filepath); return true; } } return false; } bool upgrade(bool ask = true) { // Upgrade preferences Preferences::instance()->upgrade(); QString backupFolderPath = Utils::Fs::expandPathAbs(Utils::Fs::QDesktopServicesDataLocation() + "BT_backup"); QDir backupFolderDir(backupFolderPath); // **************************************************************************************** // Silently converts old v3.3.x .fastresume files QStringList backupFiles_3_3 = backupFolderDir.entryList( QStringList(QLatin1String("*.fastresume.*")), QDir::Files, QDir::Unsorted); foreach (const QString &backupFile, backupFiles_3_3) upgradeResumeFile(backupFolderDir.absoluteFilePath(backupFile)); // **************************************************************************************** #ifdef Q_OS_MAC // native .plist QSettings *oldResumeSettings = new QSettings("qBittorrent", "qBittorrent-resume"); #else QIniSettings *oldResumeSettings = new QIniSettings("qBittorrent", "qBittorrent-resume"); #endif QString oldResumeFilename = oldResumeSettings->fileName(); QVariantHash oldResumeData = oldResumeSettings->value("torrents").toHash(); delete oldResumeSettings; if (oldResumeData.isEmpty()) { Utils::Fs::forceRemove(oldResumeFilename); return true; } if (ask && !userAcceptsUpgrade()) return false; QStringList backupFiles = backupFolderDir.entryList( QStringList(QLatin1String("*.fastresume")), QDir::Files, QDir::Unsorted); QRegExp rx(QLatin1String("^([A-Fa-f0-9]{40})\\.fastresume$")); foreach (QString backupFile, backupFiles) { if (rx.indexIn(backupFile) != -1) { if (upgradeResumeFile(backupFolderDir.absoluteFilePath(backupFile), oldResumeData[rx.cap(1)].toHash())) oldResumeData.remove(rx.cap(1)); else Logger::instance()->addMessage(QObject::tr("Couldn't migrate torrent with hash: %1").arg(rx.cap(1)), Log::WARNING); } else { Logger::instance()->addMessage(QObject::tr("Couldn't migrate torrent. Invalid fastresume file name: %1").arg(backupFile), Log::WARNING); } } foreach (const QString &hash, oldResumeData.keys()) { QVariantHash oldTorrent = oldResumeData[hash].toHash(); if (oldTorrent.value("is_magnet", false).toBool()) { libtorrent::entry resumeData; resumeData["qBt-magnetUri"] = Utils::String::toStdString(oldTorrent.value("magnet_uri").toString()); resumeData["qBt-paused"] = false; resumeData["qBt-forced"] = false; resumeData["qBt-savePath"] = Utils::String::toStdString(oldTorrent.value("save_path").toString()); resumeData["qBt-ratioLimit"] = Utils::String::toStdString(QString::number(oldTorrent.value("max_ratio", -2).toReal())); resumeData["qBt-label"] = Utils::String::toStdString(oldTorrent.value("label").toString()); resumeData["qBt-name"] = Utils::String::toStdString(oldTorrent.value("name").toString()); resumeData["qBt-seedStatus"] = oldTorrent.value("seed").toBool(); resumeData["qBt-tempPathDisabled"] = false; int queuePosition = oldTorrent.value("priority", 0).toInt(); resumeData["qBt-queuePosition"] = (queuePosition >= 0 ? queuePosition : 0); QString filename = QString("%1.fastresume").arg(hash); QString filepath = backupFolderDir.absoluteFilePath(filename); QFile resumeFile(filepath); QVector out; libtorrent::bencode(std::back_inserter(out), resumeData); if (resumeFile.open(QIODevice::WriteOnly)) resumeFile.write(&out[0], out.size()); } } int counter = 0; QString backupResumeFilename = oldResumeFilename + ".bak"; while (QFile::exists(backupResumeFilename)) { ++counter; backupResumeFilename = oldResumeFilename + ".bak" + QString::number(counter); } QFile::rename(oldResumeFilename, backupResumeFilename); return true; } #ifdef Q_OS_MAC void migratePlistToIni(const QString &application) { QIniSettings iniFile("qBittorrent", application); if (!iniFile.allKeys().isEmpty()) return; // We copy the contents of plist, only if inifile does not exist(is empty). QSettings *plistFile = new QSettings("qBittorrent", application); plistFile->setFallbacksEnabled(false); const QStringList plist = plistFile->allKeys(); if (!plist.isEmpty()) { foreach (const QString &key, plist) iniFile.setValue(key, plistFile->value(key)); plistFile->clear(); } QString plistPath = plistFile->fileName(); delete plistFile; Utils::Fs::forceRemove(plistPath); } void macMigratePlists() { migratePlistToIni("qBittorrent-data"); migratePlistToIni("qBittorrent-rss"); migratePlistToIni("qBittorrent"); } #endif // Q_OS_MAC #ifndef DISABLE_GUI void migrateRSS() { // Copy old feed items to new file if needed QIniSettings qBTRSS("qBittorrent", "qBittorrent-rss-feeds"); if (!qBTRSS.allKeys().isEmpty()) return; // We move the contents of RSS old_items only if inifile does not exist (is empty). QIniSettings qBTRSSLegacy("qBittorrent", "qBittorrent-rss"); QHash allOldItems = qBTRSSLegacy.value("old_items", QHash()).toHash(); if (!allOldItems.empty()) { qDebug("Moving %d old items for feeds to qBittorrent-rss-feeds", allOldItems.size()); qBTRSS.setValue("old_items", allOldItems); qBTRSSLegacy.remove("old_items"); } } #endif #endif // UPGRADE_H