1
0
mirror of https://github.com/d47081/qBittorrent.git synced 2025-01-12 15:57:57 +00:00

Use QSaveFile wherever applicable

expected.hpp was fetched from:
b803e3c07b/include/nonstd/expected.hpp
This commit is contained in:
Chocobo1 2021-09-05 12:29:32 +08:00
parent 81139c0098
commit 21f72baae2
No known key found for this signature in database
GPG Key ID: 210D9C873253A68C
17 changed files with 2597 additions and 88 deletions

View File

@ -28,7 +28,6 @@
#include "upgrade.h" #include "upgrade.h"
#include <QFile>
#include <QMetaEnum> #include <QMetaEnum>
#include <QVector> #include <QVector>
@ -37,6 +36,7 @@
#include "base/profile.h" #include "base/profile.h"
#include "base/settingsstorage.h" #include "base/settingsstorage.h"
#include "base/utils/fs.h" #include "base/utils/fs.h"
#include "base/utils/io.h"
#include "base/utils/string.h" #include "base/utils/string.h"
namespace namespace
@ -53,17 +53,10 @@ namespace
if (!newData.isEmpty() || oldData.isEmpty()) if (!newData.isEmpty() || oldData.isEmpty())
return; return;
QFile file(savePath); const nonstd::expected<void, QString> result = Utils::IO::saveToFile(savePath, oldData);
if (!file.open(QIODevice::WriteOnly)) if (!result)
{ {
LogMsg(errorMsgFormat.arg(savePath, file.errorString()) , Log::WARNING); LogMsg(errorMsgFormat.arg(savePath, result.error()) , Log::WARNING);
return;
}
if (file.write(oldData) != oldData.size())
{
file.close();
Utils::Fs::forceRemove(savePath);
LogMsg(errorMsgFormat.arg(savePath, QObject::tr("Write incomplete")) , Log::WARNING);
return; return;
} }

2517
src/base/3rdparty/expected.hpp vendored Normal file

File diff suppressed because it is too large Load Diff

View File

