Browse Source

- Made torrent deletion from hard-drive safer

adaptive-webui-19844
Christophe Dumez 18 years ago
parent
commit
e82cb2188e
  1. 1
      Changelog
  2. 25
      TODO
  3. 6
      src/bittorrent.cpp
  4. 58
      src/deleteThread.h
  5. 155
      src/misc.h
  6. 15
      src/qtorrenthandle.cpp
  7. 3
      src/qtorrenthandle.h

1
Changelog

@ -49,6 +49,7 @@
- BUGFIX: Improved the way menu icons are installed to avoid problems on some systems - BUGFIX: Improved the way menu icons are installed to avoid problems on some systems
- BUGFIX: Improved incremental download - BUGFIX: Improved incremental download
- BUGFIX: Improved unicode support - BUGFIX: Improved unicode support
- BUGFIX: Made torrent deletion from hard-drive safer
- COSMETIC: Redesigned torrent properties a little - COSMETIC: Redesigned torrent properties a little
- COSMETIC: Redesigned options a little - COSMETIC: Redesigned options a little
- COSMETIC: Display more logs messages concerning features - COSMETIC: Display more logs messages concerning features

25
TODO

@ -71,26 +71,5 @@ LANGUAGES UPDATED:
- Russian *BETA5* - Russian *BETA5*
- Korean *BETA5* - Korean *BETA5*
beta4->beta5 changelog: beta5->beta6 changelog:
- FEATURE: Supports Bittorrent FAST extension - BUGFIX: Made torrent deletion from hard-drive safer
- FEATURE: Improved code handling torrents that have just finished checking
- FEATURE: Improved progress column sorting code
- FEATURE: Allow to remove url seeds, even hard-coded ones
- FEATURE: Improved code for handling of finished torrents
- FEATURE: Optimized download list refreshing a little
- FEATURE: Big code cleanup
- BUGFIX: Wait for torrent_paused_alert before saving fast resume data on exit
- BUGFIX: Wait for torrent_paused_alert before reloading a torrent for full allocation mode
- BUFFIG: Fixed overflow causing ratio data to be negative
- BUGFIX: Fixed progress column delayed sorting (after torrent finished checking)
- BUGFIX: Finished torrents were still displayed as checking when paused by libtorrent on full disk (hitted an assert)
- BUGFIX: Fixed the way icons are installed to avoid problems on some systems
- BUGFIX: Fixed qBittorrent version in .desktop file
- BUGFIX: Fixed session ratio value (was either 10. or 1.)
- BUGFIX: Improved incremental download
- BUGFIX: Fixed preview from seeding list
- BUGFIX: Fixed Alt+3 & Ctrl+F keyboard shortcuts for third tab
- BUGFIX: Improved unicode support
- BUGFIX: Add torrents in pause before applying settings to avoid race conditions
- I18N: Updated Italian, Polish, Portuguese, Brazilian, German, Russian, Korean and Spanish translations
- COSMETIC: Improved the way progress bars are rendered

6
src/bittorrent.cpp

@ -166,6 +166,10 @@ void bittorrent::deleteTorrent(QString hash, bool permanent) {
} }
QString savePath = h.save_path(); QString savePath = h.save_path();
QString fileName = h.name(); QString fileName = h.name();
QStringList files_path;
if(permanent){
files_path = h.files_path();
}
// Remove it from session // Remove it from session
s->remove_torrent(h.get_torrent_handle()); s->remove_torrent(h.get_torrent_handle());
// Remove it from torrent backup directory // Remove it from torrent backup directory
@ -208,7 +212,7 @@ void bittorrent::deleteTorrent(QString hash, bool permanent) {
// Remove from Hard drive // Remove from Hard drive
qDebug("Removing this on hard drive: %s", qPrintable(savePath+QDir::separator()+fileName)); qDebug("Removing this on hard drive: %s", qPrintable(savePath+QDir::separator()+fileName));
// Deleting in a thread to avoid GUI freeze // Deleting in a thread to avoid GUI freeze
deleter->deletePath(savePath+QDir::separator()+fileName); deleter->deleteTorrent(savePath, files_path);
} }
} }

58
src/deleteThread.h

