• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * nghttp2 - HTTP/2 C Library
3  *
4  * Copyright (c) 2015 Tatsuhiro Tsujikawa
5  *
6  * Permission is hereby granted, free of charge, to any person obtaining
7  * a copy of this software and associated documentation files (the
8  * "Software"), to deal in the Software without restriction, including
9  * without limitation the rights to use, copy, modify, merge, publish,
10  * distribute, sublicense, and/or sell copies of the Software, and to
11  * permit persons to whom the Software is furnished to do so, subject to
12  * the following conditions:
13  *
14  * The above copyright notice and this permission notice shall be
15  * included in all copies or substantial portions of the Software.
16  *
17  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
18  * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
19  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
20  * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
21  * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
22  * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
23  * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
24  */
25 #ifndef ASIO_HTTP2_CLIENT_H
26 #define ASIO_HTTP2_CLIENT_H
27 
28 #include <nghttp2/asio_http2.h>
29 
30 namespace nghttp2 {
31 
32 namespace asio_http2 {
33 
34 namespace client {
35 
36 class response_impl;
37 
38 class response {
39 public:
40   // Application must not call this directly.
41   response();
42   ~response();
43 
44   // Sets callback which is invoked when chunk of response body is
45   // received.
46   void on_data(data_cb cb) const;
47 
48   // Returns status code.
49   int status_code() const;
50 
51   // Returns content-length.  -1 if it is unknown.
52   int64_t content_length() const;
53 
54   // Returns the response header fields.  The pusedo header fields,
55   // which start with colon (:), are exluced from this list.
56   const header_map &header() const;
57 
58   // Application must not call this directly.
59   response_impl &impl() const;
60 
61 private:
62   std::unique_ptr<response_impl> impl_;
63 };
64 
65 class request;
66 
67 using response_cb = std::function<void(const response &)>;
68 using request_cb = std::function<void(const request &)>;
69 using connect_cb =
70     std::function<void(boost::asio::ip::tcp::resolver::iterator)>;
71 
72 class request_impl;
73 
74 class request {
75 public:
76   // Application must not call this directly.
77   request();
78   ~request();
79 
80   // Sets callback which is invoked when response header is received.
81   void on_response(response_cb cb) const;
82 
83   // Sets callback which is invoked when push request header is
84   // received.
85   void on_push(request_cb cb) const;
86 
87   // Sets callback which is invoked when this request and response are
88   // finished.  After the invocation of this callback, the application
89   // must not access request and response object.
90   void on_close(close_cb cb) const;
91 
92   // Write trailer part.  This must be called after setting both
93   // NGHTTP2_DATA_FLAG_EOF and NGHTTP2_DATA_FLAG_NO_END_STREAM set in
94   // *data_flag parameter in generator_cb passed to session::submit()
95   // function.
96   void write_trailer(header_map h) const;
97 
98   // Cancels this request and response with given error code.
99   void cancel(uint32_t error_code = NGHTTP2_INTERNAL_ERROR) const;
100 
101   // Resumes deferred uploading.
102   void resume() const;
103 
104   // Returns method (e.g., GET).
105   const std::string &method() const;
106 
107   // Returns request URI, split into components.
108   const uri_ref &uri() const;
109 
110   // Returns request header fields.  The pusedo header fields, which
111   // start with colon (:), are exluced from this list.
112   const header_map &header() const;
113 
114   // Application must not call this directly.
115   request_impl &impl() const;
116 
117 private:
118   std::unique_ptr<request_impl> impl_;
119 };
120 
121 // Wrapper around an nghttp2_priority_spec.
122 class priority_spec {
123 public:
124   // The default ctor is used only by sentinel values.
125   priority_spec() = default;
126 
127   // Create a priority spec with the given priority settings.
128   explicit priority_spec(const int32_t stream_id, const int32_t weight,
129                          const bool exclusive = false);
130 
131   // Return a pointer to a valid nghttp2 priority spec, or null.
132   const nghttp2_priority_spec *get() const;
133 
134   // Indicates whether or not this spec is valid (i.e. was constructed with
135   // values).
136   bool valid() const;
137 
138 private:
139   nghttp2_priority_spec spec_;
140   bool valid_ = false;
141 };
142 
143 class session_impl;
144 
145 class session {
146 public:
147   // Starts HTTP/2 session by connecting to |host| and |service|
148   // (e.g., "80") using clear text TCP connection with connect timeout
149   // 60 seconds.
150   session(boost::asio::io_service &io_service, const std::string &host,
151           const std::string &service);
152 
153   // Same as previous but with pegged local endpoint
154   session(boost::asio::io_service &io_service,
155           const boost::asio::ip::tcp::endpoint &local_endpoint,
156           const std::string &host, const std::string &service);
157 
158   // Starts HTTP/2 session by connecting to |host| and |service|
159   // (e.g., "80") using clear text TCP connection with given connect
160   // timeout.
161   session(boost::asio::io_service &io_service, const std::string &host,
162           const std::string &service,
163           const boost::posix_time::time_duration &connect_timeout);
164 
165   // Same as previous but with pegged local endpoint
166   session(boost::asio::io_service &io_service,
167           const boost::asio::ip::tcp::endpoint &local_endpoint,
168           const std::string &host, const std::string &service,
169           const boost::posix_time::time_duration &connect_timeout);
170 
171   // Starts HTTP/2 session by connecting to |host| and |service|
172   // (e.g., "443") using encrypted SSL/TLS connection with connect
173   // timeout 60 seconds.
174   session(boost::asio::io_service &io_service,
175           boost::asio::ssl::context &tls_context, const std::string &host,
176           const std::string &service);
177 
178   // Starts HTTP/2 session by connecting to |host| and |service|
179   // (e.g., "443") using encrypted SSL/TLS connection with given
180   // connect timeout.
181   session(boost::asio::io_service &io_service,
182           boost::asio::ssl::context &tls_context, const std::string &host,
183           const std::string &service,
184           const boost::posix_time::time_duration &connect_timeout);
185 
186   ~session();
187 
188   session(session &&other) noexcept;
189   session &operator=(session &&other) noexcept;
190 
191   // Sets callback which is invoked after connection is established.
192   void on_connect(connect_cb cb) const;
193 
194   // Sets callback which is invoked there is connection level error
195   // and session is terminated.
196   void on_error(error_cb cb) const;
197 
198   // Sets read timeout, which defaults to 60 seconds.
199   void read_timeout(const boost::posix_time::time_duration &t);
200 
201   // Shutdowns connection.
202   void shutdown() const;
203 
204   // Returns underlying io_service object.
205   boost::asio::io_service &io_service() const;
206 
207   // Submits request to server using |method| (e.g., "GET"), |uri|
208   // (e.g., "http://localhost/") and optionally additional header
209   // fields.  This function returns pointer to request object if it
210   // succeeds, or nullptr and |ec| contains error message.
211   const request *submit(boost::system::error_code &ec,
212                         const std::string &method, const std::string &uri,
213                         header_map h = header_map{},
214                         priority_spec prio = priority_spec()) const;
215 
216   // Submits request to server using |method| (e.g., "GET"), |uri|
217   // (e.g., "http://localhost/") and optionally additional header
218   // fields.  The |data| is request body.  This function returns
219   // pointer to request object if it succeeds, or nullptr and |ec|
220   // contains error message.
221   const request *submit(boost::system::error_code &ec,
222                         const std::string &method, const std::string &uri,
223                         std::string data, header_map h = header_map{},
224                         priority_spec prio = priority_spec()) const;
225 
226   // Submits request to server using |method| (e.g., "GET"), |uri|
227   // (e.g., "http://localhost/") and optionally additional header
228   // fields.  The |cb| is used to generate request body.  This
229   // function returns pointer to request object if it succeeds, or
230   // nullptr and |ec| contains error message.
231   const request *submit(boost::system::error_code &ec,
232                         const std::string &method, const std::string &uri,
233                         generator_cb cb, header_map h = header_map{},
234                         priority_spec prio = priority_spec()) const;
235 
236 private:
237   std::shared_ptr<session_impl> impl_;
238 };
239 
240 // configure |tls_ctx| for client use.  Currently, we just set NPN
241 // callback for HTTP/2.
242 boost::system::error_code
243 configure_tls_context(boost::system::error_code &ec,
244                       boost::asio::ssl::context &tls_ctx);
245 
246 } // namespace client
247 
248 } // namespace asio_http2
249 
250 } // namespace nghttp2
251 
252 #endif // ASIO_HTTP2_CLIENT_H
253