193 lines
5.5 KiB
C++
193 lines
5.5 KiB
C++
/*
|
|
* Copyright 2011-2013 Nikhil Marathe <nsm.nikhil@gmail.com>
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
* of this software and associated documentation files (the "Software"), to
|
|
* deal in the Software without restriction, including without limitation the
|
|
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
|
|
* sell copies of the Software, and to permit persons to whom the Software is
|
|
* furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in
|
|
* all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
|
|
* IN THE SOFTWARE.
|
|
*/
|
|
|
|
#include "qhttpresponse.h"
|
|
|
|
#include <QDateTime>
|
|
#include <QLocale>
|
|
|
|
#include "qhttpserver.h"
|
|
#include "qhttpconnection.h"
|
|
|
|
QHttpResponse::QHttpResponse(QHttpConnection *connection)
|
|
// TODO: parent child relation
|
|
: QObject(0)
|
|
, m_connection(connection)
|
|
, m_headerWritten(false)
|
|
, m_sentConnectionHeader(false)
|
|
, m_sentContentLengthHeader(false)
|
|
, m_sentTransferEncodingHeader(false)
|
|
, m_sentDate(false)
|
|
, m_keepAlive(true)
|
|
, m_last(false)
|
|
, m_useChunkedEncoding(false)
|
|
, m_finished(false)
|
|
{
|
|
}
|
|
|
|
QHttpResponse::~QHttpResponse()
|
|
{
|
|
}
|
|
|
|
void QHttpResponse::setHeader(const QString &field, const QString &value)
|
|
{
|
|
if (!m_finished)
|
|
m_headers[field] = value;
|
|
else
|
|
qWarning() << "QHttpResponse::setHeader() Cannot set headers after response has finished.";
|
|
}
|
|
|
|
void QHttpResponse::writeHeader(const char *field, const QString &value)
|
|
{
|
|
if (!m_finished)
|
|
{
|
|
m_connection->write(field);
|
|
m_connection->write(": ");
|
|
m_connection->write(value.toUtf8());
|
|
m_connection->write("\r\n");
|
|
}
|
|
else
|
|
qWarning() << "QHttpResponse::writeHeader() Cannot write headers after response has finished.";
|
|
}
|
|
|
|
void QHttpResponse::writeHeaders()
|
|
{
|
|
if (m_finished)
|
|
return;
|
|
|
|
foreach(const QString &name, m_headers.keys())
|
|
{
|
|
QString value = m_headers[name];
|
|
if (name.compare("connection", Qt::CaseInsensitive) == 0)
|
|
{
|
|
m_sentConnectionHeader = true;
|
|
if (value.compare("close", Qt::CaseInsensitive) == 0)
|
|
m_last = true;
|
|
else
|
|
m_keepAlive = true;
|
|
}
|
|
else if (name.compare("transfer-encoding", Qt::CaseInsensitive) == 0)
|
|
{
|
|
m_sentTransferEncodingHeader = true;
|
|
if (value.compare("chunked", Qt::CaseInsensitive) == 0)
|
|
m_useChunkedEncoding = true;
|
|
}
|
|
else if (name.compare("content-length", Qt::CaseInsensitive) == 0)
|
|
m_sentContentLengthHeader = true;
|
|
else if (name.compare("date", Qt::CaseInsensitive) == 0)
|
|
m_sentDate = true;
|
|
|
|
/// @todo Expect case (??)
|
|
|
|
writeHeader(name.toLatin1(), value.toLatin1());
|
|
}
|
|
|
|
if (!m_sentConnectionHeader)
|
|
{
|
|
if (m_keepAlive && (m_sentContentLengthHeader || m_useChunkedEncoding))
|
|
writeHeader("Connection", "keep-alive");
|
|
else
|
|
{
|
|
m_last = true;
|
|
writeHeader("Connection", "close");
|
|
}
|
|
}
|
|
|
|
if (!m_sentContentLengthHeader && !m_sentTransferEncodingHeader)
|
|
{
|
|
if (m_useChunkedEncoding)
|
|
writeHeader("Transfer-Encoding", "chunked");
|
|
else
|
|
m_last = true;
|
|
}
|
|
|
|
// Sun, 06 Nov 1994 08:49:37 GMT - RFC 822. Use QLocale::c() so english is used for month and day.
|
|
if (!m_sentDate)
|
|
writeHeader("Date", QLocale::c().toString(QDateTime::currentDateTimeUtc(), "ddd, dd MMM yyyy hh:mm:ss") + " GMT");
|
|
}
|
|
|
|
void QHttpResponse::writeHead(int status)
|
|
{
|
|
if (m_finished)
|
|
{
|
|
qWarning() << "QHttpResponse::writeHead() Cannot write headers after response has finished.";
|
|
return;
|
|
}
|
|
|
|
if (m_headerWritten)
|
|
{
|
|
qWarning() << "QHttpResponse::writeHead() Already called once for this response.";
|
|
return;
|
|
}
|
|
|
|
m_connection->write(QString("HTTP/1.1 %1 %2\r\n").arg(status).arg(STATUS_CODES[status]).toLatin1());
|
|
writeHeaders();
|
|
m_connection->write("\r\n");
|
|
|
|
m_headerWritten = true;
|
|
}
|
|
|
|
void QHttpResponse::writeHead(StatusCode statusCode)
|
|
{
|
|
writeHead(static_cast<int>(statusCode));
|
|
}
|
|
|
|
void QHttpResponse::write(const QByteArray &data)
|
|
{
|
|
if (m_finished)
|
|
{
|
|
qWarning() << "QHttpResponse::write() Cannot write body after response has finished.";
|
|
return;
|
|
}
|
|
|
|
if (!m_headerWritten)
|
|
{
|
|
qWarning() << "QHttpResponse::write() You must call writeHead() before writing body data.";
|
|
return;
|
|
}
|
|
|
|
m_connection->write(data);
|
|
}
|
|
|
|
void QHttpResponse::end(const QByteArray &data)
|
|
{
|
|
if (m_finished)
|
|
{
|
|
qWarning() << "QHttpResponse::end() Cannot write end after response has finished.";
|
|
return;
|
|
}
|
|
|
|
if (data.size() > 0)
|
|
write(data);
|
|
m_finished = true;
|
|
|
|
emit done();
|
|
|
|
/// @todo End connection and delete ourselves. Is this a still valid note?
|
|
deleteLater();
|
|
}
|
|
|
|
void QHttpResponse::connectionClosed()
|
|
{
|
|
m_finished = true;
|
|
deleteLater();
|
|
}
|