qBittorrent/src/base/bittorrent/torrentcreatorthread.cpp

255 lines
8.5 KiB
C++
Raw Normal View History

2015-04-19 18:17:47 +03:00
/*
* Bittorrent Client using Qt and libtorrent.
2018-04-14 22:53:45 +03:00
* Copyright (C) 2010 Christophe Dumez <chris@qbittorrent.org>
2015-04-19 18:17:47 +03:00
*
* 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.
*/
2016-04-28 19:02:38 +08:00
#include "torrentcreatorthread.h"
#include <fstream>
2015-04-19 18:17:47 +03:00
#include <libtorrent/bencode.hpp>
2016-04-28 19:02:38 +08:00
#include <libtorrent/create_torrent.hpp>
#include <libtorrent/file_storage.hpp>
2017-11-21 18:23:39 +08:00
#include <libtorrent/torrent_info.hpp>
2015-04-19 18:17:47 +03:00
#include <QDirIterator>
2016-04-28 19:02:38 +08:00
#include <QFile>
#include <QFileInfo>
#include <QHash>
2015-04-19 18:17:47 +03:00
#include "base/exceptions.h"
#include "base/global.h"
#include "base/utils/compare.h"
2015-09-25 11:10:05 +03:00
#include "base/utils/fs.h"
#include "base/utils/io.h"
#include "base/version.h"
#include "ltunderlyingtype.h"
2015-04-19 18:17:47 +03:00
2017-11-21 18:23:39 +08:00
namespace
2015-04-19 18:17:47 +03:00
{
2017-11-21 18:23:39 +08:00
// do not include files and folders whose
// name starts with a .
bool fileFilter(const std::string &f)
{
return !Utils::Fs::fileName(QString::fromStdString(f)).startsWith('.');
}
2020-09-09 15:08:14 +08:00
#if (LIBTORRENT_VERSION_NUM >= 20000)
lt::create_flags_t toNativeTorrentFormatFlag(const BitTorrent::TorrentFormat torrentFormat)
{
2020-11-16 10:02:11 +03:00
switch (torrentFormat)
{
2020-09-09 15:08:14 +08:00
case BitTorrent::TorrentFormat::V1:
return lt::create_torrent::v1_only;
case BitTorrent::TorrentFormat::Hybrid:
return {};
case BitTorrent::TorrentFormat::V2:
return lt::create_torrent::v2_only;
}
return {};
}
#endif
2015-04-19 18:17:47 +03:00
}
2017-11-21 18:23:39 +08:00
using namespace BitTorrent;
2015-04-19 18:17:47 +03:00
TorrentCreatorThread::TorrentCreatorThread(QObject *parent)
: QThread(parent)
{
}
TorrentCreatorThread::~TorrentCreatorThread()
{
requestInterruption();
wait();
2015-04-19 18:17:47 +03:00
}
void TorrentCreatorThread::create(const TorrentCreatorParams &params)
2015-04-19 18:17:47 +03:00
{
m_params = params;
2015-04-19 18:17:47 +03:00
start();
}
void TorrentCreatorThread::sendProgressSignal(int currentPieceIdx, int totalPieces)
2015-04-19 18:17:47 +03:00
{
emit updateProgress(static_cast<int>((currentPieceIdx * 100.) / totalPieces));
2015-04-19 18:17:47 +03:00
}
void TorrentCreatorThread::checkInterruptionRequested() const
{
if (isInterruptionRequested())
throw RuntimeError(tr("Operation aborted"));
}
2015-04-19 18:17:47 +03:00
void TorrentCreatorThread::run()
{
emit updateProgress(0);
2020-11-16 10:02:11 +03:00
try
{
const QString parentPath = Utils::Fs::branchPath(m_params.inputPath) + '/';
const Utils::Compare::NaturalLessThan<Qt::CaseInsensitive> naturalLessThan {};
2015-04-19 18:17:47 +03:00
// Adding files to the torrent
lt::file_storage fs;
2020-11-16 10:02:11 +03:00
if (QFileInfo(m_params.inputPath).isFile())
{
lt::add_files(fs, Utils::Fs::toNativePath(m_params.inputPath).toStdString(), fileFilter);
}
2020-11-16 10:02:11 +03:00
else
{
// need to sort the file names by natural sort order
QStringList dirs = {m_params.inputPath};
QDirIterator dirIter {m_params.inputPath, (QDir::AllDirs | QDir::NoDotAndDotDot), QDirIterator::Subdirectories};
2020-11-16 10:02:11 +03:00
while (dirIter.hasNext())
{
dirIter.next();
dirs += dirIter.filePath();
}
std::sort(dirs.begin(), dirs.end(), naturalLessThan);
QStringList fileNames;
2019-01-06 18:31:32 +08:00
QHash<QString, qint64> fileSizeMap;
2020-11-16 10:02:11 +03:00
for (const auto &dir : asConst(dirs))
{
QStringList tmpNames; // natural sort files within each dir
QDirIterator fileIter(dir, QDir::Files);
2020-11-16 10:02:11 +03:00
while (fileIter.hasNext())
{
fileIter.next();
const QString relFilePath = fileIter.filePath().mid(parentPath.length());
tmpNames += relFilePath;
fileSizeMap[relFilePath] = fileIter.fileInfo().size();
}
std::sort(tmpNames.begin(), tmpNames.end(), naturalLessThan);
fileNames += tmpNames;
}
for (const auto &fileName : asConst(fileNames))
fs.add_file(fileName.toStdString(), fileSizeMap[fileName]);
}
checkInterruptionRequested();
2015-04-19 18:17:47 +03:00
2020-09-09 15:08:14 +08:00
#if (LIBTORRENT_VERSION_NUM >= 20000)
lt::create_torrent newTorrent {fs, m_params.pieceSize, toNativeTorrentFormatFlag(m_params.torrentFormat)};
#else
lt::create_torrent newTorrent(fs, m_params.pieceSize, m_params.paddedFileSizeLimit
, (m_params.isAlignmentOptimized ? lt::create_torrent::optimize_alignment : lt::create_flags_t {}));
2020-09-09 15:08:14 +08:00
#endif
2015-04-19 18:17:47 +03:00
// Add url seeds
2020-11-16 10:02:11 +03:00
for (QString seed : asConst(m_params.urlSeeds))
{
seed = seed.trimmed();
if (!seed.isEmpty())
newTorrent.add_url_seed(seed.toStdString());
}
2015-04-19 18:17:47 +03:00
int tier = 0;
2020-11-16 10:02:11 +03:00
for (const QString &tracker : asConst(m_params.trackers))
{
if (tracker.isEmpty())
2015-04-19 18:17:47 +03:00
++tier;
else
newTorrent.add_tracker(tracker.trimmed().toStdString(), tier);
2015-04-19 18:17:47 +03:00
}
2015-04-19 18:17:47 +03:00
// calculate the hash for all pieces
lt::set_piece_hashes(newTorrent, Utils::Fs::toNativePath(parentPath).toStdString()
, [this, &newTorrent](const lt::piece_index_t n)
{
checkInterruptionRequested();
sendProgressSignal(static_cast<LTUnderlyingType<lt::piece_index_t>>(n), newTorrent.num_pieces());
});
2015-04-19 18:17:47 +03:00
// Set qBittorrent as creator and add user comment to
// torrent_info structure
newTorrent.set_creator("qBittorrent " QBT_VERSION);
newTorrent.set_comment(m_params.comment.toUtf8().constData());
2015-04-19 18:17:47 +03:00
// Is private ?
newTorrent.set_priv(m_params.isPrivate);
checkInterruptionRequested();
2015-04-19 18:17:47 +03:00
lt::entry entry = newTorrent.generate();
// add source field
if (!m_params.source.isEmpty())
entry["info"]["source"] = m_params.source.toStdString();
checkInterruptionRequested();
// create the torrent
QFile outfile {m_params.savePath};
2020-11-16 10:02:11 +03:00
if (!outfile.open(QIODevice::WriteOnly))
throw RuntimeError(outfile.errorString());
2015-04-19 18:17:47 +03:00
checkInterruptionRequested();
lt::bencode(Utils::IO::FileDeviceOutputIterator {outfile}, entry);
2020-11-16 10:02:11 +03:00
if (outfile.error() != QFileDevice::NoError)
throw RuntimeError(outfile.errorString());
2015-04-19 18:17:47 +03:00
outfile.close();
emit updateProgress(100);
emit creationSuccess(m_params.savePath, parentPath);
2015-04-19 18:17:47 +03:00
}
catch (const RuntimeError &err)
{
emit creationFailure(tr("Create new torrent file failed. Reason: %1.").arg(err.message()));
}
catch (const std::exception &err)
2020-11-16 10:02:11 +03:00
{
emit creationFailure(tr("Create new torrent file failed. Reason: %1.").arg(QString::fromLocal8Bit(err.what())));
2015-04-19 18:17:47 +03:00
}
}
2020-09-09 15:08:14 +08:00
#if (LIBTORRENT_VERSION_NUM >= 20000)
int TorrentCreatorThread::calculateTotalPieces(const QString &inputPath, const int pieceSize, const TorrentFormat torrentFormat)
#else
int TorrentCreatorThread::calculateTotalPieces(const QString &inputPath, const int pieceSize, const bool isAlignmentOptimized, const int paddedFileSizeLimit)
2020-09-09 15:08:14 +08:00
#endif
{
if (inputPath.isEmpty())
return 0;
lt::file_storage fs;
lt::add_files(fs, Utils::Fs::toNativePath(inputPath).toStdString(), fileFilter);
2020-09-09 15:08:14 +08:00
#if (LIBTORRENT_VERSION_NUM >= 20000)
return lt::create_torrent {fs, pieceSize, toNativeTorrentFormatFlag(torrentFormat)}.num_pieces();
#else
return lt::create_torrent(fs, pieceSize, paddedFileSizeLimit
, (isAlignmentOptimized ? lt::create_torrent::optimize_alignment : lt::create_flags_t {})).num_pieces();
2020-09-09 15:08:14 +08:00
#endif
}