qhttpresponse.cpp 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. /*
  2. * Copyright 2011-2014 Nikhil Marathe <nsm.nikhil@gmail.com>
  3. *
  4. * Permission is hereby granted, free of charge, to any person obtaining a copy
  5. * of this software and associated documentation files (the "Software"), to
  6. * deal in the Software without restriction, including without limitation the
  7. * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
  8. * sell copies of the Software, and to permit persons to whom the Software is
  9. * furnished to do so, subject to the following conditions:
  10. *
  11. * The above copyright notice and this permission notice shall be included in
  12. * all copies or substantial portions of the Software.
  13. *
  14. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  15. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  16. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  17. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  18. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
  19. * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
  20. * IN THE SOFTWARE.
  21. */
  22. #include "qhttpresponse.h"
  23. #include <QDateTime>
  24. #include <QLocale>
  25. #include "qhttpserver.h"
  26. #include "qhttpconnection.h"
  27. QHttpResponse::QHttpResponse(QHttpConnection *connection)
  28. // TODO: parent child relation
  29. : QObject(0),
  30. m_connection(connection),
  31. m_headerWritten(false),
  32. m_sentConnectionHeader(false),
  33. m_sentContentLengthHeader(false),
  34. m_sentTransferEncodingHeader(false),
  35. m_sentDate(false),
  36. m_keepAlive(true),
  37. m_last(false),
  38. m_useChunkedEncoding(false),
  39. m_finished(false)
  40. {
  41. connect(m_connection, SIGNAL(allBytesWritten()), this, SIGNAL(allBytesWritten()));
  42. }
  43. QHttpResponse::~QHttpResponse()
  44. {
  45. }
  46. void QHttpResponse::setHeader(const QString &field, const QString &value)
  47. {
  48. if (!m_finished)
  49. m_headers[field] = value;
  50. else
  51. qWarning() << "QHttpResponse::setHeader() Cannot set headers after response has finished.";
  52. }
  53. void QHttpResponse::writeHeader(const char *field, const QString &value)
  54. {
  55. if (!m_finished) {
  56. m_connection->write(field);
  57. m_connection->write(": ");
  58. m_connection->write(value.toUtf8());
  59. m_connection->write("\r\n");
  60. } else
  61. qWarning()
  62. << "QHttpResponse::writeHeader() Cannot write headers after response has finished.";
  63. }
  64. void QHttpResponse::writeHeaders()
  65. {
  66. if (m_finished)
  67. return;
  68. foreach(const QString & name, m_headers.keys()) {
  69. QString value = m_headers[name];
  70. if (name.compare("connection", Qt::CaseInsensitive) == 0) {
  71. m_sentConnectionHeader = true;
  72. if (value.compare("close", Qt::CaseInsensitive) == 0)
  73. m_last = true;
  74. else
  75. m_keepAlive = true;
  76. } else if (name.compare("transfer-encoding", Qt::CaseInsensitive) == 0) {
  77. m_sentTransferEncodingHeader = true;
  78. if (value.compare("chunked", Qt::CaseInsensitive) == 0)
  79. m_useChunkedEncoding = true;
  80. } else if (name.compare("content-length", Qt::CaseInsensitive) == 0)
  81. m_sentContentLengthHeader = true;
  82. else if (name.compare("date", Qt::CaseInsensitive) == 0)
  83. m_sentDate = true;
  84. /// @todo Expect case (??)
  85. writeHeader(name.toLatin1(), value.toLatin1());
  86. }
  87. if (!m_sentConnectionHeader) {
  88. if (m_keepAlive && (m_sentContentLengthHeader || m_useChunkedEncoding)) {
  89. writeHeader("Connection", "keep-alive");
  90. } else {
  91. m_last = true;
  92. writeHeader("Connection", "close");
  93. }
  94. }
  95. if (!m_sentContentLengthHeader && !m_sentTransferEncodingHeader) {
  96. if (m_useChunkedEncoding)
  97. writeHeader("Transfer-Encoding", "chunked");
  98. else
  99. m_last = true;
  100. }
  101. // Sun, 06 Nov 1994 08:49:37 GMT - RFC 822. Use QLocale::c() so english is used for month and
  102. // day.
  103. if (!m_sentDate)
  104. writeHeader("Date",
  105. QLocale::c().toString(QDateTime::currentDateTimeUtc(),
  106. "ddd, dd MMM yyyy hh:mm:ss") + " GMT");
  107. }
  108. void QHttpResponse::writeHead(int status)
  109. {
  110. if (m_finished) {
  111. qWarning()
  112. << "QHttpResponse::writeHead() Cannot write headers after response has finished.";
  113. return;
  114. }
  115. if (m_headerWritten) {
  116. qWarning() << "QHttpResponse::writeHead() Already called once for this response.";
  117. return;
  118. }
  119. m_connection->write(
  120. QString("HTTP/1.1 %1 %2\r\n").arg(status).arg(STATUS_CODES[status]).toLatin1());
  121. writeHeaders();
  122. m_connection->write("\r\n");
  123. m_headerWritten = true;
  124. }
  125. void QHttpResponse::writeHead(StatusCode statusCode)
  126. {
  127. writeHead(static_cast<int>(statusCode));
  128. }
  129. void QHttpResponse::write(const QByteArray &data)
  130. {
  131. if (m_finished) {
  132. qWarning() << "QHttpResponse::write() Cannot write body after response has finished.";
  133. return;
  134. }
  135. if (!m_headerWritten) {
  136. qWarning() << "QHttpResponse::write() You must call writeHead() before writing body data.";
  137. return;
  138. }
  139. m_connection->write(data);
  140. }
  141. void QHttpResponse::flush()
  142. {
  143. m_connection->flush();
  144. }
  145. void QHttpResponse::waitForBytesWritten()
  146. {
  147. m_connection->waitForBytesWritten();
  148. }
  149. void QHttpResponse::end(const QByteArray &data)
  150. {
  151. if (m_finished) {
  152. qWarning() << "QHttpResponse::end() Cannot write end after response has finished.";
  153. return;
  154. }
  155. if (data.size() > 0)
  156. write(data);
  157. m_finished = true;
  158. Q_EMIT done();
  159. /// @todo End connection and delete ourselves. Is this a still valid note?
  160. deleteLater();
  161. }
  162. void QHttpResponse::connectionClosed()
  163. {
  164. m_finished = true;
  165. Q_EMIT done();
  166. deleteLater();
  167. }