TidalStreamURLRequest: Fix parsing manifest urls

This commit is contained in:
Jonas Kvinge
2025-07-19 21:38:55 +02:00
parent 2ddacf2f98
commit e614626913
2 changed files with 59 additions and 29 deletions

View File

@@ -142,6 +142,23 @@ void TidalStreamURLRequest::GetStreamURL() {
} }
QList<QUrl> TidalStreamURLRequest::ParseUrls(const QJsonObject &json_object) {
const QJsonValue json_urls = json_object["urls"_L1];
if (!json_urls.isArray()) {
return QList<QUrl>();
}
const QJsonArray json_array_urls = json_urls.toArray();
QList<QUrl> urls;
urls.reserve(json_array_urls.count());
for (const QJsonValue &value : json_array_urls) {
urls << QUrl(value.toString());
}
return urls;
}
void TidalStreamURLRequest::StreamURLReceived() { void TidalStreamURLRequest::StreamURLReceived() {
if (!reply_) return; if (!reply_) return;
@@ -170,7 +187,7 @@ void TidalStreamURLRequest::StreamURLReceived() {
qLog(Debug) << "Tidal returned track ID" << track_id << "for" << media_url_; qLog(Debug) << "Tidal returned track ID" << track_id << "for" << media_url_;
} }
Song::FileType filetype(Song::FileType::Stream); Song::FileType filetype = Song::FileType::Stream;
if (json_object.contains("codec"_L1) || json_object.contains("codecs"_L1)) { if (json_object.contains("codec"_L1) || json_object.contains("codecs"_L1)) {
QString codec; QString codec;
@@ -183,11 +200,21 @@ void TidalStreamURLRequest::StreamURLReceived() {
} }
} }
int samplerate = -1;
if (json_object.contains("sampleRate"_L1)) {
samplerate = json_object["sampleRate"_L1].toInt();
}
int bit_depth = -1;
if (json_object.contains("bitDepth"_L1)) {
bit_depth = json_object["bitDepth"_L1].toInt();
}
QList<QUrl> urls; QList<QUrl> urls;
if (json_object.contains("manifest"_L1)) { if (json_object.contains("manifest"_L1)) {
const QString manifest(json_object["manifest"_L1].toString()); const QString manifest = json_object["manifest"_L1].toString();
const QByteArray data_manifest = QByteArray::fromBase64(manifest.toUtf8()); const QByteArray data_manifest = QByteArray::fromBase64(manifest.toUtf8());
QXmlStreamReader xml_reader(data_manifest); QXmlStreamReader xml_reader(data_manifest);
@@ -207,46 +234,46 @@ void TidalStreamURLRequest::StreamURLReceived() {
} }
const QJsonObject &object_manifest = json_object_result_manifest.json_object; const QJsonObject &object_manifest = json_object_result_manifest.json_object;
if (object_manifest.contains("encryptionType"_L1) && object_manifest.contains("keyId"_L1)) { if (object_manifest.contains("encryptionType"_L1)) {
QString encryption_type = object_manifest["encryptionType"_L1].toString(); const QString encryption_type = object_manifest["encryptionType"_L1].toString();
QString key_id = object_manifest["keyId"_L1].toString(); if (!encryption_type.isEmpty() && encryption_type != "NONE"_L1) {
if (!encryption_type.isEmpty() && !key_id.isEmpty()) {
Q_EMIT StreamURLFailure(id_, media_url_, tr("Received URL with %1 encrypted stream from Tidal. Strawberry does not currently support encrypted streams.").arg(encryption_type)); Q_EMIT StreamURLFailure(id_, media_url_, tr("Received URL with %1 encrypted stream from Tidal. Strawberry does not currently support encrypted streams.").arg(encryption_type));
return; return;
} }
} }
if (!object_manifest.contains("mimeType"_L1)) { if (object_manifest.contains("urls"_L1)) {
Q_EMIT StreamURLFailure(id_, media_url_, u"Invalid Json reply, stream url reply manifest is missing mimeType."_s); urls << ParseUrls(object_manifest);
return;
} }
const QString mimetype = object_manifest["mimeType"_L1].toString(); if (object_manifest.contains("mimeType_"_L1)) {
QMimeDatabase mimedb; const QString mimetype = object_manifest["mimeType"_L1].toString();
const QStringList suffixes = mimedb.mimeTypeForName(mimetype).suffixes(); QMimeDatabase mimedb;
for (const QString &suffix : suffixes) { const QStringList suffixes = mimedb.mimeTypeForName(mimetype).suffixes();
filetype = Song::FiletypeByExtension(suffix); for (const QString &suffix : suffixes) {
if (filetype != Song::FileType::Unknown) break; filetype = Song::FiletypeByExtension(suffix);
if (filetype != Song::FileType::Unknown) break;
}
if (filetype == Song::FileType::Unknown) {
qLog(Debug) << "Tidal: Unknown mimetype" << mimetype;
filetype = Song::FileType::Stream;
}
} }
if (filetype == Song::FileType::Unknown) { else if (object_manifest.contains("codecs"_L1)) {
qLog(Debug) << "Tidal: Unknown mimetype" << mimetype; const QString codec = object_manifest["codecs"_L1].toString().toLower();
filetype = Song::FileType::Stream; filetype = Song::FiletypeByExtension(codec);
if (filetype == Song::FileType::Unknown) {
qLog(Debug) << "Tidal: Unknown codec" << codec;
filetype = Song::FileType::Stream;
}
} }
} }
} }
if (json_object.contains("urls"_L1)) { if (json_object.contains("urls"_L1)) {
const QJsonValue json_urls = json_object["urls"_L1]; urls << ParseUrls(json_object);
if (!json_urls.isArray()) {
Q_EMIT StreamURLFailure(id_, media_url_, u"Invalid Json reply, urls is not an array."_s);
return;
}
const QJsonArray json_array_urls = json_urls.toArray();
urls.reserve(json_array_urls.count());
for (const QJsonValue &value : json_array_urls) {
urls << QUrl(value.toString());
}
} }
else if (json_object.contains("url"_L1)) { else if (json_object.contains("url"_L1)) {
const QUrl new_url(json_object["url"_L1].toString()); const QUrl new_url(json_object["url"_L1].toString());
@@ -280,6 +307,6 @@ void TidalStreamURLRequest::StreamURLReceived() {
return; return;
} }
Q_EMIT StreamURLSuccess(id_, media_url_, urls.first(), filetype); Q_EMIT StreamURLSuccess(id_, media_url_, urls.first(), filetype, samplerate, bit_depth);
} }

View File

@@ -58,6 +58,9 @@ class TidalStreamURLRequest : public TidalBaseRequest {
private Q_SLOTS: private Q_SLOTS:
void StreamURLReceived(); void StreamURLReceived();
private:
static QList<QUrl> ParseUrls(const QJsonObject &json_object);
private: private:
TidalService *service_; TidalService *service_;
QNetworkReply *reply_; QNetworkReply *reply_;