NuriaProject Framework  0.1
The NuriaProject Framework
httpclient.hpp
1 /* Copyright (c) 2014-2015, The Nuria Project
2  * The NuriaProject Framework is free software: you can redistribute it and/or
3  * modify it under the terms of the GNU Lesser General Public License as
4  * published by the Free Software Foundation, either version 3 of the License,
5  * or (at your option) any later version.
6  *
7  * The NuriaProject Framework is distributed in the hope that it will be useful,
8  * but WITHOUT ANY WARRANTY; without even the implied warranty of
9  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10  * GNU Lesser General Public License for more details.
11  *
12  * You should have received a copy of the GNU Lesser General Public License
13  * along with The NuriaProject Framework.
14  * If not, see <http://www.gnu.org/licenses/>.
15  */
16 
17 #ifndef NURIA_HTTPCLIENT_HPP
18 #define NURIA_HTTPCLIENT_HPP
19 
20 #include "httppostbodyreader.hpp"
21 #include "network_global.hpp"
22 #include <QNetworkCookie>
23 #include <QHostAddress>
24 #include <QIODevice>
25 #include <QMetaType>
26 #include <QMap>
27 #include <QUrl>
28 
29 namespace Nuria {
30 
31 class HttpClientPrivate;
32 class HttpTransport;
33 class HttpFilter;
34 class HttpServer;
35 class HttpNode;
36 class SlotInfo;
37 
118 class NURIA_NETWORK_EXPORT HttpClient : public QIODevice {
119  Q_OBJECT
120  Q_ENUMS(HttpVerb HttpHeader)
121  Q_FLAGS(HttpVerbs)
122 public:
123 
127  enum HttpVersion {
128  HttpUnknown,
130  Http1_1
131  };
132 
137  enum HttpVerb {
142  InvalidVerb = 0,
143  GET = 1,
144  POST = 2,
145  HEAD = 4,
146  PUT = 8,
147  DELETE = 16,
148 
150  AllVerbs = GET | POST | HEAD | PUT | DELETE
151  };
152 
153  Q_DECLARE_FLAGS(HttpVerbs, HttpVerb)
154 
155 
160  enum HttpHeader {
161 
162  /* Request and response */
163  HeaderCacheControl = 0,
164  HeaderContentLength,
165  HeaderContentType,
166  HeaderConnection,
167  HeaderDate,
168 
169  /* Request only */
170  HeaderHost = 1000,
172  HeaderAccept,
173  HeaderAcceptCharset,
174  HeaderAcceptEncoding,
175  HeaderAcceptLanguage,
176  HeaderAuthorization,
177  HeaderCookie,
178  HeaderRange,
179  HeaderReferer,
180  HeaderDoNotTrack,
181  HeaderExpect,
182 
183  /* Response only */
184  HeaderContentEncoding = 2000,
185  HeaderContentLanguage,
186  HeaderContentDisposition,
187  HeaderContentRange,
188  HeaderLastModified,
189  HeaderRefresh,
190  HeaderSetCookie,
191  HeaderTransferEncoding,
192  HeaderLocation
193 
194  };
195 
198 
206  Streaming = 0,
207 
220 
227  ChunkedStreaming
228  };
229 
232 
234  ConnectionClose = 0,
235 
237  ConnectionKeepAlive
238  };
239 
242 
244  DeflateFilter = 0,
245 
247  GzipFilter = 1
248 
249  };
250 
254  enum class RedirectMode {
255 
260  Keep = 0,
261 
263  ForceSecure,
264 
266  ForceUnsecure
267  };
268 
270  typedef QMultiMap< QByteArray, QByteArray > HeaderMap;
271 
278  explicit HttpClient (HttpTransport *transport, HttpServer *server);
279 
281  virtual ~HttpClient ();
282 
284  HttpTransport *transport () const;
285 
290  static QByteArray httpStatusCodeName (int code);
291 
295  static QByteArray httpHeaderName (HttpHeader header);
296 
298  bool isHeaderReady () const;
299 
305  bool responseHeaderSent () const;
306 
308  HttpVerb verb () const;
309 
311  QUrl path () const;
312 
317  bool hasRequestHeader (const QByteArray &key) const;
318 
320  bool hasRequestHeader (HttpHeader header) const;
321 
329  QByteArray requestHeader (const QByteArray &key) const;
330 
332  QByteArray requestHeader (HttpHeader header) const;
333 
340  QList< QByteArray > requestHeaders (const QByteArray &key) const;
341 
343  QList< QByteArray > requestHeaders (HttpHeader header) const;
344 
346  const HeaderMap &requestHeaders () const;
347 
349  bool hasResponseHeader (const QByteArray &key) const;
350 
352  bool hasResponseHeader (HttpHeader header) const;
353 
357  const HeaderMap &responseHeaders () const;
358 
362  QList< QByteArray > responseHeaders (const QByteArray &key) const;
363 
365  QList< QByteArray > responseHeaders (HttpHeader header) const;
366 
380  bool setResponseHeader (const QByteArray &key, const QByteArray &value, bool append = false);
381 
383  bool setResponseHeader (HttpHeader header, const QByteArray &value, bool append = false);
384 
389  bool setResponseHeaders (const HeaderMap &headers);
390 
399  bool pipeToClient (QIODevice *device, qint64 maxlen = -1);
400 
413  bool pipeFromPostBody (QIODevice *device, bool takeOwnership = false);
414 
420  qint64 rangeStart () const;
421 
427  qint64 rangeEnd () const;
428 
433  qint64 contentLength () const;
434 
440  bool setRangeStart (qint64 pos);
441 
447  bool setRangeEnd (qint64 pos);
448 
454  bool setContentLength (qint64 length);
455 
461  virtual bool isSequential () const;
462 
467  qint64 postBodyLength () const;
468 
472  qint64 postBodyTransferred ();
473 
477  QHostAddress localAddress () const;
478 
482  quint16 localPort () const;
483 
487  QHostAddress peerAddress () const;
488 
492  quint16 peerPort () const;
493 
498  int responseCode () const;
499 
503  void setResponseCode (int code);
504 
509  bool isConnectionSecure () const;
510 
514  HttpServer *httpServer () const;
515 
517  typedef QMap< QByteArray, QNetworkCookie > Cookies;
518 
524  Cookies cookies ();
525 
532  QNetworkCookie cookie (const QByteArray &name);
533 
539  bool hasCookie (const QByteArray &name);
540 
549  void setCookie (const QByteArray &name, const QByteArray &value,
550  const QDateTime &expires, bool secure = false);
551 
566  void setCookie (const QByteArray &name, const QByteArray &value,
567  qint64 maxAge = 0, bool secure = false);
568 
576  void setCookie (const QNetworkCookie &cookie);
577 
586  void removeCookie (const QByteArray &name);
587 
592  bool keepConnectionOpen () const;
593 
604  void setKeepConnectionOpen (bool keepOpen);
605 
609  SlotInfo slotInfo () const;
610 
615  void setSlotInfo (const SlotInfo &info);
616 
624  bool hasReadablePostBody () const;
625 
633  HttpPostBodyReader *postBodyReader ();
634 
636  TransferMode transferMode () const;
637 
642  bool setTransferMode (TransferMode mode);
643 
645  ConnectionMode connectionMode () const;
646 
651  bool requestCompletelyReceived () const;
652 
654  bool requestHasPostBody () const;
655 
662  void addFilter (StandardFilter filter);
663 
665  void addFilter (HttpFilter *filter);
666 
668  void removeFilter (StandardFilter filter);
669 
671  void removeFilter (HttpFilter *filter);
672 
686  bool redirectClient (const QString &localPath, RedirectMode mode = RedirectMode::Keep,
687  int statusCode = 307);
688 
697  bool redirectClient (const QUrl &remoteUrl, int statusCode = 307);
698 
704  bool atEnd () const override;
705 
721  bool manualInit (HttpVerb verb, HttpVersion version, const QByteArray &path, const HeaderMap &headers);
722 
723  //
724  qint64 bytesAvailable () const override;
725  qint64 pos () const override;
726  qint64 size () const override;
727  bool seek (qint64 pos) override;
728  bool reset () override;
729 
730 signals:
731 
733  void disconnected ();
734 
736  void headerReady ();
737 
739  void postBodyComplete ();
740 
741 public slots:
742 
752  bool killConnection (int error, const QString &cause = QString());
753 
760  bool sendResponseHeader ();
761 
771  void close () override;
772 
778  void forceClose ();
779 
780 private slots:
781 
783  void clientDisconnected ();
784 
786  void pipeToClientReadyRead ();
787 
788 protected:
789 
791  virtual qint64 readData (char *data, qint64 maxlen);
792 
794  virtual qint64 writeData (const char *data, qint64 len);
795 
801  bool resolveUrl (const QUrl &url);
802 
806  bool bufferPostBody (QByteArray &data);
807 
808 private:
809  friend class HttpTransport;
810  friend class HttpServer;
811  friend class HttpNode;
812 
816  bool readRangeRequestHeader ();
817 
824  void readRequestCookies ();
825 
826  bool sendRedirectResponse (const QByteArray &location, const QByteArray &display, int code);
827  void updateRequestedUrl ();
828 
829  void bytesSent (qint64 bytes);
830  void processData (QByteArray &data);
831 
832  bool sendChunkedData (const QByteArray &data);
833  qint64 parseIntegerHeaderValue (const QByteArray &value);
834  bool readPostBodyContentLength ();
835  bool send100ContinueIfClientExpectsIt ();
836  bool readAllAvailableHeaderLines (QByteArray &data);
837  bool readFirstLine (const QByteArray &line);
838  bool readHeader (const QByteArray &line);
839  bool isReceivedHeaderHttp11Compliant ();
840  bool verifyPostRequestCompliance ();
841  bool verifyCompleteHeader ();
842  bool invokeRequestedPath ();
843  bool readConnectionHeader ();
844  bool closeConnectionIfNoLongerNeeded ();
845  bool postProcessRequestHeader ();
846  bool contentTypeIsMultipart (const QByteArray &value) const;
847  bool contentTypeIsUrlEncoded (const QByteArray &value) const;
848  HttpPostBodyReader *createHttpMultiPartReader (const QByteArray &header);
849  HttpPostBodyReader *createUrlEncodedPartReader (const QByteArray &header);
850  qint64 writeDataInternal (QByteArray data);
851  bool sendData (const QByteArray &data);
852  void closeInternal ();
853  QByteArray filterInit ();
854  QByteArray filterDeinit ();
855  bool filterData (QByteArray &data);
856  bool filterHeaders (HeaderMap &headers);
857  void addFilterNameToHeader (HeaderMap &headers, const QByteArray &name);
858  void initPath (QByteArray path);
859 
864  bool sendPipeChunkToClient ();
865 
866  //
867  HttpClientPrivate *d_ptr;
868 
869 };
870 
871 }
872 
873 Q_DECLARE_OPERATORS_FOR_FLAGS(Nuria::HttpClient::HttpVerbs)
874 Q_DECLARE_METATYPE(Nuria::HttpClient::HttpVersion)
875 Q_DECLARE_METATYPE(Nuria::HttpClient::HttpVerbs)
876 Q_DECLARE_METATYPE(Nuria::HttpClient::HttpVerb)
877 Q_DECLARE_METATYPE(Nuria::HttpClient*)
878 
879 #endif // NURIA_HTTPCLIENT_HPP
TransferMode
Definition: httpclient.hpp:197
Http 1.0.
Definition: httpclient.hpp:129
User-Agent, must be present in a HTTP/1.1 session.
Definition: httpclient.hpp:171
HttpVerb
Definition: httpclient.hpp:137
HttpVersion
Definition: httpclient.hpp:127
ConnectionMode
Definition: httpclient.hpp:231
Definition: abstractsessionmanager.hpp:24
QMultiMap< QByteArray, QByteArray > HeaderMap
Definition: httpclient.hpp:270
RedirectMode
Definition: httpclient.hpp:254
HttpHeader
Definition: httpclient.hpp:160
Abstract class for readers of the body of HTTP POST requests.
Definition: httppostbodyreader.hpp:40
Definition: httpnode.hpp:38
Definition: httpclient.hpp:219
StandardFilter
Definition: httpclient.hpp:241
Abstract data transport for HttpClient.
Definition: httptransport.hpp:53
Server for the HyperText Transfer Protocol.
Definition: httpserver.hpp:71
Virtual directory for HttpServer.
Definition: httpnode.hpp:176
A filter can modify a HttpClients outgoing stream.
Definition: httpfilter.hpp:36
QMap< QByteArray, QNetworkCookie > Cookies
Definition: httpclient.hpp:517
The HttpClient class represents a connection to a client.
Definition: httpclient.hpp:118