/* * Copyright 2011-2014 Nikhil Marathe * * 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 #include #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) { connect(m_connection, SIGNAL(allBytesWritten()), this, SIGNAL(allBytesWritten())); } 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(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::flush() { m_connection->flush(); } void QHttpResponse::waitForBytesWritten() { m_connection->waitForBytesWritten(); } 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; Q_EMIT done(); /// @todo End connection and delete ourselves. Is this a still valid note? deleteLater(); } void QHttpResponse::connectionClosed() { m_finished = true; Q_EMIT done(); deleteLater(); }