1
0
mirror of https://github.com/tomahawk-player/tomahawk.git synced 2025-08-07 06:36:55 +02:00
Files
tomahawk/thirdparty/qxt/qxtweb-standalone/qxtweb/qxtabstracthttpconnector.cpp
2011-02-21 15:25:43 +01:00

250 lines
8.2 KiB
C++

/****************************************************************************
**
** Copyright (C) Qxt Foundation. Some rights reserved.
**
** This file is part of the QxtWeb module of the Qxt library.
**
** This library is free software; you can redistribute it and/or modify it
** under the terms of the Common Public License, version 1.0, as published
** by IBM, and/or under the terms of the GNU Lesser General Public License,
** version 2.1, as published by the Free Software Foundation.
**
** This file is provided "AS IS", without WARRANTIES OR CONDITIONS OF ANY
** KIND, EITHER EXPRESS OR IMPLIED INCLUDING, WITHOUT LIMITATION, ANY
** WARRANTIES OR CONDITIONS OF TITLE, NON-INFRINGEMENT, MERCHANTABILITY OR
** FITNESS FOR A PARTICULAR PURPOSE.
**
** You should have received a copy of the CPL and the LGPL along with this
** file. See the LICENSE file and the cpl1.0.txt/lgpl-2.1.txt files
** included with the source distribution for more information.
** If you did not receive a copy of the licenses, contact the Qxt Foundation.
**
** <http://libqxt.org> <foundation@libqxt.org>
**
****************************************************************************/
/*!
\class QxtAbstractHttpConnector
\inmodule QxtWeb
\brief The QxtAbstractHttpConnector class is a base class for defining
HTTP-based protocols for use with QxtHttpSessionManager
QxtHttpSessionManager does the work of managing sessions and state for the
otherwise stateless HTTP protocol, but it relies on QxtAbstractHttpConnector
subclasses to implement the protocol used to communicate with the web server.
Subclasses are responsible for accepting new connections (by implementing
listen(const QHostAddress&, quint16) and invoking addConnection(QIODevice*)),
for informing the session manager when request headers are available (by
implementing canParseRequest(const QByteArray&)), for parsing the request
headers (by implementing parseRequest(QByteArray&)), and for writing response
headers (by implementing writeHeaders(QIODevice*, const QHttpResponseHeader&)).
\sa QxtHttpSessionManager
*/
#include "qxthttpsessionmanager.h"
#include "qxtwebcontent.h"
#include <QReadWriteLock>
#include <QHash>
#include <QIODevice>
#include <QByteArray>
#ifndef QXT_DOXYGEN_RUN
class QxtAbstractHttpConnectorPrivate : public QxtPrivate<QxtAbstractHttpConnector>
{
public:
QxtHttpSessionManager* manager;
QReadWriteLock bufferLock, requestLock;
QHash<QIODevice*, QByteArray> buffers; // connection->buffer
QHash<quint32, QIODevice*> requests; // requestID->connection
QHash<quint32, QSharedPointer<QIODevice> > dataSources; // iodevices providing result data
quint32 nextRequestID;
inline quint32 getNextRequestID(QIODevice* connection)
{
QWriteLocker locker(&requestLock);
do
{
nextRequestID++;
if (nextRequestID == 0xFFFFFFFF) nextRequestID = 1;
}
while (requests.contains(nextRequestID)); // yeah, right
requests[nextRequestID] = connection;
return nextRequestID;
}
inline void doneWithRequest(quint32 requestID)
{
QWriteLocker locker(&requestLock);
requests.remove(requestID);
dataSources.remove(requestID);
}
inline QIODevice* getRequestConnection(quint32 requestID)
{
QReadLocker locker(&requestLock);
return requests[requestID];
}
inline void setRequestDataSource( quint32 requestID, QSharedPointer<QIODevice>& dataSource )
{
QWriteLocker locker(&requestLock);
dataSources.insert( requestID, dataSource );
}
inline QSharedPointer<QIODevice>& getRequestDataSource(quint32 requestID)
{
QReadLocker locker(&requestLock);
return dataSources[requestID];
}
};
#endif
/*!
* Creates a QxtAbstractHttpConnector with the specified \a parent.
*
* Note that this is an abstract class and cannot be instantiated directly.
*/
QxtAbstractHttpConnector::QxtAbstractHttpConnector(QObject* parent) : QObject(parent)
{
QXT_INIT_PRIVATE(QxtAbstractHttpConnector);
qxt_d().nextRequestID = 0;
}
/*!
* \internal
*/
void QxtAbstractHttpConnector::setSessionManager(QxtHttpSessionManager* manager)
{
qxt_d().manager = manager;
}
/*!
* Returns the session manager into which the connector is installed.
*
* \sa QxtHttpSessionManager::setConnector
*/
QxtHttpSessionManager* QxtAbstractHttpConnector::sessionManager() const
{
return qxt_d().manager;
}
/*!
* \internal
* Returns the QIODevice associated with a \a requestID.
*
* The request ID is generated internally and used by the session manager.
*/
QIODevice* QxtAbstractHttpConnector::getRequestConnection(quint32 requestID)
{
return qxt_d().getRequestConnection(requestID);
}
void QxtAbstractHttpConnector::setRequestDataSource( quint32 requestID, QSharedPointer<QIODevice>& dataSource )
{
qxt_d().setRequestDataSource( requestID, dataSource );
}
QSharedPointer<QIODevice>& QxtAbstractHttpConnector::getRequestDataSource(quint32 requestID)
{
return qxt_d().getRequestDataSource( requestID );
}
void QxtAbstractHttpConnector::doneWithRequest(quint32 requestID)
{
qxt_d().doneWithRequest( requestID );
}
/*!
* Starts managing a new connection from \a device.
*
* This function should be invoked by a subclass to attach incoming connections
* to the session manager.
*/
void QxtAbstractHttpConnector::addConnection(QIODevice* device)
{
QWriteLocker locker(&qxt_d().bufferLock);
qxt_d().buffers[device] = QByteArray();
QObject::connect(device, SIGNAL(readyRead()), this, SLOT(incomingData()));
QObject::connect(device, SIGNAL(aboutToClose()), this, SLOT(disconnected()));
QObject::connect(device, SIGNAL(disconnected()), this, SLOT(disconnected()));
QObject::connect(device, SIGNAL(destroyed()), this, SLOT(disconnected()));
}
/*!
* \internal
*/
void QxtAbstractHttpConnector::incomingData(QIODevice* device)
{
if (!device)
{
device = qobject_cast<QIODevice*>(sender());
if (!device) return;
}
QReadLocker locker(&qxt_d().bufferLock);
QByteArray& buffer = qxt_d().buffers[device];
buffer.append(device->readAll());
if (!canParseRequest(buffer)) return;
QHttpRequestHeader header = parseRequest(buffer);
QxtWebContent* content = 0;
QByteArray start;
if (header.contentLength() > 0)
{
start = buffer.left(header.value("content-length").toInt());
buffer = buffer.mid(header.value("content-length").toInt());
content = new QxtWebContent(header.contentLength(), start, device);
}
else if (header.hasKey("connection") && header.value("connection").toLower() == "close")
{
start = buffer;
buffer.clear();
content = new QxtWebContent(header.contentLength(), start, device);
} // else no content
quint32 requestID = qxt_d().getNextRequestID(device);
sessionManager()->incomingRequest(requestID, header, content);
}
/*!
* \internal
*/
void QxtAbstractHttpConnector::disconnected()
{
QIODevice* device = qobject_cast<QIODevice*>(sender());
if (!device) return;
QWriteLocker locker(&qxt_d().bufferLock);
qxt_d().buffers.remove(device);
sessionManager()->disconnected(device);
}
/*!
* \fn virtual bool QxtAbstractHttpConnector::listen(const QHostAddress& interface, quint16 port)
* Invoked by the session manager to indicate that the connector should listen
* for incoming connections on the specified \a interface and \a port.
*
* If the interface is QHostAddress::Any, the server will listen on all network interfaces.
*
* Returns true on success, or false if the server could not begin listening.
*
* \sa addConnection(QIODevice*)
*/
/*!
* \fn virtual bool QxtAbstractHttpConnector::canParseRequest(const QByteArray& buffer)
* Returns true if a complete set of request headers can be extracted from the provided \a buffer.
*/
/*!
* \fn virtual QHttpRequestHeader QxtAbstractHttpConnector::parseRequest(QByteArray& buffer)
* Extracts a set of request headers from the provided \a buffer.
*
* Subclasses implementing this function must be sure to remove the parsed data from the buffer.
*/
/*!
* \fn virtual void QxtAbstractHttpConnector::writeHeaders(QIODevice* device, const QHttpResponseHeader& header)
* Writes a the response \a header to the specified \a device.
*/