mirror of
https://github.com/tomahawk-player/tomahawk.git
synced 2025-08-05 21:57:41 +02:00
246 lines
8.2 KiB
C++
246 lines
8.2 KiB
C++
|
|
/****************************************************************************
|
|
** Copyright (c) 2006 - 2011, the LibQxt project.
|
|
** See the Qxt AUTHORS file for a list of authors and copyright holders.
|
|
** All rights reserved.
|
|
**
|
|
** Redistribution and use in source and binary forms, with or without
|
|
** modification, are permitted provided that the following conditions are met:
|
|
** * Redistributions of source code must retain the above copyright
|
|
** notice, this list of conditions and the following disclaimer.
|
|
** * Redistributions in binary form must reproduce the above copyright
|
|
** notice, this list of conditions and the following disclaimer in the
|
|
** documentation and/or other materials provided with the distribution.
|
|
** * Neither the name of the LibQxt project nor the
|
|
** names of its contributors may be used to endorse or promote products
|
|
** derived from this software without specific prior written permission.
|
|
**
|
|
** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
** ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
** WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
** DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
|
|
** DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
** (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
** LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
** ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
** SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
**
|
|
** <http://libqxt.org> <foundation@libqxt.org>
|
|
*****************************************************************************/
|
|
|
|
/*!
|
|
\class QxtWebServiceDirectory
|
|
|
|
\inmodule QxtWeb
|
|
|
|
\brief The QxtWebServiceDirectory class provides Path-based web service dispatcher
|
|
|
|
QxtWebServiceDirectory allows multiple services to be associated with a single
|
|
session. Selection between services is determined by the first path component
|
|
in the URL. For example, the URL "/site/request?param=true" would relay the
|
|
URL "/request?param=true" to the service named "site".
|
|
|
|
This class can be used recursively to declare a hierarchy of services. For
|
|
example:
|
|
\code
|
|
QxtWebServiceDirectory* top = new QxtWebServiceDirectory(sm, sm);
|
|
QxtWebServiceDirectory* service1 = new QxtWebServiceDirectory(sm, top);
|
|
QxtWebServiceDirectory* service2 = new QxtWebServiceDirectory(sm, top);
|
|
QxtWebServiceDirectory* service1a = new QxtWebServiceDirectory(sm, service1);
|
|
QxtWebServiceDirectory* service1b = new QxtWebServiceDirectory(sm, service1);
|
|
top->addService("1", service1);
|
|
top->addService("2", service2);
|
|
service1->addService("a", service1a);
|
|
service1->addService("b", service1b);
|
|
\endcode
|
|
This accepts the URLs "/1/a/", "/1/b/", and "/2/".
|
|
*/
|
|
|
|
#include "qxtwebservicedirectory.h"
|
|
#include "qxtwebservicedirectory_p.h"
|
|
#include "qxtwebevent.h"
|
|
#include <QUrl>
|
|
#include <QtDebug>
|
|
|
|
#ifndef QXT_DOXYGEN_RUN
|
|
QxtWebServiceDirectoryPrivate::QxtWebServiceDirectoryPrivate() : QObject(0)
|
|
{
|
|
// initializers only
|
|
}
|
|
|
|
void QxtWebServiceDirectoryPrivate::serviceDestroyed()
|
|
{
|
|
QxtAbstractWebService* service = qobject_cast<QxtAbstractWebService*>(sender());
|
|
if (!service) return; // this shouldn't happen
|
|
QString path;
|
|
while (!(path = services.key(service)).isNull())
|
|
{
|
|
services.remove(path);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/*!
|
|
* Constructs a QxtWebServiceDirectory object with the specified session manager \a sm and \a parent.
|
|
*
|
|
* Often, the session manager will also be the parent, but this is not a requirement.
|
|
*/
|
|
QxtWebServiceDirectory::QxtWebServiceDirectory(QxtAbstractWebSessionManager* sm, QObject* parent) : QxtAbstractWebService(sm, parent)
|
|
{
|
|
QXT_INIT_PRIVATE(QxtWebServiceDirectory);
|
|
}
|
|
|
|
/*!
|
|
* Adds a \a service to the directory at the given \a path.
|
|
* \sa removeService(), service()
|
|
*/
|
|
void QxtWebServiceDirectory::addService(const QString& path, QxtAbstractWebService* service)
|
|
{
|
|
if (qxt_d().services.contains(path))
|
|
{
|
|
qWarning() << "QxtWebServiceDirectory::addService:" << path << "already registered";
|
|
}
|
|
|
|
qxt_d().services[path] = service;
|
|
if (qxt_d().defaultRedirect.isEmpty())
|
|
setDefaultRedirect(path);
|
|
connect(service, SIGNAL(destroyed()), &qxt_d(), SLOT(serviceDestroyed()));
|
|
}
|
|
|
|
/*!
|
|
* Removes the service at the given \a path.
|
|
*
|
|
* Note that the service object is not destroyed.
|
|
*/
|
|
void QxtWebServiceDirectory::removeService(const QString& path)
|
|
{
|
|
if (!qxt_d().services.contains(path))
|
|
{
|
|
qWarning() << "QxtWebServiceDirectory::removeService:" << path << "not registered";
|
|
}
|
|
else
|
|
{
|
|
qxt_d().services.remove(path);
|
|
}
|
|
}
|
|
|
|
/*!
|
|
* Returns the service at the given \a path.
|
|
*/
|
|
QxtAbstractWebService* QxtWebServiceDirectory::service(const QString& path) const
|
|
{
|
|
if (!qxt_d().services.contains(path))
|
|
return 0;
|
|
return qxt_d().services[path];
|
|
}
|
|
|
|
/*!
|
|
* \internal
|
|
* Returns the first path segment from the URL in the \a event object.
|
|
* (i.e. "a" from "/a/b/c") This also removes the path segment from the
|
|
* event object. (in the previous example, the event's URL is now "/b/c")
|
|
*/
|
|
static QString extractPathLevel(QxtWebRequestEvent* event)
|
|
{
|
|
QString path = event->url.path();
|
|
int pos = path.indexOf("/", 1); // the path always starts with /
|
|
if (pos == -1)
|
|
event->url.setPath(""); // cue to redirect to /service/
|
|
else
|
|
event->url.setPath(path.mid(pos));
|
|
return path.mid(1, pos - 1);
|
|
}
|
|
|
|
/*!
|
|
* \reimp
|
|
*/
|
|
void QxtWebServiceDirectory::pageRequestedEvent(QxtWebRequestEvent* event)
|
|
{
|
|
QString path = extractPathLevel(event);
|
|
if (path.isEmpty())
|
|
{
|
|
indexRequested(event);
|
|
}
|
|
else if (!qxt_d().services.contains(path))
|
|
{
|
|
unknownServiceRequested(event, path);
|
|
}
|
|
else if (event->url.path().isEmpty())
|
|
{
|
|
postEvent(new QxtWebRedirectEvent(event->sessionID, event->requestID, path + '/', 307));
|
|
}
|
|
else
|
|
{
|
|
qxt_d().services[path]->pageRequestedEvent(event);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* \reimp unimplemented
|
|
*/
|
|
/*
|
|
void QxtWebServiceDirectory::functionInvokedEvent(QxtWebRequestEvent* event) {
|
|
QString path = extractPathLevel(event);
|
|
if(path == "") {
|
|
indexRequested(event);
|
|
} else if(!qxt_d().services.contains(path)) {
|
|
unknownServiceRequested(event, path);
|
|
} else {
|
|
qxt_d().services[path]->functionInvokedEvent(event);
|
|
}
|
|
}
|
|
*/
|
|
|
|
/*!
|
|
* This \a event handler is called whenever the URL requests a service with \a name that has
|
|
* not been added to the directory.
|
|
*
|
|
* The default implementation returns a 404 "Service not known" error.
|
|
* Subclasses may reimplement this event handler to customize this behavior.
|
|
*/
|
|
void QxtWebServiceDirectory::unknownServiceRequested(QxtWebRequestEvent* event, const QString& name)
|
|
{
|
|
postEvent(new QxtWebErrorEvent(event->sessionID, event->requestID, 404, ("Service "" + QString(name).replace('<', "<") + "" not known").toUtf8()));
|
|
}
|
|
|
|
/*!
|
|
* This \a event handler is called whenever the URL does not contain a path, that
|
|
* is, the URL is "/" or empty.
|
|
*
|
|
* The default implementation redirects to the service specified by
|
|
* setDefaultRedirect(), or invokes unknownServiceRequested() if no default
|
|
* redirect has been set.
|
|
*/
|
|
void QxtWebServiceDirectory::indexRequested(QxtWebRequestEvent* event)
|
|
{
|
|
if (defaultRedirect().isEmpty())
|
|
{
|
|
unknownServiceRequested(event, "/");
|
|
}
|
|
else
|
|
{
|
|
postEvent(new QxtWebRedirectEvent(event->sessionID, event->requestID, defaultRedirect() + '/', 307));
|
|
}
|
|
}
|
|
|
|
/*!
|
|
* Returns the path that will be used by default by the indexRequested event.
|
|
* \sa indexRequested(), setDefaultRedirect()
|
|
*/
|
|
QString QxtWebServiceDirectory::defaultRedirect() const
|
|
{
|
|
return qxt_d().defaultRedirect;
|
|
}
|
|
|
|
/*!
|
|
* Sets the \a path that will be used by default by the indexRequested event.
|
|
* \sa indexRequested(), defaultRedirect()
|
|
*/
|
|
void QxtWebServiceDirectory::setDefaultRedirect(const QString& path)
|
|
{
|
|
if (!qxt_d().services.contains(path))
|
|
qWarning() << "QxtWebServiceDirectory::setDefaultRedirect:" << path << "not registered";
|
|
qxt_d().defaultRedirect = path;
|
|
}
|