FETPAPI 0.3.0.0
This project provides C++ classes which facilitate the developement of ETP1.2 clients and servers.
All Classes Namespaces Functions Variables Pages
HttpClientSession.h
1/*-----------------------------------------------------------------------
2Licensed to the Apache Software Foundation (ASF) under one
3or more contributor license agreements. See the NOTICE file
4distributed with this work for additional information
5regarding copyright ownership. The ASF licenses this file
6to you under the Apache License, Version 2.0 (the
7"License"; you may not use this file except in compliance
8with the License. You may obtain a copy of the License at
9
10http://www.apache.org/licenses/LICENSE-2.0
11
12Unless required by applicable law or agreed to in writing,
13software distributed under the License is distributed on an
14"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15KIND, either express or implied. See the License for the
16specific language governing permissions and limitations
17under the License.
18-----------------------------------------------------------------------*/
19#pragma once
20
21#include <iostream>
22
23#include <boost/beast/core.hpp>
24#include <boost/beast/http.hpp>
25#include <boost/beast/version.hpp>
26#include <boost/asio/connect.hpp>
27#include <boost/asio/ip/tcp.hpp>
28
29using tcp = boost::asio::ip::tcp; // from <boost/asio/ip/tcp.hpp>
30namespace http = boost::beast::http; // from <boost/beast/http.hpp>
31
32namespace ETP_NS
33{
34 // Performs an HTTP GET and stores the response
35 class HttpClientSession : public std::enable_shared_from_this<HttpClientSession>
36 {
37 tcp::resolver resolver_;
38 tcp::socket socket_;
39 boost::beast::flat_buffer buffer_; // (Must persist between reads)
40 http::request<http::empty_body> req_;
41 http::response<http::string_body> res_;
42
43 public:
44 // Resolver and socket require an io_context
45 explicit
46 HttpClientSession(boost::asio::io_context& ioc)
47 : resolver_(ioc)
48 , socket_(ioc)
49 {
50 }
51
52 // Start the asynchronous operation
53 void
54 run(
55 const std::string& etpServerHost,
56 uint16_t etpServerPort,
57 const std::string& etpServerTarget,
58 uint32_t version,
59 const std::string& etpServerAuthorization = "",
60 const std::string& proxyHost = "",
61 uint16_t proxyPort = 80,
62 const std::string& proxyAuthorization = "")
63 {
64 // Set up an HTTP GET request message
65 req_.version(version);
66 req_.method(http::verb::get);
67 req_.target(etpServerTarget);
68 req_.set(http::field::host, etpServerHost + ':' + std::to_string(etpServerPort));
69 req_.set(http::field::user_agent, BOOST_BEAST_VERSION_STRING);
70 if (!etpServerAuthorization.empty()) {
71 req_.set(http::field::authorization, etpServerAuthorization);
72 }
73 if (!proxyAuthorization.empty()) {
74 req_.set(http::field::proxy_authorization, proxyAuthorization);
75 }
76
77 // Look up the domain name
78 resolver_.async_resolve(
79 proxyHost.empty() ? etpServerHost : proxyHost,
80 std::to_string(proxyHost.empty() ? etpServerPort : proxyPort),
81 std::bind(
82 &HttpClientSession::on_resolve,
83 shared_from_this(),
84 std::placeholders::_1,
85 std::placeholders::_2));
86 }
87
88 void
89 on_resolve(
90 boost::system::error_code ec,
91 tcp::resolver::results_type results)
92 {
93 if (ec) {
94 std::cerr << "resolve : " << ec.message() << std::endl;
95 return;
96 }
97
98 // Make the connection on the IP address we get from a lookup
99 boost::asio::async_connect(
100 socket_,
101 results.begin(),
102 results.end(),
103 std::bind(
104 &HttpClientSession::on_connect,
105 shared_from_this(),
106 std::placeholders::_1));
107 }
108
109 void
110 on_connect(boost::system::error_code ec)
111 {
112 if (ec) {
113 std::cerr << "connect : " << ec.message() << std::endl;
114 return;
115 }
116
117 // Send the HTTP request to the remote host
118 http::async_write(socket_, req_,
119 std::bind(
120 &HttpClientSession::on_write,
121 shared_from_this(),
122 std::placeholders::_1,
123 std::placeholders::_2));
124 }
125
126 void
127 on_write(
128 boost::system::error_code ec,
129 std::size_t bytes_transferred)
130 {
131 boost::ignore_unused(bytes_transferred);
132
133 if (ec) {
134 std::cerr << "write : " << ec.message() << std::endl;
135 return;
136 }
137
138 // Receive the HTTP response
139 http::async_read(socket_, buffer_, res_,
140 std::bind(
141 &HttpClientSession::on_read,
142 shared_from_this(),
143 std::placeholders::_1,
144 std::placeholders::_2));
145 }
146
147 void
148 on_read(
149 boost::system::error_code ec,
150 std::size_t bytes_transferred)
151 {
152 boost::ignore_unused(bytes_transferred);
153
154 if (ec) {
155 std::cerr << "read : " << ec.message() << std::endl;
156 return;
157 }
158
159 // Gracefully close the socket
160 socket_.shutdown(tcp::socket::shutdown_both, ec);
161
162 // not_connected happens sometimes so don't bother reporting it.
163 if (ec && ec != boost::system::errc::not_connected) {
164 std::cerr << "shutdown : " << ec.message() << std::endl;
165 return;
166 }
167
168 // If we get here then the connection is closed gracefully
169 }
170
171 const http::response<http::string_body>& getResponse() const {
172 return res_;
173 }
174 };
175}
Definition HttpClientSession.h:36