blob: 38bebb5c8a41939946cf6e65735304cd4880e7cb [file] [log] [blame]
Ed Tanous7045c8d2017-04-03 10:04:37 -07001#pragma once
Ed Tanous04e438c2020-10-03 08:06:26 -07002#include "logging.hpp"
Ed Tanous1abe55e2018-09-05 08:30:59 -07003#include "nlohmann/json.hpp"
4
Ed Tanousd43cd0c2020-09-30 20:46:53 -07005#include <boost/beast/http/message.hpp>
Ed Tanousd4b6c662021-03-10 13:29:30 -08006#include <boost/beast/http/string_body.hpp>
Ed Tanous89f18002022-03-24 18:38:24 -07007#include <utils/hex_utils.hpp>
Ed Tanous7045c8d2017-04-03 10:04:37 -07008
Ed Tanous8a9a25c2021-05-11 14:50:58 -07009#include <optional>
Gunnar Mills1214b7e2020-06-04 10:11:30 -050010#include <string>
Ed Tanous8a9a25c2021-05-11 14:50:58 -070011#include <string_view>
Ed Tanous7045c8d2017-04-03 10:04:37 -070012
Ed Tanous1abe55e2018-09-05 08:30:59 -070013namespace crow
14{
Ed Tanouse0d918b2018-03-27 17:41:04 -070015
Ed Tanous52cc1122020-07-18 13:51:21 -070016template <typename Adaptor, typename Handler>
Ed Tanous7045c8d2017-04-03 10:04:37 -070017class Connection;
Borawski.Lukasz9d8fd302018-01-05 14:56:09 +010018
Ed Tanous1abe55e2018-09-05 08:30:59 -070019struct Response
20{
Ed Tanous52cc1122020-07-18 13:51:21 -070021 template <typename Adaptor, typename Handler>
Ed Tanous1abe55e2018-09-05 08:30:59 -070022 friend class crow::Connection;
23 using response_type =
24 boost::beast::http::response<boost::beast::http::string_body>;
Ed Tanous7045c8d2017-04-03 10:04:37 -070025
Ed Tanousa24526d2018-12-10 15:17:59 -080026 std::optional<response_type> stringResponse;
Ed Tanouse0d918b2018-03-27 17:41:04 -070027
Ed Tanous1abe55e2018-09-05 08:30:59 -070028 nlohmann::json jsonValue;
Ed Tanous7045c8d2017-04-03 10:04:37 -070029
Ed Tanous39e77502019-03-04 17:35:53 -080030 void addHeader(const std::string_view key, const std::string_view value)
Ed Tanous1abe55e2018-09-05 08:30:59 -070031 {
32 stringResponse->set(key, value);
Ed Tanous7045c8d2017-04-03 10:04:37 -070033 }
Ed Tanous1abe55e2018-09-05 08:30:59 -070034
Ed Tanous39e77502019-03-04 17:35:53 -080035 void addHeader(boost::beast::http::field key, std::string_view value)
Ed Tanous1abe55e2018-09-05 08:30:59 -070036 {
37 stringResponse->set(key, value);
Ed Tanous2cd4cc12018-07-25 10:51:19 -070038 }
Ed Tanous7045c8d2017-04-03 10:04:37 -070039
Ed Tanous1abe55e2018-09-05 08:30:59 -070040 Response() : stringResponse(response_type{})
Gunnar Mills1214b7e2020-06-04 10:11:30 -050041 {}
Ed Tanous7045c8d2017-04-03 10:04:37 -070042
Ed Tanous13548d82022-07-22 09:50:44 -070043 Response(Response&& res) noexcept :
44 stringResponse(std::move(res.stringResponse)), completed(res.completed)
45 {
46 jsonValue = std::move(res.jsonValue);
47 // See note in operator= move handler for why this is needed.
48 if (!res.completed)
49 {
50 completeRequestHandler = std::move(res.completeRequestHandler);
51 res.completeRequestHandler = nullptr;
52 }
53 isAliveHelper = res.isAliveHelper;
54 res.isAliveHelper = nullptr;
55 }
56
Ed Tanousecd6a3a2022-01-07 09:18:40 -080057 ~Response() = default;
58
59 Response(const Response&) = delete;
Nan Zhou72374eb2022-01-27 17:06:51 -080060
Ed Tanous1abe55e2018-09-05 08:30:59 -070061 Response& operator=(const Response& r) = delete;
62
63 Response& operator=(Response&& r) noexcept
64 {
Nan Zhou72374eb2022-01-27 17:06:51 -080065 BMCWEB_LOG_DEBUG << "Moving response containers; this: " << this
66 << "; other: " << &r;
67 if (this == &r)
68 {
69 return *this;
70 }
Ed Tanous1abe55e2018-09-05 08:30:59 -070071 stringResponse = std::move(r.stringResponse);
72 r.stringResponse.emplace(response_type{});
73 jsonValue = std::move(r.jsonValue);
Ed Tanous13548d82022-07-22 09:50:44 -070074
75 // Only need to move completion handler if not already completed
76 // Note, there are cases where we might move out of a Response object
77 // while in a completion handler for that response object. This check
78 // is intended to prevent destructing the functor we are currently
79 // executing from in that case.
80 if (!r.completed)
81 {
82 completeRequestHandler = std::move(r.completeRequestHandler);
83 r.completeRequestHandler = nullptr;
84 }
85 else
86 {
87 completeRequestHandler = nullptr;
88 }
Ed Tanous1abe55e2018-09-05 08:30:59 -070089 completed = r.completed;
Nan Zhou72374eb2022-01-27 17:06:51 -080090 isAliveHelper = std::move(r.isAliveHelper);
Nan Zhou72374eb2022-01-27 17:06:51 -080091 r.isAliveHelper = nullptr;
Ed Tanous1abe55e2018-09-05 08:30:59 -070092 return *this;
93 }
94
95 void result(boost::beast::http::status v)
96 {
97 stringResponse->result(v);
98 }
99
Ed Tanousbb60f4d2022-06-27 10:39:09 -0700100 boost::beast::http::status result() const
Ed Tanous1abe55e2018-09-05 08:30:59 -0700101 {
102 return stringResponse->result();
103 }
104
Carson Labrado039a47e2022-04-05 16:03:20 +0000105 unsigned resultInt() const
Ed Tanous1abe55e2018-09-05 08:30:59 -0700106 {
107 return stringResponse->result_int();
108 }
109
Ed Tanousbb60f4d2022-06-27 10:39:09 -0700110 std::string_view reason() const
Ed Tanous1abe55e2018-09-05 08:30:59 -0700111 {
112 return stringResponse->reason();
113 }
114
115 bool isCompleted() const noexcept
116 {
117 return completed;
118 }
119
120 std::string& body()
121 {
122 return stringResponse->body();
123 }
124
125 void keepAlive(bool k)
126 {
127 stringResponse->keep_alive(k);
128 }
129
Ed Tanousbb60f4d2022-06-27 10:39:09 -0700130 bool keepAlive() const
Ed Tanousceac6f72018-12-02 11:58:47 -0800131 {
132 return stringResponse->keep_alive();
133 }
134
Ed Tanous1abe55e2018-09-05 08:30:59 -0700135 void preparePayload()
136 {
137 stringResponse->prepare_payload();
Ed Tanous23a21a12020-07-25 04:45:05 +0000138 }
Ed Tanous1abe55e2018-09-05 08:30:59 -0700139
140 void clear()
141 {
142 BMCWEB_LOG_DEBUG << this << " Clearing response containers";
143 stringResponse.emplace(response_type{});
144 jsonValue.clear();
145 completed = false;
146 }
147
Ed Tanous81ce6092020-12-17 16:54:55 +0000148 void write(std::string_view bodyPart)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700149 {
Ed Tanous81ce6092020-12-17 16:54:55 +0000150 stringResponse->body() += std::string(bodyPart);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700151 }
152
153 void end()
154 {
Ed Tanous89f18002022-03-24 18:38:24 -0700155 // Only set etag if this request succeeded
156 if (result() == boost::beast::http::status::ok)
157 {
158 // and the json response isn't empty
159 if (!jsonValue.empty())
160 {
161 size_t hashval = std::hash<nlohmann::json>{}(jsonValue);
162 std::string hexVal = "\"" + intToHexString(hashval, 8) + "\"";
163 addHeader(boost::beast::http::field::etag, hexVal);
164 }
165 }
Ed Tanous1abe55e2018-09-05 08:30:59 -0700166 if (completed)
167 {
Nan Zhou72374eb2022-01-27 17:06:51 -0800168 BMCWEB_LOG_ERROR << this << " Response was ended twice";
Ed Tanous1abe55e2018-09-05 08:30:59 -0700169 return;
170 }
171 completed = true;
Nan Zhou72374eb2022-01-27 17:06:51 -0800172 BMCWEB_LOG_DEBUG << this << " calling completion handler";
Ed Tanous1abe55e2018-09-05 08:30:59 -0700173 if (completeRequestHandler)
174 {
Nan Zhou72374eb2022-01-27 17:06:51 -0800175 BMCWEB_LOG_DEBUG << this << " completion handler was valid";
176 completeRequestHandler(*this);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700177 }
178 }
179
Ed Tanousbb60f4d2022-06-27 10:39:09 -0700180 bool isAlive() const
Ed Tanous1abe55e2018-09-05 08:30:59 -0700181 {
182 return isAliveHelper && isAliveHelper();
183 }
184
Nan Zhou72374eb2022-01-27 17:06:51 -0800185 void setCompleteRequestHandler(std::function<void(Response&)>&& handler)
John Edward Broadbent4147b8a2021-07-19 16:52:24 -0700186 {
Nan Zhou72374eb2022-01-27 17:06:51 -0800187 BMCWEB_LOG_DEBUG << this << " setting completion handler";
188 completeRequestHandler = std::move(handler);
Ed Tanous13548d82022-07-22 09:50:44 -0700189
190 // Now that we have a new completion handler attached, we're no longer
191 // complete
192 completed = false;
Nan Zhou72374eb2022-01-27 17:06:51 -0800193 }
194
195 std::function<void(Response&)> releaseCompleteRequestHandler()
196 {
197 BMCWEB_LOG_DEBUG << this << " releasing completion handler"
198 << static_cast<bool>(completeRequestHandler);
199 std::function<void(Response&)> ret = completeRequestHandler;
200 completeRequestHandler = nullptr;
Ed Tanous13548d82022-07-22 09:50:44 -0700201 completed = true;
Nan Zhou72374eb2022-01-27 17:06:51 -0800202 return ret;
203 }
204
205 void setIsAliveHelper(std::function<bool()>&& handler)
206 {
207 isAliveHelper = std::move(handler);
208 }
209
210 std::function<bool()> releaseIsAliveHelper()
211 {
212 std::function<bool()> ret = std::move(isAliveHelper);
213 isAliveHelper = nullptr;
214 return ret;
John Edward Broadbent4147b8a2021-07-19 16:52:24 -0700215 }
216
Ed Tanous1abe55e2018-09-05 08:30:59 -0700217 private:
Nan Zhou72374eb2022-01-27 17:06:51 -0800218 bool completed = false;
219 std::function<void(Response&)> completeRequestHandler;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700220 std::function<bool()> isAliveHelper;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700221};
Ed Tanous1abe55e2018-09-05 08:30:59 -0700222} // namespace crow