687 lines
21 KiB
C++
687 lines
21 KiB
C++
/* This file is part of Clementine.
|
|
Copyright 2014, Chocobozzz <djidane14ff@hotmail.fr>
|
|
Copyright 2014, Krzysztof Sobiecki <sobkas@gmail.com>
|
|
Copyright 2014, David Sansome <me@davidsansome.com>
|
|
Copyright 2014, John Maguire <john.maguire@gmail.com>
|
|
|
|
Clementine 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.
|
|
|
|
Clementine 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 Clementine. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "seafileservice.h"
|
|
|
|
#include <QJsonArray>
|
|
#include <QJsonDocument>
|
|
#include <QJsonObject>
|
|
#include <QTimer>
|
|
#include <QUrlQuery>
|
|
#include <cmath>
|
|
|
|
#include "core/application.h"
|
|
#include "core/player.h"
|
|
#include "core/taskmanager.h"
|
|
#include "core/waitforsignal.h"
|
|
#include "internet/core/oauthenticator.h"
|
|
#include "internet/seafile/seafileurlhandler.h"
|
|
#include "library/librarybackend.h"
|
|
#include "ui/iconloader.h"
|
|
|
|
const char* SeafileService::kServiceName = "Seafile";
|
|
const char* SeafileService::kSettingsGroup = "Seafile";
|
|
|
|
namespace {
|
|
|
|
static const char* kAuthTokenUrl = "/api2/auth-token/";
|
|
|
|
static const char* kFolderItemsUrl = "/api2/repos/%1/dir/";
|
|
static const char* kListReposUrl = "/api2/repos/";
|
|
|
|
static const char* kFileUrl = "/api2/repos/%1/file/";
|
|
static const char* kFileContentUrl = "/api2/repos/%1/file/detail/";
|
|
|
|
static const int kMaxTries = 10;
|
|
} // namespace
|
|
|
|
SeafileService::SeafileService(Application* app, InternetModel* parent)
|
|
: CloudFileService(app, parent, kServiceName, kSettingsGroup,
|
|
IconLoader::Load("seafile", IconLoader::Provider),
|
|
SettingsDialog::Page_Seafile),
|
|
indexing_task_id_(-1),
|
|
indexing_task_max_(0),
|
|
indexing_task_progress_(0),
|
|
changing_libary_(false) {
|
|
QSettings s;
|
|
s.beginGroup(kSettingsGroup);
|
|
access_token_ = s.value("access_token").toString();
|
|
server_ = s.value("server").toString();
|
|
|
|
QByteArray tree_bytes = s.value("tree").toByteArray();
|
|
|
|
if (!tree_bytes.isEmpty()) {
|
|
QDataStream stream(&tree_bytes, QIODevice::ReadOnly);
|
|
stream >> tree_;
|
|
}
|
|
|
|
app->player()->RegisterUrlHandler(new SeafileUrlHandler(this, this));
|
|
|
|
connect(&tree_, SIGNAL(ToAdd(QString, QString, SeafileTree::Entry)), this,
|
|
SLOT(AddEntry(QString, QString, SeafileTree::Entry)));
|
|
connect(&tree_, SIGNAL(ToDelete(QString, QString, SeafileTree::Entry)), this,
|
|
SLOT(DeleteEntry(QString, QString, SeafileTree::Entry)));
|
|
connect(&tree_, SIGNAL(ToUpdate(QString, QString, SeafileTree::Entry)), this,
|
|
SLOT(UpdateEntry(QString, QString, SeafileTree::Entry)));
|
|
}
|
|
|
|
bool SeafileService::has_credentials() const {
|
|
return !access_token_.isEmpty();
|
|
}
|
|
|
|
void SeafileService::AddAuthorizationHeader(QNetworkRequest* request) const {
|
|
request->setRawHeader("Authorization",
|
|
QString("Token %1").arg(access_token_).toLatin1());
|
|
}
|
|
|
|
void SeafileService::ForgetCredentials() {
|
|
QSettings s;
|
|
s.beginGroup(kSettingsGroup);
|
|
|
|
s.remove("access_token");
|
|
s.remove("tree");
|
|
access_token_.clear();
|
|
tree_.Clear();
|
|
|
|
server_.clear();
|
|
}
|
|
|
|
bool SeafileService::GetToken(const QString& mail, const QString& password,
|
|
const QString& server) {
|
|
QUrl url(server + kAuthTokenUrl);
|
|
QUrlQuery url_query;
|
|
|
|
url_query.addQueryItem("username", mail);
|
|
url_query.addQueryItem("password", password);
|
|
|
|
QNetworkRequest request(url);
|
|
AddAuthorizationHeader(&request);
|
|
|
|
QNetworkReply* reply =
|
|
network_->post(request, url_query.toString().toLatin1());
|
|
WaitForSignal(reply, SIGNAL(finished()));
|
|
|
|
if (!CheckReply(&reply)) {
|
|
qLog(Warning) << "Something wrong with the reply... (GetToken)";
|
|
return false;
|
|
}
|
|
|
|
reply->deleteLater();
|
|
|
|
QJsonObject json_response =
|
|
QJsonDocument::fromJson(reply->readAll()).object();
|
|
|
|
// Because the server responds "token"
|
|
access_token_ = json_response["token"].toString().replace("\"", "");
|
|
|
|
if (access_token_.isEmpty()) {
|
|
return false;
|
|
}
|
|
|
|
QSettings s;
|
|
s.beginGroup(kSettingsGroup);
|
|
s.setValue("access_token", access_token_);
|
|
|
|
server_ = server;
|
|
|
|
emit Connected();
|
|
|
|
return true;
|
|
}
|
|
|
|
void SeafileService::GetLibraries() {
|
|
QUrl url(server_ + kListReposUrl);
|
|
QNetworkRequest request(url);
|
|
AddAuthorizationHeader(&request);
|
|
QNetworkReply* reply = network_->get(request);
|
|
|
|
NewClosure(reply, SIGNAL(finished()), this,
|
|
SLOT(GetLibrariesFinished(QNetworkReply*)), reply);
|
|
}
|
|
|
|
void SeafileService::GetLibrariesFinished(QNetworkReply* reply) {
|
|
if (!CheckReply(&reply)) {
|
|
qLog(Warning) << "Something wrong with the reply... (GetLibraries)";
|
|
return;
|
|
}
|
|
|
|
reply->deleteLater();
|
|
|
|
// key : id, value : name
|
|
QMap<QString, QString> libraries;
|
|
QByteArray data = reply->readAll();
|
|
|
|
QJsonArray json_repos = QJsonDocument::fromJson(data).array();
|
|
|
|
for (const QJsonValue& json_repo : json_repos) {
|
|
QJsonObject repo = json_repo.toObject();
|
|
QString repo_name = repo["name"].toString(),
|
|
repo_id = repo["id"].toString();
|
|
|
|
// One library can appear several times and we don't add encrypted libraries
|
|
// (not supported yet)
|
|
if (!libraries.contains(repo_id) && !repo["encrypted"].toBool()) {
|
|
libraries.insert(repo_id, repo_name);
|
|
}
|
|
}
|
|
|
|
emit GetLibrariesFinishedSignal(libraries);
|
|
}
|
|
|
|
void SeafileService::ChangeLibrary(const QString& new_library) {
|
|
if (new_library == library_updated_ || changing_libary_) return;
|
|
|
|
if (indexing_task_id_ != -1) {
|
|
qLog(Debug) << "Want to change the Seafile library, but Clementine waits "
|
|
"the previous indexing...";
|
|
changing_libary_ = true;
|
|
NewClosure(this, SIGNAL(UpdatingLibrariesFinishedSignal()), this,
|
|
SLOT(ChangeLibrary(QString)), new_library);
|
|
return;
|
|
}
|
|
|
|
AbortReadTagsReplies();
|
|
|
|
qLog(Debug) << "Change the Seafile library";
|
|
|
|
// Every other libraries have to be destroyed from the tree
|
|
if (new_library != "all") {
|
|
for (SeafileTree::TreeItem* library : tree_.libraries()) {
|
|
if (new_library != library->entry().id()) {
|
|
DeleteEntry(library->entry().id(), "/", library->entry());
|
|
}
|
|
}
|
|
}
|
|
|
|
changing_libary_ = false;
|
|
UpdateLibraries();
|
|
}
|
|
|
|
void SeafileService::Connect() {
|
|
if (has_credentials()) {
|
|
UpdateLibraries();
|
|
} else {
|
|
ShowSettingsDialog();
|
|
}
|
|
}
|
|
|
|
void SeafileService::UpdateLibraries() {
|
|
// Quit if we are already updating the libraries
|
|
if (indexing_task_id_ != -1) {
|
|
return;
|
|
}
|
|
|
|
indexing_task_id_ =
|
|
app_->task_manager()->StartTask(tr("Building Seafile index..."));
|
|
|
|
connect(this, SIGNAL(GetLibrariesFinishedSignal(QMap<QString, QString>)),
|
|
this, SLOT(UpdateLibrariesInProgress(QMap<QString, QString>)));
|
|
|
|
GetLibraries();
|
|
}
|
|
|
|
void SeafileService::UpdateLibrariesInProgress(
|
|
const QMap<QString, QString>& libraries) {
|
|
disconnect(this, SIGNAL(GetLibrariesFinishedSignal(QMap<QString, QString>)),
|
|
this, SLOT(UpdateLibrariesInProgress(QMap<QString, QString>)));
|
|
|
|
QSettings s;
|
|
s.beginGroup(kSettingsGroup);
|
|
QString library_to_update = s.value("library").toString();
|
|
|
|
// If the library didn't change, we don't need to update
|
|
if (!library_updated_.isNull() && library_updated_ == library_to_update) {
|
|
app_->task_manager()->SetTaskFinished(indexing_task_id_);
|
|
indexing_task_id_ = -1;
|
|
UpdatingLibrariesFinishedSignal();
|
|
return;
|
|
}
|
|
|
|
library_updated_ = library_to_update;
|
|
|
|
if (library_to_update.isEmpty() || library_to_update == "none") {
|
|
app_->task_manager()->SetTaskFinished(indexing_task_id_);
|
|
indexing_task_id_ = -1;
|
|
UpdatingLibrariesFinishedSignal();
|
|
return;
|
|
}
|
|
|
|
QMapIterator<QString, QString> library(libraries);
|
|
while (library.hasNext()) {
|
|
library.next();
|
|
|
|
// Need to check this library ?
|
|
if (library_to_update == "all" || library.key() == library_to_update) {
|
|
FetchAndCheckFolderItems(
|
|
SeafileTree::Entry(library.value(), library.key(),
|
|
SeafileTree::Entry::LIBRARY),
|
|
"/");
|
|
// If not, we can destroy the library from the tree
|
|
} else {
|
|
// If the library was not in the tree, it's not a problem because
|
|
// DeleteEntry won't do anything
|
|
DeleteEntry(library.key(), "/",
|
|
SeafileTree::Entry(library.value(), library.key(),
|
|
SeafileTree::Entry::LIBRARY));
|
|
}
|
|
}
|
|
|
|
// If we didn't do anything, set the task finished
|
|
if (indexing_task_max_ == 0) {
|
|
app_->task_manager()->SetTaskFinished(indexing_task_id_);
|
|
indexing_task_id_ = -1;
|
|
UpdatingLibrariesFinishedSignal();
|
|
}
|
|
}
|
|
|
|
QNetworkReply* SeafileService::PrepareFetchFolderItems(const QString& library,
|
|
const QString& path) {
|
|
QUrl url(server_ + QString(kFolderItemsUrl).arg(library));
|
|
QUrlQuery url_query;
|
|
url_query.addQueryItem("p", path);
|
|
url.setQuery(url_query);
|
|
|
|
QNetworkRequest request(url);
|
|
AddAuthorizationHeader(&request);
|
|
QNetworkReply* reply = network_->get(request);
|
|
|
|
return reply;
|
|
}
|
|
|
|
void SeafileService::FetchAndCheckFolderItems(const SeafileTree::Entry& library,
|
|
const QString& path) {
|
|
StartTaskInProgress();
|
|
|
|
QNetworkReply* reply = PrepareFetchFolderItems(library.id(), path);
|
|
NewClosure(reply, SIGNAL(finished()), this,
|
|
SLOT(FetchAndCheckFolderItemsFinished(
|
|
QNetworkReply*, SeafileTree::Entry, QString)),
|
|
reply, library, path);
|
|
}
|
|
|
|
void SeafileService::FetchAndCheckFolderItemsFinished(
|
|
QNetworkReply* reply, const SeafileTree::Entry& library,
|
|
const QString& path) {
|
|
if (!CheckReply(&reply)) {
|
|
qLog(Warning)
|
|
<< "Something wrong with the reply... (FetchFolderItemsToList)";
|
|
FinishedTaskInProgress();
|
|
return;
|
|
}
|
|
|
|
reply->deleteLater();
|
|
|
|
QByteArray data = reply->readAll();
|
|
|
|
QJsonArray json_entries = QJsonDocument::fromJson(data).array();
|
|
|
|
SeafileTree::Entries entries;
|
|
for (const QJsonValue& e : json_entries) {
|
|
QJsonObject entry = e.toObject();
|
|
SeafileTree::Entry::Type entry_type =
|
|
SeafileTree::Entry::StringToType(entry["type"].toString());
|
|
QString entry_name = entry["name"].toString();
|
|
|
|
// We just want libraries/directories and files which could be songs.
|
|
if (entry_type == SeafileTree::Entry::NONE) {
|
|
qLog(Warning) << "Type entry unknown for this entry";
|
|
} else if (entry_type == SeafileTree::Entry::FILE &&
|
|
GuessMimeTypeForFile(entry_name).isNull()) {
|
|
continue;
|
|
}
|
|
|
|
entries.append(
|
|
SeafileTree::Entry(entry_name, entry["id"].toString(), entry_type));
|
|
}
|
|
|
|
tree_.CheckEntries(entries, library, path);
|
|
|
|
FinishedTaskInProgress();
|
|
}
|
|
|
|
void SeafileService::AddRecursivelyFolderItems(const QString& library,
|
|
const QString& path) {
|
|
StartTaskInProgress();
|
|
|
|
QNetworkReply* reply = PrepareFetchFolderItems(library, path);
|
|
NewClosure(
|
|
reply, SIGNAL(finished()), this,
|
|
SLOT(AddRecursivelyFolderItemsFinished(QNetworkReply*, QString, QString)),
|
|
reply, library, path);
|
|
}
|
|
|
|
void SeafileService::AddRecursivelyFolderItemsFinished(QNetworkReply* reply,
|
|
const QString& library,
|
|
const QString& path) {
|
|
if (!CheckReply(&reply)) {
|
|
qLog(Warning) << "Something wrong with the reply... (FetchFolderItems)";
|
|
FinishedTaskInProgress();
|
|
return;
|
|
}
|
|
|
|
reply->deleteLater();
|
|
|
|
QByteArray data = reply->readAll();
|
|
QJsonArray json_entries = QJsonDocument::fromJson(data).array();
|
|
|
|
for (const QJsonValue& e : json_entries) {
|
|
QJsonObject json_entry = e.toObject();
|
|
SeafileTree::Entry::Type entry_type =
|
|
SeafileTree::Entry::StringToType(json_entry["type"].toString());
|
|
QString entry_name = json_entry["name"].toString();
|
|
|
|
// We just want libraries/directories and files which could be songs.
|
|
if (entry_type == SeafileTree::Entry::NONE) {
|
|
qLog(Warning) << "Type entry unknown for this entry";
|
|
} else if (entry_type == SeafileTree::Entry::FILE &&
|
|
GuessMimeTypeForFile(entry_name).isNull()) {
|
|
continue;
|
|
}
|
|
|
|
SeafileTree::Entry entry(entry_name, json_entry["id"].toString(),
|
|
entry_type);
|
|
|
|
// If AddEntry was not successful we stop
|
|
if (!tree_.AddEntry(library, path, entry)) {
|
|
FinishedTaskInProgress();
|
|
return;
|
|
}
|
|
|
|
if (entry.is_dir()) {
|
|
AddRecursivelyFolderItems(library, path + entry.name() + "/");
|
|
} else {
|
|
MaybeAddFileEntry(entry.name(), library, path);
|
|
}
|
|
}
|
|
|
|
FinishedTaskInProgress();
|
|
}
|
|
|
|
QNetworkReply* SeafileService::PrepareFetchContentForFile(
|
|
const QString& library, const QString& filepath) {
|
|
QUrl content_url(server_ + QString(kFileContentUrl).arg(library));
|
|
QUrlQuery content_url_query;
|
|
content_url_query.addQueryItem("p", filepath);
|
|
content_url.setQuery(content_url_query);
|
|
|
|
QNetworkRequest request(content_url);
|
|
AddAuthorizationHeader(&request);
|
|
QNetworkReply* reply = network_->get(request);
|
|
|
|
return reply;
|
|
}
|
|
|
|
void SeafileService::MaybeAddFileEntry(const QString& entry_name,
|
|
const QString& library,
|
|
const QString& path) {
|
|
QString mime_type = GuessMimeTypeForFile(entry_name);
|
|
|
|
if (mime_type.isNull()) return;
|
|
|
|
// Get the details of the entry
|
|
QNetworkReply* reply = PrepareFetchContentForFile(library, path + entry_name);
|
|
NewClosure(reply, SIGNAL(finished()), this,
|
|
SLOT(MaybeAddFileEntryInProgress(QNetworkReply*, QString, QString,
|
|
QString)),
|
|
reply, library, path, mime_type);
|
|
}
|
|
|
|
void SeafileService::MaybeAddFileEntryInProgress(QNetworkReply* reply,
|
|
const QString& library,
|
|
const QString& path,
|
|
const QString& mime_type) {
|
|
if (!CheckReply(&reply)) {
|
|
qLog(Warning) << "Something wrong with the reply... (MaybeAddFileEntry)";
|
|
return;
|
|
}
|
|
|
|
reply->deleteLater();
|
|
|
|
QByteArray data = reply->readAll();
|
|
|
|
QJsonObject json_entry_detail = QJsonDocument::fromJson(data).object();
|
|
|
|
QUrl url;
|
|
url.setScheme("seafile");
|
|
url.setPath("/" + library + path + json_entry_detail["name"].toString());
|
|
|
|
Song song;
|
|
song.set_url(url);
|
|
song.set_ctime(0);
|
|
song.set_mtime(json_entry_detail["mtime"].toInt());
|
|
song.set_filesize(json_entry_detail["size"].toInt());
|
|
song.set_title(json_entry_detail["name"].toString());
|
|
|
|
// Get the download url of the entry
|
|
reply = PrepareFetchContentUrlForFile(
|
|
library, path + json_entry_detail["name"].toString());
|
|
NewClosure(
|
|
reply, SIGNAL(finished()), this,
|
|
SLOT(FetchContentUrlForFileFinished(QNetworkReply*, Song, QString)),
|
|
reply, song, mime_type);
|
|
}
|
|
|
|
QNetworkReply* SeafileService::PrepareFetchContentUrlForFile(
|
|
const QString& library, const QString& filepath) {
|
|
QUrl content_url(server_ + QString(kFileUrl).arg(library));
|
|
QUrlQuery content_url_query;
|
|
content_url_query.addQueryItem("p", filepath);
|
|
// See https://github.com/haiwen/seahub/issues/677
|
|
content_url_query.addQueryItem("reuse", "1");
|
|
content_url.setQuery(content_url_query);
|
|
|
|
QNetworkRequest request(content_url);
|
|
AddAuthorizationHeader(&request);
|
|
QNetworkReply* reply = network_->get(request);
|
|
|
|
return reply;
|
|
}
|
|
|
|
void SeafileService::FetchContentUrlForFileFinished(QNetworkReply* reply,
|
|
const Song& song,
|
|
const QString& mime_type) {
|
|
if (!CheckReply(&reply)) {
|
|
qLog(Warning)
|
|
<< "Something wrong with the reply... (FetchContentUrlForFile)";
|
|
return;
|
|
}
|
|
|
|
reply->deleteLater();
|
|
|
|
// Because server response is "http://..."
|
|
QString real_url = QString(reply->readAll()).replace("\"", "");
|
|
|
|
MaybeAddFileToDatabase(song, mime_type, QUrl(real_url),
|
|
QString("Token %1").arg(access_token_));
|
|
}
|
|
|
|
QUrl SeafileService::GetStreamingUrlFromSongId(const QString& library,
|
|
const QString& filepath) {
|
|
QNetworkReply* reply = PrepareFetchContentUrlForFile(library, filepath);
|
|
WaitForSignal(reply, SIGNAL(finished()));
|
|
|
|
if (!CheckReply(&reply)) {
|
|
qLog(Warning)
|
|
<< "Something wrong with the reply... (GetStreamingUrlFromSongId)";
|
|
return QUrl("");
|
|
}
|
|
reply->deleteLater();
|
|
|
|
QString response = QString(reply->readAll()).replace("\"", "");
|
|
|
|
return QUrl(response);
|
|
}
|
|
|
|
void SeafileService::AddEntry(const QString& library, const QString& path,
|
|
const SeafileTree::Entry& entry) {
|
|
if (entry.is_library()) {
|
|
tree_.AddLibrary(entry.name(), entry.id());
|
|
AddRecursivelyFolderItems(library, "/");
|
|
} else {
|
|
// If AddEntry was not successful we stop
|
|
// It could happen when the user changes the library to update while an
|
|
// update was in progress
|
|
if (!tree_.AddEntry(library, path, entry)) {
|
|
return;
|
|
}
|
|
|
|
if (entry.is_file()) {
|
|
MaybeAddFileEntry(entry.name(), library, path);
|
|
} else {
|
|
AddRecursivelyFolderItems(library, path + entry.name() + "/");
|
|
}
|
|
}
|
|
}
|
|
|
|
void SeafileService::UpdateEntry(const QString& library, const QString& path,
|
|
const SeafileTree::Entry& entry) {
|
|
if (entry.is_file()) {
|
|
DeleteEntry(library, path, entry);
|
|
AddEntry(library, path, entry);
|
|
} else {
|
|
QString entry_path = path;
|
|
|
|
if (entry.is_dir()) {
|
|
entry_path += entry.name() + "/";
|
|
}
|
|
|
|
FetchAndCheckFolderItems(
|
|
SeafileTree::Entry("", library, SeafileTree::Entry::LIBRARY),
|
|
entry_path);
|
|
}
|
|
}
|
|
|
|
void SeafileService::DeleteEntry(const QString& library, const QString& path,
|
|
const SeafileTree::Entry& entry) {
|
|
// For the QPair -> 1 : path, 2 : entry
|
|
QList<QPair<QString, SeafileTree::Entry>> files_to_delete;
|
|
if (entry.is_library()) {
|
|
SeafileTree::TreeItem* item = tree_.FindLibrary(library);
|
|
files_to_delete = tree_.GetRecursiveFilesOfDir("/", item);
|
|
tree_.DeleteLibrary(library);
|
|
} else {
|
|
if (entry.is_dir()) {
|
|
SeafileTree::TreeItem* item =
|
|
tree_.FindFromAbsolutePath(library, path + entry.name() + "/");
|
|
files_to_delete =
|
|
tree_.GetRecursiveFilesOfDir(path + entry.name() + "/", item);
|
|
} else {
|
|
files_to_delete.append(qMakePair(path, entry));
|
|
}
|
|
|
|
if (!tree_.DeleteEntry(library, path, entry)) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Delete songs from the library of Clementine
|
|
for (const QPair<QString, SeafileTree::Entry>& file_to_delete :
|
|
files_to_delete) {
|
|
if (!GuessMimeTypeForFile(file_to_delete.second.name()).isEmpty()) {
|
|
QUrl song_url("seafile:/" + library + file_to_delete.first +
|
|
file_to_delete.second.name());
|
|
Song song = library_backend_->GetSongByUrl(song_url);
|
|
|
|
if (song.is_valid()) {
|
|
library_backend_->DeleteSongs(SongList() << song);
|
|
} else {
|
|
qLog(Warning) << "Can't delete song from the Clementine's library : "
|
|
<< song_url;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
bool SeafileService::CheckReply(QNetworkReply** reply, int tries) {
|
|
if (!(*reply)) {
|
|
return false;
|
|
} else if (tries > kMaxTries) {
|
|
(*reply)->deleteLater();
|
|
return false;
|
|
}
|
|
|
|
QVariant status_code_variant =
|
|
(*reply)->attribute(QNetworkRequest::HttpStatusCodeAttribute);
|
|
if (status_code_variant.isValid()) {
|
|
int status_code = status_code_variant.toInt();
|
|
|
|
if (status_code == NO_ERROR) {
|
|
return true;
|
|
} else if (status_code == TOO_MANY_REQUESTS) {
|
|
qLog(Debug) << "Too many requests, wait...";
|
|
|
|
int seconds_to_wait;
|
|
if ((*reply)->hasRawHeader("X-Throttle-Wait-Seconds")) {
|
|
seconds_to_wait =
|
|
((*reply)->rawHeader("X-Throttle-Wait-Seconds").toInt() + 1) * 1000;
|
|
} else {
|
|
seconds_to_wait = std::pow(tries, 2) * 1000;
|
|
}
|
|
|
|
QTimer timer;
|
|
timer.start(seconds_to_wait);
|
|
WaitForSignal(&timer, SIGNAL(timeout()));
|
|
|
|
(*reply)->deleteLater();
|
|
|
|
// We execute the reply again
|
|
*reply = network_->get((*reply)->request());
|
|
WaitForSignal(*reply, SIGNAL(finished()));
|
|
|
|
return CheckReply(reply, ++tries);
|
|
}
|
|
}
|
|
|
|
// Unknown, 404 ...
|
|
(*reply)->deleteLater();
|
|
qLog(Warning) << "Error with the reply : " << status_code_variant.toInt();
|
|
return false;
|
|
}
|
|
|
|
void SeafileService::StartTaskInProgress() {
|
|
indexing_task_max_++;
|
|
task_manager_->SetTaskProgress(indexing_task_id_, indexing_task_progress_,
|
|
indexing_task_max_);
|
|
}
|
|
|
|
void SeafileService::FinishedTaskInProgress() {
|
|
indexing_task_progress_++;
|
|
if (indexing_task_progress_ == indexing_task_max_) {
|
|
task_manager_->SetTaskFinished(indexing_task_id_);
|
|
indexing_task_id_ = -1;
|
|
UpdatingLibrariesFinishedSignal();
|
|
} else {
|
|
task_manager_->SetTaskProgress(indexing_task_id_, indexing_task_progress_,
|
|
indexing_task_max_);
|
|
}
|
|
}
|
|
|
|
SeafileService::~SeafileService() {
|
|
// Save the tree !
|
|
QSettings s;
|
|
s.beginGroup(kSettingsGroup);
|
|
|
|
QByteArray tree_byte;
|
|
QDataStream stream(&tree_byte, QIODevice::WriteOnly);
|
|
stream << tree_;
|
|
|
|
s.setValue("tree", tree_byte);
|
|
}
|