2015-08-27 14:25:14 +03:00
|
|
|
/*
|
|
|
|
* Bittorrent Client using Qt and libtorrent.
|
|
|
|
* Copyright (C) 2015 Vladimir Golovnev <glassez@yandex.ru>
|
|
|
|
* Copyright (C) 2006 Christophe Dumez <chris@qbittorrent.org>
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <QDomDocument>
|
|
|
|
#include <QDomNode>
|
|
|
|
#include <QDomElement>
|
|
|
|
#include <QDir>
|
2015-09-24 10:33:02 +03:00
|
|
|
#include <QProcess>
|
2015-08-27 14:25:14 +03:00
|
|
|
#include <QDebug>
|
|
|
|
|
|
|
|
#include "base/utils/fs.h"
|
|
|
|
#include "base/utils/misc.h"
|
|
|
|
#include "base/preferences.h"
|
2016-05-11 13:25:29 +02:00
|
|
|
#include "base/profile.h"
|
2015-08-27 14:25:14 +03:00
|
|
|
#include "base/net/downloadmanager.h"
|
|
|
|
#include "base/net/downloadhandler.h"
|
|
|
|
#include "searchengine.h"
|
|
|
|
|
|
|
|
enum SearchResultColumn
|
|
|
|
{
|
|
|
|
PL_DL_LINK,
|
|
|
|
PL_NAME,
|
|
|
|
PL_SIZE,
|
|
|
|
PL_SEEDS,
|
|
|
|
PL_LEECHS,
|
|
|
|
PL_ENGINE_URL,
|
|
|
|
PL_DESC_LINK,
|
|
|
|
NB_PLUGIN_COLUMNS
|
|
|
|
};
|
|
|
|
|
|
|
|
static inline void removePythonScriptIfExists(const QString &scriptPath)
|
|
|
|
{
|
|
|
|
Utils::Fs::forceRemove(scriptPath);
|
|
|
|
Utils::Fs::forceRemove(scriptPath + "c");
|
|
|
|
}
|
|
|
|
|
|
|
|
const QHash<QString, QString> SearchEngine::m_categoryNames = SearchEngine::initializeCategoryNames();
|
|
|
|
|
|
|
|
SearchEngine::SearchEngine()
|
2016-09-23 23:14:19 +03:00
|
|
|
: m_updateUrl(QString("http://searchplugins.qbittorrent.org/%1/engines/").arg(Utils::Misc::pythonVersion() >= 3 ? "nova3" : "nova"))
|
2015-09-24 10:33:02 +03:00
|
|
|
, m_searchStopped(false)
|
2015-08-27 14:25:14 +03:00
|
|
|
{
|
|
|
|
updateNova();
|
|
|
|
|
|
|
|
m_searchProcess = new QProcess(this);
|
|
|
|
m_searchProcess->setEnvironment(QProcess::systemEnvironment());
|
2017-06-14 01:59:57 +03:00
|
|
|
connect(m_searchProcess, &QProcess::started, this, &SearchEngine::searchStarted);
|
|
|
|
connect(m_searchProcess, &QProcess::readyReadStandardOutput, this, &SearchEngine::readSearchOutput);
|
|
|
|
connect(m_searchProcess, static_cast<void (QProcess::*)(int)>(&QProcess::finished), this, &SearchEngine::processFinished);
|
2015-08-27 14:25:14 +03:00
|
|
|
|
|
|
|
m_searchTimeout = new QTimer(this);
|
|
|
|
m_searchTimeout->setSingleShot(true);
|
2017-06-14 01:59:57 +03:00
|
|
|
connect(m_searchTimeout, &QTimer::timeout, this, &SearchEngine::onTimeout);
|
2015-08-27 14:25:14 +03:00
|
|
|
|
|
|
|
update();
|
|
|
|
}
|
|
|
|
|
|
|
|
SearchEngine::~SearchEngine()
|
|
|
|
{
|
|
|
|
qDeleteAll(m_plugins.values());
|
|
|
|
cancelSearch();
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList SearchEngine::allPlugins() const
|
|
|
|
{
|
|
|
|
return m_plugins.keys();
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList SearchEngine::enabledPlugins() const
|
|
|
|
{
|
|
|
|
QStringList plugins;
|
|
|
|
foreach (const PluginInfo *plugin, m_plugins.values()) {
|
|
|
|
if (plugin->enabled)
|
|
|
|
plugins << plugin->name;
|
|
|
|
}
|
|
|
|
|
|
|
|
return plugins;
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList SearchEngine::supportedCategories() const
|
|
|
|
{
|
|
|
|
QStringList result;
|
|
|
|
foreach (const PluginInfo *plugin, m_plugins.values()) {
|
|
|
|
if (plugin->enabled) {
|
|
|
|
foreach (QString cat, plugin->supportedCategories) {
|
|
|
|
if (!result.contains(cat))
|
|
|
|
result << cat;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
PluginInfo *SearchEngine::pluginInfo(const QString &name) const
|
|
|
|
{
|
|
|
|
return m_plugins.value(name, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SearchEngine::isActive() const
|
|
|
|
{
|
|
|
|
return (m_searchProcess->state() != QProcess::NotRunning);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SearchEngine::enablePlugin(const QString &name, bool enabled)
|
|
|
|
{
|
|
|
|
PluginInfo *plugin = m_plugins.value(name, 0);
|
|
|
|
if (plugin) {
|
|
|
|
plugin->enabled = enabled;
|
|
|
|
// Save to Hard disk
|
|
|
|
Preferences *const pref = Preferences::instance();
|
|
|
|
QStringList disabledPlugins = pref->getSearchEngDisabled();
|
|
|
|
if (enabled)
|
|
|
|
disabledPlugins.removeAll(name);
|
|
|
|
else if (!disabledPlugins.contains(name))
|
|
|
|
disabledPlugins.append(name);
|
|
|
|
pref->setSearchEngDisabled(disabledPlugins);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Updates shipped plugin
|
|
|
|
void SearchEngine::updatePlugin(const QString &name)
|
|
|
|
{
|
|
|
|
installPlugin(QString("%1%2.py").arg(m_updateUrl).arg(name));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Install or update plugin from file or url
|
|
|
|
void SearchEngine::installPlugin(const QString &source)
|
|
|
|
{
|
|
|
|
qDebug("Asked to install plugin at %s", qPrintable(source));
|
|
|
|
|
|
|
|
if (Utils::Misc::isUrl(source)) {
|
2017-06-14 01:59:57 +03:00
|
|
|
using namespace Net;
|
|
|
|
DownloadHandler *handler = DownloadManager::instance()->downloadUrl(source, true);
|
|
|
|
connect(handler, static_cast<void (DownloadHandler::*)(const QString &, const QString &)>(&DownloadHandler::downloadFinished)
|
|
|
|
, this, &SearchEngine::pluginDownloaded);
|
|
|
|
connect(handler, &DownloadHandler::downloadFailed, this, &SearchEngine::pluginDownloadFailed);
|
2015-08-27 14:25:14 +03:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
QString path = source;
|
|
|
|
if (path.startsWith("file:", Qt::CaseInsensitive))
|
|
|
|
path = QUrl(path).toLocalFile();
|
|
|
|
|
|
|
|
QString pluginName = Utils::Fs::fileName(path);
|
|
|
|
pluginName.chop(pluginName.size() - pluginName.lastIndexOf("."));
|
|
|
|
|
|
|
|
if (!path.endsWith(".py", Qt::CaseInsensitive))
|
|
|
|
emit pluginInstallationFailed(pluginName, tr("Unknown search engine plugin file format."));
|
|
|
|
else
|
|
|
|
installPlugin_impl(pluginName, path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SearchEngine::installPlugin_impl(const QString &name, const QString &path)
|
|
|
|
{
|
2016-05-14 16:46:40 +02:00
|
|
|
PluginVersion newVersion = getPluginVersion(path);
|
|
|
|
qDebug() << "Version to be installed:" << newVersion;
|
2015-08-27 14:25:14 +03:00
|
|
|
|
|
|
|
PluginInfo *plugin = pluginInfo(name);
|
2016-05-14 16:46:40 +02:00
|
|
|
if (plugin && !(plugin->version < newVersion)) {
|
2015-08-27 14:25:14 +03:00
|
|
|
qDebug("Apparently update is not needed, we have a more recent version");
|
|
|
|
emit pluginUpdateFailed(name, tr("A more recent version of this plugin is already installed."));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Process with install
|
|
|
|
QString destPath = pluginPath(name);
|
|
|
|
bool updated = false;
|
|
|
|
if (QFile::exists(destPath)) {
|
|
|
|
// Backup in case install fails
|
|
|
|
QFile::copy(destPath, destPath + ".bak");
|
|
|
|
Utils::Fs::forceRemove(destPath);
|
|
|
|
Utils::Fs::forceRemove(destPath + "c");
|
|
|
|
updated = true;
|
|
|
|
}
|
|
|
|
// Copy the plugin
|
|
|
|
QFile::copy(path, destPath);
|
|
|
|
// Update supported plugins
|
|
|
|
update();
|
|
|
|
// Check if this was correctly installed
|
|
|
|
if (!m_plugins.contains(name)) {
|
|
|
|
// Remove broken file
|
|
|
|
Utils::Fs::forceRemove(destPath);
|
|
|
|
if (updated) {
|
|
|
|
// restore backup
|
|
|
|
QFile::copy(destPath + ".bak", destPath);
|
|
|
|
Utils::Fs::forceRemove(destPath + ".bak");
|
|
|
|
// Update supported plugins
|
|
|
|
update();
|
|
|
|
emit pluginUpdateFailed(name, tr("Plugin is not supported."));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
emit pluginInstallationFailed(name, tr("Plugin is not supported."));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Install was successful, remove backup
|
|
|
|
if (updated)
|
|
|
|
Utils::Fs::forceRemove(destPath + ".bak");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SearchEngine::uninstallPlugin(const QString &name)
|
|
|
|
{
|
|
|
|
// remove it from hard drive
|
|
|
|
QDir pluginsFolder(pluginsLocation());
|
|
|
|
QStringList filters;
|
|
|
|
filters << name + ".*";
|
|
|
|
QStringList files = pluginsFolder.entryList(filters, QDir::Files, QDir::Unsorted);
|
|
|
|
QString file;
|
|
|
|
foreach (file, files)
|
|
|
|
Utils::Fs::forceRemove(pluginsFolder.absoluteFilePath(file));
|
|
|
|
// Remove it from supported engines
|
|
|
|
delete m_plugins.take(name);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:55:48 +03:00
|
|
|
void SearchEngine::updateIconPath(PluginInfo * const plugin)
|
|
|
|
{
|
|
|
|
if (!plugin) return;
|
|
|
|
QString iconPath = QString("%1/%2.png").arg(pluginsLocation()).arg(plugin->name);
|
|
|
|
if (QFile::exists(iconPath)) {
|
|
|
|
plugin->iconPath = iconPath;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
iconPath = QString("%1/%2.ico").arg(pluginsLocation()).arg(plugin->name);
|
|
|
|
if (QFile::exists(iconPath))
|
|
|
|
plugin->iconPath = iconPath;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-27 14:25:14 +03:00
|
|
|
void SearchEngine::checkForUpdates()
|
|
|
|
{
|
2017-06-14 01:59:57 +03:00
|
|
|
// Download version file from update server
|
|
|
|
using namespace Net;
|
|
|
|
DownloadHandler *handler = DownloadManager::instance()->downloadUrl(m_updateUrl + "versions.txt");
|
|
|
|
connect(handler, static_cast<void (DownloadHandler::*)(const QString &, const QByteArray &)>(&DownloadHandler::downloadFinished)
|
|
|
|
, this, &SearchEngine::versionInfoDownloaded);
|
|
|
|
connect(handler, &DownloadHandler::downloadFailed, this, &SearchEngine::versionInfoDownloadFailed);
|
2015-08-27 14:25:14 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void SearchEngine::cancelSearch()
|
|
|
|
{
|
|
|
|
if (m_searchProcess->state() != QProcess::NotRunning) {
|
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
m_searchProcess->kill();
|
|
|
|
#else
|
|
|
|
m_searchProcess->terminate();
|
|
|
|
#endif
|
|
|
|
m_searchStopped = true;
|
|
|
|
m_searchTimeout->stop();
|
|
|
|
|
|
|
|
m_searchProcess->waitForFinished(1000);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-06 12:32:44 +03:00
|
|
|
void SearchEngine::downloadTorrent(const QString &siteUrl, const QString &url)
|
|
|
|
{
|
|
|
|
QProcess *downloadProcess = new QProcess(this);
|
|
|
|
downloadProcess->setEnvironment(QProcess::systemEnvironment());
|
2017-06-14 01:59:57 +03:00
|
|
|
connect(downloadProcess, static_cast<void (QProcess::*)(int)>(&QProcess::finished), this, &SearchEngine::torrentFileDownloadFinished);
|
2016-04-06 12:32:44 +03:00
|
|
|
m_downloaders << downloadProcess;
|
|
|
|
QStringList params {
|
|
|
|
Utils::Fs::toNativePath(engineLocation() + "/nova2dl.py"),
|
|
|
|
siteUrl,
|
|
|
|
url
|
|
|
|
};
|
|
|
|
// Launch search
|
|
|
|
downloadProcess->start(Utils::Misc::pythonExecutable(), params, QIODevice::ReadOnly);
|
|
|
|
}
|
|
|
|
|
2015-08-27 14:25:14 +03:00
|
|
|
void SearchEngine::startSearch(const QString &pattern, const QString &category, const QStringList &usedPlugins)
|
|
|
|
{
|
|
|
|
// Search process already running or
|
|
|
|
// No search pattern entered
|
|
|
|
if ((m_searchProcess->state() != QProcess::NotRunning) || pattern.isEmpty()) {
|
|
|
|
emit searchFailed();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reload environment variables (proxy)
|
|
|
|
m_searchProcess->setEnvironment(QProcess::systemEnvironment());
|
|
|
|
|
|
|
|
QStringList params;
|
|
|
|
m_searchStopped = false;
|
|
|
|
params << Utils::Fs::toNativePath(engineLocation() + "/nova2.py");
|
|
|
|
params << usedPlugins.join(",");
|
|
|
|
params << category;
|
|
|
|
params << pattern.split(" ");
|
|
|
|
|
|
|
|
// Launch search
|
|
|
|
m_searchProcess->start(Utils::Misc::pythonExecutable(), params, QIODevice::ReadOnly);
|
|
|
|
m_searchTimeout->start(180000); // 3min
|
|
|
|
}
|
|
|
|
|
|
|
|
QString SearchEngine::categoryFullName(const QString &categoryName)
|
|
|
|
{
|
|
|
|
return tr(m_categoryNames.value(categoryName).toUtf8().constData());
|
|
|
|
}
|
|
|
|
|
2016-02-29 14:46:39 +08:00
|
|
|
QString SearchEngine::pluginFullName(const QString &pluginName)
|
|
|
|
{
|
|
|
|
return pluginInfo(pluginName) ? pluginInfo(pluginName)->fullName : QString();
|
|
|
|
}
|
|
|
|
|
2015-08-27 14:25:14 +03:00
|
|
|
QString SearchEngine::pluginsLocation()
|
|
|
|
{
|
|
|
|
return QString("%1/engines").arg(engineLocation());
|
|
|
|
}
|
|
|
|
|
|
|
|
QString SearchEngine::engineLocation()
|
|
|
|
{
|
|
|
|
QString folder = "nova";
|
|
|
|
if (Utils::Misc::pythonVersion() >= 3)
|
|
|
|
folder = "nova3";
|
2016-05-11 13:25:29 +02:00
|
|
|
const QString location = Utils::Fs::expandPathAbs(specialFolderLocation(SpecialFolder::Data) + folder);
|
2015-08-27 14:25:14 +03:00
|
|
|
QDir locationDir(location);
|
|
|
|
if (!locationDir.exists())
|
|
|
|
locationDir.mkpath(locationDir.absolutePath());
|
|
|
|
return location;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Slot called when QProcess is Finished
|
|
|
|
// QProcess can be finished for 3 reasons :
|
|
|
|
// Error | Stopped by user | Finished normally
|
2015-09-24 10:33:02 +03:00
|
|
|
void SearchEngine::processFinished(int exitcode)
|
2015-08-27 14:25:14 +03:00
|
|
|
{
|
|
|
|
m_searchTimeout->stop();
|
|
|
|
|
|
|
|
if (exitcode == 0)
|
|
|
|
emit searchFinished(m_searchStopped);
|
|
|
|
else
|
|
|
|
emit searchFailed();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SearchEngine::versionInfoDownloaded(const QString &url, const QByteArray &data)
|
|
|
|
{
|
|
|
|
Q_UNUSED(url)
|
|
|
|
parseVersionInfo(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SearchEngine::versionInfoDownloadFailed(const QString &url, const QString &reason)
|
|
|
|
{
|
|
|
|
Q_UNUSED(url)
|
|
|
|
emit checkForUpdatesFailed(tr("Update server is temporarily unavailable. %1").arg(reason));
|
|
|
|
}
|
|
|
|
|
|
|
|
void SearchEngine::pluginDownloaded(const QString &url, QString filePath)
|
|
|
|
{
|
|
|
|
filePath = Utils::Fs::fromNativePath(filePath);
|
|
|
|
|
|
|
|
QString pluginName = Utils::Fs::fileName(url);
|
|
|
|
pluginName.chop(pluginName.size() - pluginName.lastIndexOf(".")); // Remove extension
|
|
|
|
installPlugin_impl(pluginName, filePath);
|
|
|
|
Utils::Fs::forceRemove(filePath);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SearchEngine::pluginDownloadFailed(const QString &url, const QString &reason)
|
|
|
|
{
|
|
|
|
QString pluginName = url.split('/').last();
|
|
|
|
pluginName.replace(".py", "", Qt::CaseInsensitive);
|
|
|
|
if (pluginInfo(pluginName))
|
|
|
|
emit pluginUpdateFailed(pluginName, tr("Failed to download the plugin file. %1").arg(reason));
|
|
|
|
else
|
|
|
|
emit pluginInstallationFailed(pluginName, tr("Failed to download the plugin file. %1").arg(reason));
|
|
|
|
}
|
|
|
|
|
2016-04-06 12:32:44 +03:00
|
|
|
void SearchEngine::torrentFileDownloadFinished(int exitcode)
|
|
|
|
{
|
|
|
|
QProcess *downloadProcess = static_cast<QProcess*>(sender());
|
|
|
|
if (exitcode == 0) {
|
|
|
|
QString line = QString::fromUtf8(downloadProcess->readAllStandardOutput()).trimmed();
|
|
|
|
QStringList parts = line.split(' ');
|
|
|
|
if (parts.size() == 2)
|
|
|
|
emit torrentFileDownloaded(parts[0]);
|
|
|
|
}
|
|
|
|
|
|
|
|
qDebug() << "Deleting downloadProcess";
|
|
|
|
m_downloaders.removeOne(downloadProcess);
|
|
|
|
downloadProcess->deleteLater();
|
|
|
|
}
|
|
|
|
|
2015-08-27 14:25:14 +03:00
|
|
|
// Update nova.py search plugin if necessary
|
|
|
|
void SearchEngine::updateNova()
|
|
|
|
{
|
|
|
|
qDebug("Updating nova");
|
|
|
|
|
|
|
|
// create nova directory if necessary
|
|
|
|
QDir searchDir(engineLocation());
|
|
|
|
QString novaFolder = Utils::Misc::pythonVersion() >= 3 ? "searchengine/nova3" : "searchengine/nova";
|
|
|
|
QFile packageFile(searchDir.absoluteFilePath("__init__.py"));
|
|
|
|
packageFile.open(QIODevice::WriteOnly | QIODevice::Text);
|
|
|
|
packageFile.close();
|
|
|
|
if (!searchDir.exists("engines"))
|
|
|
|
searchDir.mkdir("engines");
|
|
|
|
Utils::Fs::removeDirRecursive(searchDir.absoluteFilePath("__pycache__"));
|
|
|
|
|
|
|
|
QFile packageFile2(searchDir.absolutePath() + "/engines/__init__.py");
|
|
|
|
packageFile2.open(QIODevice::WriteOnly | QIODevice::Text);
|
|
|
|
packageFile2.close();
|
|
|
|
|
|
|
|
// Copy search plugin files (if necessary)
|
|
|
|
QString filePath = searchDir.absoluteFilePath("nova2.py");
|
|
|
|
if (getPluginVersion(":/" + novaFolder + "/nova2.py") > getPluginVersion(filePath)) {
|
|
|
|
removePythonScriptIfExists(filePath);
|
|
|
|
QFile::copy(":/" + novaFolder + "/nova2.py", filePath);
|
|
|
|
}
|
|
|
|
|
2016-04-06 12:32:44 +03:00
|
|
|
filePath = searchDir.absoluteFilePath("nova2dl.py");
|
|
|
|
if (getPluginVersion(":/" + novaFolder + "/nova2dl.py") > getPluginVersion(filePath)) {
|
|
|
|
removePythonScriptIfExists(filePath);
|
|
|
|
QFile::copy(":/" + novaFolder + "/nova2dl.py", filePath);
|
|
|
|
}
|
|
|
|
|
2015-08-27 14:25:14 +03:00
|
|
|
filePath = searchDir.absoluteFilePath("fix_encoding.py");
|
|
|
|
QFile::copy(":/" + novaFolder + "/fix_encoding.py", filePath);
|
|
|
|
|
|
|
|
filePath = searchDir.absoluteFilePath("novaprinter.py");
|
|
|
|
if (getPluginVersion(":/" + novaFolder + "/novaprinter.py") > getPluginVersion(filePath)) {
|
|
|
|
removePythonScriptIfExists(filePath);
|
|
|
|
QFile::copy(":/" + novaFolder + "/novaprinter.py", filePath);
|
|
|
|
}
|
|
|
|
|
|
|
|
filePath = searchDir.absoluteFilePath("helpers.py");
|
|
|
|
if (getPluginVersion(":/" + novaFolder + "/helpers.py") > getPluginVersion(filePath)) {
|
|
|
|
removePythonScriptIfExists(filePath);
|
|
|
|
QFile::copy(":/" + novaFolder + "/helpers.py", filePath);
|
|
|
|
}
|
|
|
|
|
|
|
|
filePath = searchDir.absoluteFilePath("socks.py");
|
|
|
|
removePythonScriptIfExists(filePath);
|
|
|
|
QFile::copy(":/" + novaFolder + "/socks.py", filePath);
|
|
|
|
|
|
|
|
if (novaFolder.endsWith("nova")) {
|
|
|
|
filePath = searchDir.absoluteFilePath("fix_encoding.py");
|
|
|
|
removePythonScriptIfExists(filePath);
|
|
|
|
QFile::copy(":/" + novaFolder + "/fix_encoding.py", filePath);
|
|
|
|
}
|
|
|
|
else if (novaFolder.endsWith("nova3")) {
|
|
|
|
filePath = searchDir.absoluteFilePath("sgmllib3.py");
|
|
|
|
removePythonScriptIfExists(filePath);
|
|
|
|
QFile::copy(":/" + novaFolder + "/sgmllib3.py", filePath);
|
|
|
|
}
|
|
|
|
|
|
|
|
QDir destDir(pluginsLocation());
|
|
|
|
Utils::Fs::removeDirRecursive(destDir.absoluteFilePath("__pycache__"));
|
|
|
|
}
|
|
|
|
|
|
|
|
void SearchEngine::onTimeout()
|
|
|
|
{
|
|
|
|
cancelSearch();
|
|
|
|
}
|
|
|
|
|
|
|
|
// search QProcess return output as soon as it gets new
|
|
|
|
// stuff to read. We split it into lines and parse each
|
|
|
|
// line to SearchResult calling parseSearchResult().
|
|
|
|
void SearchEngine::readSearchOutput()
|
|
|
|
{
|
|
|
|
QByteArray output = m_searchProcess->readAllStandardOutput();
|
|
|
|
output.replace("\r", "");
|
|
|
|
QList<QByteArray> lines = output.split('\n');
|
|
|
|
if (!m_searchResultLineTruncated.isEmpty())
|
|
|
|
lines.prepend(m_searchResultLineTruncated + lines.takeFirst());
|
|
|
|
m_searchResultLineTruncated = lines.takeLast().trimmed();
|
|
|
|
|
|
|
|
QList<SearchResult> searchResultList;
|
|
|
|
foreach (const QByteArray &line, lines) {
|
|
|
|
SearchResult searchResult;
|
|
|
|
if (parseSearchResult(QString::fromUtf8(line), searchResult))
|
|
|
|
searchResultList << searchResult;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!searchResultList.isEmpty())
|
|
|
|
emit newSearchResults(searchResultList);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SearchEngine::update()
|
|
|
|
{
|
|
|
|
QProcess nova;
|
|
|
|
nova.setEnvironment(QProcess::systemEnvironment());
|
|
|
|
QStringList params;
|
|
|
|
params << Utils::Fs::toNativePath(engineLocation() + "/nova2.py");
|
|
|
|
params << "--capabilities";
|
|
|
|
nova.start(Utils::Misc::pythonExecutable(), params, QIODevice::ReadOnly);
|
|
|
|
nova.waitForStarted();
|
|
|
|
nova.waitForFinished();
|
|
|
|
|
|
|
|
QString capabilities = QString(nova.readAll());
|
|
|
|
QDomDocument xmlDoc;
|
|
|
|
if (!xmlDoc.setContent(capabilities)) {
|
|
|
|
qWarning() << "Could not parse Nova search engine capabilities, msg: " << capabilities.toLocal8Bit().data();
|
|
|
|
qWarning() << "Error: " << nova.readAllStandardError().constData();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QDomElement root = xmlDoc.documentElement();
|
|
|
|
if (root.tagName() != "capabilities") {
|
|
|
|
qWarning() << "Invalid XML file for Nova search engine capabilities, msg: " << capabilities.toLocal8Bit().data();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (QDomNode engineNode = root.firstChild(); !engineNode.isNull(); engineNode = engineNode.nextSibling()) {
|
|
|
|
QDomElement engineElem = engineNode.toElement();
|
|
|
|
if (!engineElem.isNull()) {
|
|
|
|
QString pluginName = engineElem.tagName();
|
|
|
|
|
|
|
|
PluginInfo *plugin = new PluginInfo;
|
|
|
|
plugin->name = pluginName;
|
|
|
|
plugin->version = getPluginVersion(pluginPath(pluginName));
|
|
|
|
plugin->fullName = engineElem.elementsByTagName("name").at(0).toElement().text();
|
|
|
|
plugin->url = engineElem.elementsByTagName("url").at(0).toElement().text();
|
|
|
|
|
|
|
|
foreach (QString cat, engineElem.elementsByTagName("categories").at(0).toElement().text().split(" ")) {
|
|
|
|
cat = cat.trimmed();
|
|
|
|
if (!cat.isEmpty())
|
|
|
|
plugin->supportedCategories << cat;
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList disabledEngines = Preferences::instance()->getSearchEngDisabled();
|
|
|
|
plugin->enabled = !disabledEngines.contains(pluginName);
|
|
|
|
|
2017-06-13 17:55:48 +03:00
|
|
|
updateIconPath(plugin);
|
2015-08-27 14:25:14 +03:00
|
|
|
|
|
|
|
if (!m_plugins.contains(pluginName)) {
|
|
|
|
m_plugins[pluginName] = plugin;
|
|
|
|
emit pluginInstalled(pluginName);
|
|
|
|
}
|
|
|
|
else if (m_plugins[pluginName]->version != plugin->version) {
|
|
|
|
delete m_plugins.take(pluginName);
|
|
|
|
m_plugins[pluginName] = plugin;
|
|
|
|
emit pluginUpdated(pluginName);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parse one line of search results list
|
|
|
|
// Line is in the following form:
|
|
|
|
// file url | file name | file size | nb seeds | nb leechers | Search engine url
|
|
|
|
bool SearchEngine::parseSearchResult(const QString &line, SearchResult &searchResult)
|
|
|
|
{
|
|
|
|
const QStringList parts = line.split("|");
|
|
|
|
const int nbFields = parts.size();
|
|
|
|
if (nbFields < (NB_PLUGIN_COLUMNS - 1)) return false; // -1 because desc_link is optional
|
|
|
|
|
|
|
|
searchResult = SearchResult();
|
|
|
|
searchResult.fileUrl = parts.at(PL_DL_LINK).trimmed(); // download URL
|
|
|
|
searchResult.fileName = parts.at(PL_NAME).trimmed(); // Name
|
|
|
|
searchResult.fileSize = parts.at(PL_SIZE).trimmed().toLongLong(); // Size
|
|
|
|
bool ok = false;
|
|
|
|
searchResult.nbSeeders = parts.at(PL_SEEDS).trimmed().toLongLong(&ok); // Seeders
|
|
|
|
if (!ok || (searchResult.nbSeeders < 0))
|
|
|
|
searchResult.nbSeeders = -1;
|
|
|
|
searchResult.nbLeechers = parts.at(PL_LEECHS).trimmed().toLongLong(&ok); // Leechers
|
|
|
|
if (!ok || (searchResult.nbLeechers < 0))
|
|
|
|
searchResult.nbLeechers = -1;
|
|
|
|
searchResult.siteUrl = parts.at(PL_ENGINE_URL).trimmed(); // Search site URL
|
|
|
|
if (nbFields == NB_PLUGIN_COLUMNS)
|
|
|
|
searchResult.descrLink = parts.at(PL_DESC_LINK).trimmed(); // Description Link
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SearchEngine::parseVersionInfo(const QByteArray &info)
|
|
|
|
{
|
|
|
|
qDebug("Checking if update is needed");
|
|
|
|
|
2016-05-14 16:46:40 +02:00
|
|
|
QHash<QString, PluginVersion> updateInfo;
|
2015-08-27 14:25:14 +03:00
|
|
|
bool dataCorrect = false;
|
|
|
|
QList<QByteArray> lines = info.split('\n');
|
|
|
|
foreach (QByteArray line, lines) {
|
|
|
|
line = line.trimmed();
|
|
|
|
if (line.isEmpty()) continue;
|
|
|
|
if (line.startsWith("#")) continue;
|
|
|
|
|
|
|
|
QList<QByteArray> list = line.split(' ');
|
|
|
|
if (list.size() != 2) continue;
|
|
|
|
|
|
|
|
QString pluginName = QString(list.first());
|
|
|
|
if (!pluginName.endsWith(":")) continue;
|
|
|
|
|
|
|
|
pluginName.chop(1); // remove trailing ':'
|
2016-05-14 16:46:40 +02:00
|
|
|
PluginVersion version = PluginVersion::tryParse(list.last(), {});
|
2017-06-02 01:57:28 +03:00
|
|
|
if (version == PluginVersion()) continue;
|
|
|
|
|
|
|
|
dataCorrect = true;
|
|
|
|
if (isUpdateNeeded(pluginName, version)) {
|
|
|
|
qDebug("Plugin: %s is outdated", qPrintable(pluginName));
|
|
|
|
updateInfo[pluginName] = version;
|
2015-08-27 14:25:14 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!dataCorrect)
|
|
|
|
emit checkForUpdatesFailed(tr("An incorrect update info received."));
|
|
|
|
else
|
|
|
|
emit checkForUpdatesFinished(updateInfo);
|
|
|
|
}
|
|
|
|
|
2016-05-14 16:46:40 +02:00
|
|
|
bool SearchEngine::isUpdateNeeded(QString pluginName, PluginVersion newVersion) const
|
2015-08-27 14:25:14 +03:00
|
|
|
{
|
|
|
|
PluginInfo *plugin = pluginInfo(pluginName);
|
|
|
|
if (!plugin) return true;
|
|
|
|
|
2016-05-14 16:46:40 +02:00
|
|
|
PluginVersion oldVersion = plugin->version;
|
|
|
|
qDebug() << "IsUpdate needed? to be installed:" << newVersion << ", already installed:" << oldVersion;
|
2015-08-27 14:25:14 +03:00
|
|
|
return (newVersion > oldVersion);
|
|
|
|
}
|
|
|
|
|
|
|
|
QString SearchEngine::pluginPath(const QString &name)
|
|
|
|
{
|
|
|
|
return QString("%1/%2.py").arg(pluginsLocation()).arg(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
QHash<QString, QString> SearchEngine::initializeCategoryNames()
|
|
|
|
{
|
|
|
|
QHash<QString, QString> result;
|
|
|
|
|
|
|
|
result["all"] = QT_TRANSLATE_NOOP("SearchEngine", "All categories");
|
|
|
|
result["movies"] = QT_TRANSLATE_NOOP("SearchEngine", "Movies");
|
|
|
|
result["tv"] = QT_TRANSLATE_NOOP("SearchEngine", "TV shows");
|
|
|
|
result["music"] = QT_TRANSLATE_NOOP("SearchEngine", "Music");
|
|
|
|
result["games"] = QT_TRANSLATE_NOOP("SearchEngine", "Games");
|
|
|
|
result["anime"] = QT_TRANSLATE_NOOP("SearchEngine", "Anime");
|
|
|
|
result["software"] = QT_TRANSLATE_NOOP("SearchEngine", "Software");
|
|
|
|
result["pictures"] = QT_TRANSLATE_NOOP("SearchEngine", "Pictures");
|
|
|
|
result["books"] = QT_TRANSLATE_NOOP("SearchEngine", "Books");
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-05-14 16:46:40 +02:00
|
|
|
PluginVersion SearchEngine::getPluginVersion(QString filePath)
|
2015-08-27 14:25:14 +03:00
|
|
|
{
|
|
|
|
QFile plugin(filePath);
|
|
|
|
if (!plugin.exists()) {
|
|
|
|
qDebug("%s plugin does not exist, returning 0.0", qPrintable(filePath));
|
2016-05-14 16:46:40 +02:00
|
|
|
return {};
|
2015-08-27 14:25:14 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!plugin.open(QIODevice::ReadOnly | QIODevice::Text))
|
2016-05-14 16:46:40 +02:00
|
|
|
return {};
|
2015-08-27 14:25:14 +03:00
|
|
|
|
2016-05-14 16:46:40 +02:00
|
|
|
PluginVersion version;
|
2015-08-27 14:25:14 +03:00
|
|
|
while (!plugin.atEnd()) {
|
|
|
|
QByteArray line = plugin.readLine();
|
|
|
|
if (line.startsWith("#VERSION: ")) {
|
|
|
|
line = line.split(' ').last().trimmed();
|
2016-05-14 16:46:40 +02:00
|
|
|
version = PluginVersion::tryParse(line, {});
|
|
|
|
qDebug() << "plugin" << filePath << "version: " << version;
|
2015-08-27 14:25:14 +03:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return version;
|
|
|
|
}
|