SNode.C
Loading...
Searching...
No Matches
SocketContext.cpp
Go to the documentation of this file.
1/*
2 * SNode.C - A Slim Toolkit for Network Communication
3 * Copyright (C) Volker Christian <me@vchrist.at>
4 * 2020, 2021, 2022, 2023, 2024, 2025
5 *
6 * This program is free software: you can redistribute it and/or modify
7 * it under the terms of the GNU Lesser General Public License as published
8 * by the Free Software Foundation, either version 3 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public License
17 * along with this program. If not, see <http://www.gnu.org/licenses/>.
18 */
19
20/*
21 * MIT License
22 *
23 * Permission is hereby granted, free of charge, to any person obtaining a copy
24 * of this software and associated documentation files (the "Software"), to deal
25 * in the Software without restriction, including without limitation the rights
26 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
27 * copies of the Software, and to permit persons to whom the Software is
28 * furnished to do so, subject to the following conditions:
29 *
30 * The above copyright notice and this permission notice shall be included in
31 * all copies or substantial portions of the Software.
32 *
33 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
34 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
35 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
36 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
37 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
38 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
39 * THE SOFTWARE.
40 */
41
42#include "web/http/client/SocketContext.h" // IWYU pragma: export
43
44#include "core/EventReceiver.h"
45#include "core/socket/stream/SocketConnection.h"
46#include "web/http/CookieOptions.h"
47
48#ifndef DOXYGEN_SHOULD_SKIP_THIS
49
50#include "log/Logger.h"
51#include "utils/Timeval.h"
52#include "web/http/http_utils.h"
53
54#include <utility>
55
56#endif /* DOXYGEN_SHOULD_SKIP_THIS */
57
58// #define TRUE 1
59// #define FALSE 0
60
61namespace web::http::client {
62
63 SocketContext::SocketContext(core::socket::stream::SocketConnection* socketConnection,
64 const std::function<void(const std::shared_ptr<MasterRequest>&)>& onHttpConnected,
65 const std::function<void(const std::shared_ptr<MasterRequest>&)>& onHttpDisconnected,
66 const std::string& hostHeader,
67 bool pipelinedRequests)
68 : Super(socketConnection)
69 , onHttpConnected(onHttpConnected)
70 , onHttpDisconnected(onHttpDisconnected)
71 , pipelinedRequests(pipelinedRequests)
72 , masterRequest(std::make_shared<MasterRequest>(this, hostHeader))
73 , parser(
74 this,
75 [this]() {
77 },
78 [this](web::http::client::Response& response) {
79 deliverResponse(std::make_shared<Response>(std::move(response)));
80 },
81 [this](int status, const std::string& reason) {
83 }) {
85 }
86
88 if (!deliveredRequests.empty()) {
89 LOG(DEBUG) << getSocketConnection()->getConnectionName() << " HTTP: Responses missed";
90 for (const std::shared_ptr<MasterRequest>& request : deliveredRequests) {
91 LOG(DEBUG) << " " << request->method << " " << request->url << " HTTP/" << request->httpMajor << "." << request->httpMinor;
92 }
93 }
94
95 if (!pendingRequests.empty()) {
96 LOG(DEBUG) << getSocketConnection()->getConnectionName() << " HTTP: Requests ignored";
97 for (const std::shared_ptr<MasterRequest>& request : pendingRequests) {
98 LOG(DEBUG) << " " << request->method << " " << request->url << " HTTP/" << request->httpMajor << "." << request->httpMinor;
99 }
100 }
101 }
102
103 void SocketContext::requestPrepared(const std::shared_ptr<MasterRequest>& request) {
104 const std::string requestLine = std::string(request->method)
105 .append(" ")
106 .append(request->url)
107 .append(" HTTP/")
108 .append(std::to_string(request->httpMajor))
109 .append(".")
110 .append(std::to_string(request->httpMinor));
111
114 LOG(INFO) << getSocketConnection()->getConnectionName() << " HTTP: Request (" << request->count
115 << ") accepted: " << requestLine;
116 flags = (flags & Flags::HTTP11) | ((request->httpMajor == 1 && request->httpMinor == 1) ? Flags::HTTP11 : Flags::NONE);
117 flags = (flags & Flags::HTTP10) | ((request->httpMajor == 1 && request->httpMinor == 0) ? Flags::HTTP10 : Flags::NONE);
119 (web::http::ciContains(request->header("Connection"), "keep-alive") ? Flags::KEEPALIVE : Flags::NONE);
120 flags = (flags & Flags::CLOSE) | (web::http::ciContains(request->header("Connection"), "close") ? Flags::CLOSE : Flags::NONE);
121
122 pendingRequests.push_back(request);
123
124 LOG(DEBUG) << getSocketConnection()->getConnectionName() << " HTTP: Request (" << request->count << ") queued: " << requestLine
125 << " - QueueSize = " << pendingRequests.size() << " - Flags: " << flags << " - "
126 << web::http::ciContains(request->header("Connection"), "close");
127
128 if (pendingRequests.size() == 1) {
130 }
131 } else {
132 LOG(WARNING) << getSocketConnection()->getConnectionName() << " HTTP: Request (" << request->count
133 << ") rejected: " << requestLine;
134
135 LOG(WARNING) << httputils::toString(request->method,
136 request->url,
137 "HTTP/" + std::to_string(request->httpMajor) + "." + std::to_string(request->httpMinor),
138 request->getQueries(),
139 request->getHeaders(),
140 request->getTrailer(),
141 request->getCookies(),
142 {});
143 }
144 }
145
147 if (!pendingRequests.empty()) {
148 const std::shared_ptr<MasterRequest>& request = pendingRequests.front();
149
150 const std::string requestLine = std::string(request->method)
151 .append(" ")
152 .append(request->url)
153 .append(" HTTP/")
154 .append(std::to_string(request->httpMajor))
155 .append(".")
156 .append(std::to_string(request->httpMinor));
157
158 LOG(DEBUG) << getSocketConnection()->getConnectionName() << " HTTP: Request (" << request->count << ") start: " << requestLine;
159
160 if (!request->initiate(request)) {
161 LOG(WARNING) << getSocketConnection()->getConnectionName() << " HTTP: Request (" << request->count
162 << ") delivering failed: " << requestLine;
163
164 core::EventReceiver::atNextTick([this, masterRequest = static_cast<std::weak_ptr<Request>>(masterRequest)]() {
165 pendingRequests.pop_front();
166 if (!masterRequest.expired() && !pendingRequests.empty()) {
167 const std::shared_ptr<Request>& request = pendingRequests.front();
168
169 LOG(DEBUG) << getSocketConnection()->getConnectionName() << " HTTP: Request (" << request->count
170 << ") dequeued: " << request->method << " " << request->url << " HTTP/" << request->httpMajor << "."
171 << request->httpMinor;
172
174 }
175 });
176 }
177 }
178 }
179
180 void SocketContext::requestDelivered(bool success) {
181 const std::shared_ptr<MasterRequest> currentRequest = std::move(pendingRequests.front());
182 pendingRequests.pop_front();
183
184 const std::string requestLine = std::string(currentRequest->method)
185 .append(" ")
186 .append(currentRequest->url)
187 .append(" HTTP/")
188 .append(std::to_string(currentRequest->httpMajor))
189 .append(".")
190 .append(std::to_string(currentRequest->httpMinor));
191
192 if (success) {
193 LOG(DEBUG) << getSocketConnection()->getConnectionName() << " HTTP: Request (" << currentRequest->count
194 << ") delivered: " << requestLine << " " << pendingRequests.size();
195
196 deliveredRequests.push_back(currentRequest);
197
198 if (pipelinedRequests && !pendingRequests.empty()) {
199 core::EventReceiver::atNextTick([this, masterRequest = static_cast<std::weak_ptr<Request>>(masterRequest)]() {
200 if (!masterRequest.expired()) {
201 const std::shared_ptr<Request>& request = pendingRequests.front();
202
203 LOG(DEBUG) << getSocketConnection()->getConnectionName() << " HTTP: Request (" << request->count
204 << ") dequeued: " << request->method << " " << request->url << " HTTP/" << request->httpMajor << "."
205 << request->httpMinor;
206
208 }
209 });
210 }
211 } else {
212 LOG(WARNING) << getSocketConnection()->getConnectionName() << " HTTP: Request (" << currentRequest->count
213 << ") deliver failed: " << requestLine;
214
216 }
217 }
218
220 if (deliveredRequests.empty()) {
221 LOG(ERROR) << getSocketConnection()->getConnectionName() << " HTTP: Response without delivered request";
222
223 shutdownWrite(true);
224 }
225 }
226
227 void SocketContext::deliverResponse(const std::shared_ptr<Response>& response) {
228 const std::shared_ptr<MasterRequest> request = std::move(deliveredRequests.front());
229 deliveredRequests.pop_front();
230
231 const std::string requestLine = std::string(request->method)
232 .append(" ")
233 .append(request->url)
234 .append(" HTTP/")
235 .append(std::to_string(request->httpMajor))
236 .append(".")
237 .append(std::to_string(request->httpMinor));
238
239 LOG(INFO) << getSocketConnection()->getConnectionName() << " HTTP: Response received for request (" << request->count
240 << "): " << requestLine;
241
242 LOG(INFO) << getSocketConnection()->getConnectionName() << " HTTP/" << response->httpMajor << "." << response->httpMinor << " "
243 << response->statusCode << " " << response->reason;
244
245 request->deliverResponse(request, response);
246
247 LOG(INFO) << getSocketConnection()->getConnectionName() << " HTTP: Request (" << request->count << ") completed: " << requestLine;
248
249 requestCompleted(response);
250 }
251
252 void SocketContext::deliverResponseParseError(int status, const std::string& reason) {
253 const std::shared_ptr<MasterRequest> request = std::move(deliveredRequests.front());
254 deliveredRequests.pop_front();
255
256 const std::string requestLine = std::string(request->method)
257 .append(" ")
258 .append(request->url)
259 .append(" HTTP/")
260 .append(std::to_string(request->httpMajor))
261 .append(".")
262 .append(std::to_string(request->httpMinor));
263
264 LOG(WARNING) << getSocketConnection()->getConnectionName() << " HTTP: Response parse error: " << reason << " (" << status
265 << ") for request (" << request->count << "): " << requestLine
266 << std::string(request->method)
267 .append(" ")
268 .append(request->url)
269 .append(" HTTP/")
270 .append(std::to_string(request->httpMajor))
271 .append(".")
272 .append(std::to_string(request->httpMinor));
273
274 request->deliverResponseParseError(request, reason);
275
276 shutdownWrite(true);
277 }
278
279 void SocketContext::requestCompleted(const std::shared_ptr<Response>& response) {
282 ((response->httpMajor == 0 && response->httpMinor == 0) || (response->httpMajor == 1 && response->httpMinor == 0)));
283
284 if (httpClose) {
285 LOG(DEBUG) << getSocketConnection()->getConnectionName() << " HTTP: Connection = Close";
286 } else {
287 LOG(DEBUG) << getSocketConnection()->getConnectionName() << " HTTP: Connection = Keep-Alive";
288
289 if (!pipelinedRequests && !pendingRequests.empty()) {
290 core::EventReceiver::atNextTick([this, masterRequest = static_cast<std::weak_ptr<Request>>(masterRequest)]() {
291 if (!masterRequest.expired()) {
292 const std::shared_ptr<Request>& request = pendingRequests.front();
293
294 LOG(DEBUG) << getSocketConnection()->getConnectionName() << " HTTP: Initiating request (" << request->count
295 << "): " << request->method << " " << request->url << " HTTP/" << request->httpMajor << "."
296 << request->httpMinor;
297
299 }
300 });
301 }
302 }
303 }
304
305 void SocketContext::setSseEventReceiver(const std::function<std::size_t()>& onServerSentEvent) {
306 this->onServerSentEvent = onServerSentEvent;
307
309 }
310
312 LOG(INFO) << getSocketConnection()->getConnectionName() << " HTTP: Connected";
313
315 }
316
318 std::size_t consumed = 0;
319
320 if (!httpClose && (!deliveredRequests.empty() || onServerSentEvent)) {
321 if (!onServerSentEvent) {
322 consumed = parser.parse();
323 } else if (onServerSentEvent) {
324 consumed = onServerSentEvent();
325 }
326 }
327
328 return consumed;
329 }
330
332 while (!deliveredRequests.empty()) {
333 const std::shared_ptr<Response> response(new Response());
334 response->httpVersion = "HTTP/1.1";
335 response->httpMajor = 1;
336 response->httpMinor = 1;
337 response->statusCode = "0";
338 response->reason = "Connection loss";
339
340 deliverResponse(response);
341 }
342
344
345 LOG(INFO) << getSocketConnection()->getConnectionName() << " HTTP: Received disconnect";
346 }
347
348 bool SocketContext::onSignal([[maybe_unused]] int signum) {
349 LOG(INFO) << getSocketConnection()->getConnectionName() << " HTTP: Received signal " << signum;
350
351 return true;
352 }
353
355 // Do nothing in case of an write error
356 }
357
358} // namespace web::http::client
static void atNextTick(const std::function< void(void)> &callBack)
virtual void setWriteTimeout(const utils::Timeval &timeout)=0
const std::string & getConnectionName() const
SocketConnection * getSocketConnection() const
SocketContext(core::socket::stream::SocketConnection *socketConnection)
void shutdownWrite(bool forceClose=false)
std::size_t parse()
Definition Parser.cpp:92
void deliverResponseParseError(const std::shared_ptr< MasterRequest > &request, const std::string &message)
Definition Request.cpp:720
void deliverResponse(const std::shared_ptr< MasterRequest > &request, const std::shared_ptr< Response > &response)
Definition Request.cpp:716
bool initiate(const std::shared_ptr< MasterRequest > &request)
Definition Request.cpp:543
void setMasterRequest(const std::shared_ptr< MasterRequest > &masterRequest)
Definition Request.cpp:106
const CiStringMap< std::string > & getCookies() const
Definition Request.cpp:253
const CiStringMap< std::string > & getQueries() const
Definition Request.cpp:241
std::string header(const std::string &field) const
Definition Request.cpp:235
const CiStringMap< std::string > & getTrailer() const
Definition Request.cpp:249
const CiStringMap< std::string > & getHeaders() const
Definition Request.cpp:245
ResponseParser(core::socket::stream::SocketContext *socketContext, const std::function< void()> &onResponseStart, const std::function< void(Response &)> &onResponseParsed, const std::function< void(int, const std::string &)> &onResponseParseError)
ConnectionState connectionState
Definition Response.h:74
void setSseEventReceiver(const std::function< std::size_t()> &onServerSentEvent)
void deliverResponseParseError(int status, const std::string &reason)
SocketContext(core::socket::stream::SocketConnection *socketConnection, const std::function< void(const std::shared_ptr< MasterRequest > &)> &onHttpConnected, const std::function< void(const std::shared_ptr< MasterRequest > &)> &onHttpDisconnected, const std::string &hostHeader, bool pipelinedRequests)
std::function< void(const std::shared_ptr< MasterRequest > &)> onHttpConnected
std::size_t onReceivedFromPeer() override
std::function< std::size_t()> onServerSentEvent
std::list< std::shared_ptr< MasterRequest > > deliveredRequests
void onWriteError(int errnum) override
std::function< void(const std::shared_ptr< MasterRequest > &)> onHttpDisconnected
bool onSignal(int signum) override
void requestPrepared(const std::shared_ptr< MasterRequest > &request)
std::shared_ptr< MasterRequest > masterRequest
void deliverResponse(const std::shared_ptr< Response > &response)
std::list< std::shared_ptr< MasterRequest > > pendingRequests
void requestCompleted(const std::shared_ptr< Response > &response)
std::string toString(const std::string &method, const std::string &url, const std::string &version, const web::http::CiStringMap< std::string > &queries, const web::http::CiStringMap< std::string > &header, const web::http::CiStringMap< std::string > &trailer, const web::http::CiStringMap< std::string > &cookies, const std::vector< char > &body)
bool ciContains(const std::string &str1, const std::string &str2)