Fix discogs album cover fetching. Remove QJSON dependency.
This commit is contained in:
parent
b2b1ba7abe
commit
2be8f88d73
@ -143,7 +143,7 @@ if(LASTFM5_INCLUDE_DIRS AND LASTFM51_INCLUDE_DIRS)
|
|||||||
endif()
|
endif()
|
||||||
|
|
||||||
# QJSON
|
# QJSON
|
||||||
CHECK_INCLUDE_FILES(qjson/parser.h QJSON_PARSER_H)
|
#CHECK_INCLUDE_FILES(qjson/parser.h QJSON_PARSER_H)
|
||||||
|
|
||||||
# CHROMAPRINT
|
# CHROMAPRINT
|
||||||
CHECK_INCLUDE_FILES(chromaprint.h CHROMAPRINT_H)
|
CHECK_INCLUDE_FILES(chromaprint.h CHROMAPRINT_H)
|
||||||
@ -164,7 +164,7 @@ endif(${CMAKE_BUILD_TYPE} MATCHES "Release")
|
|||||||
# Set up definitions and paths
|
# Set up definitions and paths
|
||||||
add_definitions(${QT_DEFINITIONS})
|
add_definitions(${QT_DEFINITIONS})
|
||||||
link_directories(${TAGLIB_LIBRARY_DIRS})
|
link_directories(${TAGLIB_LIBRARY_DIRS})
|
||||||
link_directories(${QJSON_LIBRARY_DIRS})
|
#link_directories(${QJSON_LIBRARY_DIRS})
|
||||||
link_directories(${GSTREAMER_LIBRARY_DIRS})
|
link_directories(${GSTREAMER_LIBRARY_DIRS})
|
||||||
|
|
||||||
# Don't try to use webkit if their include directories couldn't be found.
|
# Don't try to use webkit if their include directories couldn't be found.
|
||||||
@ -369,8 +369,8 @@ endif()
|
|||||||
add_subdirectory(3rdparty/qocoa)
|
add_subdirectory(3rdparty/qocoa)
|
||||||
|
|
||||||
# QJSON
|
# QJSON
|
||||||
add_subdirectory(3rdparty/qjson)
|
#add_subdirectory(3rdparty/qjson)
|
||||||
set(QJSON_INCLUDE_DIRS ${CMAKE_CURRENT_SOURCE_DIR}/3rdparty/qjson ${CMAKE_CURRENT_SOURCE_DIR}/3rdparty/qjson/include)
|
#set(QJSON_INCLUDE_DIRS ${CMAKE_CURRENT_SOURCE_DIR}/3rdparty/qjson ${CMAKE_CURRENT_SOURCE_DIR}/3rdparty/qjson/include)
|
||||||
|
|
||||||
# Subdirectories
|
# Subdirectories
|
||||||
add_subdirectory(src)
|
add_subdirectory(src)
|
||||||
|
2
dist/strawberry.spec
vendored
2
dist/strawberry.spec
vendored
@ -83,5 +83,5 @@ make clean
|
|||||||
%{_datadir}/icons/hicolor/scalable/apps/strawberry.svg
|
%{_datadir}/icons/hicolor/scalable/apps/strawberry.svg
|
||||||
|
|
||||||
%changelog
|
%changelog
|
||||||
* ma. feb. 26 2018 0.1.1
|
* fr. mars 02 2018 0.1.1
|
||||||
- Version 0.1.1
|
- Version 0.1.1
|
||||||
|
@ -56,7 +56,7 @@ include_directories(${QTSINGLEAPPLICATION_INCLUDE_DIRS})
|
|||||||
include_directories(${QXT_INCLUDE_DIRS})
|
include_directories(${QXT_INCLUDE_DIRS})
|
||||||
include_directories(${SHA2_INCLUDE_DIRS})
|
include_directories(${SHA2_INCLUDE_DIRS})
|
||||||
include_directories(${CHROMAPRINT_INCLUDE_DIRS})
|
include_directories(${CHROMAPRINT_INCLUDE_DIRS})
|
||||||
include_directories(${QJSON_INCLUDE_DIRS})
|
#include_directories(${QJSON_INCLUDE_DIRS})
|
||||||
|
|
||||||
find_package(OpenGL)
|
find_package(OpenGL)
|
||||||
include_directories(${OPENGL_INCLUDE_DIR})
|
include_directories(${OPENGL_INCLUDE_DIR})
|
||||||
@ -920,7 +920,7 @@ target_link_libraries(strawberry_lib
|
|||||||
${QTSINGLEAPPLICATION_LIBRARIES}
|
${QTSINGLEAPPLICATION_LIBRARIES}
|
||||||
${CMAKE_THREAD_LIBS_INIT}
|
${CMAKE_THREAD_LIBS_INIT}
|
||||||
${SQLITE_LIBRARIES}
|
${SQLITE_LIBRARIES}
|
||||||
${QJSON_LIBRARIES}
|
#${QJSON_LIBRARIES}
|
||||||
z
|
z
|
||||||
Qocoa
|
Qocoa
|
||||||
)
|
)
|
||||||
@ -990,8 +990,8 @@ endif (APPLE)
|
|||||||
set(3RDPARTY_SQLITE_LIBRARY qsqlite)
|
set(3RDPARTY_SQLITE_LIBRARY qsqlite)
|
||||||
target_link_libraries(strawberry_lib qsqlite)
|
target_link_libraries(strawberry_lib qsqlite)
|
||||||
|
|
||||||
set(3RDPARTY_QJSON_LIBRARY qjson)
|
#set(3RDPARTY_QJSON_LIBRARY qjson)
|
||||||
target_link_libraries(strawberry_lib qjson)
|
#target_link_libraries(strawberry_lib qjson)
|
||||||
|
|
||||||
if (WIN32)
|
if (WIN32)
|
||||||
target_link_libraries(strawberry_lib
|
target_link_libraries(strawberry_lib
|
||||||
|
@ -79,6 +79,8 @@ AlbumCoverManager::AlbumCoverManager(Application *app, CollectionBackend *collec
|
|||||||
abort_progress_(new QPushButton(this)),
|
abort_progress_(new QPushButton(this)),
|
||||||
jobs_(0),
|
jobs_(0),
|
||||||
collection_backend_(collection_backend) {
|
collection_backend_(collection_backend) {
|
||||||
|
|
||||||
|
//qLog(Debug) << __PRETTY_FUNCTION__;
|
||||||
|
|
||||||
ui_->setupUi(this);
|
ui_->setupUi(this);
|
||||||
ui_->albums->set_cover_manager(this);
|
ui_->albums->set_cover_manager(this);
|
||||||
|
@ -22,14 +22,13 @@
|
|||||||
|
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
|
|
||||||
//#include <qjson/parser.h>
|
|
||||||
#include <QJson/Parser>
|
|
||||||
|
|
||||||
#include <QVariant>
|
#include <QVariant>
|
||||||
#include <QNetworkReply>
|
#include <QNetworkReply>
|
||||||
#include <QXmlStreamReader>
|
#include <QXmlStreamReader>
|
||||||
#include <QStringList>
|
#include <QStringList>
|
||||||
#include <QUrlQuery>
|
#include <QUrlQuery>
|
||||||
|
#include <QJsonObject>
|
||||||
|
#include <QJsonDocument>
|
||||||
|
|
||||||
#include "discogscoverprovider.h"
|
#include "discogscoverprovider.h"
|
||||||
|
|
||||||
@ -38,30 +37,53 @@
|
|||||||
#include "core/network.h"
|
#include "core/network.h"
|
||||||
#include "core/utilities.h"
|
#include "core/utilities.h"
|
||||||
|
|
||||||
const char *DiscogsCoverProvider::kUrl = "https://api.discogs.com/database/search";
|
const char *DiscogsCoverProvider::kUrlSearch = "https://api.discogs.com/database/search";
|
||||||
|
const char *DiscogsCoverProvider::kUrlReleases = "https://api.discogs.com/releases";
|
||||||
|
|
||||||
const char *DiscogsCoverProvider::kAccessKeyB64 = "dGh6ZnljUGJlZ1NEeXBuSFFxSVk=";
|
const char *DiscogsCoverProvider::kAccessKeyB64 = "dGh6ZnljUGJlZ1NEeXBuSFFxSVk=";
|
||||||
const char *DiscogsCoverProvider::kSecretAccessKeyB64 = "ZkFIcmlaSER4aHhRSlF2U3d0bm5ZVmdxeXFLWUl0UXI=";
|
const char *DiscogsCoverProvider::kSecretAccessKeyB64 = "ZkFIcmlaSER4aHhRSlF2U3d0bm5ZVmdxeXFLWUl0UXI=";
|
||||||
|
|
||||||
DiscogsCoverProvider::DiscogsCoverProvider(QObject *parent) : CoverProvider("Discogs", parent), network_(new NetworkAccessManager(this)) {}
|
DiscogsCoverProvider::DiscogsCoverProvider(QObject *parent) : CoverProvider("Discogs", parent), network_(new NetworkAccessManager(this)) {}
|
||||||
|
|
||||||
bool DiscogsCoverProvider::StartSearch(const QString &artist, const QString &album, int id) {
|
bool DiscogsCoverProvider::StartSearch(const QString &artist, const QString &album, int s_id) {
|
||||||
|
|
||||||
|
//qLog(Debug) << __PRETTY_FUNCTION__ << artist << album << s_id;
|
||||||
|
|
||||||
|
DiscogsCoverSearchContext *s_ctx = new DiscogsCoverSearchContext;
|
||||||
|
if (s_ctx == nullptr) return false;
|
||||||
|
s_ctx->id = s_id;
|
||||||
|
s_ctx->artist = artist;
|
||||||
|
s_ctx->album = album;
|
||||||
|
s_ctx->r_count = 0;
|
||||||
|
requests_search_.insert(s_id, s_ctx);
|
||||||
|
SendSearchRequest(s_ctx);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
bool DiscogsCoverProvider::StartRelease(DiscogsCoverSearchContext *s_ctx, int r_id, QString resource_url) {
|
||||||
|
|
||||||
|
//qLog(Debug) << __PRETTY_FUNCTION__ << resource_url;
|
||||||
|
|
||||||
|
DiscogsCoverReleaseContext *r_ctx = new DiscogsCoverReleaseContext;
|
||||||
|
if (r_ctx == nullptr) return false;
|
||||||
|
|
||||||
|
s_ctx->r_count++;
|
||||||
|
|
||||||
|
r_ctx->id = r_id;
|
||||||
|
r_ctx->resource_url = resource_url;
|
||||||
|
|
||||||
|
r_ctx->s_id = s_ctx->id;
|
||||||
|
|
||||||
|
requests_release_.insert(r_id, r_ctx);
|
||||||
|
SendReleaseRequest(s_ctx, r_ctx);
|
||||||
|
|
||||||
//qLog(Debug) << __PRETTY_FUNCTION__;
|
|
||||||
|
|
||||||
DiscogsCoverSearchContext *ctx = new DiscogsCoverSearchContext;
|
|
||||||
ctx->id = id;
|
|
||||||
ctx->artist = artist;
|
|
||||||
ctx->album = album;
|
|
||||||
ctx->state = DiscogsCoverSearchContext::State_Init;
|
|
||||||
pending_requests_.insert(id, ctx);
|
|
||||||
SendSearchRequest(ctx);
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void DiscogsCoverProvider::SendSearchRequest(DiscogsCoverSearchContext *ctx) {
|
void DiscogsCoverProvider::SendSearchRequest(DiscogsCoverSearchContext *s_ctx) {
|
||||||
|
|
||||||
//qLog(Debug) << __PRETTY_FUNCTION__;
|
//qLog(Debug) << __PRETTY_FUNCTION__;
|
||||||
|
|
||||||
@ -71,36 +93,20 @@ void DiscogsCoverProvider::SendSearchRequest(DiscogsCoverSearchContext *ctx) {
|
|||||||
typedef QPair<QByteArray, QByteArray> EncodedArg;
|
typedef QPair<QByteArray, QByteArray> EncodedArg;
|
||||||
typedef QList<EncodedArg> EncodedArgList;
|
typedef QList<EncodedArg> EncodedArgList;
|
||||||
|
|
||||||
QString type;
|
|
||||||
|
|
||||||
switch (ctx->state) {
|
|
||||||
case DiscogsCoverSearchContext::State_Init:
|
|
||||||
type = "master";
|
|
||||||
ctx->state = DiscogsCoverSearchContext::State_MastersRequested;
|
|
||||||
break;
|
|
||||||
case DiscogsCoverSearchContext::State_MastersRequested:
|
|
||||||
type = "release";
|
|
||||||
ctx->state = DiscogsCoverSearchContext::State_ReleasesRequested;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
EndSearch(ctx);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
ArgList args = ArgList()
|
ArgList args = ArgList()
|
||||||
<< Arg("key", QByteArray::fromBase64(kAccessKeyB64))
|
<< Arg("key", QByteArray::fromBase64(kAccessKeyB64))
|
||||||
<< Arg("secret", QByteArray::fromBase64(kSecretAccessKeyB64));
|
<< Arg("secret", QByteArray::fromBase64(kSecretAccessKeyB64));
|
||||||
|
|
||||||
if (!ctx->artist.isEmpty()) {
|
args.append(Arg("type", "release"));
|
||||||
args.append(Arg("artist", ctx->artist.toLower()));
|
if (!s_ctx->artist.isEmpty()) {
|
||||||
|
args.append(Arg("artist", s_ctx->artist.toLower()));
|
||||||
}
|
}
|
||||||
if (!ctx->album.isEmpty()) {
|
if (!s_ctx->album.isEmpty()) {
|
||||||
args.append(Arg("release_title", ctx->album.toLower()));
|
args.append(Arg("release_title", s_ctx->album.toLower()));
|
||||||
}
|
}
|
||||||
args.append(Arg("type", type));
|
|
||||||
|
|
||||||
QUrlQuery url_query;
|
QUrlQuery url_query;
|
||||||
QUrl url(kUrl);
|
QUrl url(kUrlSearch);
|
||||||
QStringList query_items;
|
QStringList query_items;
|
||||||
|
|
||||||
// Encode the arguments
|
// Encode the arguments
|
||||||
@ -112,7 +118,6 @@ void DiscogsCoverProvider::SendSearchRequest(DiscogsCoverSearchContext *ctx) {
|
|||||||
|
|
||||||
// Sign the request
|
// Sign the request
|
||||||
const QByteArray data_to_sign = QString("GET\n%1\n%2\n%3").arg(url.host(), url.path(), query_items.join("&")).toLatin1();
|
const QByteArray data_to_sign = QString("GET\n%1\n%2\n%3").arg(url.host(), url.path(), query_items.join("&")).toLatin1();
|
||||||
//const QByteArray signature(Utilities::HmacSha256(kSecretAccessKey, data_to_sign));
|
|
||||||
const QByteArray signature(Utilities::HmacSha256(QByteArray::fromBase64(kSecretAccessKeyB64), data_to_sign));
|
const QByteArray signature(Utilities::HmacSha256(QByteArray::fromBase64(kSecretAccessKeyB64), data_to_sign));
|
||||||
|
|
||||||
// Add the signature to the request
|
// Add the signature to the request
|
||||||
@ -121,96 +126,286 @@ void DiscogsCoverProvider::SendSearchRequest(DiscogsCoverSearchContext *ctx) {
|
|||||||
url.setQuery(url_query);
|
url.setQuery(url_query);
|
||||||
QNetworkReply *reply = network_->get(QNetworkRequest(url));
|
QNetworkReply *reply = network_->get(QNetworkRequest(url));
|
||||||
|
|
||||||
NewClosure(reply, SIGNAL(error(QNetworkReply::NetworkError)), this, SLOT(QueryError(QNetworkReply::NetworkError, QNetworkReply*, int)), reply, ctx->id);
|
NewClosure(reply, SIGNAL(error(QNetworkReply::NetworkError)), this, SLOT(SearchRequestError(QNetworkReply::NetworkError, QNetworkReply*, int)), reply, s_ctx->id);
|
||||||
NewClosure(reply, SIGNAL(finished()), this, SLOT(HandleSearchReply(QNetworkReply*, int)), reply, ctx->id);
|
NewClosure(reply, SIGNAL(finished()), this, SLOT(HandleSearchReply(QNetworkReply*, int)), reply, s_ctx->id);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void DiscogsCoverProvider::HandleSearchReply(QNetworkReply *reply, int id) {
|
void DiscogsCoverProvider::SendReleaseRequest(DiscogsCoverSearchContext *s_ctx, DiscogsCoverReleaseContext *r_ctx) {
|
||||||
|
|
||||||
//qLog(Debug) << __PRETTY_FUNCTION__;
|
//qLog(Debug) << __PRETTY_FUNCTION__;
|
||||||
|
|
||||||
|
typedef QPair<QString, QString> Arg;
|
||||||
|
typedef QList<Arg> ArgList;
|
||||||
|
|
||||||
|
typedef QPair<QByteArray, QByteArray> EncodedArg;
|
||||||
|
typedef QList<EncodedArg> EncodedArgList;
|
||||||
|
|
||||||
|
QUrlQuery url_query;
|
||||||
|
QStringList query_items;
|
||||||
|
|
||||||
|
ArgList args = ArgList()
|
||||||
|
<< Arg("key", QByteArray::fromBase64(kAccessKeyB64))
|
||||||
|
<< Arg("secret", QByteArray::fromBase64(kSecretAccessKeyB64));
|
||||||
|
// Encode the arguments
|
||||||
|
for (const Arg &arg : args) {
|
||||||
|
EncodedArg encoded_arg(QUrl::toPercentEncoding(arg.first), QUrl::toPercentEncoding(arg.second));
|
||||||
|
query_items << QString(encoded_arg.first + "=" + encoded_arg.second);
|
||||||
|
url_query.addQueryItem(encoded_arg.first, encoded_arg.second);
|
||||||
|
}
|
||||||
|
|
||||||
|
//QString urlstr = QString("%1/%2").arg(kUrlReleases).arg(r_ctx->id);
|
||||||
|
QUrl url(r_ctx->resource_url);
|
||||||
|
|
||||||
|
//qLog(Debug) << "Send: " << url;
|
||||||
|
|
||||||
|
// Sign the request
|
||||||
|
const QByteArray data_to_sign = QString("GET\n%1\n%2\n%3").arg(url.host(), url.path(), query_items.join("&")).toLatin1();
|
||||||
|
const QByteArray signature(Utilities::HmacSha256(QByteArray::fromBase64(kSecretAccessKeyB64), data_to_sign));
|
||||||
|
|
||||||
|
// Add the signature to the request
|
||||||
|
url_query.addQueryItem("Signature", QUrl::toPercentEncoding(signature.toBase64()));
|
||||||
|
|
||||||
|
url.setQuery(url_query);
|
||||||
|
QNetworkReply *reply = network_->get(QNetworkRequest(url));
|
||||||
|
|
||||||
|
NewClosure(reply, SIGNAL(error(QNetworkReply::NetworkError)), this, SLOT(ReleaseRequestError(QNetworkReply::NetworkError, QNetworkReply*, int, int)), reply, s_ctx->id, r_ctx->id);
|
||||||
|
NewClosure(reply, SIGNAL(finished()), this, SLOT(HandleReleaseReply(QNetworkReply*, int, int)), reply, s_ctx->id, r_ctx->id);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void DiscogsCoverProvider::HandleSearchReply(QNetworkReply *reply, int s_id) {
|
||||||
|
|
||||||
|
//qLog(Debug) << __PRETTY_FUNCTION__;
|
||||||
|
|
||||||
//QString text(reply->readAll());
|
//QString text(reply->readAll());
|
||||||
//qLog(Debug) << text;
|
//qLog(Debug) << "text: " << text << "\n";
|
||||||
|
|
||||||
reply->deleteLater();
|
reply->deleteLater();
|
||||||
|
|
||||||
DiscogsCoverSearchContext *ctx;
|
DiscogsCoverSearchContext *s_ctx;
|
||||||
if (!pending_requests_.contains(id)) {
|
if (!requests_search_.contains(s_id)) {
|
||||||
// the request was cancelled while we were waiting for the reply
|
qLog(Error) << "Discogs: Got reply for cancelled request: " << s_id;
|
||||||
qLog(Debug) << "Discogs: got reply for cancelled request" << id;
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
ctx = pending_requests_.value(id);
|
s_ctx = requests_search_.value(s_id);
|
||||||
|
if (s_ctx == nullptr) return;
|
||||||
QJson::Parser parser;
|
|
||||||
bool ok;
|
QString json_string;
|
||||||
bool found = false;
|
json_string = reply->readAll();
|
||||||
QVariantMap reply_map = parser.parse(reply, &ok).toMap();
|
QByteArray json_bytes = json_string.toLocal8Bit();
|
||||||
|
auto json_doc = QJsonDocument::fromJson(json_bytes);
|
||||||
if (!ok || !reply_map.contains("results")) {
|
if (json_doc.isNull()) {
|
||||||
// this is an error; either parse error or bad response from the server
|
qLog(Error) << "Discogs: Failed to create JSON doc.";
|
||||||
EndSearch(ctx);
|
EndSearch(s_ctx);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (!json_doc.isObject()) {
|
||||||
|
qLog(Error) << "Discogs: JSON is not an object.";
|
||||||
|
EndSearch(s_ctx);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QJsonObject json_obj = json_doc.object();
|
||||||
|
if (json_obj.isEmpty()) {
|
||||||
|
qLog(Error) << "Discogs: JSON object is empty.";
|
||||||
|
EndSearch(s_ctx);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
QVariantMap reply_map = json_obj.toVariantMap();
|
||||||
|
if (!reply_map.contains("results")) {
|
||||||
|
qLog(Error) << "Discogs: Search reply from server is missing JSON results.";
|
||||||
|
//qLog(Error) << "Discogs: Map dump:";
|
||||||
|
//qLog(Error) << reply_map;
|
||||||
|
EndSearch(s_ctx);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
QVariantList results = reply_map["results"].toList();
|
QVariantList results = reply_map["results"].toList();
|
||||||
|
int i = 0;
|
||||||
|
|
||||||
for (const QVariant &result : results) {
|
for (const QVariant &result : results) {
|
||||||
QVariantMap result_map = result.toMap();
|
QVariantMap result_map = result.toMap();
|
||||||
// In order to use less round-trips, we cheat here. Instead of
|
int r_id;
|
||||||
// following the "resource_url", and then scan all images in the
|
QString title;
|
||||||
// resource, we go directly to the largest primary image by
|
QString resource_url;
|
||||||
// constructing the primary image's url from the thmub's url.
|
if ((!result_map.contains("id")) || (!result_map.contains("resource_url"))) continue;
|
||||||
if (result_map.contains("thumb")) {
|
|
||||||
CoverSearchResult cover_result;
|
if (result_map.contains("id")) {
|
||||||
cover_result.image_url = QUrl(result_map["thumb"].toString().replace("R-90-", "R-"));
|
r_id = result_map["id"].toInt();
|
||||||
if (result_map.contains("title")) {
|
//qLog(Debug) << "id: " << r_id;
|
||||||
cover_result.description = result_map["title"].toString();
|
|
||||||
}
|
|
||||||
ctx->results.append(cover_result);
|
|
||||||
found = true;
|
|
||||||
}
|
}
|
||||||
|
if (result_map.contains("title")) {
|
||||||
|
title = result_map["title"].toString();
|
||||||
|
}
|
||||||
|
if (result_map.contains("resource_url")) {
|
||||||
|
resource_url = result_map["resource_url"].toString();
|
||||||
|
//qLog(Debug) << "resource_url: " << resource_url;
|
||||||
|
}
|
||||||
|
StartRelease(s_ctx, r_id, resource_url);
|
||||||
|
i++;
|
||||||
}
|
}
|
||||||
if (found) {
|
if (i <= 0) EndSearch(s_ctx);
|
||||||
EndSearch(ctx);
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void DiscogsCoverProvider::HandleReleaseReply(QNetworkReply *reply, int s_id, int r_id) {
|
||||||
|
|
||||||
|
//qLog(Debug) << __PRETTY_FUNCTION__;
|
||||||
|
|
||||||
|
//QString text(reply->readAll());
|
||||||
|
//qLog(Debug) << "text: " << text << "\n";
|
||||||
|
|
||||||
|
reply->deleteLater();
|
||||||
|
|
||||||
|
DiscogsCoverReleaseContext *r_ctx;
|
||||||
|
if (!requests_release_.contains(r_id)) {
|
||||||
|
qLog(Error) << "Discogs: Got reply for cancelled request: " << r_id;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
r_ctx = requests_release_.value(r_id);
|
||||||
|
if (r_ctx == nullptr) return;
|
||||||
|
|
||||||
|
DiscogsCoverSearchContext *s_ctx;
|
||||||
|
if (!requests_search_.contains(s_id)) {
|
||||||
|
qLog(Error) << "Discogs: Got reply for cancelled request: " << s_id << " " << r_id;
|
||||||
|
EndSearch(nullptr, r_ctx);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
s_ctx = requests_search_.value(s_id);
|
||||||
|
if (s_ctx == nullptr) return;
|
||||||
|
|
||||||
|
QString json_string;
|
||||||
|
json_string = reply->readAll();
|
||||||
|
QByteArray json_bytes = json_string.toLocal8Bit();
|
||||||
|
auto json_doc = QJsonDocument::fromJson(json_bytes);
|
||||||
|
if (json_doc.isNull()) {
|
||||||
|
qLog(Error) << "Discogs: Failed to create JSON doc.";
|
||||||
|
EndSearch(s_ctx, r_ctx);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (!json_doc.isObject()) {
|
||||||
|
qLog(Error) << "Discogs: JSON is not an object.";
|
||||||
|
EndSearch(s_ctx, r_ctx);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// otherwise, no results
|
QJsonObject json_obj = json_doc.object();
|
||||||
switch (ctx->state) {
|
if (json_obj.isEmpty()) {
|
||||||
case DiscogsCoverSearchContext::State_MastersRequested:
|
qLog(Error) << "Discogs: JSON object is empty.";
|
||||||
// search again, this time for releases
|
EndSearch(s_ctx, r_ctx);
|
||||||
SendSearchRequest(ctx);
|
return;
|
||||||
break;
|
|
||||||
default:
|
|
||||||
EndSearch(ctx);
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QVariantMap reply_map = json_obj.toVariantMap();
|
||||||
|
if (!reply_map.contains("images")) {
|
||||||
|
//qLog(Error) << "Discogs: Search reply from server is missing JSON images.";
|
||||||
|
//qLog(Error) << "Discogs: Map dump:";
|
||||||
|
//qLog(Error) << reply_map;
|
||||||
|
EndSearch(s_ctx, r_ctx);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
QVariantList results = reply_map["images"].toList();
|
||||||
|
|
||||||
|
for (const QVariant &result : results) {
|
||||||
|
QVariantMap result_map = result.toMap();
|
||||||
|
CoverSearchResult cover_result;
|
||||||
|
cover_result.description = s_ctx->title;
|
||||||
|
|
||||||
|
if (result_map.contains("type")) {
|
||||||
|
QString type = result_map["type"].toString();
|
||||||
|
if (type != "primary") continue;
|
||||||
|
}
|
||||||
|
if (result_map.contains("height")) {
|
||||||
|
}
|
||||||
|
if (result_map.contains("width")) {
|
||||||
|
}
|
||||||
|
if (result_map.contains("resource_url")) {
|
||||||
|
cover_result.image_url = QUrl(result_map["resource_url"].toString());
|
||||||
|
}
|
||||||
|
s_ctx->results.append(cover_result);
|
||||||
|
}
|
||||||
|
|
||||||
|
EndSearch(s_ctx, r_ctx);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void DiscogsCoverProvider::CancelSearch(int id) {
|
void DiscogsCoverProvider::CancelSearch(int id) {
|
||||||
|
|
||||||
//qLog(Debug) << __PRETTY_FUNCTION__ << id;
|
//qLog(Debug) << __PRETTY_FUNCTION__ << id;
|
||||||
|
|
||||||
delete pending_requests_.take(id);
|
delete requests_search_.take(id);
|
||||||
}
|
}
|
||||||
|
|
||||||
void DiscogsCoverProvider::QueryError(QNetworkReply::NetworkError error, QNetworkReply *reply, int id) {
|
void DiscogsCoverProvider::SearchRequestError(QNetworkReply::NetworkError error, QNetworkReply *reply, int s_id) {
|
||||||
|
|
||||||
//qLog(Debug) << __PRETTY_FUNCTION__;
|
//qLog(Debug) << __PRETTY_FUNCTION__;
|
||||||
|
|
||||||
|
DiscogsCoverSearchContext *s_ctx;
|
||||||
|
if (!requests_search_.contains(s_id)) {
|
||||||
|
qLog(Error) << "Discogs: got reply for cancelled request: " << s_id;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
s_ctx = requests_search_.value(s_id);
|
||||||
|
if (s_ctx == nullptr) return;
|
||||||
|
|
||||||
|
EndSearch(s_ctx);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void DiscogsCoverProvider::EndSearch(DiscogsCoverSearchContext *ctx) {
|
void DiscogsCoverProvider::ReleaseRequestError(QNetworkReply::NetworkError error, QNetworkReply *reply, int s_id, int r_id) {
|
||||||
|
|
||||||
//qLog(Debug) << __PRETTY_FUNCTION__;
|
//qLog(Debug) << __PRETTY_FUNCTION__;
|
||||||
|
|
||||||
(void)pending_requests_.remove(ctx->id);
|
DiscogsCoverSearchContext *s_ctx;
|
||||||
emit SearchFinished(ctx->id, ctx->results);
|
if (!requests_search_.contains(s_id)) {
|
||||||
delete ctx;
|
qLog(Error) << "Discogs: got reply for cancelled request: " << s_id << " " << r_id;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
s_ctx = requests_search_.value(s_id);
|
||||||
|
if (s_ctx == nullptr) return;
|
||||||
|
|
||||||
|
DiscogsCoverReleaseContext *r_ctx;
|
||||||
|
if (!requests_release_.contains(r_id)) {
|
||||||
|
qLog(Error) << "Discogs: got reply for cancelled request: " << s_id << r_id;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
r_ctx = requests_release_.value(r_id);
|
||||||
|
if (r_ctx == nullptr) return;
|
||||||
|
|
||||||
|
EndSearch(s_ctx, r_ctx);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void DiscogsCoverProvider::EndSearch(DiscogsCoverSearchContext *s_ctx, DiscogsCoverReleaseContext *r_ctx) {
|
||||||
|
|
||||||
|
//qLog(Debug) << __PRETTY_FUNCTION__;
|
||||||
|
|
||||||
|
(void)requests_release_.remove(r_ctx->id);
|
||||||
|
delete r_ctx;
|
||||||
|
|
||||||
|
if (s_ctx == nullptr) return;
|
||||||
|
|
||||||
|
s_ctx->r_count--;
|
||||||
|
|
||||||
|
//qLog(Debug) << "r_count: " << s_ctx->r_count;
|
||||||
|
|
||||||
|
if (s_ctx->r_count <= 0) EndSearch(s_ctx);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void DiscogsCoverProvider::EndSearch(DiscogsCoverSearchContext *s_ctx) {
|
||||||
|
|
||||||
|
//qLog(Debug) << __PRETTY_FUNCTION__;
|
||||||
|
|
||||||
|
(void)requests_search_.remove(s_ctx->id);
|
||||||
|
emit SearchFinished(s_ctx->id, s_ctx->results);
|
||||||
|
delete s_ctx;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -34,7 +34,7 @@ class QNetworkAccessManager;
|
|||||||
|
|
||||||
// This struct represents a single search-for-cover request. It identifies and describes the request.
|
// This struct represents a single search-for-cover request. It identifies and describes the request.
|
||||||
struct DiscogsCoverSearchContext {
|
struct DiscogsCoverSearchContext {
|
||||||
enum State { State_Init, State_MastersRequested, State_ReleasesRequested };
|
enum State { State_Init, State_Release };
|
||||||
|
|
||||||
// the unique request identifier
|
// the unique request identifier
|
||||||
int id;
|
int id;
|
||||||
@ -42,6 +42,8 @@ struct DiscogsCoverSearchContext {
|
|||||||
// the search query
|
// the search query
|
||||||
QString artist;
|
QString artist;
|
||||||
QString album;
|
QString album;
|
||||||
|
QString title;
|
||||||
|
int r_count;
|
||||||
|
|
||||||
State state;
|
State state;
|
||||||
|
|
||||||
@ -49,13 +51,32 @@ struct DiscogsCoverSearchContext {
|
|||||||
};
|
};
|
||||||
Q_DECLARE_METATYPE(DiscogsCoverSearchContext)
|
Q_DECLARE_METATYPE(DiscogsCoverSearchContext)
|
||||||
|
|
||||||
|
struct DiscogsCoverReleaseContext {
|
||||||
|
//enum State { State_Init, State_MastersRequested, State_ReleasesRequested };
|
||||||
|
|
||||||
|
int id; // the unique request identifier
|
||||||
|
int s_id; // the search request identifier
|
||||||
|
|
||||||
|
// the search query
|
||||||
|
//QString artist;
|
||||||
|
//QString album;
|
||||||
|
//QString title;
|
||||||
|
QString resource_url;
|
||||||
|
|
||||||
|
//State state;
|
||||||
|
|
||||||
|
//CoverSearchResults results;
|
||||||
|
};
|
||||||
|
Q_DECLARE_METATYPE(DiscogsCoverReleaseContext)
|
||||||
|
|
||||||
class DiscogsCoverProvider : public CoverProvider {
|
class DiscogsCoverProvider : public CoverProvider {
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
|
||||||
public:
|
public:
|
||||||
explicit DiscogsCoverProvider(QObject *parent = nullptr);
|
explicit DiscogsCoverProvider(QObject *parent = nullptr);
|
||||||
|
|
||||||
static const char *kUrl;
|
static const char *kUrlSearch;
|
||||||
|
static const char *kUrlReleases;
|
||||||
|
|
||||||
static const char *kRequestTokenURL;
|
static const char *kRequestTokenURL;
|
||||||
static const char *kAuthorizeURL;
|
static const char *kAuthorizeURL;
|
||||||
@ -67,22 +88,27 @@ class DiscogsCoverProvider : public CoverProvider {
|
|||||||
static const char *kAccessKeyB64;
|
static const char *kAccessKeyB64;
|
||||||
static const char *kSecretAccessKeyB64;
|
static const char *kSecretAccessKeyB64;
|
||||||
|
|
||||||
bool StartSearch(const QString &artist, const QString &album, int id);
|
bool StartSearch(const QString &artist, const QString &album, int s_id);
|
||||||
|
|
||||||
void CancelSearch(int id);
|
void CancelSearch(int id);
|
||||||
|
|
||||||
private slots:
|
private slots:
|
||||||
void QueryError(QNetworkReply::NetworkError error, QNetworkReply *reply, int id);
|
void SearchRequestError(QNetworkReply::NetworkError error, QNetworkReply *reply, int s_id);
|
||||||
void HandleSearchReply(QNetworkReply* reply, int id);
|
void ReleaseRequestError(QNetworkReply::NetworkError error, QNetworkReply *reply, int s_id, int r_id);
|
||||||
|
void HandleSearchReply(QNetworkReply *reply, int s_id);
|
||||||
|
void HandleReleaseReply(QNetworkReply *reply, int sa_id, int si_id);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QNetworkAccessManager *network_;
|
QNetworkAccessManager *network_;
|
||||||
QHash<int, DiscogsCoverSearchContext*> pending_requests_;
|
QHash<int, DiscogsCoverSearchContext*> requests_search_;
|
||||||
|
QHash<int, DiscogsCoverReleaseContext*> requests_release_;
|
||||||
|
|
||||||
|
bool StartRelease(DiscogsCoverSearchContext *s_ctx, int r_id, QString resource_url);
|
||||||
|
|
||||||
void SendSearchRequest(DiscogsCoverSearchContext *ctx);
|
void SendSearchRequest(DiscogsCoverSearchContext *s_ctx);
|
||||||
void ReadItem(QXmlStreamReader *reader, CoverSearchResults *results);
|
void SendReleaseRequest(DiscogsCoverSearchContext *s_ctx, DiscogsCoverReleaseContext *r_ctx);
|
||||||
void ReadLargeImage(QXmlStreamReader *reader, CoverSearchResults *results);
|
void EndSearch(DiscogsCoverSearchContext *s_ctx, DiscogsCoverReleaseContext *r_ctx);
|
||||||
void QueryFinished(QNetworkReply *reply, int id);
|
void EndSearch(DiscogsCoverSearchContext *s_ctx);
|
||||||
void EndSearch(DiscogsCoverSearchContext *ctx);
|
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -53,11 +53,11 @@ bool MusicbrainzCoverProvider::StartSearch(const QString &artist, const QString
|
|||||||
QNetworkRequest request(url);
|
QNetworkRequest request(url);
|
||||||
|
|
||||||
QNetworkReply *reply = network_->get(request);
|
QNetworkReply *reply = network_->get(request);
|
||||||
NewClosure(reply, SIGNAL(finished()), this, SLOT(ReleaseSearchFinished(QNetworkReply*, int)), reply, id);
|
NewClosure(reply, SIGNAL(finished()), this, SLOT(ReleaseSearchFinished(QNetworkReply *, int)), reply, id);
|
||||||
|
|
||||||
cover_names_[id] = QString("%1 - %2").arg(artist, album);
|
cover_names_[id] = QString("%1 - %2").arg(artist, album);
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void MusicbrainzCoverProvider::ReleaseSearchFinished(QNetworkReply *reply, int id) {
|
void MusicbrainzCoverProvider::ReleaseSearchFinished(QNetworkReply *reply, int id) {
|
||||||
@ -77,22 +77,24 @@ void MusicbrainzCoverProvider::ReleaseSearchFinished(QNetworkReply *reply, int i
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (const QString& release_id : releases) {
|
for (const QString &release_id : releases) {
|
||||||
QUrl url(QString(kAlbumCoverUrl).arg(release_id));
|
QUrl url(QString(kAlbumCoverUrl).arg(release_id));
|
||||||
QNetworkReply *reply = network_->head(QNetworkRequest(url));
|
QNetworkReply *reply = network_->head(QNetworkRequest(url));
|
||||||
image_checks_.insert(id, reply);
|
image_checks_.insert(id, reply);
|
||||||
NewClosure(reply, SIGNAL(finished()), this, SLOT(ImageCheckFinished(int)), id);
|
NewClosure(reply, SIGNAL(finished()), this, SLOT(ImageCheckFinished(int)), id);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void MusicbrainzCoverProvider::ImageCheckFinished(int id) {
|
void MusicbrainzCoverProvider::ImageCheckFinished(int id) {
|
||||||
|
|
||||||
QList<QNetworkReply*> replies = image_checks_.values(id);
|
QList<QNetworkReply *> replies = image_checks_.values(id);
|
||||||
|
|
||||||
int finished_count = std::count_if(replies.constBegin(), replies.constEnd(), mem_fun(&QNetworkReply::isFinished));
|
int finished_count = std::count_if(replies.constBegin(), replies.constEnd(), mem_fun(&QNetworkReply::isFinished));
|
||||||
if (finished_count == replies.size()) {
|
if (finished_count == replies.size()) {
|
||||||
QString cover_name = cover_names_.take(id);
|
QString cover_name = cover_names_.take(id);
|
||||||
QList<CoverSearchResult> results;
|
QList<CoverSearchResult> results;
|
||||||
for (QNetworkReply* reply : replies) {
|
for (QNetworkReply *reply : replies) {
|
||||||
reply->deleteLater();
|
reply->deleteLater();
|
||||||
if (reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt() < 400) {
|
if (reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt() < 400) {
|
||||||
CoverSearchResult result;
|
CoverSearchResult result;
|
||||||
@ -109,8 +111,9 @@ void MusicbrainzCoverProvider::ImageCheckFinished(int id) {
|
|||||||
|
|
||||||
void MusicbrainzCoverProvider::CancelSearch(int id) {
|
void MusicbrainzCoverProvider::CancelSearch(int id) {
|
||||||
|
|
||||||
QList<QNetworkReply*> replies = image_checks_.values(id);
|
QList<QNetworkReply *> replies = image_checks_.values(id);
|
||||||
for (QNetworkReply* reply : replies) {
|
|
||||||
|
for (QNetworkReply *reply : replies) {
|
||||||
reply->abort();
|
reply->abort();
|
||||||
reply->deleteLater();
|
reply->deleteLater();
|
||||||
}
|
}
|
||||||
|
@ -50,4 +50,3 @@ class MusicbrainzCoverProvider : public CoverProvider {
|
|||||||
};
|
};
|
||||||
|
|
||||||
#endif // MUSICBRAINZCOVERPROVIDER_H
|
#endif // MUSICBRAINZCOVERPROVIDER_H
|
||||||
|
|
||||||
|
@ -20,9 +20,6 @@
|
|||||||
|
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
|
|
||||||
//#include <qjson/parser.h>
|
|
||||||
#include <QJson/Parser>
|
|
||||||
|
|
||||||
#include <QCoreApplication>
|
#include <QCoreApplication>
|
||||||
#include <QNetworkReply>
|
#include <QNetworkReply>
|
||||||
#include <QStringList>
|
#include <QStringList>
|
||||||
|
Loading…
x
Reference in New Issue
Block a user