|
|
@ -47,6 +47,7 @@ |
|
|
|
#include "algorithm.h" |
|
|
|
#include "algorithm.h" |
|
|
|
#include "base/bittorrent/magneturi.h" |
|
|
|
#include "base/bittorrent/magneturi.h" |
|
|
|
#include "base/bittorrent/torrentinfo.h" |
|
|
|
#include "base/bittorrent/torrentinfo.h" |
|
|
|
|
|
|
|
#include "base/global.h" |
|
|
|
#include "base/preferences.h" |
|
|
|
#include "base/preferences.h" |
|
|
|
|
|
|
|
|
|
|
|
namespace |
|
|
|
namespace |
|
|
@ -62,37 +63,33 @@ namespace |
|
|
|
FileSystemWatcher::FileSystemWatcher(QObject *parent) |
|
|
|
FileSystemWatcher::FileSystemWatcher(QObject *parent) |
|
|
|
: QFileSystemWatcher(parent) |
|
|
|
: QFileSystemWatcher(parent) |
|
|
|
{ |
|
|
|
{ |
|
|
|
m_filters << "*.torrent" << "*.magnet"; |
|
|
|
|
|
|
|
connect(this, &QFileSystemWatcher::directoryChanged, this, &FileSystemWatcher::scanLocalFolder); |
|
|
|
connect(this, &QFileSystemWatcher::directoryChanged, this, &FileSystemWatcher::scanLocalFolder); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
FileSystemWatcher::~FileSystemWatcher() |
|
|
|
FileSystemWatcher::~FileSystemWatcher() |
|
|
|
{ |
|
|
|
{ |
|
|
|
#ifndef Q_OS_WIN |
|
|
|
#ifndef Q_OS_WIN |
|
|
|
if (m_watchTimer) |
|
|
|
|
|
|
|
delete m_watchTimer; |
|
|
|
delete m_watchTimer; |
|
|
|
#endif |
|
|
|
#endif |
|
|
|
if (m_partialTorrentTimer) |
|
|
|
|
|
|
|
delete m_partialTorrentTimer; |
|
|
|
delete m_partialTorrentTimer; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
QStringList FileSystemWatcher::directories() const |
|
|
|
QStringList FileSystemWatcher::directories() const |
|
|
|
{ |
|
|
|
{ |
|
|
|
QStringList dirs; |
|
|
|
QStringList dirs = QFileSystemWatcher::directories(); |
|
|
|
#ifndef Q_OS_WIN |
|
|
|
#ifndef Q_OS_WIN |
|
|
|
if (m_watchTimer) { |
|
|
|
for (const QDir &dir : qAsConst(m_watchedFolders)) |
|
|
|
foreach (const QDir &dir, m_watchedFolders) |
|
|
|
|
|
|
|
dirs << dir.canonicalPath(); |
|
|
|
dirs << dir.canonicalPath(); |
|
|
|
} |
|
|
|
|
|
|
|
#endif |
|
|
|
#endif |
|
|
|
dirs << QFileSystemWatcher::directories(); |
|
|
|
|
|
|
|
return dirs; |
|
|
|
return dirs; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
void FileSystemWatcher::addPath(const QString &path) |
|
|
|
void FileSystemWatcher::addPath(const QString &path) |
|
|
|
{ |
|
|
|
{ |
|
|
|
|
|
|
|
if (path.isEmpty()) return; |
|
|
|
|
|
|
|
|
|
|
|
#if !defined Q_OS_WIN && !defined Q_OS_HAIKU |
|
|
|
#if !defined Q_OS_WIN && !defined Q_OS_HAIKU |
|
|
|
QDir dir(path); |
|
|
|
const QDir dir(path); |
|
|
|
if (!dir.exists()) return; |
|
|
|
if (!dir.exists()) return; |
|
|
|
|
|
|
|
|
|
|
|
// Check if the path points to a network file system or not
|
|
|
|
// Check if the path points to a network file system or not
|
|
|
@ -105,67 +102,43 @@ void FileSystemWatcher::addPath(const QString &path) |
|
|
|
if (!m_watchTimer) { |
|
|
|
if (!m_watchTimer) { |
|
|
|
m_watchTimer = new QTimer(this); |
|
|
|
m_watchTimer = new QTimer(this); |
|
|
|
connect(m_watchTimer, &QTimer::timeout, this, &FileSystemWatcher::scanNetworkFolders); |
|
|
|
connect(m_watchTimer, &QTimer::timeout, this, &FileSystemWatcher::scanNetworkFolders); |
|
|
|
m_watchTimer->start(WATCH_INTERVAL); // 5 sec
|
|
|
|
m_watchTimer->start(WATCH_INTERVAL); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
return; |
|
|
|
} |
|
|
|
} |
|
|
|
else { |
|
|
|
|
|
|
|
#endif |
|
|
|
#endif |
|
|
|
|
|
|
|
|
|
|
|
// Normal mode
|
|
|
|
// Normal mode
|
|
|
|
qDebug("FS Watching is watching %s in normal mode", qUtf8Printable(path)); |
|
|
|
qDebug("FS Watching is watching %s in normal mode", qUtf8Printable(path)); |
|
|
|
QFileSystemWatcher::addPath(path); |
|
|
|
QFileSystemWatcher::addPath(path); |
|
|
|
scanLocalFolder(path); |
|
|
|
scanLocalFolder(path); |
|
|
|
#if !defined Q_OS_WIN && !defined Q_OS_HAIKU |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
#endif |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
void FileSystemWatcher::removePath(const QString &path) |
|
|
|
void FileSystemWatcher::removePath(const QString &path) |
|
|
|
{ |
|
|
|
{ |
|
|
|
#ifndef Q_OS_WIN |
|
|
|
#ifndef Q_OS_WIN |
|
|
|
QDir dir(path); |
|
|
|
if (m_watchedFolders.removeOne(path)) { |
|
|
|
for (int i = 0; i < m_watchedFolders.count(); ++i) { |
|
|
|
|
|
|
|
if (QDir(m_watchedFolders.at(i)) == dir) { |
|
|
|
|
|
|
|
m_watchedFolders.removeAt(i); |
|
|
|
|
|
|
|
if (m_watchedFolders.isEmpty()) |
|
|
|
if (m_watchedFolders.isEmpty()) |
|
|
|
delete m_watchTimer; |
|
|
|
delete m_watchTimer; |
|
|
|
return; |
|
|
|
return; |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
#endif |
|
|
|
#endif |
|
|
|
// Normal mode
|
|
|
|
// Normal mode
|
|
|
|
QFileSystemWatcher::removePath(path); |
|
|
|
QFileSystemWatcher::removePath(path); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
void FileSystemWatcher::scanLocalFolder(QString path) |
|
|
|
void FileSystemWatcher::scanLocalFolder(const QString &path) |
|
|
|
{ |
|
|
|
{ |
|
|
|
qDebug("scanLocalFolder(%s) called", qUtf8Printable(path)); |
|
|
|
processTorrentsInDir(path); |
|
|
|
QStringList torrents; |
|
|
|
|
|
|
|
// Local folders scan
|
|
|
|
|
|
|
|
addTorrentsFromDir(QDir(path), torrents); |
|
|
|
|
|
|
|
// Report detected torrent files
|
|
|
|
|
|
|
|
if (!torrents.empty()) { |
|
|
|
|
|
|
|
qDebug("The following files are being reported: %s", qUtf8Printable(torrents.join("\n"))); |
|
|
|
|
|
|
|
emit torrentsAdded(torrents); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef Q_OS_WIN |
|
|
|
void FileSystemWatcher::scanNetworkFolders() |
|
|
|
void FileSystemWatcher::scanNetworkFolders() |
|
|
|
{ |
|
|
|
{ |
|
|
|
#ifndef Q_OS_WIN |
|
|
|
for (const QDir &dir : qAsConst(m_watchedFolders)) |
|
|
|
qDebug("scanNetworkFolders() called"); |
|
|
|
processTorrentsInDir(dir); |
|
|
|
QStringList torrents; |
|
|
|
|
|
|
|
// Network folders scan
|
|
|
|
|
|
|
|
foreach (const QDir &dir, m_watchedFolders) { |
|
|
|
|
|
|
|
//qDebug("FSWatcher: Polling manually folder %s", qUtf8Printable(dir.path()));
|
|
|
|
|
|
|
|
addTorrentsFromDir(dir, torrents); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
// Report detected torrent files
|
|
|
|
|
|
|
|
if (!torrents.empty()) { |
|
|
|
|
|
|
|
qDebug("The following files are being reported: %s", qUtf8Printable(torrents.join("\n"))); |
|
|
|
|
|
|
|
emit torrentsAdded(torrents); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
#endif |
|
|
|
#endif |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void FileSystemWatcher::processPartialTorrents() |
|
|
|
void FileSystemWatcher::processPartialTorrents() |
|
|
|
{ |
|
|
|
{ |
|
|
@ -207,41 +180,33 @@ void FileSystemWatcher::processPartialTorrents() |
|
|
|
emit torrentsAdded(noLongerPartial); |
|
|
|
emit torrentsAdded(noLongerPartial); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
void FileSystemWatcher::startPartialTorrentTimer() |
|
|
|
void FileSystemWatcher::processTorrentsInDir(const QDir &dir) |
|
|
|
{ |
|
|
|
{ |
|
|
|
Q_ASSERT(!m_partialTorrents.isEmpty()); |
|
|
|
QStringList torrents; |
|
|
|
if (!m_partialTorrentTimer) { |
|
|
|
const QStringList files = dir.entryList({"*.torrent", "*.magnet"}, QDir::Files); |
|
|
|
m_partialTorrentTimer = new QTimer(); |
|
|
|
for (const QString &file : files) { |
|
|
|
connect(m_partialTorrentTimer, &QTimer::timeout, this, &FileSystemWatcher::processPartialTorrents); |
|
|
|
|
|
|
|
m_partialTorrentTimer->setSingleShot(true); |
|
|
|
|
|
|
|
m_partialTorrentTimer->start(WATCH_INTERVAL); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void FileSystemWatcher::addTorrentsFromDir(const QDir &dir, QStringList &torrents) |
|
|
|
|
|
|
|
{ |
|
|
|
|
|
|
|
const QStringList files = dir.entryList(m_filters, QDir::Files, QDir::Unsorted); |
|
|
|
|
|
|
|
foreach (const QString &file, files) { |
|
|
|
|
|
|
|
const QString fileAbsPath = dir.absoluteFilePath(file); |
|
|
|
const QString fileAbsPath = dir.absoluteFilePath(file); |
|
|
|
if (fileAbsPath.endsWith(".magnet")) { |
|
|
|
if (file.endsWith(".magnet")) |
|
|
|
torrents << fileAbsPath; |
|
|
|
torrents << fileAbsPath; |
|
|
|
} |
|
|
|
else if (BitTorrent::TorrentInfo::loadFromFile(fileAbsPath).isValid()) |
|
|
|
else if (BitTorrent::TorrentInfo::loadFromFile(fileAbsPath).isValid()) { |
|
|
|
|
|
|
|
torrents << fileAbsPath; |
|
|
|
torrents << fileAbsPath; |
|
|
|
|
|
|
|
else if (!m_partialTorrents.contains(fileAbsPath)) |
|
|
|
|
|
|
|
m_partialTorrents[fileAbsPath] = 0; |
|
|
|
} |
|
|
|
} |
|
|
|
else if (!m_partialTorrents.contains(fileAbsPath)) { |
|
|
|
|
|
|
|
qDebug("Partial torrent detected at: %s", qUtf8Printable(fileAbsPath)); |
|
|
|
|
|
|
|
qDebug("Delay the file's processing..."); |
|
|
|
|
|
|
|
m_partialTorrents.insert(fileAbsPath, 0); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!m_partialTorrents.empty()) |
|
|
|
if (!torrents.empty()) |
|
|
|
startPartialTorrentTimer(); |
|
|
|
emit torrentsAdded(torrents); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!m_partialTorrents.empty() && !m_partialTorrentTimer) { |
|
|
|
|
|
|
|
m_partialTorrentTimer = new QTimer(this); |
|
|
|
|
|
|
|
connect(m_partialTorrentTimer, &QTimer::timeout, this, &FileSystemWatcher::processPartialTorrents); |
|
|
|
|
|
|
|
m_partialTorrentTimer->setSingleShot(true); |
|
|
|
|
|
|
|
m_partialTorrentTimer->start(WATCH_INTERVAL); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
#if !defined Q_OS_WIN && !defined Q_OS_HAIKU |
|
|
|
#if !defined Q_OS_WIN && !defined Q_OS_HAIKU |
|
|
|
bool FileSystemWatcher::isNetworkFileSystem(QString path) |
|
|
|
bool FileSystemWatcher::isNetworkFileSystem(const QString &path) |
|
|
|
{ |
|
|
|
{ |
|
|
|
QString file = path; |
|
|
|
QString file = path; |
|
|
|
if (!file.endsWith("/")) |
|
|
|
if (!file.endsWith("/")) |
|
|
|