/***************************************************************************** * gta5view Grand Theft Auto V Profile Viewer * Copyright (C) 2016-2017 Syping * * 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 3 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, see <http://www.gnu.org/licenses/>. *****************************************************************************/ #include "DatabaseThread.h" #include "CrewDatabase.h" #include "AppEnv.h" #include <QNetworkAccessManager> #include <QNetworkRequest> #include <QNetworkReply> #include <QJsonDocument> #include <QJsonObject> #include <QStringList> #include <QVariantMap> #include <QEventLoop> #include <QTimer> #include <QDebug> #include <QUrl> #define crewMaxPages 83 #define maxLoadFails 3 DatabaseThread::DatabaseThread(CrewDatabase *crewDB, QObject *parent) : QThread(parent), crewDB(crewDB) { continueLastCrew = true; threadRunning = true; } void DatabaseThread::run() { QEventLoop threadLoop; QStringList crewList; QStringList crewListR; // Register thread loop end signal QObject::connect(this, SIGNAL(threadTerminated()), &threadLoop, SLOT(quit())); // Setup crewList for Quick time scan crewList = crewDB->getCrews(); if (!crewList.isEmpty()) { crewListR = deleteCompatibleCrews(crewList); } else { while (crewList.isEmpty() && threadRunning) { QTimer::singleShot(1000, &threadLoop, SLOT(quit())); threadLoop.exec(); if (!crewDB->isAddingCrews()) { crewList = crewDB->getCrews(); } } if (threadRunning) { crewListR = deleteCompatibleCrews(crewList); } } // Only do QTS when Thread should be run if (threadRunning) { // Quick time scan #ifdef GTA5SYNC_DEBUG qDebug() << "Start QTS"; #endif if (crewListR.length() <= 5) { scanCrewReference(crewListR, 2500); emit crewNameUpdated(); } if (crewList.length() <= 3) { scanCrewMembersList(crewList, 3, 2500); emit playerNameUpdated(); } else if (crewList.length() <= 5) { scanCrewMembersList(crewList, 2, 2500); emit playerNameUpdated(); } if (threadRunning) { QTimer::singleShot(10000, &threadLoop, SLOT(quit())); threadLoop.exec(); } } while (threadRunning) { crewList = crewDB->getCrews(); crewListR = deleteCompatibleCrews(crewList); // Long time scan #ifdef GTA5SYNC_DEBUG qDebug() << "Start LTS"; #endif scanCrewReference(crewListR, 10000); emit crewNameUpdated(); scanCrewMembersList(crewList, crewMaxPages, 10000); emit playerNameUpdated(); if (threadRunning) { QTimer::singleShot(300000, &threadLoop, SLOT(quit())); threadLoop.exec(); } } } void DatabaseThread::scanCrewReference(const QStringList &crewList, const int &requestDelay) { for (QString crewID : crewList) { if (threadRunning && crewID != QLatin1String("0")) { QNetworkAccessManager *netManager = new QNetworkAccessManager(); QNetworkRequest netRequest(AppEnv::getCrewFetchingUrl(crewID)); #if QT_VERSION >= 0x050600 netRequest.setAttribute(QNetworkRequest::FollowRedirectsAttribute, true); #endif netRequest.setRawHeader("User-Agent", AppEnv::getUserAgent()); netRequest.setRawHeader("Accept", "text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8"); netRequest.setRawHeader("Accept-Language", "en-US;q=0.5,en;q=0.3"); netRequest.setRawHeader("Connection", "keep-alive"); QNetworkReply *netReply = netManager->get(netRequest); QEventLoop *downloadLoop = new QEventLoop(); QObject::connect(netReply, SIGNAL(finished()), downloadLoop, SLOT(quit())); if (!continueLastCrew) { QObject::connect(this, SIGNAL(threadTerminated()), downloadLoop, SLOT(quit())); } QTimer::singleShot(30000, downloadLoop, SLOT(quit())); downloadLoop->exec(); downloadLoop->disconnect(); delete downloadLoop; if (netReply->isFinished()) { QString crewName; QByteArray crewHtml = netReply->readAll(); QStringList crewHtmlSplit1 = QString::fromUtf8(crewHtml).split("<title>Rockstar Games Social Club - Crew : "); if (crewHtmlSplit1.length() >= 2) { QStringList crewHtmlSplit2 = QString(crewHtmlSplit1.at(1)).split("</title>"); if (crewHtmlSplit2.length() >= 1) { crewName = crewHtmlSplit2.at(0); } } if (!crewName.isEmpty()) { emit crewNameFound(crewID.toInt(), crewName); } } if (threadRunning) { QEventLoop *waitingLoop = new QEventLoop(); QTimer::singleShot(requestDelay, waitingLoop, SLOT(quit())); if (!continueLastCrew) { QObject::connect(this, SIGNAL(threadTerminated()), waitingLoop, SLOT(quit())); } waitingLoop->exec(); waitingLoop->disconnect(); delete waitingLoop; } delete netReply; delete netManager; } } } void DatabaseThread::scanCrewMembersList(const QStringList &crewList, const int &maxPages, const int &requestDelay) { for (QString crewID : crewList) { if (threadRunning && crewID != QLatin1String("0")) { int currentFail = 0; int currentPage = 0; int foundPlayers = 0; int totalPlayers = 1000; while(foundPlayers < totalPlayers && currentPage < maxPages && (continueLastCrew ? true : threadRunning)) { QNetworkAccessManager *netManager = new QNetworkAccessManager(); QNetworkRequest netRequest(AppEnv::getPlayerFetchingUrl(crewID, currentPage)); #if QT_VERSION >= 0x050600 netRequest.setAttribute(QNetworkRequest::FollowRedirectsAttribute, true); #endif netRequest.setRawHeader("User-Agent", AppEnv::getUserAgent()); netRequest.setRawHeader("Accept", "text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8"); netRequest.setRawHeader("Accept-Language", "en-US;q=0.5,en;q=0.3"); netRequest.setRawHeader("Connection", "keep-alive"); QNetworkReply *netReply = netManager->get(netRequest); QEventLoop *downloadLoop = new QEventLoop(); QObject::connect(netReply, SIGNAL(finished()), downloadLoop, SLOT(quit())); if (!continueLastCrew) { QObject::connect(this, SIGNAL(threadTerminated()), downloadLoop, SLOT(quit())); } QTimer::singleShot(30000, downloadLoop, SLOT(quit())); downloadLoop->exec(); downloadLoop->disconnect(); delete downloadLoop; if (netReply->isFinished()) { QByteArray crewJson = netReply->readAll(); QJsonDocument crewDocument = QJsonDocument::fromJson(crewJson); QJsonObject crewObject = crewDocument.object(); QVariantMap crewMap = crewObject.toVariantMap(); if (crewMap.contains("Total")) { totalPlayers = crewMap["Total"].toInt(); } if (crewMap.contains("Members")) { const QList<QVariant> memberList = crewMap["Members"].toList(); for (QVariant memberVariant : memberList) { QMap<QString, QVariant> memberMap = memberVariant.toMap(); if (memberMap.contains("RockstarId") && memberMap.contains("Name")) { int RockstarId = memberMap["RockstarId"].toInt(); QString memberName = memberMap["Name"].toString(); if (!memberName.isEmpty() && RockstarId != 0) { foundPlayers++; emit playerNameFound(RockstarId, memberName); } } } } currentPage++; } else { currentFail++; if (currentFail == maxLoadFails) { currentFail = 0; currentPage++; } } delete netReply; delete netManager; if (foundPlayers < totalPlayers && currentPage < maxPages && (continueLastCrew ? true : threadRunning)) { QEventLoop *waitingLoop = new QEventLoop(); QTimer::singleShot(requestDelay, waitingLoop, SLOT(quit())); if (!continueLastCrew) { QObject::connect(this, SIGNAL(threadTerminated()), waitingLoop, SLOT(quit())); } waitingLoop->exec(); waitingLoop->disconnect(); delete waitingLoop; } } } } } void DatabaseThread::deleteCompatibleCrews(QStringList *crewList) { for (QString crewNID : *crewList) { if (crewDB->isCompatibleCrew(crewNID)) { crewList->removeAll(crewNID); } } } QStringList DatabaseThread::deleteCompatibleCrews(const QStringList &crewList) { QStringList crewListR = crewList; for (QString crewNID : crewListR) { if (crewDB->isCompatibleCrew(crewNID)) { crewListR.removeAll(crewNID); } } return crewListR; } void DatabaseThread::terminateThread() { threadRunning = false; emit threadTerminated(); }