strawberry-audio-player-win.../src/tidal/tidalfavoriterequest.cpp

310 lines
9.0 KiB
C++
Raw Normal View History

2020-04-13 19:04:06 +02:00
/*
* Strawberry Music Player
2021-03-20 21:14:47 +01:00
* Copyright 2018-2021, Jonas Kvinge <jonas@jkvinge.net>
2020-04-13 19:04:06 +02:00
*
* Strawberry 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.
*
* Strawberry 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 Strawberry. If not, see <http://www.gnu.org/licenses/>.
*
*/
#include "config.h"
#include <QtGlobal>
#include <QObject>
#include <QMap>
#include <QMultiMap>
#include <QByteArray>
#include <QString>
#include <QStringList>
#include <QUrl>
#include <QUrlQuery>
#include <QNetworkRequest>
#include <QNetworkReply>
#include "core/logging.h"
2021-01-11 16:48:46 +01:00
#include "core/networkaccessmanager.h"
2020-04-13 19:04:06 +02:00
#include "core/song.h"
#include "tidalservice.h"
#include "tidalbaserequest.h"
#include "tidalfavoriterequest.h"
TidalFavoriteRequest::TidalFavoriteRequest(TidalService *service, NetworkAccessManager *network, QObject *parent)
: TidalBaseRequest(service, network, parent),
2021-07-11 07:40:57 +02:00
service_(service),
network_(network),
need_login_(false) {}
2020-04-13 19:04:06 +02:00
TidalFavoriteRequest::~TidalFavoriteRequest() {
while (!replies_.isEmpty()) {
QNetworkReply *reply = replies_.takeFirst();
2021-01-26 16:48:04 +01:00
QObject::disconnect(reply, nullptr, this, nullptr);
2020-04-13 19:04:06 +02:00
reply->abort();
reply->deleteLater();
}
}
QString TidalFavoriteRequest::FavoriteText(const FavoriteType type) {
switch (type) {
case FavoriteType_Artists:
return "artists";
case FavoriteType_Albums:
return "albums";
case FavoriteType_Songs:
return "tracks";
}
return QString();
}
QString TidalFavoriteRequest::FavoriteMethod(const FavoriteType type) {
switch (type) {
case FavoriteType_Artists:
return "artistIds";
case FavoriteType_Albums:
return "albumIds";
case FavoriteType_Songs:
return "trackIds";
}
return QString();
2020-04-13 19:04:06 +02:00
}
void TidalFavoriteRequest::AddArtists(const SongList &songs) {
AddFavorites(FavoriteType_Artists, songs);
}
void TidalFavoriteRequest::AddAlbums(const SongList &songs) {
AddFavorites(FavoriteType_Albums, songs);
}
void TidalFavoriteRequest::AddSongs(const SongList &songs) {
AddFavorites(FavoriteType_Songs, songs);
}
void TidalFavoriteRequest::AddSongs(const SongMap &songs) {
AddFavoritesRequest(FavoriteType_Songs, songs.keys(), songs.values());
2020-04-13 19:04:06 +02:00
}
void TidalFavoriteRequest::AddFavorites(const FavoriteType type, const SongList &songs) {
2020-04-23 21:05:57 +02:00
QStringList id_list;
2020-04-13 19:04:06 +02:00
for (const Song &song : songs) {
QString id;
switch (type) {
case FavoriteType_Artists:
if (song.artist_id().isEmpty()) continue;
id = song.artist_id();
break;
2021-07-11 09:49:38 +02:00
case FavoriteType_Albums:
2020-04-13 19:04:06 +02:00
if (song.album_id().isEmpty()) continue;
id = song.album_id();
break;
2021-07-11 09:49:38 +02:00
case FavoriteType_Songs:
2020-04-13 19:04:06 +02:00
if (song.song_id().isEmpty()) continue;
id = song.song_id();
break;
}
if (!id.isEmpty() && !id_list.contains(id)) {
2020-04-23 21:05:57 +02:00
id_list << id;
2020-04-13 19:04:06 +02:00
}
}
2020-04-23 21:05:57 +02:00
if (id_list.isEmpty()) return;
2020-04-13 19:04:06 +02:00
AddFavoritesRequest(type, id_list, songs);
}
void TidalFavoriteRequest::AddFavoritesRequest(const FavoriteType type, const QStringList &id_list, const SongList &songs) {
2020-04-13 19:04:06 +02:00
ParamList params = ParamList() << Param("countryCode", country_code())
<< Param(FavoriteMethod(type), id_list.join(','));
2020-04-13 19:04:06 +02:00
QUrlQuery url_query;
2021-06-12 20:53:23 +02:00
for (const Param &param : params) {
2020-04-23 21:05:57 +02:00
url_query.addQueryItem(QUrl::toPercentEncoding(param.first), QUrl::toPercentEncoding(param.second));
2020-04-13 19:04:06 +02:00
}
2022-10-31 06:12:02 +01:00
QUrl url(QString(TidalService::kApiUrl) + QString("/") + "users/" + QString::number(service_->user_id()) + "/favorites/" + FavoriteText(type));
2020-04-13 19:04:06 +02:00
QNetworkRequest req(url);
req.setAttribute(QNetworkRequest::RedirectPolicyAttribute, QNetworkRequest::NoLessSafeRedirectPolicy);
2020-04-13 19:04:06 +02:00
req.setHeader(QNetworkRequest::ContentTypeHeader, "application/x-www-form-urlencoded");
if (oauth() && !access_token().isEmpty()) req.setRawHeader("authorization", "Bearer " + access_token().toUtf8());
else if (!session_id().isEmpty()) req.setRawHeader("X-Tidal-SessionId", session_id().toUtf8());
2020-04-13 19:04:06 +02:00
QByteArray query = url_query.toString(QUrl::FullyEncoded).toUtf8();
QNetworkReply *reply = network_->post(req, query);
2021-03-21 00:37:17 +01:00
QObject::connect(reply, &QNetworkReply::finished, this, [this, reply, type, songs]() { AddFavoritesReply(reply, type, songs); });
2020-04-13 19:04:06 +02:00
replies_ << reply;
qLog(Debug) << "Tidal: Sending request" << url << query;
}
void TidalFavoriteRequest::AddFavoritesReply(QNetworkReply *reply, const FavoriteType type, const SongList &songs) {
if (replies_.contains(reply)) {
replies_.removeAll(reply);
2021-01-26 16:48:04 +01:00
QObject::disconnect(reply, nullptr, this, nullptr);
2020-04-13 19:04:06 +02:00
reply->deleteLater();
}
else {
return;
}
2021-03-21 04:43:55 +01:00
GetReplyData(reply, false);
2020-04-13 19:04:06 +02:00
if (reply->error() != QNetworkReply::NoError) {
return;
}
qLog(Debug) << "Tidal:" << songs.count() << "songs added to" << FavoriteText(type) << "favorites.";
switch (type) {
case FavoriteType_Artists:
emit ArtistsAdded(songs);
break;
2021-07-11 09:49:38 +02:00
case FavoriteType_Albums:
2020-04-13 19:04:06 +02:00
emit AlbumsAdded(songs);
break;
2021-07-11 09:49:38 +02:00
case FavoriteType_Songs:
2020-04-13 19:04:06 +02:00
emit SongsAdded(songs);
break;
}
}
void TidalFavoriteRequest::RemoveArtists(const SongList &songs) {
RemoveFavorites(FavoriteType_Artists, songs);
}
void TidalFavoriteRequest::RemoveAlbums(const SongList &songs) {
RemoveFavorites(FavoriteType_Albums, songs);
}
void TidalFavoriteRequest::RemoveSongs(const SongList &songs) {
RemoveFavorites(FavoriteType_Songs, songs);
}
void TidalFavoriteRequest::RemoveSongs(const SongMap &songs) {
SongList songs_list = songs.values();
for (const Song &song : songs_list) {
RemoveFavoritesRequest(FavoriteType_Songs, song.song_id(), SongList() << song);
}
2020-04-13 19:04:06 +02:00
}
void TidalFavoriteRequest::RemoveFavorites(const FavoriteType type, const SongList &songs) {
2020-04-13 19:04:06 +02:00
QMultiMap<QString, Song> songs_map;
for (const Song &song : songs) {
QString id;
switch (type) {
case FavoriteType_Artists:
2020-04-23 21:05:57 +02:00
if (song.artist_id().isEmpty()) continue;
2020-04-13 19:04:06 +02:00
id = song.artist_id();
break;
2021-07-11 09:49:38 +02:00
case FavoriteType_Albums:
2020-04-13 19:04:06 +02:00
if (song.album_id().isEmpty()) continue;
2020-04-23 21:05:57 +02:00
id = song.album_id();
2020-04-13 19:04:06 +02:00
break;
2021-07-11 09:49:38 +02:00
case FavoriteType_Songs:
2020-04-23 21:05:57 +02:00
if (song.song_id().isEmpty()) continue;
2020-04-13 19:04:06 +02:00
id = song.song_id();
break;
}
if (!id.isEmpty()) {
songs_map.insert(id, song);
}
2020-04-13 19:04:06 +02:00
}
QStringList ids = songs_map.uniqueKeys();
2020-04-13 19:04:06 +02:00
for (const QString &id : ids) {
RemoveFavoritesRequest(type, id, songs_map.values(id));
2020-04-13 19:04:06 +02:00
}
}
void TidalFavoriteRequest::RemoveFavoritesRequest(const FavoriteType type, const QString &id, const SongList &songs) {
2020-04-13 19:04:06 +02:00
ParamList params = ParamList() << Param("countryCode", country_code());
QUrlQuery url_query;
2021-06-12 20:53:23 +02:00
for (const Param &param : params) {
2020-04-23 21:05:57 +02:00
url_query.addQueryItem(QUrl::toPercentEncoding(param.first), QUrl::toPercentEncoding(param.second));
2020-04-13 19:04:06 +02:00
}
2022-10-31 06:12:02 +01:00
QUrl url(QString(TidalService::kApiUrl) + QString("/") + "users/" + QString::number(service_->user_id()) + "/favorites/" + FavoriteText(type) + QString("/") + id);
2020-04-13 19:04:06 +02:00
url.setQuery(url_query);
QNetworkRequest req(url);
#if QT_VERSION >= QT_VERSION_CHECK(5, 9, 0)
req.setAttribute(QNetworkRequest::RedirectPolicyAttribute, QNetworkRequest::NoLessSafeRedirectPolicy);
#else
2020-04-13 19:04:06 +02:00
req.setAttribute(QNetworkRequest::FollowRedirectsAttribute, true);
#endif
2020-04-13 19:04:06 +02:00
req.setHeader(QNetworkRequest::ContentTypeHeader, "application/x-www-form-urlencoded");
if (oauth() && !access_token().isEmpty()) req.setRawHeader("authorization", "Bearer " + access_token().toUtf8());
else if (!session_id().isEmpty()) req.setRawHeader("X-Tidal-SessionId", session_id().toUtf8());
2020-04-13 19:04:06 +02:00
QNetworkReply *reply = network_->deleteResource(req);
2021-03-21 00:37:17 +01:00
QObject::connect(reply, &QNetworkReply::finished, this, [this, reply, type, songs]() { RemoveFavoritesReply(reply, type, songs); });
2020-04-13 19:04:06 +02:00
replies_ << reply;
qLog(Debug) << "Tidal: Sending request" << url << "with" << songs.count() << "songs";
}
void TidalFavoriteRequest::RemoveFavoritesReply(QNetworkReply *reply, const FavoriteType type, const SongList &songs) {
if (replies_.contains(reply)) {
replies_.removeAll(reply);
2021-01-26 16:48:04 +01:00
QObject::disconnect(reply, nullptr, this, nullptr);
2020-04-13 19:04:06 +02:00
reply->deleteLater();
}
else {
return;
}
2021-03-21 04:43:55 +01:00
GetReplyData(reply, false);
2020-04-13 19:04:06 +02:00
if (reply->error() != QNetworkReply::NoError) {
return;
}
qLog(Debug) << "Tidal:" << songs.count() << "songs removed from" << FavoriteText(type) << "favorites.";
switch (type) {
case FavoriteType_Artists:
emit ArtistsRemoved(songs);
break;
2021-07-11 09:49:38 +02:00
case FavoriteType_Albums:
2020-04-13 19:04:06 +02:00
emit AlbumsRemoved(songs);
break;
2021-07-11 09:49:38 +02:00
case FavoriteType_Songs:
2020-04-13 19:04:06 +02:00
emit SongsRemoved(songs);
break;
}
}
void TidalFavoriteRequest::Error(const QString &error, const QVariant &debug) {
qLog(Error) << "Tidal:" << error;
if (debug.isValid()) qLog(Debug) << debug;
}