/***************************************************************************
* Copyright (C) 2010-2016 by Ilya Kotov *
* forkotov02@ya.ru *
* *
* 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 2 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, write to the *
* Free Software Foundation, Inc., *
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *
***************************************************************************/
#include <QNetworkAccessManager>
#include <QNetworkRequest>
#include <QNetworkProxy>
#include <QNetworkReply>
#include <QByteArray>
#include <QCryptographicHash>
#include <QXmlStreamReader>
#include <QUrl>
#include <QTime>
#include <QTimer>
#include <QDateTime>
#include <QDir>
#include <QDesktopServices>
#include <QSettings>
#include <qmmp/soundcore.h>
#include <qmmp/qmmpsettings.h>
#include <qmmp/qmmp.h>
#include "scrobbler.h"
#define API_KEY "d71c6f01b2ea562d7042bd5f5970041f"
#define SECRET "32d47bc0010473d40e1d38bdcff20968"
void ScrobblerResponse::parse(QIODevice *device)
{
QXmlStreamReader reader(device);
QStringList tags;
while(!reader.atEnd())
{
reader.readNext();
if(reader.isStartElement())
{
tags << reader.name().toString();
if(tags.last() == "lfm")
status = reader.attributes().value("status").toString();
else if(tags.last() == "error")
code = reader.attributes().value("code").toString();
}
else if(reader.isCharacters() && !reader.isWhitespace())
{
if(tags.last() == "token")
token = reader.text().toString();
else if(tags.last() == "error")
error = reader.text().toString();
if(tags.count() >= 2 && tags.at(tags.count() - 2) == "session")
{
if(tags.last() == "key")
key = reader.text().toString();
else if(tags.last() == "name")
name = reader.text().toString();
else if(tags.last() == "subscriber")
subscriber = reader.text().toString();
}
}
else if(reader.isEndElement())
{
tags.takeLast();
}
}
}
Scrobbler::Scrobbler(const QString &scrobblerUrl, const QString &name, QObject *parent)
: QObject(parent)
{
m_notificationReply = 0;
m_submitedSongs = 0;
m_submitReply = 0;
m_previousState = Qmmp::Stopped;
m_elapsed = 0;
m_scrobblerUrl = scrobblerUrl;
m_name = name;
m_time = new QTime();
m_cache = new ScrobblerCache(Qmmp::configDir() +"scrobbler_"+name+".cache");
m_ua = QString("qmmp-plugins/%1").arg(Qmmp::strVersion().toLower()).toLatin1();
m_http = new QNetworkAccessManager(this);
m_core = SoundCore::instance();
QSettings settings(Qmmp::configFile(), QSettings::IniFormat);
m_session = settings.value("Scrobbler/"+name+"_session").toString();
connect(m_http, SIGNAL(finished (QNetworkReply *)), SLOT(processResponse(QNetworkReply *)));
connect(QmmpSettings::instance(), SIGNAL(networkSettingsChanged()), SLOT(setupProxy()));
connect(m_core, SIGNAL(metaDataChanged()), SLOT(updateMetaData()));
connect(m_core, SIGNAL(stateChanged (Qmmp::State)), SLOT(setState(Qmmp::State)));
setupProxy();
m_cachedSongs = m_cache->load();
if(!m_session.isEmpty())
{
submit();
if(m_core->state() == Qmmp::Playing)
{
setState(Qmmp::Playing);
updateMetaData();
}
}
}
Scrobbler::~Scrobbler()
{
m_cache->save(m_cachedSongs);
delete m_time;
delete m_cache;
}
void Scrobbler::setState(Qmmp::State state)
{
if(state == Qmmp::Playing && m_previousState == Qmmp::Paused)
{
qDebug("Scrobbler[%s]: resuming from %d seconds played", qPrintable(m_name), m_elapsed / 1000);
m_time->restart();
}
else if(state == Qmmp::Paused)
{
m_elapsed += m_time->elapsed();
qDebug("Scrobbler[%s]: pausing after %d seconds played", qPrintable(m_name), m_elapsed / 1000);
}
else if(state == Qmmp::Stopped && !m_song.metaData().isEmpty())
{
if(m_previousState == Qmmp::Playing)
m_elapsed += m_time->elapsed();
m_elapsed /= 1000; //convert to seconds
if((m_elapsed > 240) || (m_elapsed > MIN_SONG_LENGTH && m_song.length() == 0) ||
(m_elapsed > int(m_song.length()/2) && m_song.length() > MIN_SONG_LENGTH))
{
m_cachedSongs << m_song;
m_cache->save(m_cachedSongs);
}
submit();
m_song.clear();
m_elapsed = 0;
}
m_previousState = state;
}
void Scrobbler::updateMetaData()
{
QMap <Qmmp::MetaData, QString> metadata = m_core->metaData();
if(m_core->state() != Qmmp::Playing)
return;
if(!m_song.metaData().isEmpty() && m_song.metaData() != metadata)
{
int elapsed = (m_elapsed + m_time->elapsed()) / 1000;
if((elapsed > 240) || (elapsed > MIN_SONG_LENGTH && m_song.length() == 0) ||
(elapsed > int(m_song.length()/2) && m_song.length() > MIN_SONG_LENGTH))
{
m_cachedSongs << m_song;
m_cache->save(m_cachedSongs);
}
submit();
m_song.clear();
}
if(!metadata.value(Qmmp::TITLE).isEmpty() && !metadata.value(Qmmp::ARTIST).isEmpty())
{
m_song = SongInfo(metadata, m_core->totalTime()/1000);
m_song.setTimeStamp(QDateTime::currentDateTime().toTime_t());
sendNotification(m_song);
}
m_time->restart();
m_elapsed = 0;
}
void Scrobbler::processResponse(QNetworkReply *reply)
{
if (reply->error() != QNetworkReply::NoError)
{
qWarning("Scrobbler[%s]: http error: %s", qPrintable(m_name), qPrintable(reply->errorString()));
}
ScrobblerResponse response;
response.parse(reply);
QString error_code;
if(response.status != "ok" && !response.status.isEmpty())
{
if(!response.error.isEmpty())
{
qWarning("Scrobbler[%s]: status=%s, %s-%s", qPrintable(m_name), qPrintable(response.status),
qPrintable(response.code), qPrintable(response.error));
error_code = response.code;
}
else
qWarning("Scrobbler[%s]: invalid content", qPrintable(m_name));
}
if (reply == m_submitReply)
{
m_submitReply = 0;
if (response.status == "ok")
{
qDebug("Scrobbler[%s]: submited %d song(s)", qPrintable(m_name), m_submitedSongs);
while (m_submitedSongs)
{
m_submitedSongs--;
m_cachedSongs.removeFirst ();
}
if (!m_cachedSongs.isEmpty()) //submit remaining songs
{
submit();
}
else
{
m_cache->save(m_cachedSongs); // update the cache file to reflect the empty cache
updateMetaData();
}
}
else if(error_code == "9") //invalid session key
{
m_session.clear();
qWarning("Scrobbler[%s]: invalid session key, scrobbling disabled", qPrintable(m_name));
}
else if(error_code == "11" || error_code == "16" || error_code.isEmpty()) //unavailable
{
QTimer::singleShot(120000, this, SLOT(submit()));
}
else
{
m_session.clear();
qWarning("Scrobbler[%s]: service returned unrecoverable error, scrobbling disabled",
qPrintable(m_name));
}
}
else if (reply == m_notificationReply)
{
m_notificationReply = 0;
if(response.status == "ok")
{
qDebug("Scrobbler[%s]: Now-Playing notification done", qPrintable(m_name));
}
else if(error_code == "9") //invalid session key
{
m_session.clear();
qWarning("Scrobbler[%s]: invalid session key, scrobbling disabled", qPrintable(m_name));
}
}
reply->deleteLater();
}
void Scrobbler::setupProxy()
{
QmmpSettings *gs = QmmpSettings::instance();
if (gs->isProxyEnabled())
{
QNetworkProxy proxy(QNetworkProxy::HttpProxy, gs->proxy().host(), gs->proxy().port());
if(gs->useProxyAuth())
{
proxy.setUser(gs->proxy().userName());
proxy.setPassword(gs->proxy().password());
}
m_http->setProxy(proxy);
}
else
m_http->setProxy(QNetworkProxy::NoProxy);
}
void Scrobbler::submit()
{
if (m_cachedSongs.isEmpty() || m_session.isEmpty() || m_submitReply)
return;
qDebug("Scrobbler[%s]: submit request", qPrintable(m_name));
m_submitedSongs = qMin(m_cachedSongs.size(),25);
QMap <QString, QString> params;
for (int i = 0; i < m_submitedSongs; ++i)
{
SongInfo info = m_cachedSongs[i];
params.insert(QString("track[%1]").arg(i),info.metaData(Qmmp::TITLE));
params.insert(QString("timestamp[%1]").arg(i),QString("%1").arg(info.timeStamp()));
params.insert(QString("artist[%1]").arg(i),info.metaData(Qmmp::ARTIST));
params.insert(QString("album[%1]").arg(i),info.metaData(Qmmp::ALBUM));
params.insert(QString("trackNumber[%1]").arg(i),info.metaData(Qmmp::TRACK));
if(info.length() > 0)
params.insert(QString("duration[%1]").arg(i),QString("%1").arg(info.length()));
}
params.insert("api_key", API_KEY);
params.insert("method", "track.scrobble");
params.insert("sk", m_session);
QStringList keys = params.keys();
foreach (QString key, keys) //removes empty keys
{
if(params.value(key).isEmpty() || params.value(key) == "0")
params.remove(key);
}
QUrl url(m_scrobblerUrl);
url.setPort(m_scrobblerUrl.startsWith("https") ? 443 : 80);
QUrl body("");
QByteArray data;
foreach (QString key, params.keys())
{
body.addQueryItem(key, params.value(key));
data.append(key.toUtf8() + params.value(key).toUtf8());
}
data.append(SECRET);
body.addQueryItem("api_sig", QCryptographicHash::hash(data, QCryptographicHash::Md5).toHex());
QByteArray bodyData = body.toEncoded().remove(0,1);
bodyData.replace("+", QUrl::toPercentEncoding("+"));
QNetworkRequest request(url);
request.setRawHeader("User-Agent", m_ua);
request.setRawHeader("Host", url.host().toLatin1());
request.setRawHeader("Accept", "*/*");
request.setRawHeader("Content-Type", "application/x-www-form-urlencoded");
request.setHeader(QNetworkRequest::ContentLengthHeader, bodyData.size());
m_submitReply = m_http->post(request, bodyData);
}
void Scrobbler::sendNotification(const SongInfo &info)
{
if(m_session.isEmpty() || m_notificationReply)
return;
qDebug("Scrobbler[%s]: sending notification", qPrintable(m_name));
QMap <QString, QString> params;
params.insert("track", info.metaData(Qmmp::TITLE));
params.insert("artist", info.metaData(Qmmp::ARTIST));
if(!info.metaData(Qmmp::ALBUM).isEmpty())
params.insert("album", info.metaData(Qmmp::ALBUM));
if(!info.metaData(Qmmp::TRACK).isEmpty())
params.insert("trackNumber", info.metaData(Qmmp::TRACK));
if(info.length() > 0)
params.insert("duration", QString("%1").arg(info.length()));
params.insert("api_key", API_KEY);
params.insert("method", "track.updateNowPlaying");
params.insert("sk", m_session);
foreach (QString key, params) //removes empty keys
{
if(params.value(key).isEmpty())
params.remove(key);
}
QUrl url(m_scrobblerUrl);
url.setPort(m_scrobblerUrl.startsWith("https") ? 443 : 80);
QUrl body("");
QByteArray data;
foreach (QString key, params.keys())
{
body.addQueryItem(key, params.value(key));
data.append(key.toUtf8() + params.value(key).toUtf8());
}
data.append(SECRET);
body.addQueryItem("api_sig", QCryptographicHash::hash(data, QCryptographicHash::Md5).toHex());
QByteArray bodyData = body.toEncoded().remove(0,1);
bodyData.replace("+", QUrl::toPercentEncoding("+"));
QNetworkRequest request(url);
request.setRawHeader("User-Agent", m_ua);
request.setRawHeader("Host", url.host().toLatin1());
request.setRawHeader("Accept", "*/*");
request.setRawHeader("Content-Type", "application/x-www-form-urlencoded");
request.setHeader(QNetworkRequest::ContentLengthHeader, bodyData.size());
m_notificationReply = m_http->post(request, bodyData);
}
ScrobblerAuth::ScrobblerAuth(const QString &scrobblerUrl, const QString &authUrl,
const QString &name, QObject *parent) : QObject(parent)
{
m_getTokenReply = 0;
m_getSessionReply = 0;
m_scrobblerUrl = scrobblerUrl;
m_authUrl = authUrl;
m_name = name;
m_ua = QString("qmmp-plugins/%1").arg(Qmmp::strVersion().toLower()).toLatin1();
m_http = new QNetworkAccessManager(this);
connect(m_http, SIGNAL(finished (QNetworkReply *)), SLOT(processResponse(QNetworkReply *)));
QmmpSettings *gs = QmmpSettings::instance();
if (gs->isProxyEnabled())
{
QNetworkProxy proxy(QNetworkProxy::HttpProxy, gs->proxy().host(), gs->proxy().port());
if(gs->useProxyAuth())
{
proxy.setUser(gs->proxy().userName());
proxy.setPassword(gs->proxy().password());
}
m_http->setProxy(proxy);
}
else
m_http->setProxy(QNetworkProxy::NoProxy);
}
void ScrobblerAuth::getToken()
{
qDebug("ScrobblerAuth[%s]: new token request", qPrintable(m_name));
m_session.clear();
QUrl url(m_scrobblerUrl + "?");
url.setPort(m_scrobblerUrl.startsWith("https") ? 443 : 80);
url.addQueryItem("method", "auth.getToken");
url.addQueryItem("api_key", API_KEY);
QByteArray data;
data.append("api_key" API_KEY);
data.append("methodauth.getToken");
data.append(SECRET);
url.addQueryItem("api_sig", QCryptographicHash::hash(data,QCryptographicHash::Md5).toHex());
QNetworkRequest request(url);
request.setRawHeader("User-Agent", m_ua);
request.setRawHeader("Host",url.host().toLatin1());
request.setRawHeader("Accept", "*/*");
m_getTokenReply = m_http->get(request);
}
void ScrobblerAuth::getSession()
{
qDebug("ScrobblerAuth[%s]: new session request", qPrintable(m_name));
QUrl url(m_scrobblerUrl + "?");
url.setPort(m_scrobblerUrl.startsWith("https") ? 443 : 80);
url.addQueryItem("api_key", API_KEY);
url.addQueryItem("method", "auth.getSession");
url.addQueryItem("token", m_token);
QByteArray data;
data.append("api_key" API_KEY);
data.append("methodauth.getSession");
data.append("token" + m_token.toUtf8());
data.append(SECRET);
url.addQueryItem("api_sig", QCryptographicHash::hash(data, QCryptographicHash::Md5).toHex());
QNetworkRequest request(url);
request.setRawHeader("User-Agent", m_ua);
request.setRawHeader("Host",url.host().toLatin1());
request.setRawHeader("Accept", "*/*");
m_getSessionReply = m_http->get(request);
}
void ScrobblerAuth::checkSession(const QString &session)
{
qDebug("ScrobblerAuth[%s]: checking session...", qPrintable(m_name));
m_session = session;
QMap <QString, QString> params;
params.insert("api_key", API_KEY);
params.insert("sk", session);
params.insert("method", "user.getInfo");
QUrl url(m_scrobblerUrl);
url.setPort(m_scrobblerUrl.startsWith("https") ? 443 : 80);
QUrl body("");
QByteArray data;
foreach (QString key, params.keys())
{
body.addQueryItem(key, params.value(key));
data.append(key.toUtf8() + params.value(key).toUtf8());
}
data.append(SECRET);
body.addQueryItem("api_sig", QCryptographicHash::hash(data, QCryptographicHash::Md5).toHex());
QByteArray bodyData = body.toEncoded().remove(0,1);
bodyData.replace("+", QUrl::toPercentEncoding("+"));
QNetworkRequest request(url);
request.setRawHeader("User-Agent", m_ua);
request.setRawHeader("Host", url.host().toLatin1());
request.setRawHeader("Accept", "*/*");
request.setRawHeader("Content-Type", "application/x-www-form-urlencoded");
request.setHeader(QNetworkRequest::ContentLengthHeader, bodyData.size());
m_checkSessionReply = m_http->post(request, bodyData);
}
QString ScrobblerAuth::session() const
{
return m_session;
}
void ScrobblerAuth::processResponse(QNetworkReply *reply)
{
if (reply->error() != QNetworkReply::NoError)
{
qWarning("ScrobblerAuth[%s]: http error: %s", qPrintable(m_name), qPrintable(reply->errorString()));
}
ScrobblerResponse response;
response.parse(reply);
QString error_code;
if(response.status != "ok" && !response.status.isEmpty())
{
if(!response.error.isEmpty())
{
qWarning("ScrobblerAuth[%s]: status=%s, %s-%s", qPrintable(m_name), qPrintable(response.status),
qPrintable(response.code), qPrintable(response.error));
error_code = response.code;
}
else
qWarning("ScrobblerAuth[%s]: invalid content", qPrintable(m_name));
}
if (reply == m_getTokenReply)
{
m_getTokenReply = 0;
if(response.status == "ok")
{
m_token = response.token;
qDebug("ScrobblerAuth[%s]: token: %s", qPrintable(m_name), qPrintable(m_token));
QDesktopServices::openUrl(m_authUrl + "?api_key=" API_KEY "&token="+m_token);
emit(tokenRequestFinished(NO_ERROR));
}
else if(error_code.isEmpty())
{
m_token.clear();
emit tokenRequestFinished(NETWORK_ERROR);
}
else if(error_code == "8" || error_code == "7" || error_code == "11" || error_code.isEmpty())
{
m_token.clear();
emit tokenRequestFinished(LASTFM_ERROR);
}
else
{
m_token.clear();
emit tokenRequestFinished(LASTFM_ERROR);
}
}
else if(reply == m_getSessionReply)
{
m_getSessionReply = 0;
m_session.clear();
if(response.status == "ok")
{
m_session = response.key;
qDebug("ScrobblerAuth[%s]: name: %s", qPrintable(m_name), qPrintable(response.name));
qDebug("ScrobblerAuth[%s]: key: %s", qPrintable(m_name), qPrintable(m_session));
qDebug("ScrobblerAuth[%s]: subscriber: %s", qPrintable(m_name), qPrintable(response.subscriber));
emit sessionRequestFinished(NO_ERROR);
}
else if(error_code == "4" || error_code == "15") //invalid token
{
m_token.clear();
emit sessionRequestFinished(LASTFM_ERROR);
}
else if(error_code == "11") //service offline
{
m_token.clear();
emit sessionRequestFinished(LASTFM_ERROR);
}
else if(error_code == "14") // unauthorized token
{
m_token.clear();
emit sessionRequestFinished(LASTFM_ERROR);
}
else if (error_code.isEmpty()) //network error
{
m_token.clear();
emit sessionRequestFinished(NETWORK_ERROR);
}
else
{
m_token.clear();
emit sessionRequestFinished(LASTFM_ERROR);
}
}
else if(reply == m_checkSessionReply)
{
m_checkSessionReply = 0;
if(response.status == "ok")
{
qDebug("ScrobblerAuth[%s]: session ok", qPrintable(m_name));
emit checkSessionFinished(NO_ERROR);
}
else if(error_code.isEmpty())
{
qWarning("ScrobblerAuth[%s]: network error", qPrintable(m_name));
emit checkSessionFinished(NETWORK_ERROR);
}
else
{
qWarning("ScrobblerAuth[%s]: received last.fm error (code=%s)",
qPrintable(m_name), qPrintable(error_code));
emit checkSessionFinished(LASTFM_ERROR);
}
}
reply->deleteLater();
}