blob: 1dcf87a758323f590c2e90310ef2a382d2943cbc [file] [log] [blame]
Ed Tanousfca2cbe2021-01-28 14:49:59 -08001#pragma once
2#include "bmcweb_config.h"
3
4#include "async_resp.hpp"
5#include "authentication.hpp"
6#include "complete_response_fields.hpp"
Ed Tanous325310d2024-03-15 09:05:04 -07007#include "http_body.hpp"
Ed Tanousfca2cbe2021-01-28 14:49:59 -08008#include "http_response.hpp"
9#include "http_utility.hpp"
10#include "logging.hpp"
11#include "mutual_tls.hpp"
12#include "nghttp2_adapters.hpp"
13#include "ssl_key_handler.hpp"
14#include "utility.hpp"
15
Ed Tanousfca2cbe2021-01-28 14:49:59 -080016#include <boost/asio/io_context.hpp>
17#include <boost/asio/ip/tcp.hpp>
18#include <boost/asio/ssl/stream.hpp>
19#include <boost/asio/steady_timer.hpp>
Ed Tanousfca2cbe2021-01-28 14:49:59 -080020#include <boost/beast/http/error.hpp>
21#include <boost/beast/http/parser.hpp>
22#include <boost/beast/http/read.hpp>
23#include <boost/beast/http/serializer.hpp>
Ed Tanousfca2cbe2021-01-28 14:49:59 -080024#include <boost/beast/http/write.hpp>
Ed Tanousfca2cbe2021-01-28 14:49:59 -080025#include <boost/beast/websocket.hpp>
Ed Tanousd0882182024-01-26 23:45:25 -080026#include <boost/system/error_code.hpp>
Ed Tanousfca2cbe2021-01-28 14:49:59 -080027
Ed Tanousd0882182024-01-26 23:45:25 -080028#include <array>
Ed Tanousfca2cbe2021-01-28 14:49:59 -080029#include <atomic>
30#include <chrono>
Ed Tanousd0882182024-01-26 23:45:25 -080031#include <functional>
32#include <memory>
Ed Tanous89cda632024-04-16 08:45:54 -070033#include <string>
Ed Tanousfca2cbe2021-01-28 14:49:59 -080034#include <vector>
35
36namespace crow
37{
38
39struct Http2StreamData
40{
Jonathan Doman102a4cd2024-04-15 16:56:23 -070041 std::shared_ptr<Request> req = std::make_shared<Request>();
Ed Tanous325310d2024-03-15 09:05:04 -070042 std::optional<bmcweb::HttpBody::reader> reqReader;
Ed Tanous89cda632024-04-16 08:45:54 -070043 std::string accept;
Ed Tanous47f29342024-03-19 12:18:06 -070044 Response res;
Ed Tanousb2896142024-01-31 15:25:47 -080045 std::optional<bmcweb::HttpBody::writer> writer;
Ed Tanousfca2cbe2021-01-28 14:49:59 -080046};
47
48template <typename Adaptor, typename Handler>
49class HTTP2Connection :
50 public std::enable_shared_from_this<HTTP2Connection<Adaptor, Handler>>
51{
52 using self_type = HTTP2Connection<Adaptor, Handler>;
53
54 public:
55 HTTP2Connection(Adaptor&& adaptorIn, Handler* handlerIn,
Ed Tanous52e31622024-01-23 16:31:11 -080056 std::function<std::string()>& getCachedDateStrF) :
Patrick Williamsbd79bce2024-08-16 15:22:20 -040057 adaptor(std::move(adaptorIn)), ngSession(initializeNghttp2Session()),
58 handler(handlerIn), getCachedDateStr(getCachedDateStrF)
Ed Tanousfca2cbe2021-01-28 14:49:59 -080059 {}
60
61 void start()
62 {
63 // Create the control stream
Ed Tanousf42e8592023-08-25 10:47:44 -070064 streams[0];
Ed Tanousfca2cbe2021-01-28 14:49:59 -080065
66 if (sendServerConnectionHeader() != 0)
67 {
Ed Tanous62598e32023-07-17 17:06:25 -070068 BMCWEB_LOG_ERROR("send_server_connection_header failed");
Ed Tanousfca2cbe2021-01-28 14:49:59 -080069 return;
70 }
71 doRead();
72 }
73
74 int sendServerConnectionHeader()
75 {
Ed Tanous62598e32023-07-17 17:06:25 -070076 BMCWEB_LOG_DEBUG("send_server_connection_header()");
Ed Tanousfca2cbe2021-01-28 14:49:59 -080077
78 uint32_t maxStreams = 4;
79 std::array<nghttp2_settings_entry, 2> iv = {
80 {{NGHTTP2_SETTINGS_MAX_CONCURRENT_STREAMS, maxStreams},
81 {NGHTTP2_SETTINGS_ENABLE_PUSH, 0}}};
82 int rv = ngSession.submitSettings(iv);
83 if (rv != 0)
84 {
Ed Tanous62598e32023-07-17 17:06:25 -070085 BMCWEB_LOG_ERROR("Fatal error: {}", nghttp2_strerror(rv));
Ed Tanousfca2cbe2021-01-28 14:49:59 -080086 return -1;
87 }
Ed Tanousd0882182024-01-26 23:45:25 -080088 writeBuffer();
Ed Tanousfca2cbe2021-01-28 14:49:59 -080089 return 0;
90 }
91
Patrick Williamsbd79bce2024-08-16 15:22:20 -040092 static ssize_t
93 fileReadCallback(nghttp2_session* /* session */, int32_t streamId,
94 uint8_t* buf, size_t length, uint32_t* dataFlags,
95 nghttp2_data_source* /*source*/, void* userPtr)
Ed Tanousfca2cbe2021-01-28 14:49:59 -080096 {
Ed Tanousf42e8592023-08-25 10:47:44 -070097 self_type& self = userPtrToSelf(userPtr);
98
99 auto streamIt = self.streams.find(streamId);
100 if (streamIt == self.streams.end())
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800101 {
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800102 return NGHTTP2_ERR_TEMPORAL_CALLBACK_FAILURE;
103 }
Ed Tanousf42e8592023-08-25 10:47:44 -0700104 Http2StreamData& stream = streamIt->second;
Ed Tanous62598e32023-07-17 17:06:25 -0700105 BMCWEB_LOG_DEBUG("File read callback length: {}", length);
Ed Tanousd547d8d2024-03-16 18:04:41 -0700106 if (!stream.writer)
107 {
108 return NGHTTP2_ERR_TEMPORAL_CALLBACK_FAILURE;
109 }
Ed Tanous52e31622024-01-23 16:31:11 -0800110 boost::beast::error_code ec;
111 boost::optional<std::pair<boost::asio::const_buffer, bool>> out =
112 stream.writer->getWithMaxSize(ec, length);
113 if (ec)
Ed Tanous27b0cf92023-08-07 12:02:40 -0700114 {
Ed Tanous325310d2024-03-15 09:05:04 -0700115 BMCWEB_LOG_CRITICAL("Failed to get buffer");
Ed Tanous27b0cf92023-08-07 12:02:40 -0700116 return NGHTTP2_ERR_TEMPORAL_CALLBACK_FAILURE;
117 }
Ed Tanous52e31622024-01-23 16:31:11 -0800118 if (!out)
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800119 {
Ed Tanous325310d2024-03-15 09:05:04 -0700120 BMCWEB_LOG_ERROR("Empty file, setting EOF");
Ed Tanous52e31622024-01-23 16:31:11 -0800121 *dataFlags |= NGHTTP2_DATA_FLAG_EOF;
122 return 0;
123 }
124
125 BMCWEB_LOG_DEBUG("Send chunk of size: {}", out->first.size());
126 if (length < out->first.size())
127 {
Ed Tanous325310d2024-03-15 09:05:04 -0700128 BMCWEB_LOG_CRITICAL(
129 "Buffer overflow that should never happen happened");
Ed Tanous52e31622024-01-23 16:31:11 -0800130 // Should never happen because of length limit on get() above
131 return NGHTTP2_ERR_TEMPORAL_CALLBACK_FAILURE;
132 }
Ed Tanous325310d2024-03-15 09:05:04 -0700133 boost::asio::mutable_buffer writeableBuf(buf, length);
Ed Tanous52e31622024-01-23 16:31:11 -0800134 BMCWEB_LOG_DEBUG("Copying {} bytes to buf", out->first.size());
Ed Tanous325310d2024-03-15 09:05:04 -0700135 size_t copied = boost::asio::buffer_copy(writeableBuf, out->first);
136 if (copied != out->first.size())
137 {
138 BMCWEB_LOG_ERROR(
139 "Couldn't copy all {} bytes into buffer, only copied {}",
140 out->first.size(), copied);
141 return NGHTTP2_ERR_TEMPORAL_CALLBACK_FAILURE;
142 }
Ed Tanous52e31622024-01-23 16:31:11 -0800143
144 if (!out->second)
145 {
Ed Tanous325310d2024-03-15 09:05:04 -0700146 BMCWEB_LOG_DEBUG("Setting EOF flag");
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800147 *dataFlags |= NGHTTP2_DATA_FLAG_EOF;
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800148 }
Ed Tanous325310d2024-03-15 09:05:04 -0700149 return static_cast<ssize_t>(copied);
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800150 }
151
152 nghttp2_nv headerFromStringViews(std::string_view name,
Ed Tanous52e31622024-01-23 16:31:11 -0800153 std::string_view value, uint8_t flags)
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800154 {
155 uint8_t* nameData = std::bit_cast<uint8_t*>(name.data());
156 uint8_t* valueData = std::bit_cast<uint8_t*>(value.data());
Ed Tanous52e31622024-01-23 16:31:11 -0800157 return {nameData, valueData, name.size(), value.size(), flags};
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800158 }
159
160 int sendResponse(Response& completedRes, int32_t streamId)
161 {
Ed Tanous62598e32023-07-17 17:06:25 -0700162 BMCWEB_LOG_DEBUG("send_response stream_id:{}", streamId);
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800163
164 auto it = streams.find(streamId);
165 if (it == streams.end())
166 {
167 close();
168 return -1;
169 }
Ed Tanous499b5b42024-04-06 08:39:18 -0700170 Http2StreamData& stream = it->second;
171 Response& res = stream.res;
172 res = std::move(completedRes);
Ed Tanous499b5b42024-04-06 08:39:18 -0700173
Ed Tanous89cda632024-04-16 08:45:54 -0700174 completeResponseFields(stream.accept, res);
Ed Tanous499b5b42024-04-06 08:39:18 -0700175 res.addHeader(boost::beast::http::field::date, getCachedDateStr());
176 res.preparePayload();
177
178 boost::beast::http::fields& fields = res.fields();
179 std::string code = std::to_string(res.resultInt());
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800180 std::vector<nghttp2_nv> hdr;
Ed Tanous52e31622024-01-23 16:31:11 -0800181 hdr.emplace_back(
182 headerFromStringViews(":status", code, NGHTTP2_NV_FLAG_NONE));
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800183 for (const boost::beast::http::fields::value_type& header : fields)
184 {
Ed Tanous52e31622024-01-23 16:31:11 -0800185 hdr.emplace_back(headerFromStringViews(
Ed Tanousd0882182024-01-26 23:45:25 -0800186 header.name_string(), header.value(), NGHTTP2_NV_FLAG_NONE));
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800187 }
Ed Tanousb2896142024-01-31 15:25:47 -0800188 http::response<bmcweb::HttpBody>& fbody = res.response;
Ed Tanous52e31622024-01-23 16:31:11 -0800189 stream.writer.emplace(fbody.base(), fbody.body());
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800190
191 nghttp2_data_provider dataPrd{
Ed Tanousf42e8592023-08-25 10:47:44 -0700192 .source = {.fd = 0},
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800193 .read_callback = fileReadCallback,
194 };
195
196 int rv = ngSession.submitResponse(streamId, hdr, &dataPrd);
197 if (rv != 0)
198 {
Ed Tanous62598e32023-07-17 17:06:25 -0700199 BMCWEB_LOG_ERROR("Fatal error: {}", nghttp2_strerror(rv));
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800200 close();
201 return -1;
202 }
Ed Tanousd0882182024-01-26 23:45:25 -0800203 writeBuffer();
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800204
205 return 0;
206 }
207
208 nghttp2_session initializeNghttp2Session()
209 {
210 nghttp2_session_callbacks callbacks;
211 callbacks.setOnFrameRecvCallback(onFrameRecvCallbackStatic);
212 callbacks.setOnStreamCloseCallback(onStreamCloseCallbackStatic);
213 callbacks.setOnHeaderCallback(onHeaderCallbackStatic);
214 callbacks.setOnBeginHeadersCallback(onBeginHeadersCallbackStatic);
Ed Tanous325310d2024-03-15 09:05:04 -0700215 callbacks.setOnDataChunkRecvCallback(onDataChunkRecvStatic);
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800216
217 nghttp2_session session(callbacks);
218 session.setUserData(this);
219
220 return session;
221 }
222
223 int onRequestRecv(int32_t streamId)
224 {
Ed Tanous62598e32023-07-17 17:06:25 -0700225 BMCWEB_LOG_DEBUG("on_request_recv");
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800226
227 auto it = streams.find(streamId);
228 if (it == streams.end())
229 {
230 close();
231 return -1;
232 }
Ed Tanousd547d8d2024-03-16 18:04:41 -0700233 auto& reqReader = it->second.reqReader;
234 if (reqReader)
Ed Tanous325310d2024-03-15 09:05:04 -0700235 {
236 boost::beast::error_code ec;
Ed Tanousd547d8d2024-03-16 18:04:41 -0700237 reqReader->finish(ec);
Ed Tanous325310d2024-03-15 09:05:04 -0700238 if (ec)
239 {
240 BMCWEB_LOG_CRITICAL("Failed to finalize payload");
241 close();
242 return NGHTTP2_ERR_TEMPORAL_CALLBACK_FAILURE;
243 }
244 }
Jonathan Doman102a4cd2024-04-15 16:56:23 -0700245 crow::Request& thisReq = *it->second.req;
Ed Tanous8e8245d2024-04-11 22:21:38 -0700246 thisReq.ioService = static_cast<decltype(thisReq.ioService)>(
247 &adaptor.get_executor().context());
Ed Tanous89cda632024-04-16 08:45:54 -0700248
249 it->second.accept = thisReq.getHeaderValue("Accept");
250
Ed Tanous62598e32023-07-17 17:06:25 -0700251 BMCWEB_LOG_DEBUG("Handling {} \"{}\"", logPtr(&thisReq),
252 thisReq.url().encoded_path());
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800253
Ed Tanousf42e8592023-08-25 10:47:44 -0700254 crow::Response& thisRes = it->second.res;
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800255
256 thisRes.setCompleteRequestHandler(
257 [this, streamId](Response& completeRes) {
Patrick Williamsbd79bce2024-08-16 15:22:20 -0400258 BMCWEB_LOG_DEBUG("res.completeRequestHandler called");
259 if (sendResponse(completeRes, streamId) != 0)
260 {
261 close();
262 return;
263 }
264 });
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800265 auto asyncResp =
Ed Tanousf42e8592023-08-25 10:47:44 -0700266 std::make_shared<bmcweb::AsyncResp>(std::move(it->second.res));
Ed Tanous83328312024-05-09 15:48:09 -0700267 if constexpr (!BMCWEB_INSECURE_DISABLE_AUTH)
Ed Tanous325310d2024-03-15 09:05:04 -0700268 {
Ed Tanous83328312024-05-09 15:48:09 -0700269 thisReq.session = crow::authentication::authenticate(
270 {}, asyncResp->res, thisReq.method(), thisReq.req, nullptr);
271 if (!crow::authentication::isOnAllowlist(thisReq.url().path(),
272 thisReq.method()) &&
273 thisReq.session == nullptr)
Ed Tanous499b5b42024-04-06 08:39:18 -0700274 {
Ed Tanous83328312024-05-09 15:48:09 -0700275 BMCWEB_LOG_WARNING("Authentication failed");
276 forward_unauthorized::sendUnauthorized(
277 thisReq.url().encoded_path(),
278 thisReq.getHeaderValue("X-Requested-With"),
279 thisReq.getHeaderValue("Accept"), asyncResp->res);
280 return 0;
Ed Tanous499b5b42024-04-06 08:39:18 -0700281 }
Ed Tanous325310d2024-03-15 09:05:04 -0700282 }
Ed Tanous83328312024-05-09 15:48:09 -0700283 std::string_view expected =
284 thisReq.getHeaderValue(boost::beast::http::field::if_none_match);
285 BMCWEB_LOG_DEBUG("Setting expected hash {}", expected);
286 if (!expected.empty())
287 {
288 asyncResp->res.setExpectedHash(expected);
289 }
290 handler->handle(it->second.req, asyncResp);
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800291 return 0;
292 }
293
Ed Tanous325310d2024-03-15 09:05:04 -0700294 int onDataChunkRecvCallback(uint8_t /*flags*/, int32_t streamId,
295 const uint8_t* data, size_t len)
296 {
297 auto thisStream = streams.find(streamId);
298 if (thisStream == streams.end())
299 {
300 BMCWEB_LOG_ERROR("Unknown stream{}", streamId);
301 close();
302 return -1;
303 }
Ed Tanousd547d8d2024-03-16 18:04:41 -0700304
305 std::optional<bmcweb::HttpBody::reader>& reqReader =
306 thisStream->second.reqReader;
307 if (!reqReader)
Ed Tanous325310d2024-03-15 09:05:04 -0700308 {
Ed Tanous8e5cc7b2024-04-02 11:00:54 -0700309 reqReader.emplace(
Jonathan Doman102a4cd2024-04-15 16:56:23 -0700310 bmcweb::HttpBody::reader(thisStream->second.req->req.base(),
311 thisStream->second.req->req.body()));
Ed Tanous325310d2024-03-15 09:05:04 -0700312 }
313 boost::beast::error_code ec;
Ed Tanousd547d8d2024-03-16 18:04:41 -0700314 reqReader->put(boost::asio::const_buffer(data, len), ec);
Ed Tanous325310d2024-03-15 09:05:04 -0700315 if (ec)
316 {
317 BMCWEB_LOG_CRITICAL("Failed to write payload");
318 return NGHTTP2_ERR_TEMPORAL_CALLBACK_FAILURE;
319 }
320 return 0;
321 }
322
Patrick Williamsbd79bce2024-08-16 15:22:20 -0400323 static int onDataChunkRecvStatic(
324 nghttp2_session* /* session */, uint8_t flags, int32_t streamId,
325 const uint8_t* data, size_t len, void* userData)
Ed Tanous325310d2024-03-15 09:05:04 -0700326 {
327 BMCWEB_LOG_DEBUG("on_frame_recv_callback");
328 if (userData == nullptr)
329 {
330 BMCWEB_LOG_CRITICAL("user data was null?");
331 return NGHTTP2_ERR_CALLBACK_FAILURE;
332 }
Patrick Williamsbd79bce2024-08-16 15:22:20 -0400333 return userPtrToSelf(userData).onDataChunkRecvCallback(
334 flags, streamId, data, len);
Ed Tanous325310d2024-03-15 09:05:04 -0700335 }
336
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800337 int onFrameRecvCallback(const nghttp2_frame& frame)
338 {
Ed Tanous62598e32023-07-17 17:06:25 -0700339 BMCWEB_LOG_DEBUG("frame type {}", static_cast<int>(frame.hd.type));
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800340 switch (frame.hd.type)
341 {
342 case NGHTTP2_DATA:
343 case NGHTTP2_HEADERS:
344 // Check that the client request has finished
345 if ((frame.hd.flags & NGHTTP2_FLAG_END_STREAM) != 0)
346 {
347 return onRequestRecv(frame.hd.stream_id);
348 }
349 break;
350 default:
351 break;
352 }
353 return 0;
354 }
355
356 static int onFrameRecvCallbackStatic(nghttp2_session* /* session */,
357 const nghttp2_frame* frame,
358 void* userData)
359 {
Ed Tanous62598e32023-07-17 17:06:25 -0700360 BMCWEB_LOG_DEBUG("on_frame_recv_callback");
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800361 if (userData == nullptr)
362 {
Ed Tanous62598e32023-07-17 17:06:25 -0700363 BMCWEB_LOG_CRITICAL("user data was null?");
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800364 return NGHTTP2_ERR_CALLBACK_FAILURE;
365 }
366 if (frame == nullptr)
367 {
Ed Tanous62598e32023-07-17 17:06:25 -0700368 BMCWEB_LOG_CRITICAL("frame was null?");
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800369 return NGHTTP2_ERR_CALLBACK_FAILURE;
370 }
371 return userPtrToSelf(userData).onFrameRecvCallback(*frame);
372 }
373
374 static self_type& userPtrToSelf(void* userData)
375 {
376 // This method exists to keep the unsafe reinterpret cast in one
377 // place.
378 // NOLINTNEXTLINE(cppcoreguidelines-pro-type-reinterpret-cast)
379 return *reinterpret_cast<self_type*>(userData);
380 }
381
382 static int onStreamCloseCallbackStatic(nghttp2_session* /* session */,
383 int32_t streamId,
384 uint32_t /*unused*/, void* userData)
385 {
Ed Tanous62598e32023-07-17 17:06:25 -0700386 BMCWEB_LOG_DEBUG("on_stream_close_callback stream {}", streamId);
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800387 if (userData == nullptr)
388 {
Ed Tanous62598e32023-07-17 17:06:25 -0700389 BMCWEB_LOG_CRITICAL("user data was null?");
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800390 return NGHTTP2_ERR_CALLBACK_FAILURE;
391 }
Ed Tanousf42e8592023-08-25 10:47:44 -0700392 if (userPtrToSelf(userData).streams.erase(streamId) <= 0)
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800393 {
394 return -1;
395 }
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800396 return 0;
397 }
398
399 int onHeaderCallback(const nghttp2_frame& frame,
400 std::span<const uint8_t> name,
401 std::span<const uint8_t> value)
402 {
403 // NOLINTNEXTLINE(cppcoreguidelines-pro-type-reinterpret-cast)
404 std::string_view nameSv(reinterpret_cast<const char*>(name.data()),
405 name.size());
406 // NOLINTNEXTLINE(cppcoreguidelines-pro-type-reinterpret-cast)
407 std::string_view valueSv(reinterpret_cast<const char*>(value.data()),
408 value.size());
409
Ed Tanous62598e32023-07-17 17:06:25 -0700410 BMCWEB_LOG_DEBUG("on_header_callback name: {} value {}", nameSv,
411 valueSv);
Ed Tanousa07e9812024-03-19 10:31:13 -0700412 if (frame.hd.type != NGHTTP2_HEADERS)
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800413 {
Ed Tanousa07e9812024-03-19 10:31:13 -0700414 return 0;
415 }
416 if (frame.headers.cat != NGHTTP2_HCAT_REQUEST)
417 {
418 return 0;
419 }
420 auto thisStream = streams.find(frame.hd.stream_id);
421 if (thisStream == streams.end())
422 {
423 BMCWEB_LOG_ERROR("Unknown stream{}", frame.hd.stream_id);
424 close();
425 return -1;
426 }
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800427
Jonathan Doman102a4cd2024-04-15 16:56:23 -0700428 crow::Request& thisReq = *thisStream->second.req;
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800429
Ed Tanousa07e9812024-03-19 10:31:13 -0700430 if (nameSv == ":path")
431 {
432 thisReq.target(valueSv);
433 }
434 else if (nameSv == ":method")
435 {
436 boost::beast::http::verb verb =
437 boost::beast::http::string_to_verb(valueSv);
438 if (verb == boost::beast::http::verb::unknown)
439 {
440 BMCWEB_LOG_ERROR("Unknown http verb {}", valueSv);
441 close();
442 return -1;
443 }
Myung Bae1873a042024-04-01 09:27:39 -0500444 thisReq.method(verb);
Ed Tanousa07e9812024-03-19 10:31:13 -0700445 }
446 else if (nameSv == ":scheme")
447 {
448 // Nothing to check on scheme
449 }
450 else
451 {
Myung Bae1873a042024-04-01 09:27:39 -0500452 thisReq.addHeader(nameSv, valueSv);
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800453 }
454 return 0;
455 }
456
Patrick Williamsbd79bce2024-08-16 15:22:20 -0400457 static int onHeaderCallbackStatic(
458 nghttp2_session* /* session */, const nghttp2_frame* frame,
459 const uint8_t* name, size_t namelen, const uint8_t* value,
460 size_t vallen, uint8_t /* flags */, void* userData)
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800461 {
462 if (userData == nullptr)
463 {
Ed Tanous62598e32023-07-17 17:06:25 -0700464 BMCWEB_LOG_CRITICAL("user data was null?");
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800465 return NGHTTP2_ERR_CALLBACK_FAILURE;
466 }
467 if (frame == nullptr)
468 {
Ed Tanous62598e32023-07-17 17:06:25 -0700469 BMCWEB_LOG_CRITICAL("frame was null?");
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800470 return NGHTTP2_ERR_CALLBACK_FAILURE;
471 }
472 if (name == nullptr)
473 {
Ed Tanous62598e32023-07-17 17:06:25 -0700474 BMCWEB_LOG_CRITICAL("name was null?");
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800475 return NGHTTP2_ERR_CALLBACK_FAILURE;
476 }
477 if (value == nullptr)
478 {
Ed Tanous62598e32023-07-17 17:06:25 -0700479 BMCWEB_LOG_CRITICAL("value was null?");
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800480 return NGHTTP2_ERR_CALLBACK_FAILURE;
481 }
482 return userPtrToSelf(userData).onHeaderCallback(*frame, {name, namelen},
483 {value, vallen});
484 }
485
486 int onBeginHeadersCallback(const nghttp2_frame& frame)
487 {
488 if (frame.hd.type == NGHTTP2_HEADERS &&
489 frame.headers.cat == NGHTTP2_HCAT_REQUEST)
490 {
Ed Tanous62598e32023-07-17 17:06:25 -0700491 BMCWEB_LOG_DEBUG("create stream for id {}", frame.hd.stream_id);
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800492
Ed Tanous3a499412024-06-05 16:05:35 -0700493 streams.emplace(frame.hd.stream_id, Http2StreamData());
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800494 }
495 return 0;
496 }
497
498 static int onBeginHeadersCallbackStatic(nghttp2_session* /* session */,
499 const nghttp2_frame* frame,
500 void* userData)
501 {
Ed Tanous62598e32023-07-17 17:06:25 -0700502 BMCWEB_LOG_DEBUG("on_begin_headers_callback");
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800503 if (userData == nullptr)
504 {
Ed Tanous62598e32023-07-17 17:06:25 -0700505 BMCWEB_LOG_CRITICAL("user data was null?");
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800506 return NGHTTP2_ERR_CALLBACK_FAILURE;
507 }
508 if (frame == nullptr)
509 {
Ed Tanous62598e32023-07-17 17:06:25 -0700510 BMCWEB_LOG_CRITICAL("frame was null?");
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800511 return NGHTTP2_ERR_CALLBACK_FAILURE;
512 }
513 return userPtrToSelf(userData).onBeginHeadersCallback(*frame);
514 }
515
516 static void afterWriteBuffer(const std::shared_ptr<self_type>& self,
517 const boost::system::error_code& ec,
518 size_t sendLength)
519 {
520 self->isWriting = false;
Ed Tanous62598e32023-07-17 17:06:25 -0700521 BMCWEB_LOG_DEBUG("Sent {}", sendLength);
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800522 if (ec)
523 {
524 self->close();
525 return;
526 }
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800527 self->writeBuffer();
528 }
529
530 void writeBuffer()
531 {
532 if (isWriting)
533 {
534 return;
535 }
Ed Tanousd0882182024-01-26 23:45:25 -0800536 std::span<const uint8_t> data = ngSession.memSend();
537 if (data.empty())
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800538 {
539 return;
540 }
541 isWriting = true;
Ed Tanous325310d2024-03-15 09:05:04 -0700542 boost::asio::async_write(
543 adaptor, boost::asio::const_buffer(data.data(), data.size()),
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800544 std::bind_front(afterWriteBuffer, shared_from_this()));
545 }
546
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800547 void close()
548 {
549 if constexpr (std::is_same_v<Adaptor,
Ed Tanous003301a2024-04-16 09:59:19 -0700550 boost::asio::ssl::stream<
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800551 boost::asio::ip::tcp::socket>>)
552 {
553 adaptor.next_layer().close();
554 }
555 else
556 {
557 adaptor.close();
558 }
559 }
560
Ed Tanousd0882182024-01-26 23:45:25 -0800561 void afterDoRead(const std::shared_ptr<self_type>& /*self*/,
562 const boost::system::error_code& ec,
563 size_t bytesTransferred)
564 {
565 BMCWEB_LOG_DEBUG("{} async_read_some {} Bytes", logPtr(this),
566 bytesTransferred);
567
568 if (ec)
569 {
570 BMCWEB_LOG_ERROR("{} Error while reading: {}", logPtr(this),
571 ec.message());
572 close();
573 BMCWEB_LOG_DEBUG("{} from read(1)", logPtr(this));
574 return;
575 }
576 std::span<uint8_t> bufferSpan{inBuffer.data(), bytesTransferred};
577
578 ssize_t readLen = ngSession.memRecv(bufferSpan);
579 if (readLen < 0)
580 {
581 BMCWEB_LOG_ERROR("nghttp2_session_mem_recv returned {}", readLen);
582 close();
583 return;
584 }
585 writeBuffer();
586
587 doRead();
588 }
589
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800590 void doRead()
591 {
Ed Tanous62598e32023-07-17 17:06:25 -0700592 BMCWEB_LOG_DEBUG("{} doRead", logPtr(this));
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800593 adaptor.async_read_some(
Ed Tanousd0882182024-01-26 23:45:25 -0800594 boost::asio::buffer(inBuffer),
595 std::bind_front(&self_type::afterDoRead, this, shared_from_this()));
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800596 }
597
598 // A mapping from http2 stream ID to Stream Data
Ed Tanous52e31622024-01-23 16:31:11 -0800599 std::map<int32_t, Http2StreamData> streams;
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800600
Ed Tanousd0882182024-01-26 23:45:25 -0800601 std::array<uint8_t, 8192> inBuffer{};
Ed Tanousfca2cbe2021-01-28 14:49:59 -0800602
603 Adaptor adaptor;
604 bool isWriting = false;
605
606 nghttp2_session ngSession;
607
608 Handler* handler;
609 std::function<std::string()>& getCachedDateStr;
610
611 using std::enable_shared_from_this<
612 HTTP2Connection<Adaptor, Handler>>::shared_from_this;
613
614 using std::enable_shared_from_this<
615 HTTP2Connection<Adaptor, Handler>>::weak_from_this;
616};
617} // namespace crow