@ -1,5 +1,6 @@
add_library(qbt_base STATIC add_library(qbt_base STATIC
# headers # headers
3rdparty/expected.hpp
algorithm.h algorithm.h
asyncfilestorage.h asyncfilestorage.h
bittorrent/abstractfilestorage.h bittorrent/abstractfilestorage.h

View File

@ -30,7 +30,8 @@
#include <QDebug> #include <QDebug>
#include <QMetaObject> #include <QMetaObject>
#include <QSaveFile>
#include "base/utils/io.h"
AsyncFileStorage::AsyncFileStorage(const QString &storageFolderPath, QObject *parent) AsyncFileStorage::AsyncFileStorage(const QString &storageFolderPath, QObject *parent)
: QObject(parent) : QObject(parent)
@ -69,10 +70,10 @@ void AsyncFileStorage::store_impl(const QString &fileName, const QByteArray &dat
const QString filePath = m_storageDir.absoluteFilePath(fileName); const QString filePath = m_storageDir.absoluteFilePath(fileName);
qDebug() << "AsyncFileStorage: Saving data to" << filePath; qDebug() << "AsyncFileStorage: Saving data to" << filePath;
QSaveFile file(filePath); const nonstd::expected<void, QString> result = Utils::IO::saveToFile(filePath, data);
if (!file.open(QIODevice::WriteOnly) || (file.write(data) != data.length()) || !file.commit()) if (!result)
{ {
qDebug() << "AsyncFileStorage: Failed to save data"; qDebug() << "AsyncFileStorage: Failed to save data";
emit failed(filePath, file.errorString()); emit failed(filePath, result.error());
} }
} }

View File

@ -1,4 +1,5 @@
HEADERS += \ HEADERS += \
$$PWD/3rdparty/expected.hpp \
$$PWD/algorithm.h \ $$PWD/algorithm.h \
$$PWD/asyncfilestorage.h \ $$PWD/asyncfilestorage.h \
$$PWD/bittorrent/abstractfilestorage.h \ $$PWD/bittorrent/abstractfilestorage.h \

View File

@ -29,7 +29,6 @@
#include "bencoderesumedatastorage.h" #include "bencoderesumedatastorage.h"
#include <libtorrent/bdecode.hpp> #include <libtorrent/bdecode.hpp>
#include <libtorrent/bencode.hpp>
#include <libtorrent/entry.hpp> #include <libtorrent/entry.hpp>
#include <libtorrent/read_resume_data.hpp> #include <libtorrent/read_resume_data.hpp>
#include <libtorrent/torrent_info.hpp> #include <libtorrent/torrent_info.hpp>
@ -37,7 +36,6 @@
#include <QByteArray> #include <QByteArray>
#include <QRegularExpression> #include <QRegularExpression>
#include <QSaveFile>
#include <QThread> #include <QThread>
#include "base/algorithm.h" #include "base/algorithm.h"
@ -88,17 +86,6 @@ namespace
entryList.emplace_back(setValue.toStdString()); entryList.emplace_back(setValue.toStdString());
return entryList; return entryList;
} }
void writeEntryToFile(const QString &filepath, const lt::entry &data)
{
QSaveFile file {filepath};
if (!file.open(QIODevice::WriteOnly))
throw RuntimeError(file.errorString());
lt::bencode(Utils::IO::FileDeviceOutputIterator {file}, data);
if (file.error() != QFileDevice::NoError || !file.commit())
throw RuntimeError(file.errorString());
}
} }
BitTorrent::BencodeResumeDataStorage::BencodeResumeDataStorage(const QString &path, QObject *parent) BitTorrent::BencodeResumeDataStorage::BencodeResumeDataStorage(const QString &path, QObject *parent)
@ -355,14 +342,11 @@ void BitTorrent::BencodeResumeDataStorage::Worker::store(const TorrentID &id, co
metadataDict.insert(dataDict.extract("comment")); metadataDict.insert(dataDict.extract("comment"));
const QString torrentFilepath = m_resumeDataDir.absoluteFilePath(QString::fromLatin1("%1.torrent").arg(id.toString())); const QString torrentFilepath = m_resumeDataDir.absoluteFilePath(QString::fromLatin1("%1.torrent").arg(id.toString()));
try const nonstd::expected<void, QString> result = Utils::IO::saveToFile(torrentFilepath, metadata);
{ if (!result)
writeEntryToFile(torrentFilepath, metadata);
}
catch (const RuntimeError &err)
{ {
LogMsg(tr("Couldn't save torrent metadata to '%1'. Error: %2.") LogMsg(tr("Couldn't save torrent metadata to '%1'. Error: %2.")
.arg(torrentFilepath, err.message()), Log::CRITICAL); .arg(torrentFilepath, result.error()), Log::CRITICAL);
return; return;
} }
} }
@ -378,14 +362,11 @@ void BitTorrent::BencodeResumeDataStorage::Worker::store(const TorrentID &id, co
data["qBt-firstLastPiecePriority"] = resumeData.firstLastPiecePriority; data["qBt-firstLastPiecePriority"] = resumeData.firstLastPiecePriority;
const QString resumeFilepath = m_resumeDataDir.absoluteFilePath(QString::fromLatin1("%1.fastresume").arg(id.toString())); const QString resumeFilepath = m_resumeDataDir.absoluteFilePath(QString::fromLatin1("%1.fastresume").arg(id.toString()));
try const nonstd::expected<void, QString> result = Utils::IO::saveToFile(resumeFilepath, data);
{ if (!result)
writeEntryToFile(resumeFilepath, data);
}
catch (const RuntimeError &err)
{ {
LogMsg(tr("Couldn't save torrent resume data to '%1'. Error: %2.") LogMsg(tr("Couldn't save torrent resume data to '%1'. Error: %2.")
.arg(resumeFilepath, err.message()), Log::CRITICAL); .arg(resumeFilepath, result.error()), Log::CRITICAL);
} }
} }
@ -406,10 +387,10 @@ void BitTorrent::BencodeResumeDataStorage::Worker::storeQueue(const QVector<Torr
data += (torrentID.toString().toLatin1() + '\n'); data += (torrentID.toString().toLatin1() + '\n');
const QString filepath = m_resumeDataDir.absoluteFilePath(QLatin1String("queue")); const QString filepath = m_resumeDataDir.absoluteFilePath(QLatin1String("queue"));
QSaveFile file {filepath}; const nonstd::expected<void, QString> result = Utils::IO::saveToFile(filepath, data);
if (!file.open(QIODevice::WriteOnly) || (file.write(data) != data.size()) || !file.commit()) if (!result)
{ {
LogMsg(tr("Couldn't save data to '%1'. Error: %2") LogMsg(tr("Couldn't save data to '%1'. Error: %2")
.arg(filepath, file.errorString()), Log::CRITICAL); .arg(filepath, result.error()), Log::CRITICAL);
} }
} }

