blob: 7fd2041904a2545dc310403f48afdbcca48171af [file] [log] [blame]
AppaRao Pulibd030d02020-03-20 03:34:29 +05301/*
2// Copyright (c) 2020 Intel Corporation
3//
4// Licensed under the Apache License, Version 2.0 (the "License");
5// you may not use this file except in compliance with the License.
6// You may obtain a copy of the License at
7//
8// http://www.apache.org/licenses/LICENSE-2.0
9//
10// Unless required by applicable law or agreed to in writing, software
11// distributed under the License is distributed on an "AS IS" BASIS,
12// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13// See the License for the specific language governing permissions and
14// limitations under the License.
15*/
16#pragma once
Sunitha Harish29a82b02021-02-18 15:54:16 +053017#include <boost/asio/ip/address.hpp>
18#include <boost/asio/ip/basic_endpoint.hpp>
Ed Tanousd43cd0c2020-09-30 20:46:53 -070019#include <boost/asio/steady_timer.hpp>
20#include <boost/beast/core/flat_buffer.hpp>
21#include <boost/beast/core/tcp_stream.hpp>
22#include <boost/beast/http/message.hpp>
AppaRao Pulibd030d02020-03-20 03:34:29 +053023#include <boost/beast/version.hpp>
Sunitha Harish29a82b02021-02-18 15:54:16 +053024#include <include/async_resolve.hpp>
Gunnar Mills1214b7e2020-06-04 10:11:30 -050025
AppaRao Pulibd030d02020-03-20 03:34:29 +053026#include <cstdlib>
27#include <functional>
28#include <iostream>
29#include <memory>
AppaRao Puli2a5689a2020-04-29 15:24:31 +053030#include <queue>
AppaRao Pulibd030d02020-03-20 03:34:29 +053031#include <string>
32
33namespace crow
34{
35
AppaRao Puli2a5689a2020-04-29 15:24:31 +053036static constexpr uint8_t maxRequestQueueSize = 50;
Sunitha Harish7de9f812021-08-24 02:50:30 -050037static constexpr unsigned int httpReadBodyLimit = 8192;
AppaRao Puli2a5689a2020-04-29 15:24:31 +053038
AppaRao Pulibd030d02020-03-20 03:34:29 +053039enum class ConnState
40{
AppaRao Puli2a5689a2020-04-29 15:24:31 +053041 initialized,
Sunitha Harish29a82b02021-02-18 15:54:16 +053042 resolveInProgress,
43 resolveFailed,
AppaRao Puli2a5689a2020-04-29 15:24:31 +053044 connectInProgress,
45 connectFailed,
AppaRao Pulibd030d02020-03-20 03:34:29 +053046 connected,
AppaRao Puli2a5689a2020-04-29 15:24:31 +053047 sendInProgress,
48 sendFailed,
Sunitha Harish6eaa1d22021-02-19 13:38:31 +053049 recvInProgress,
AppaRao Puli2a5689a2020-04-29 15:24:31 +053050 recvFailed,
51 idle,
Sunitha Harish6eaa1d22021-02-19 13:38:31 +053052 closeInProgress,
Ayushi Smritife44eb02020-05-15 15:24:45 +053053 closed,
Sunitha Harish6eaa1d22021-02-19 13:38:31 +053054 suspended,
55 terminated,
56 abortConnection,
57 retry
AppaRao Pulibd030d02020-03-20 03:34:29 +053058};
59
60class HttpClient : public std::enable_shared_from_this<HttpClient>
61{
62 private:
Sunitha Harish29a82b02021-02-18 15:54:16 +053063 crow::async_resolve::Resolver resolver;
AppaRao Pulibd030d02020-03-20 03:34:29 +053064 boost::beast::tcp_stream conn;
Ayushi Smritife44eb02020-05-15 15:24:45 +053065 boost::asio::steady_timer timer;
Sunitha Harish7de9f812021-08-24 02:50:30 -050066 boost::beast::flat_static_buffer<httpReadBodyLimit> buffer;
AppaRao Pulibd030d02020-03-20 03:34:29 +053067 boost::beast::http::request<boost::beast::http::string_body> req;
Sunitha Harish6eaa1d22021-02-19 13:38:31 +053068 std::optional<
69 boost::beast::http::response_parser<boost::beast::http::string_body>>
70 parser;
Krzysztof Grobelny116c1842021-12-06 11:56:37 +010071 boost::circular_buffer_space_optimized<std::string> requestDataQueue{
72 maxRequestQueueSize};
Sunitha Harish6eaa1d22021-02-19 13:38:31 +053073
Ed Tanous84b35602021-09-08 20:06:32 -070074 ConnState state = ConnState::initialized;
75
Ayushi Smritife44eb02020-05-15 15:24:45 +053076 std::string subId;
AppaRao Pulibd030d02020-03-20 03:34:29 +053077 std::string host;
78 std::string port;
Ed Tanous84b35602021-09-08 20:06:32 -070079 uint32_t retryCount = 0;
80 uint32_t maxRetryAttempts = 5;
81 uint32_t retryIntervalSecs = 0;
82 std::string retryPolicyAction = "TerminateAfterRetries";
83 bool runningTimer = false;
AppaRao Pulibd030d02020-03-20 03:34:29 +053084
Sunitha Harish29a82b02021-02-18 15:54:16 +053085 void doResolve()
86 {
Sunitha Harish29a82b02021-02-18 15:54:16 +053087 state = ConnState::resolveInProgress;
Sunitha Harish29a82b02021-02-18 15:54:16 +053088 BMCWEB_LOG_DEBUG << "Trying to resolve: " << host << ":" << port;
89
90 auto respHandler =
91 [self(shared_from_this())](
92 const boost::beast::error_code ec,
93 const std::vector<boost::asio::ip::tcp::endpoint>&
94 endpointList) {
Sunitha Harish6eaa1d22021-02-19 13:38:31 +053095 if (ec || (endpointList.size() == 0))
Sunitha Harish29a82b02021-02-18 15:54:16 +053096 {
97 BMCWEB_LOG_ERROR << "Resolve failed: " << ec.message();
98 self->state = ConnState::resolveFailed;
Sunitha Harish6eaa1d22021-02-19 13:38:31 +053099 self->handleConnState();
Sunitha Harish29a82b02021-02-18 15:54:16 +0530100 return;
101 }
102 BMCWEB_LOG_DEBUG << "Resolved";
103 self->doConnect(endpointList);
104 };
105 resolver.asyncResolve(host, port, std::move(respHandler));
106 }
107
108 void doConnect(
109 const std::vector<boost::asio::ip::tcp::endpoint>& endpointList)
AppaRao Pulibd030d02020-03-20 03:34:29 +0530110 {
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530111 state = ConnState::connectInProgress;
112
113 BMCWEB_LOG_DEBUG << "Trying to connect to: " << host << ":" << port;
Sunitha Harish29a82b02021-02-18 15:54:16 +0530114
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530115 conn.expires_after(std::chrono::seconds(30));
Sunitha Harish29a82b02021-02-18 15:54:16 +0530116 conn.async_connect(
117 endpointList, [self(shared_from_this())](
118 const boost::beast::error_code ec,
119 const boost::asio::ip::tcp::endpoint& endpoint) {
120 if (ec)
121 {
122 BMCWEB_LOG_ERROR << "Connect " << endpoint
123 << " failed: " << ec.message();
124 self->state = ConnState::connectFailed;
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530125 self->handleConnState();
Sunitha Harish29a82b02021-02-18 15:54:16 +0530126 return;
127 }
Sunitha Harish29a82b02021-02-18 15:54:16 +0530128 BMCWEB_LOG_DEBUG << "Connected to: " << endpoint;
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530129 self->state = ConnState::connected;
130 self->handleConnState();
Sunitha Harish29a82b02021-02-18 15:54:16 +0530131 });
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530132 }
133
134 void sendMessage(const std::string& data)
135 {
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530136 state = ConnState::sendInProgress;
137
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530138 req.body() = data;
139 req.prepare_payload();
AppaRao Pulibd030d02020-03-20 03:34:29 +0530140
141 // Set a timeout on the operation
142 conn.expires_after(std::chrono::seconds(30));
143
144 // Send the HTTP request to the remote host
145 boost::beast::http::async_write(
146 conn, req,
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530147 [self(shared_from_this())](const boost::beast::error_code& ec,
AppaRao Pulibd030d02020-03-20 03:34:29 +0530148 const std::size_t& bytesTransferred) {
149 if (ec)
150 {
151 BMCWEB_LOG_ERROR << "sendMessage() failed: "
152 << ec.message();
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530153 self->state = ConnState::sendFailed;
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530154 self->handleConnState();
AppaRao Pulibd030d02020-03-20 03:34:29 +0530155 return;
156 }
157 BMCWEB_LOG_DEBUG << "sendMessage() bytes transferred: "
158 << bytesTransferred;
159 boost::ignore_unused(bytesTransferred);
160
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530161 self->recvMessage();
AppaRao Pulibd030d02020-03-20 03:34:29 +0530162 });
163 }
164
165 void recvMessage()
166 {
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530167 state = ConnState::recvInProgress;
168
169 parser.emplace(std::piecewise_construct, std::make_tuple());
170 parser->body_limit(httpReadBodyLimit);
171
AppaRao Pulibd030d02020-03-20 03:34:29 +0530172 // Receive the HTTP response
173 boost::beast::http::async_read(
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530174 conn, buffer, *parser,
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530175 [self(shared_from_this())](const boost::beast::error_code& ec,
AppaRao Pulibd030d02020-03-20 03:34:29 +0530176 const std::size_t& bytesTransferred) {
177 if (ec)
178 {
179 BMCWEB_LOG_ERROR << "recvMessage() failed: "
180 << ec.message();
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530181 self->state = ConnState::recvFailed;
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530182 self->handleConnState();
AppaRao Pulibd030d02020-03-20 03:34:29 +0530183 return;
184 }
185 BMCWEB_LOG_DEBUG << "recvMessage() bytes transferred: "
186 << bytesTransferred;
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530187 BMCWEB_LOG_DEBUG << "recvMessage() data: "
188 << self->parser->get();
AppaRao Pulibd030d02020-03-20 03:34:29 +0530189
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530190 unsigned int respCode = self->parser->get().result_int();
191 BMCWEB_LOG_DEBUG << "recvMessage() Header Response Code: "
192 << respCode;
193
194 // 2XX response is considered to be successful
195 if ((respCode < 200) || (respCode >= 300))
196 {
197 // The listener failed to receive the Sent-Event
Sunitha Harish7adb85a2021-10-26 03:10:04 -0500198 BMCWEB_LOG_ERROR
199 << "recvMessage() Listener Failed to "
200 "receive Sent-Event. Header Response Code: "
201 << respCode;
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530202 self->state = ConnState::recvFailed;
203 self->handleConnState();
204 return;
205 }
AppaRao Pulibd030d02020-03-20 03:34:29 +0530206
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530207 // Send is successful, Lets remove data from queue
208 // check for next request data in queue.
Sunitha Harish7de9f812021-08-24 02:50:30 -0500209 if (!self->requestDataQueue.empty())
210 {
211 self->requestDataQueue.pop_front();
212 }
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530213 self->state = ConnState::idle;
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530214
215 // Keep the connection alive if server supports it
216 // Else close the connection
217 BMCWEB_LOG_DEBUG << "recvMessage() keepalive : "
218 << self->parser->keep_alive();
219 if (!self->parser->keep_alive())
220 {
221 // Abort the connection since server is not keep-alive
222 // enabled
223 self->state = ConnState::abortConnection;
224 }
225
226 self->handleConnState();
AppaRao Pulibd030d02020-03-20 03:34:29 +0530227 });
228 }
229
230 void doClose()
231 {
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530232 state = ConnState::closeInProgress;
AppaRao Pulibd030d02020-03-20 03:34:29 +0530233 boost::beast::error_code ec;
234 conn.socket().shutdown(boost::asio::ip::tcp::socket::shutdown_both, ec);
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530235 conn.close();
AppaRao Pulibd030d02020-03-20 03:34:29 +0530236
AppaRao Pulibd030d02020-03-20 03:34:29 +0530237 // not_connected happens sometimes so don't bother reporting it.
238 if (ec && ec != boost::beast::errc::not_connected)
239 {
240 BMCWEB_LOG_ERROR << "shutdown failed: " << ec.message();
241 return;
242 }
243 BMCWEB_LOG_DEBUG << "Connection closed gracefully";
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530244 if ((state != ConnState::suspended) && (state != ConnState::terminated))
245 {
246 state = ConnState::closed;
247 handleConnState();
248 }
AppaRao Pulibd030d02020-03-20 03:34:29 +0530249 }
250
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530251 void waitAndRetry()
AppaRao Pulibd030d02020-03-20 03:34:29 +0530252 {
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530253 if (retryCount >= maxRetryAttempts)
254 {
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530255 BMCWEB_LOG_ERROR << "Maximum number of retries reached.";
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530256
257 // Clear queue.
258 while (!requestDataQueue.empty())
259 {
Sunitha Harish7de9f812021-08-24 02:50:30 -0500260 requestDataQueue.pop_front();
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530261 }
262
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530263 BMCWEB_LOG_DEBUG << "Retry policy: " << retryPolicyAction;
Ayushi Smritife44eb02020-05-15 15:24:45 +0530264 if (retryPolicyAction == "TerminateAfterRetries")
265 {
266 // TODO: delete subscription
267 state = ConnState::terminated;
Ayushi Smritife44eb02020-05-15 15:24:45 +0530268 }
Ed Tanous3174e4d2020-10-07 11:41:22 -0700269 if (retryPolicyAction == "SuspendRetries")
Ayushi Smritife44eb02020-05-15 15:24:45 +0530270 {
271 state = ConnState::suspended;
Ayushi Smritife44eb02020-05-15 15:24:45 +0530272 }
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530273 // Reset the retrycount to zero so that client can try connecting
274 // again if needed
Ed Tanous3174e4d2020-10-07 11:41:22 -0700275 retryCount = 0;
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530276 handleConnState();
Ayushi Smritife44eb02020-05-15 15:24:45 +0530277 return;
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530278 }
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530279
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530280 if (runningTimer)
281 {
282 BMCWEB_LOG_DEBUG << "Retry timer is already running.";
283 return;
284 }
285 runningTimer = true;
286
287 retryCount++;
288
289 BMCWEB_LOG_DEBUG << "Attempt retry after " << retryIntervalSecs
290 << " seconds. RetryCount = " << retryCount;
291 timer.expires_after(std::chrono::seconds(retryIntervalSecs));
292 timer.async_wait(
293 [self = shared_from_this()](const boost::system::error_code ec) {
294 if (ec == boost::asio::error::operation_aborted)
295 {
296 BMCWEB_LOG_DEBUG
297 << "async_wait failed since the operation is aborted"
298 << ec.message();
299 }
300 else if (ec)
301 {
302 BMCWEB_LOG_ERROR << "async_wait failed: " << ec.message();
303 // Ignore the error and continue the retry loop to attempt
304 // sending the event as per the retry policy
305 }
306 self->runningTimer = false;
307
308 // Lets close connection and start from resolve.
309 self->doClose();
310 });
Ayushi Smritife44eb02020-05-15 15:24:45 +0530311 return;
312 }
313
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530314 void handleConnState()
Ayushi Smritife44eb02020-05-15 15:24:45 +0530315 {
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530316 switch (state)
317 {
Sunitha Harish29a82b02021-02-18 15:54:16 +0530318 case ConnState::resolveInProgress:
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530319 case ConnState::connectInProgress:
320 case ConnState::sendInProgress:
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530321 case ConnState::recvInProgress:
322 case ConnState::closeInProgress:
323 {
324 BMCWEB_LOG_DEBUG << "Async operation is already in progress";
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530325 break;
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530326 }
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530327 case ConnState::initialized:
328 case ConnState::closed:
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530329 {
330 if (requestDataQueue.empty())
331 {
332 BMCWEB_LOG_DEBUG << "requestDataQueue is empty";
333 return;
334 }
335 doResolve();
336 break;
337 }
338 case ConnState::suspended:
339 case ConnState::terminated:
340 {
341 doClose();
342 break;
343 }
344 case ConnState::resolveFailed:
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530345 case ConnState::connectFailed:
346 case ConnState::sendFailed:
AppaRao Puli92a74e52020-06-04 11:12:28 +0530347 case ConnState::recvFailed:
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530348 case ConnState::retry:
AppaRao Puli92a74e52020-06-04 11:12:28 +0530349 {
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530350 // In case of failures during connect and handshake
351 // the retry policy will be applied
352 waitAndRetry();
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530353 break;
354 }
355 case ConnState::connected:
AppaRao Puli92a74e52020-06-04 11:12:28 +0530356 case ConnState::idle:
357 {
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530358 // State idle means, previous attempt is successful
359 // State connected means, client connection is established
360 // successfully
361 if (requestDataQueue.empty())
362 {
363 BMCWEB_LOG_DEBUG << "requestDataQueue is empty";
364 return;
365 }
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530366 std::string data = requestDataQueue.front();
367 sendMessage(data);
368 break;
369 }
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530370 case ConnState::abortConnection:
371 {
372 // Server did not want to keep alive the session
373 doClose();
374 break;
375 }
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530376 }
AppaRao Pulibd030d02020-03-20 03:34:29 +0530377 }
378
379 public:
Ayushi Smritife44eb02020-05-15 15:24:45 +0530380 explicit HttpClient(boost::asio::io_context& ioc, const std::string& id,
381 const std::string& destIP, const std::string& destPort,
Ed Tanous4da04452021-09-08 19:57:44 -0700382 const std::string& destUri,
383 const boost::beast::http::fields& httpHeader) :
AppaRao Pulibd030d02020-03-20 03:34:29 +0530384 conn(ioc),
Ed Tanous4da04452021-09-08 19:57:44 -0700385 timer(ioc),
386 req(boost::beast::http::verb::post, destUri, 11, "", httpHeader),
Ed Tanous84b35602021-09-08 20:06:32 -0700387 subId(id), host(destIP), port(destPort)
AppaRao Pulibd030d02020-03-20 03:34:29 +0530388 {
Ed Tanous4da04452021-09-08 19:57:44 -0700389 req.set(boost::beast::http::field::host, host);
390 req.keep_alive(true);
AppaRao Pulibd030d02020-03-20 03:34:29 +0530391 }
392
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530393 void sendData(const std::string& data)
AppaRao Pulibd030d02020-03-20 03:34:29 +0530394 {
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530395 if ((state == ConnState::suspended) || (state == ConnState::terminated))
AppaRao Pulibd030d02020-03-20 03:34:29 +0530396 {
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530397 return;
AppaRao Pulibd030d02020-03-20 03:34:29 +0530398 }
AppaRao Pulibd030d02020-03-20 03:34:29 +0530399
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530400 if (requestDataQueue.size() <= maxRequestQueueSize)
401 {
Sunitha Harish7de9f812021-08-24 02:50:30 -0500402 requestDataQueue.push_back(data);
Sunitha Harish6eaa1d22021-02-19 13:38:31 +0530403 handleConnState();
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530404 }
405 else
406 {
407 BMCWEB_LOG_ERROR << "Request queue is full. So ignoring data.";
408 }
AppaRao Pulibd030d02020-03-20 03:34:29 +0530409
AppaRao Puli2a5689a2020-04-29 15:24:31 +0530410 return;
AppaRao Pulibd030d02020-03-20 03:34:29 +0530411 }
412
Ayushi Smritife44eb02020-05-15 15:24:45 +0530413 void setRetryConfig(const uint32_t retryAttempts,
414 const uint32_t retryTimeoutInterval)
415 {
416 maxRetryAttempts = retryAttempts;
417 retryIntervalSecs = retryTimeoutInterval;
418 }
419
420 void setRetryPolicy(const std::string& retryPolicy)
421 {
422 retryPolicyAction = retryPolicy;
423 }
AppaRao Pulibd030d02020-03-20 03:34:29 +0530424};
425
426} // namespace crow