Files
cantata/http/httpserver.cpp
2013-10-15 17:14:51 +00:00

259 lines
6.4 KiB
C++

/*
* Cantata
*
* Copyright (c) 2011-2013 Craig Drummond <craig.p.drummond@gmail.com>
*
* ----
*
* 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; see the file COPYING. If not, write to
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#include "httpserver.h"
#include "httpsocket.h"
#include "tags.h"
#include "settings.h"
#include "thread.h"
#include <QUrl>
#if QT_VERSION >= 0x050000
#include <QUrlQuery>
#endif
#ifdef ENABLE_KDE_SUPPORT
#include <KDE/KGlobal>
K_GLOBAL_STATIC(HttpServer, instance)
#endif
#include <QDebug>
static bool debugIsEnabled=false;
#define DBUG if (debugIsEnabled) qWarning() << "HttpServer" << __FUNCTION__
#ifdef Q_OS_WIN
static inline QString fixWindowsPath(const QString &f)
{
return f.length()>3 && f.startsWith(QLatin1Char('/')) && QLatin1Char(':')==f.at(2) ? f.mid(1) : f;
}
#endif
void HttpServer::enableDebug()
{
debugIsEnabled=true;
}
bool HttpServer::debugEnabled()
{
return debugIsEnabled;
}
HttpServer * HttpServer::self()
{
#ifdef ENABLE_KDE_SUPPORT
return instance;
#else
static HttpServer *instance=0;
if(!instance) {
instance=new HttpServer;
}
return instance;
#endif
}
void HttpServer::stop()
{
if (socket) {
socket->terminate();
socket=0;
}
if (thread) {
thread->stop();
thread=0;
}
}
bool HttpServer::readConfig()
{
QString iface=Settings::self()->httpInterface();
if (socket && socket->isListening() && iface==socket->configuredInterface()) {
return true;
}
if (socket) {
socket->terminate();
socket=0;
}
if (thread) {
thread->stop();
thread=0;
}
quint16 prevPort=Settings::self()->httpAllocatedPort();
thread=new Thread("HttpServer");
socket=new HttpSocket(iface, prevPort);
if (socket->serverPort()!=prevPort) {
Settings::self()->saveHttpAllocatedPort(socket->serverPort());
}
socket->moveToThread(thread);
thread->start();
return socket->isListening();
}
bool HttpServer::isAlive() const
{
return socket && socket->isListening();
}
QString HttpServer::address() const
{
return socket ? QLatin1String("http://")+socket->address()+QChar(':')+QString::number(socket->serverPort())
: QLatin1String("http://127.0.0.1:*");
}
bool HttpServer::isOurs(const QString &url) const
{
return isAlive() ? url.startsWith(address()+"/") : false;
}
QByteArray HttpServer::encodeUrl(const Song &s) const
{
DBUG << "song" << s.file << isAlive();
if (!isAlive()) {
return QByteArray();
}
#if QT_VERSION < 0x050000
QUrl url;
QUrl &query=url;
#else
QUrl url;
QUrlQuery query;
#endif
url.setScheme("http");
url.setHost(socket->address());
url.setPort(socket->serverPort());
url.setPath(s.file);
if (!s.album.isEmpty()) {
query.addQueryItem("album", s.album);
}
if (!s.artist.isEmpty()) {
query.addQueryItem("artist", s.artist);
}
if (!s.albumartist.isEmpty()) {
query.addQueryItem("albumartist", s.albumartist);
}
if (!s.title.isEmpty()) {
query.addQueryItem("title", s.title);
}
if (!s.genre.isEmpty()) {
query.addQueryItem("genre", s.genre);
}
if (s.disc) {
query.addQueryItem("disc", QString::number(s.disc));
}
if (s.year) {
query.addQueryItem("year", QString::number(s.year));
}
if (s.time) {
query.addQueryItem("time", QString::number(s.time));
}
if (s.track) {
query.addQueryItem("track", QString::number(s.track));
}
query.addQueryItem("id", QString::number(s.id));
query.addQueryItem("cantata", "song");
#if QT_VERSION >= 0x050000
url.setQuery(query);
#endif
DBUG << "encoded as" << url.toString();
return url.toEncoded();
}
QByteArray HttpServer::encodeUrl(const QString &file) const
{
#ifdef Q_OS_WIN
QString f=fixWindowsPath(file);
DBUG << "file" << f;
Song s=Tags::read(f);
s.file=f;
#else
DBUG << "file" << file;
Song s=Tags::read(file);
s.file=file;
#endif
return encodeUrl(s);
}
Song HttpServer::decodeUrl(const QString &url) const
{
return decodeUrl(QUrl(url));
}
Song HttpServer::decodeUrl(const QUrl &url) const
{
Song s;
#if QT_VERSION < 0x050000
const QUrl &q=url;
#else
QUrlQuery q(url);
#endif
if (q.hasQueryItem("cantata") && q.queryItemValue("cantata")=="song") {
if (q.hasQueryItem("album")) {
s.album=q.queryItemValue("album");
}
if (q.hasQueryItem("artist")) {
s.artist=q.queryItemValue("artist");
}
if (q.hasQueryItem("albumartist")) {
s.albumartist=q.queryItemValue("albumartist");
}
if (q.hasQueryItem("title")) {
s.title=q.queryItemValue("title");
}
if (q.hasQueryItem("genre")) {
s.genre=q.queryItemValue("genre");
}
if (q.hasQueryItem("disc")) {
s.disc=q.queryItemValue("disc").toInt();
}
if (q.hasQueryItem("year")) {
s.year=q.queryItemValue("year").toInt();
}
if (q.hasQueryItem("time")) {
s.time=q.queryItemValue("time").toInt();
}
if (q.hasQueryItem("track")) {
s.track=q.queryItemValue("track").toInt();
}
if (q.hasQueryItem("id")) {
s.id=q.queryItemValue("id").toInt();
}
s.file=url.path();
s.type=Song::CantataStream;
#ifdef Q_OS_WIN
s.file=fixWindowsPath(s.file);
#endif
#if defined CDDB_FOUND || defined MUSICBRAINZ5_FOUND
if (s.file.startsWith(QChar('/')+Song::constCddaProtocol)) {
s.file=s.file.mid(1);
s.type=Song::Cdda;
}
#endif
DBUG << s.file << s.albumArtist() << s.album << s.title;
}
return s;
}