View File

@ -30,13 +30,11 @@
#include <fstream> #include <fstream>
#include <libtorrent/bencode.hpp>
#include <libtorrent/create_torrent.hpp> #include <libtorrent/create_torrent.hpp>
#include <libtorrent/file_storage.hpp> #include <libtorrent/file_storage.hpp>
#include <libtorrent/torrent_info.hpp> #include <libtorrent/torrent_info.hpp>
#include <QDirIterator> #include <QDirIterator>
#include <QFile>
#include <QFileInfo> #include <QFileInfo>
#include <QHash> #include <QHash>
@ -209,16 +207,9 @@ void TorrentCreatorThread::run()
checkInterruptionRequested(); checkInterruptionRequested();
// create the torrent // create the torrent
QFile outfile {m_params.savePath}; const nonstd::expected<void, QString> result = Utils::IO::saveToFile(m_params.savePath, entry);
if (!outfile.open(QIODevice::WriteOnly)) if (!result)
throw RuntimeError(outfile.errorString()); throw RuntimeError(result.error());
checkInterruptionRequested();
lt::bencode(Utils::IO::FileDeviceOutputIterator {outfile}, entry);
if (outfile.error() != QFileDevice::NoError)
throw RuntimeError(outfile.errorString());
outfile.close();
emit updateProgress(100); emit updateProgress(100);
emit creationSuccess(m_params.savePath, parentPath); emit creationSuccess(m_params.savePath, parentPath);

View File