@ -26,17 +26,19 @@
#include <QMutex> #include <QMutex>
#include <QWaitCondition> #include <QWaitCondition>
#include <QMutexLocker> #include <QMutexLocker>
#include <QPair>
#include "misc.h" #include "misc.h"
class subDeleteThread : public QThread { class subDeleteThread : public QThread {
Q_OBJECT Q_OBJECT
private: private:
QString path; QString save_path;
QStringList files_path;
bool abort; bool abort;
public: public:
subDeleteThread(QObject *parent, QString path) : QThread(parent), path(path){ subDeleteThread(QObject *parent, QString save_path, QStringList files_path) : QThread(parent), save_path(save_path), files_path(files_path){
abort = false; abort = false;
} }
@ -47,16 +49,15 @@ class subDeleteThread : public QThread {
signals: signals:
// For subthreads // For subthreads
void deletionSuccessST(subDeleteThread* st, QString path); void deletionSuccessST(subDeleteThread* st);
void deletionFailureST(subDeleteThread* st, QString path); void deletionFailureST(subDeleteThread* st);
protected: protected:
void run(){ void run(){
if(misc::removePath(path)) if(misc::removeTorrentSavePath(save_path, files_path))
emit deletionSuccessST(this, path); emit deletionSuccessST(this);
else else
emit deletionFailureST(this, path); emit deletionFailureST(this);
qDebug("deletion completed for %s", (const char*)path.toUtf8());
} }
}; };
@ -64,16 +65,12 @@ class deleteThread : public QThread {
Q_OBJECT Q_OBJECT
private: private:
QStringList path_list; QList<QPair<QString, QStringList> > torrents_list;
QMutex mutex; QMutex mutex;
QWaitCondition condition; QWaitCondition condition;
bool abort; bool abort;
QList<subDeleteThread*> subThreads; QList<subDeleteThread*> subThreads;
signals:
void deletionSuccess(QString path);
void deletionFailure(QString path);
public: public:
deleteThread(QObject* parent) : QThread(parent){ deleteThread(QObject* parent) : QThread(parent){
abort = false; abort = false;
@ -88,9 +85,9 @@ class deleteThread : public QThread {
wait(); wait();
} }
void deletePath(QString path){ void deleteTorrent(QString save_path, QStringList files_path){
QMutexLocker locker(&mutex); QMutexLocker locker(&mutex);
path_list << path; torrents_list << QPair<QString, QStringList>(save_path, files_path);
if(!isRunning()){ if(!isRunning()){
start(); start();
}else{ }else{
@ -104,18 +101,14 @@ class deleteThread : public QThread {
if(abort) if(abort)
return; return;
mutex.lock(); mutex.lock();
if(path_list.size() != 0){ if(torrents_list.size() != 0){
QString path = path_list.takeFirst(); QPair<QString, QStringList> torrent = torrents_list.takeFirst();
mutex.unlock(); mutex.unlock();
if(QFile::exists(path)){ subDeleteThread *st = new subDeleteThread(0, torrent.first, torrent.second);
subDeleteThread *st = new subDeleteThread(0, path); subThreads << st;
subThreads << st; connect(st, SIGNAL(deletionSuccessST(subDeleteThread*)), this, SLOT(deleteSubThread(subDeleteThread*)));
connect(st, SIGNAL(deletionSuccessST(subDeleteThread*, QString)), this, SLOT(propagateDeletionSuccess(subDeleteThread*, QString))); connect(st, SIGNAL(deletionFailureST(subDeleteThread*)), this, SLOT(deleteSubThread(subDeleteThread*)));
connect(st, SIGNAL(deletionFailureST(subDeleteThread*, QString)), this, SLOT(propagateDeletionFailure(subDeleteThread*, QString))); st->start();
st->start();
}else{
qDebug("%s does not exist, nothing to delete", (const char*)path.toUtf8());
}
}else{ }else{
condition.wait(&mutex); condition.wait(&mutex);
mutex.unlock(); mutex.unlock();
@ -123,22 +116,11 @@ class deleteThread : public QThread {
} }
} }
protected slots: protected slots:
void propagateDeletionSuccess(subDeleteThread* st, QString path){ void deleteSubThread(subDeleteThread* st){
int index = subThreads.indexOf(st);
Q_ASSERT(index != -1);
subThreads.removeAt(index);
delete st;
emit deletionSuccess(path);
qDebug("%s was successfully deleted", (const char*)path.toUtf8());
}
void propagateDeletionFailure(subDeleteThread* st, QString path){
int index = subThreads.indexOf(st); int index = subThreads.indexOf(st);
Q_ASSERT(index != -1); Q_ASSERT(index != -1);
subThreads.removeAt(index); subThreads.removeAt(index);
delete st; delete st;
emit deletionFailure(path);
std::cerr << "Could not delete path: " << (const char*)path.toUtf8() << ". Check if qBittorrent has the required rights.\n";
} }
}; };

155
src/misc.h

@ -27,13 +27,14 @@
#include <QObject> #include <QObject>
#include <QString> #include <QString>
#include <QByteArray> #include <QByteArray>
#include <QFileInfo>
#include <QDir> #include <QDir>
#include <QList> #include <QList>
#include <QPair> #include <QPair>
#include <QThread> #include <QThread>
#include <libtorrent/torrent_info.hpp> #include <libtorrent/torrent_info.hpp>
// #include "qtorrenthandle.h" #include "qtorrenthandle.h"
using namespace libtorrent; using namespace libtorrent;
#define MAX_CHAR_TMP 128 #define MAX_CHAR_TMP 128
@ -143,89 +144,89 @@ class misc : public QObject{
return qBtPath; return qBtPath;
} }
static bool removePath(QString path) { // Not used anymore because it is not safe
qDebug((QString::fromUtf8("file to delete:") + path).toUtf8()); // static bool removePath(QString path) {
if(!QFile::remove(path)) { // qDebug((QString::fromUtf8("file to delete:") + path).toUtf8());
// Probably a folder // if(!QFile::remove(path)) {
QDir current_dir(path); // // Probably a folder
if(current_dir.exists()) { // QDir current_dir(path);
//Remove sub items // if(current_dir.exists()) {
QStringList subItems = current_dir.entryList(); // //Remove sub items
QString item; // QStringList subItems = current_dir.entryList();
foreach(item, subItems) { // QString item;
if(item != QString::fromUtf8(".") && item != QString::fromUtf8("..")) { // foreach(item, subItems) {
qDebug("-> Removing "+(path+QDir::separator()+item).toUtf8()); // if(item != QString::fromUtf8(".") && item != QString::fromUtf8("..")) {
removePath(path+QDir::separator()+item); // qDebug("-> Removing "+(path+QDir::separator()+item).toUtf8());
} // removePath(path+QDir::separator()+item);
}
// Remove empty folder
if(current_dir.rmdir(path)) {
return true;
}else{
return false;
}
}else{
return false;
}
}
return true;
}
// FIXME: Not used yet because it is buggy
// static bool removeTorrentSavePath(QTorrentHandle h) {
// bool success = true;
// QString savePath = QDir::cleanPath(h.save_path() + QDir::separator());
// unsigned int nbFiles = h.num_files();
// QDir saveDir(savePath);
// // Check how many file there are
// if(nbFiles == 1){
// // Only one file, not in a folder
// QStringList filters;
// filters << h.file_at(0);
// QFileInfoList files = saveDir.entryInfoList(filters, QDir::Files);
// QFileInfo file;
// foreach(file, files){
// if(file.fileName() == h.file_at(0) && !file.isSymLink()){
// if(saveDir.remove(h.file_at(0))){
// qDebug("Deleted only file in torrent at %s", (savePath + h.file_at(0)).toUtf8().data());
// return true;
// }else{
// return false;
// } // }
// } // }
// } // // Remove empty folder
// std::cerr << "Could not delete only file in torrent at " << (savePath + h.file_at(0)).toUtf8().data() << '\n'; // if(current_dir.rmdir(path)) {
// return false; // return true;
// } // }else{
// QDir subDir(savePath + h.name()); // return false;
// // Torrent has several files in a subFolder
// for(unsigned int i=0; i<nbFiles; ++i){
// QString fileName = h.file_at(i);
// QStringList filters;
// filters << fileName;
// QFileInfoList files = saveDir.entryInfoList(filters, QDir::Files);
// QFileInfo file;
// foreach(file, files){
// if(file.fileName() == fileName && !file.isSymLink()){
// if(!subDir.remove(h.file_at(i))){
// success = false;
// std::cerr << "Could not delete file in folder at " << (savePath + h.name() + QDir::separator() + h.file_at(i)).toUtf8().data() << '\n';
// }else{
// qDebug("Deleted file in folder at %s", (savePath + h.name() + QDir::separator() + h.file_at(i)).toUtf8().data());
// }
// } // }
// break; // }else{
// return false;
// } // }
// } // }
// // try to remove topfolder if empty // return true;
// if(saveDir.rmdir(h.name())){
// qDebug("Removed top folder %s", (savePath+h.name()).toUtf8().data());
// }else{
// std::cerr << "Could not remove top folder " << (savePath+h.name()).toUtf8().data() << ", it was not empty\n";
// }
// return success;
// } // }
// safe function to remove a torrent from hard-drive
static bool removeTorrentSavePath(QString savePath, QStringList filesPath) {
bool success = true;
QDir saveDir(savePath);
QString path;
// Check how many file there are
if(filesPath.size() == 1){
// Only one file, not in a folder
path = filesPath.first();
if(QFile::exists(path)) {
if(QFile::remove(path)){
qDebug("Deleted only file in torrent at %s", path.toUtf8().data());
} else {
std::cerr << "Could not delete only file in torrent at " << path.toUtf8().data() << '\n';
success = false;
}
}else{
// File didn't exist, nothing to do
qDebug("Only file %s did not exist, nothing to delete", path.toUtf8().data());
}
// Try to remove parent folder if empty (and not save_dir)
QFileInfo fi(path);
QDir parentFolder = fi.absoluteDir();
while(parentFolder != saveDir) {
qDebug("trying to remove parent folder: %s", parentFolder.absolutePath().toUtf8().data());
if(!saveDir.rmdir(parentFolder.absolutePath())) break;
parentFolder.cdUp();
}
return success;
}
// Torrent has several files in a subFolder
foreach(path, filesPath) {
if(QFile::exists(path)) {
if(QFile::remove(path)){
qDebug("Deleted file in torrent at %s", path.toUtf8().data());
} else {
std::cerr << "Could not delete file in torrent at " << path.toUtf8().data() << '\n';
success = false;
}
} else {
qDebug("File %s did not exist, nothing to delete", path.toUtf8().data());
}
// Try to remove parent folder if empty (and not save_dir)
QFileInfo fi(path);
QDir parentFolder = fi.absoluteDir();
while(parentFolder != saveDir) {
qDebug("trying to remove parent folder: %s", parentFolder.absolutePath().toUtf8().data());
if(!saveDir.rmdir(parentFolder.absolutePath())) break;
parentFolder.cdUp();
}
}
return success;
}
static QString findFileInDir(QString dir_path, QString fileName) { static QString findFileInDir(QString dir_path, QString fileName) {
QDir dir(dir_path); QDir dir(dir_path);
if(dir.exists(fileName)) { if(dir.exists(fileName)) {

15
src/qtorrenthandle.cpp

@ -20,7 +20,9 @@
*/ */
#include <QString> #include <QString>
#include <QStringList>
#include <QFile> #include <QFile>
#include <QDir>
#include <QByteArray> #include <QByteArray>
#include "misc.h" #include "misc.h"
#include "qtorrenthandle.h" #include "qtorrenthandle.h"
@ -202,6 +204,19 @@ size_type QTorrentHandle::total_payload_upload() {
return s.total_payload_upload; return s.total_payload_upload;
} }
// Return a list of absolute paths corresponding
// to all files in a torrent
QStringList QTorrentHandle::files_path() const {
QString saveDir = misc::toQString(h.save_path().string()) + QDir::separator();
QStringList res;
torrent_info::file_iterator fi = t.begin_files();
while(fi != t.end_files()) {
res << QDir::cleanPath(saveDir + misc::toQString(fi->path.string()));
fi++;
}
return res;
}
// //
// Setters // Setters
// //

3
src/qtorrenthandle.h

@ -23,9 +23,11 @@
#define QTORRENTHANDLE_H #define QTORRENTHANDLE_H
#include <libtorrent/torrent_handle.hpp> #include <libtorrent/torrent_handle.hpp>
#include <libtorrent/torrent_info.hpp>
using namespace libtorrent; using namespace libtorrent;
class QString; class QString;
class QStringList;
// A wrapper for torrent_handle in libtorrent // A wrapper for torrent_handle in libtorrent
// to interact well with Qt types // to interact well with Qt types
@ -81,6 +83,7 @@ class QTorrentHandle {
void file_progress(std::vector<float>& fp); void file_progress(std::vector<float>& fp);
size_type total_payload_download(); size_type total_payload_download();
size_type total_payload_upload(); size_type total_payload_upload();
QStringList files_path() const;
// //
// Setters // Setters

Loading…
Cancel
Save