@ -28,7 +28,6 @@
#include "torrentinfo.h" #include "torrentinfo.h"
#include <libtorrent/bencode.hpp>
#include <libtorrent/create_torrent.hpp> #include <libtorrent/create_torrent.hpp>
#include <libtorrent/error_code.hpp> #include <libtorrent/error_code.hpp>
@ -172,14 +171,9 @@ void TorrentInfo::saveToFile(const QString &path) const
{ {
const auto torrentCreator = lt::create_torrent(*nativeInfo()); const auto torrentCreator = lt::create_torrent(*nativeInfo());
const lt::entry torrentEntry = torrentCreator.generate(); const lt::entry torrentEntry = torrentCreator.generate();
const nonstd::expected<void, QString> result = Utils::IO::saveToFile(path, torrentEntry);
QFile torrentFile {path}; if (!result)
if (!torrentFile.open(QIODevice::WriteOnly)) throw RuntimeError(result.error());
throw RuntimeError(torrentFile.errorString());
lt::bencode(Utils::IO::FileDeviceOutputIterator {torrentFile}, torrentEntry);
if (torrentFile.error() != QFileDevice::NoError)
throw RuntimeError(torrentFile.errorString());
} }
catch (const lt::system_error &err) catch (const lt::system_error &err)
{ {

View File

@ -34,6 +34,7 @@
#include "base/utils/fs.h" #include "base/utils/fs.h"
#include "base/utils/gzip.h" #include "base/utils/gzip.h"
#include "base/utils/io.h"
#include "base/utils/misc.h" #include "base/utils/misc.h"
const int MAX_REDIRECTIONS = 20; // the common value for web browsers const int MAX_REDIRECTIONS = 20; // the common value for web browsers
@ -43,12 +44,7 @@ namespace
bool saveToFile(const QByteArray &replyData, QString &filePath) bool saveToFile(const QByteArray &replyData, QString &filePath)
{ {
if (!filePath.isEmpty()) if (!filePath.isEmpty())
{ return Utils::IO::saveToFile(filePath, replyData).has_value();
QFile file {filePath};
if (!file.open(QIODevice::WriteOnly))
return false;
return (file.write(replyData) == replyData.length());
}
QTemporaryFile tmpfile {Utils::Fs::tempPath() + "XXXXXX"}; QTemporaryFile tmpfile {Utils::Fs::tempPath() + "XXXXXX"};
tmpfile.setAutoRemove(false); tmpfile.setAutoRemove(false);

View File

@ -31,7 +31,6 @@
#include <QDateTime> #include <QDateTime>
#include <QDir> #include <QDir>
#include <QFile>
#include <QHostAddress> #include <QHostAddress>
#include <QLocale> #include <QLocale>
@ -40,6 +39,7 @@
#include "base/profile.h" #include "base/profile.h"
#include "base/utils/fs.h" #include "base/utils/fs.h"
#include "base/utils/gzip.h" #include "base/utils/gzip.h"
#include "base/utils/io.h"
#include "downloadmanager.h" #include "downloadmanager.h"
#include "geoipdatabase.h" #include "geoipdatabase.h"
@ -452,15 +452,16 @@ void GeoIPManager::downloadFinished(const DownloadResult &result)
if (!QDir(targetPath).exists()) if (!QDir(targetPath).exists())
QDir().mkpath(targetPath); QDir().mkpath(targetPath);
QFile targetFile(QString::fromLatin1("%1/%2").arg(targetPath, GEODB_FILENAME)); const auto path = QString::fromLatin1("%1/%2").arg(targetPath, GEODB_FILENAME);
if (!targetFile.open(QFile::WriteOnly) || (targetFile.write(data) != data.length())) const nonstd::expected<void, QString> result = Utils::IO::saveToFile(path, data);
if (result)
{ {
LogMsg(tr("Couldn't save downloaded IP geolocation database file. Reason: %1") LogMsg(tr("Successfully updated IP geolocation database."), Log::INFO);
.arg(targetFile.errorString()), Log::WARNING);
} }
else else
{ {
LogMsg(tr("Successfully updated IP geolocation database."), Log::INFO); LogMsg(tr("Couldn't save downloaded IP geolocation database file. Reason: %1")
.arg(result.error()), Log::WARNING);
} }
} }
else else

View File

@ -33,7 +33,6 @@
#include <QJsonDocument> #include <QJsonDocument>
#include <QJsonObject> #include <QJsonObject>
#include <QJsonValue> #include <QJsonValue>
#include <QSaveFile>
#include <QThread> #include <QThread>
#include <QTimer> #include <QTimer>
#include <QUrl> #include <QUrl>

View File

@ -34,7 +34,6 @@
#include <QJsonDocument> #include <QJsonDocument>
#include <QJsonObject> #include <QJsonObject>
#include <QJsonValue> #include <QJsonValue>
#include <QSaveFile>
#include <QString> #include <QString>
#include <QThread> #include <QThread>

View File

@ -40,7 +40,6 @@
#include <QJsonDocument> #include <QJsonDocument>
#include <QJsonObject> #include <QJsonObject>
#include <QJsonValue> #include <QJsonValue>
#include <QSaveFile>
#include <QSet> #include <QSet>
#include <QTextStream> #include <QTextStream>
#include <QThread> #include <QThread>
@ -60,6 +59,7 @@
#include "base/settingsstorage.h" #include "base/settingsstorage.h"
#include "base/tagset.h" #include "base/tagset.h"
#include "base/utils/fs.h" #include "base/utils/fs.h"
#include "base/utils/io.h"
#include "base/utils/string.h" #include "base/utils/string.h"
using namespace std::chrono_literals; using namespace std::chrono_literals;
@ -370,13 +370,13 @@ void TorrentFilesWatcher::store() const
jsonObj[watchedFolder] = serializeWatchedFolderOptions(options); jsonObj[watchedFolder] = serializeWatchedFolderOptions(options);
} }
const QString path = QDir(specialFolderLocation(SpecialFolder::Config)).absoluteFilePath(CONF_FILE_NAME);
const QByteArray data = QJsonDocument(jsonObj).toJson(); const QByteArray data = QJsonDocument(jsonObj).toJson();
const nonstd::expected<void, QString> result = Utils::IO::saveToFile(path, data);
QSaveFile confFile {QDir(specialFolderLocation(SpecialFolder::Config)).absoluteFilePath(CONF_FILE_NAME)}; if (!result)
if (!confFile.open(QIODevice::WriteOnly) || (confFile.write(data) != data.size()) || !confFile.commit())
{ {
LogMsg(tr("Couldn't store Watched Folders configuration to %1. Error: %2") LogMsg(tr("Couldn't store Watched Folders configuration to %1. Error: %2")
.arg(confFile.fileName(), confFile.errorString()), Log::WARNING); .arg(path, result.error()), Log::WARNING);
} }
} }

View File

@ -28,8 +28,13 @@
#include "io.h" #include "io.h"
#include <libtorrent/bencode.hpp>
#include <libtorrent/entry.hpp>
#include <QByteArray> #include <QByteArray>
#include <QFileDevice> #include <QFileDevice>
#include <QSaveFile>
#include <QString>
Utils::IO::FileDeviceOutputIterator::FileDeviceOutputIterator(QFileDevice &device, const int bufferSize) Utils::IO::FileDeviceOutputIterator::FileDeviceOutputIterator(QFileDevice &device, const int bufferSize)
: m_device {&device} : m_device {&device}
@ -60,3 +65,24 @@ Utils::IO::FileDeviceOutputIterator &Utils::IO::FileDeviceOutputIterator::operat
} }
return *this; return *this;
} }
nonstd::expected<void, QString> Utils::IO::saveToFile(const QString &path, const QByteArray &data)
{
QSaveFile file {path};
if (!file.open(QIODevice::WriteOnly) || (file.write(data) != data.size()) || !file.commit())
return nonstd::make_unexpected(file.errorString());
return {};
}
nonstd::expected<void, QString> Utils::IO::saveToFile(const QString &path, const lt::entry &data)
{
QSaveFile file {path};
if (!file.open(QIODevice::WriteOnly))
return nonstd::make_unexpected(file.errorString());
const int bencodedDataSize = lt::bencode(Utils::IO::FileDeviceOutputIterator {file}, data);
if ((file.size() != bencodedDataSize) || !file.commit())
return nonstd::make_unexpected(file.errorString());
return {};
}

View File

@ -31,8 +31,13 @@
#include <iterator> #include <iterator>
#include <memory> #include <memory>
#include <libtorrent/fwd.hpp>
#include "base/3rdparty/expected.hpp"
class QByteArray; class QByteArray;
class QFileDevice; class QFileDevice;
class QString;
namespace Utils::IO namespace Utils::IO
{ {
@ -74,4 +79,7 @@ namespace Utils::IO
std::shared_ptr<QByteArray> m_buffer; std::shared_ptr<QByteArray> m_buffer;
int m_bufferSize; int m_bufferSize;
}; };
nonstd::expected<void, QString> saveToFile(const QString &path, const QByteArray &data);
nonstd::expected<void, QString> saveToFile(const QString &path, const lt::entry &data);
} }

View File

@ -49,6 +49,7 @@
#include "base/rss/rss_session.h" #include "base/rss/rss_session.h"
#include "base/utils/compare.h" #include "base/utils/compare.h"
#include "base/utils/fs.h" #include "base/utils/fs.h"
#include "base/utils/io.h"
#include "base/utils/string.h" #include "base/utils/string.h"
#include "gui/autoexpandabledialog.h" #include "gui/autoexpandabledialog.h"
#include "gui/torrentcategorydialog.h" #include "gui/torrentcategorydialog.h"
@ -452,12 +453,12 @@ void AutomatedRssDownloader::on_exportBtn_clicked()
path += EXT_LEGACY; path += EXT_LEGACY;
} }
const QByteArray rules {RSS::AutoDownloader::instance()->exportRules(format)}; const QByteArray rules = RSS::AutoDownloader::instance()->exportRules(format);
QFile file {path}; const nonstd::expected<void, QString> result = Utils::IO::saveToFile(path, rules);
if (!file.open(QFile::WriteOnly) || (file.write(rules) != rules.length())) if (!result)
{ {
QMessageBox::critical(this, tr("I/O Error") QMessageBox::critical(this, tr("I/O Error")
, tr("Failed to create the destination file. Reason: %1").arg(file.errorString())); , tr("Failed to create the destination file. Reason: %1").arg(result.error()));
} }
} }