blob: 59ddf988325acd6f1a6267869a9a59169724c6dc [file] [log] [blame]
Ed Tanous7045c8d2017-04-03 10:04:37 -07001#pragma once
Ed Tanous1abe55e2018-09-05 08:30:59 -07002
Ed Tanous3ccb3ad2023-01-13 17:40:03 -08003#include "bmcweb_config.h"
4
Ed Tanous51dae672018-09-05 16:07:32 -07005#include <openssl/crypto.h>
6
Ed Tanousc867a832022-03-10 14:17:00 -08007#include <boost/callable_traits.hpp>
Ed Tanous079360a2022-06-29 10:05:19 -07008#include <boost/url/parse.hpp>
Ed Tanouseae855c2021-10-26 11:26:02 -07009#include <boost/url/url.hpp>
Ed Tanous079360a2022-06-29 10:05:19 -070010#include <boost/url/url_view.hpp>
Ed Tanous71f2db72022-05-25 12:28:09 -070011#include <nlohmann/json.hpp>
Nan Zhou1d8782e2021-11-29 22:23:18 -080012
Ed Tanous9ea15c32022-01-04 14:18:22 -080013#include <array>
Ed Tanous74849be2021-02-05 09:47:47 -080014#include <chrono>
Ed Tanousc715ec22022-03-10 15:38:01 -080015#include <cstddef>
Ed Tanous7045c8d2017-04-03 10:04:37 -070016#include <cstdint>
Ed Tanous9ea15c32022-01-04 14:18:22 -080017#include <ctime>
Ed Tanous7045c8d2017-04-03 10:04:37 -070018#include <functional>
Ed Tanous9896eae2022-07-23 15:07:33 -070019#include <iomanip>
Ed Tanous9ea15c32022-01-04 14:18:22 -080020#include <limits>
Ed Tanous7045c8d2017-04-03 10:04:37 -070021#include <stdexcept>
22#include <string>
Ed Tanous9ea15c32022-01-04 14:18:22 -080023#include <string_view>
Ed Tanous7045c8d2017-04-03 10:04:37 -070024#include <tuple>
Ed Tanous9ea15c32022-01-04 14:18:22 -080025#include <type_traits>
26#include <utility>
Szymon Dompkeca1600c2022-03-03 14:42:52 +010027#include <variant>
Ed Tanous7045c8d2017-04-03 10:04:37 -070028
Ed Tanous1abe55e2018-09-05 08:30:59 -070029namespace crow
30{
31namespace black_magic
32{
Ed Tanous7045c8d2017-04-03 10:04:37 -070033
Ed Tanousc715ec22022-03-10 15:38:01 -080034enum class TypeCode : uint8_t
35{
36 Unspecified = 0,
Ed Tanous15a42df2023-02-09 18:08:23 -080037 String = 1,
38 Path = 2,
39 Max = 3,
Ed Tanousc715ec22022-03-10 15:38:01 -080040};
41
42// Remove when we have c++23
43template <typename E>
44constexpr typename std::underlying_type<E>::type toUnderlying(E e) noexcept
45{
46 return static_cast<typename std::underlying_type<E>::type>(e);
47}
48
Ed Tanous1c30e502022-03-08 18:02:24 -080049constexpr inline uint64_t getParameterTag(std::string_view url)
Ed Tanous1abe55e2018-09-05 08:30:59 -070050{
Ed Tanous1c30e502022-03-08 18:02:24 -080051 uint64_t tagValue = 0;
52 size_t urlSegmentIndex = std::string_view::npos;
Ed Tanousb00dcc22021-02-23 12:52:50 -080053
Ed Tanous1c30e502022-03-08 18:02:24 -080054 size_t paramIndex = 0;
55
56 for (size_t urlIndex = 0; urlIndex < url.size(); urlIndex++)
57 {
58 char character = url[urlIndex];
59 if (character == '<')
60 {
61 if (urlSegmentIndex != std::string_view::npos)
62 {
63 return 0;
64 }
65 urlSegmentIndex = urlIndex;
66 }
67 if (character == '>')
68 {
69 if (urlSegmentIndex == std::string_view::npos)
70 {
71 return 0;
72 }
Patrick Williams89492a12023-05-10 07:51:34 -050073 std::string_view tag = url.substr(urlSegmentIndex,
74 urlIndex + 1 - urlSegmentIndex);
Ed Tanous1c30e502022-03-08 18:02:24 -080075
76 // Note, this is a really lame way to do std::pow(6, paramIndex)
77 // std::pow doesn't work in constexpr in clang.
78 // Ideally in the future we'd move this to use a power of 2 packing
79 // (probably 8 instead of 6) so that these just become bit shifts
80 uint64_t insertIndex = 1;
81 for (size_t unused = 0; unused < paramIndex; unused++)
82 {
Ed Tanous15a42df2023-02-09 18:08:23 -080083 insertIndex *= 3;
Ed Tanous1c30e502022-03-08 18:02:24 -080084 }
85
Ed Tanous1c30e502022-03-08 18:02:24 -080086 if (tag == "<str>" || tag == "<string>")
87 {
Ed Tanousc715ec22022-03-10 15:38:01 -080088 tagValue += insertIndex * toUnderlying(TypeCode::String);
Ed Tanous1c30e502022-03-08 18:02:24 -080089 }
90 if (tag == "<path>")
91 {
Ed Tanousc715ec22022-03-10 15:38:01 -080092 tagValue += insertIndex * toUnderlying(TypeCode::Path);
Ed Tanous1c30e502022-03-08 18:02:24 -080093 }
94 paramIndex++;
95 urlSegmentIndex = std::string_view::npos;
96 }
97 }
98 if (urlSegmentIndex != std::string_view::npos)
Ed Tanous988403c2020-08-24 11:29:49 -070099 {
100 return 0;
101 }
Ed Tanous1c30e502022-03-08 18:02:24 -0800102 return tagValue;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700103}
Ed Tanous7045c8d2017-04-03 10:04:37 -0700104
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500105template <typename... T>
106struct S
Ed Tanous1abe55e2018-09-05 08:30:59 -0700107{
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500108 template <typename U>
109 using push = S<U, T...>;
110 template <typename U>
111 using push_back = S<T..., U>;
112 template <template <typename... Args> class U>
113 using rebind = U<T...>;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700114};
Ed Tanous988403c2020-08-24 11:29:49 -0700115
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500116template <typename F, typename Set>
117struct CallHelper;
Ed Tanous988403c2020-08-24 11:29:49 -0700118
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500119template <typename F, typename... Args>
120struct CallHelper<F, S<Args...>>
Ed Tanous1abe55e2018-09-05 08:30:59 -0700121{
122 template <typename F1, typename... Args1,
123 typename = decltype(std::declval<F1>()(std::declval<Args1>()...))>
Ed Tanous2c70f802020-09-28 14:29:23 -0700124 static char test(int);
Ed Tanous7045c8d2017-04-03 10:04:37 -0700125
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500126 template <typename...>
Ed Tanous2c70f802020-09-28 14:29:23 -0700127 static int test(...);
Ed Tanous7045c8d2017-04-03 10:04:37 -0700128
Ed Tanous2c70f802020-09-28 14:29:23 -0700129 static constexpr bool value = sizeof(test<F, Args...>(0)) == sizeof(char);
Ed Tanous7045c8d2017-04-03 10:04:37 -0700130};
131
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500132template <uint64_t Tag>
133struct Arguments
Ed Tanous1abe55e2018-09-05 08:30:59 -0700134{
Ed Tanous15a42df2023-02-09 18:08:23 -0800135 using subarguments = typename Arguments<Tag / 3>::type;
136 using type = typename subarguments::template push<std::string>;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700137};
138
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500139template <>
140struct Arguments<0>
Ed Tanous1abe55e2018-09-05 08:30:59 -0700141{
142 using type = S<>;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700143};
144
Ed Tanous1abe55e2018-09-05 08:30:59 -0700145} // namespace black_magic
Ed Tanous7045c8d2017-04-03 10:04:37 -0700146
Ed Tanous1abe55e2018-09-05 08:30:59 -0700147namespace utility
148{
Ed Tanous7045c8d2017-04-03 10:04:37 -0700149
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500150template <typename T>
Ed Tanousc867a832022-03-10 14:17:00 -0800151struct FunctionTraits
Ed Tanous1abe55e2018-09-05 08:30:59 -0700152{
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500153 template <size_t i>
Ed Tanousc867a832022-03-10 14:17:00 -0800154 using arg = std::tuple_element_t<i, boost::callable_traits::args_t<T>>;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700155};
156
Ed Tanous26ccae32023-02-16 10:28:44 -0800157inline std::string base64encode(std::string_view data)
Adriana Kobylakd830ff52021-01-27 14:15:27 -0600158{
159 const std::array<char, 64> key = {
160 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M',
161 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z',
162 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm',
163 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z',
164 '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '+', '/'};
165
166 size_t size = data.size();
167 std::string ret;
168 ret.resize((size + 2) / 3 * 4);
169 auto it = ret.begin();
170
171 size_t i = 0;
172 while (i < size)
173 {
Ed Tanous543f4402022-01-06 13:12:53 -0800174 size_t keyIndex = 0;
Adriana Kobylakd830ff52021-01-27 14:15:27 -0600175
176 keyIndex = static_cast<size_t>(data[i] & 0xFC) >> 2;
177 *it++ = key[keyIndex];
178
179 if (i + 1 < size)
180 {
181 keyIndex = static_cast<size_t>(data[i] & 0x03) << 4;
182 keyIndex += static_cast<size_t>(data[i + 1] & 0xF0) >> 4;
183 *it++ = key[keyIndex];
184
185 if (i + 2 < size)
186 {
187 keyIndex = static_cast<size_t>(data[i + 1] & 0x0F) << 2;
188 keyIndex += static_cast<size_t>(data[i + 2] & 0xC0) >> 6;
189 *it++ = key[keyIndex];
190
191 keyIndex = static_cast<size_t>(data[i + 2] & 0x3F);
192 *it++ = key[keyIndex];
193 }
194 else
195 {
196 keyIndex = static_cast<size_t>(data[i + 1] & 0x0F) << 2;
197 *it++ = key[keyIndex];
198 *it++ = '=';
199 }
200 }
201 else
202 {
203 keyIndex = static_cast<size_t>(data[i] & 0x03) << 4;
204 *it++ = key[keyIndex];
205 *it++ = '=';
206 *it++ = '=';
207 }
208
209 i += 3;
210 }
211
212 return ret;
213}
214
Borawski.Lukasz9d8fd302018-01-05 14:56:09 +0100215// TODO this is temporary and should be deleted once base64 is refactored out of
216// crow
Ed Tanous26ccae32023-02-16 10:28:44 -0800217inline bool base64Decode(std::string_view input, std::string& output)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700218{
Ed Tanous271584a2019-07-09 16:24:22 -0700219 static const char nop = static_cast<char>(-1);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700220 // See note on encoding_data[] in above function
Jonathan Doman5beaf842020-08-14 11:23:33 -0700221 static const std::array<char, 256> decodingData = {
Ed Tanous1abe55e2018-09-05 08:30:59 -0700222 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
223 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
224 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
225 nop, 62, nop, nop, nop, 63, 52, 53, 54, 55, 56, 57, 58, 59,
226 60, 61, nop, nop, nop, nop, nop, nop, nop, 0, 1, 2, 3, 4,
227 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18,
228 19, 20, 21, 22, 23, 24, 25, nop, nop, nop, nop, nop, nop, 26,
229 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40,
230 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, nop, nop, nop,
231 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
232 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
233 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
234 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
235 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
236 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
237 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
238 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
239 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
240 nop, nop, nop, nop};
Borawski.Lukasz9d8fd302018-01-05 14:56:09 +0100241
Ed Tanous1abe55e2018-09-05 08:30:59 -0700242 size_t inputLength = input.size();
Borawski.Lukasz9d8fd302018-01-05 14:56:09 +0100243
Ed Tanous1abe55e2018-09-05 08:30:59 -0700244 // allocate space for output string
245 output.clear();
246 output.reserve(((inputLength + 2) / 3) * 4);
Borawski.Lukasz9d8fd302018-01-05 14:56:09 +0100247
Jonathan Doman5beaf842020-08-14 11:23:33 -0700248 auto getCodeValue = [](char c) {
249 auto code = static_cast<unsigned char>(c);
250 // Ensure we cannot index outside the bounds of the decoding array
251 static_assert(std::numeric_limits<decltype(code)>::max() <
252 decodingData.size());
253 return decodingData[code];
254 };
255
Ed Tanous1abe55e2018-09-05 08:30:59 -0700256 // for each 4-bytes sequence from the input, extract 4 6-bits sequences by
Gunnar Millscaa3ce32020-07-08 14:46:53 -0500257 // dropping first two bits
Ed Tanous1abe55e2018-09-05 08:30:59 -0700258 // and regenerate into 3 8-bits sequences
James Feist5a806642020-07-31 16:40:33 +0000259
Ed Tanous1abe55e2018-09-05 08:30:59 -0700260 for (size_t i = 0; i < inputLength; i++)
261 {
Ed Tanous543f4402022-01-06 13:12:53 -0800262 char base64code0 = 0;
263 char base64code1 = 0;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700264 char base64code2 = 0; // initialized to 0 to suppress warnings
Borawski.Lukasz9d8fd302018-01-05 14:56:09 +0100265
Jonathan Doman5beaf842020-08-14 11:23:33 -0700266 base64code0 = getCodeValue(input[i]);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700267 if (base64code0 == nop)
268 { // non base64 character
269 return false;
270 }
271 if (!(++i < inputLength))
272 { // we need at least two input bytes for first
273 // byte output
274 return false;
275 }
Jonathan Doman5beaf842020-08-14 11:23:33 -0700276 base64code1 = getCodeValue(input[i]);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700277 if (base64code1 == nop)
278 { // non base64 character
279 return false;
280 }
281 output +=
282 static_cast<char>((base64code0 << 2) | ((base64code1 >> 4) & 0x3));
Borawski.Lukasz9d8fd302018-01-05 14:56:09 +0100283
Ed Tanous1abe55e2018-09-05 08:30:59 -0700284 if (++i < inputLength)
285 {
286 char c = input[i];
287 if (c == '=')
288 { // padding , end of input
289 return (base64code1 & 0x0f) == 0;
290 }
Jonathan Doman5beaf842020-08-14 11:23:33 -0700291 base64code2 = getCodeValue(input[i]);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700292 if (base64code2 == nop)
293 { // non base64 character
294 return false;
295 }
296 output += static_cast<char>(((base64code1 << 4) & 0xf0) |
297 ((base64code2 >> 2) & 0x0f));
298 }
299
300 if (++i < inputLength)
301 {
302 char c = input[i];
303 if (c == '=')
304 { // padding , end of input
305 return (base64code2 & 0x03) == 0;
306 }
Ed Tanousf8fe53e2022-06-30 15:55:45 -0700307 char base64code3 = getCodeValue(input[i]);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700308 if (base64code3 == nop)
309 { // non base64 character
310 return false;
311 }
312 output +=
313 static_cast<char>((((base64code2 << 6) & 0xc0) | base64code3));
314 }
Borawski.Lukasz9d8fd302018-01-05 14:56:09 +0100315 }
316
Ed Tanous1abe55e2018-09-05 08:30:59 -0700317 return true;
Borawski.Lukasz9d8fd302018-01-05 14:56:09 +0100318}
319
Ed Tanous26ccae32023-02-16 10:28:44 -0800320inline bool constantTimeStringCompare(std::string_view a, std::string_view b)
Ed Tanous51dae672018-09-05 16:07:32 -0700321{
322 // Important note, this function is ONLY constant time if the two input
323 // sizes are the same
324 if (a.size() != b.size())
325 {
326 return false;
327 }
328 return CRYPTO_memcmp(a.data(), b.data(), a.size()) == 0;
329}
330
331struct ConstantTimeCompare
332{
Ed Tanous26ccae32023-02-16 10:28:44 -0800333 bool operator()(std::string_view a, std::string_view b) const
Ed Tanous51dae672018-09-05 16:07:32 -0700334 {
335 return constantTimeStringCompare(a, b);
336 }
337};
338
Ed Tanouseae855c2021-10-26 11:26:02 -0700339namespace details
340{
341inline boost::urls::url
Willy Tuc6bcedc2022-09-27 05:36:59 +0000342 appendUrlPieces(boost::urls::url& url,
343 const std::initializer_list<std::string_view> args)
Ed Tanouseae855c2021-10-26 11:26:02 -0700344{
Ed Tanous26ccae32023-02-16 10:28:44 -0800345 for (std::string_view arg : args)
Ed Tanouseae855c2021-10-26 11:26:02 -0700346 {
347 url.segments().push_back(arg);
348 }
349 return url;
350}
Willy Tuc6bcedc2022-09-27 05:36:59 +0000351
Ed Tanouseae855c2021-10-26 11:26:02 -0700352} // namespace details
353
Ed Tanous7f8d8fa2022-08-19 07:00:38 -0700354class OrMorePaths
355{};
356
Ed Tanouseae855c2021-10-26 11:26:02 -0700357template <typename... AV>
Willy Tuc6bcedc2022-09-27 05:36:59 +0000358inline void appendUrlPieces(boost::urls::url& url, const AV... args)
359{
360 details::appendUrlPieces(url, {args...});
361}
362
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100363namespace details
364{
365
366// std::reference_wrapper<std::string> - extracts segment to variable
367// std::string_view - checks if segment is equal to variable
Ed Tanous7f8d8fa2022-08-19 07:00:38 -0700368using UrlSegment = std::variant<std::reference_wrapper<std::string>,
369 std::string_view, OrMorePaths>;
370
371enum class UrlParseResult
372{
373 Continue,
374 Fail,
375 Done,
376};
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100377
378class UrlSegmentMatcherVisitor
379{
380 public:
Ed Tanous7f8d8fa2022-08-19 07:00:38 -0700381 UrlParseResult operator()(std::string& output)
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100382 {
Ed Tanous079360a2022-06-29 10:05:19 -0700383 output = segment;
Ed Tanous7f8d8fa2022-08-19 07:00:38 -0700384 return UrlParseResult::Continue;
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100385 }
386
Ed Tanous7f8d8fa2022-08-19 07:00:38 -0700387 UrlParseResult operator()(std::string_view expected)
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100388 {
Ed Tanous079360a2022-06-29 10:05:19 -0700389 if (segment == expected)
Ed Tanous7f8d8fa2022-08-19 07:00:38 -0700390 {
391 return UrlParseResult::Continue;
392 }
393 return UrlParseResult::Fail;
394 }
395
396 UrlParseResult operator()(OrMorePaths /*unused*/)
397 {
398 return UrlParseResult::Done;
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100399 }
400
Ed Tanous079360a2022-06-29 10:05:19 -0700401 explicit UrlSegmentMatcherVisitor(std::string_view segmentIn) :
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100402 segment(segmentIn)
403 {}
404
405 private:
Ed Tanous079360a2022-06-29 10:05:19 -0700406 std::string_view segment;
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100407};
408
Ed Tanousd9f466b2023-03-06 15:04:25 -0800409inline bool readUrlSegments(boost::urls::url_view url,
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100410 std::initializer_list<UrlSegment>&& segments)
411{
Ed Tanousd9f466b2023-03-06 15:04:25 -0800412 boost::urls::segments_view urlSegments = url.segments();
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100413
Ed Tanous7f8d8fa2022-08-19 07:00:38 -0700414 if (!urlSegments.is_absolute())
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100415 {
416 return false;
417 }
418
419 boost::urls::segments_view::iterator it = urlSegments.begin();
420 boost::urls::segments_view::iterator end = urlSegments.end();
421
422 for (const auto& segment : segments)
423 {
Ed Tanous7f8d8fa2022-08-19 07:00:38 -0700424 if (it == end)
425 {
426 // If the request ends with an "any" path, this was successful
427 return std::holds_alternative<OrMorePaths>(segment);
428 }
429 UrlParseResult res = std::visit(UrlSegmentMatcherVisitor(*it), segment);
430 if (res == UrlParseResult::Done)
431 {
432 return true;
433 }
434 if (res == UrlParseResult::Fail)
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100435 {
436 return false;
437 }
438 it++;
439 }
Carson Labrado4c30e222022-06-24 22:16:00 +0000440
441 // There will be an empty segment at the end if the URI ends with a "/"
442 // e.g. /redfish/v1/Chassis/
443 if ((it != end) && urlSegments.back().empty())
444 {
445 it++;
446 }
Ed Tanous7f8d8fa2022-08-19 07:00:38 -0700447 return it == end;
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100448}
449
450} // namespace details
451
452template <typename... Args>
Ed Tanousd9f466b2023-03-06 15:04:25 -0800453inline bool readUrlSegments(boost::urls::url_view url, Args&&... args)
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100454{
Ed Tanous39662a32023-02-06 15:09:46 -0800455 return details::readUrlSegments(url, {std::forward<Args>(args)...});
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100456}
457
Ed Tanousd9f466b2023-03-06 15:04:25 -0800458inline boost::urls::url replaceUrlSegment(boost::urls::url_view urlView,
Carson Labrado1c0bb5c2022-05-18 00:12:52 +0000459 const uint replaceLoc,
Ed Tanous26ccae32023-02-16 10:28:44 -0800460 std::string_view newSegment)
Carson Labrado1c0bb5c2022-05-18 00:12:52 +0000461{
Ed Tanousd9f466b2023-03-06 15:04:25 -0800462 boost::urls::segments_view urlSegments = urlView.segments();
Carson Labrado1c0bb5c2022-05-18 00:12:52 +0000463 boost::urls::url url("/");
464
465 if (!urlSegments.is_absolute())
466 {
467 return url;
468 }
469
470 boost::urls::segments_view::iterator it = urlSegments.begin();
471 boost::urls::segments_view::iterator end = urlSegments.end();
472
473 for (uint idx = 0; it != end; it++, idx++)
474 {
475 if (idx == replaceLoc)
476 {
477 url.segments().push_back(newSegment);
478 }
479 else
480 {
481 url.segments().push_back(*it);
482 }
483 }
484
485 return url;
486}
487
Ed Tanousd9f466b2023-03-06 15:04:25 -0800488inline std::string setProtocolDefaults(boost::urls::url_view urlView)
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800489{
Ed Tanous39662a32023-02-06 15:09:46 -0800490 if (urlView.scheme() == "https")
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800491 {
492 return "https";
493 }
Ed Tanous39662a32023-02-06 15:09:46 -0800494 if (urlView.scheme() == "http")
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800495 {
496 if (bmcwebInsecureEnableHttpPushStyleEventing)
497 {
498 return "http";
499 }
500 return "";
501 }
Chicago Duan3d307082020-11-26 14:12:12 +0800502 if (urlView.scheme() == "snmp")
503 {
504 return "snmp";
505 }
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800506 return "";
507}
508
Ed Tanousd9f466b2023-03-06 15:04:25 -0800509inline uint16_t setPortDefaults(boost::urls::url_view url)
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800510{
511 uint16_t port = url.port_number();
512 if (port != 0)
513 {
514 // user picked a port already.
515 return port;
516 }
517
518 // If the user hasn't explicitly stated a port, pick one explicitly for them
519 // based on the protocol defaults
520 if (url.scheme() == "http")
521 {
522 return 80;
523 }
524 if (url.scheme() == "https")
525 {
526 return 443;
527 }
Chicago Duan3d307082020-11-26 14:12:12 +0800528 if (url.scheme() == "snmp")
529 {
530 return 162;
531 }
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800532 return 0;
533}
534
Ed Tanous11baefe2022-02-09 12:14:12 -0800535inline bool validateAndSplitUrl(std::string_view destUrl, std::string& urlProto,
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800536 std::string& host, uint16_t& port,
Ed Tanous11baefe2022-02-09 12:14:12 -0800537 std::string& path)
538{
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800539 boost::urls::result<boost::urls::url_view> url =
Ed Tanous079360a2022-06-29 10:05:19 -0700540 boost::urls::parse_uri(destUrl);
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800541 if (!url)
542 {
543 return false;
544 }
545 urlProto = setProtocolDefaults(url.value());
546 if (urlProto.empty())
Ed Tanous11baefe2022-02-09 12:14:12 -0800547 {
548 return false;
549 }
550
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800551 port = setPortDefaults(url.value());
Ed Tanous11baefe2022-02-09 12:14:12 -0800552
Ed Tanous079360a2022-06-29 10:05:19 -0700553 host = url->encoded_host();
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800554
Ed Tanous079360a2022-06-29 10:05:19 -0700555 path = url->encoded_path();
Ed Tanous11baefe2022-02-09 12:14:12 -0800556 if (path.empty())
557 {
558 path = "/";
559 }
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800560 if (url->has_fragment())
561 {
562 path += '#';
Ed Tanous079360a2022-06-29 10:05:19 -0700563 path += url->encoded_fragment();
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800564 }
565
566 if (url->has_query())
567 {
568 path += '?';
Ed Tanous079360a2022-06-29 10:05:19 -0700569 path += url->encoded_query();
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800570 }
571
Ed Tanous11baefe2022-02-09 12:14:12 -0800572 return true;
573}
574
Ed Tanous1abe55e2018-09-05 08:30:59 -0700575} // namespace utility
576} // namespace crow
Ed Tanous71f2db72022-05-25 12:28:09 -0700577
578namespace nlohmann
579{
580template <>
581struct adl_serializer<boost::urls::url>
582{
583 // nlohmann requires a specific casing to look these up in adl
584 // NOLINTNEXTLINE(readability-identifier-naming)
585 static void to_json(json& j, const boost::urls::url& url)
586 {
Ed Tanous079360a2022-06-29 10:05:19 -0700587 j = url.buffer();
Ed Tanous71f2db72022-05-25 12:28:09 -0700588 }
589};
590
591template <>
592struct adl_serializer<boost::urls::url_view>
593{
594 // NOLINTNEXTLINE(readability-identifier-naming)
Ed Tanousd9f466b2023-03-06 15:04:25 -0800595 static void to_json(json& j, boost::urls::url_view url)
Ed Tanous71f2db72022-05-25 12:28:09 -0700596 {
Ed Tanous079360a2022-06-29 10:05:19 -0700597 j = url.buffer();
Ed Tanous71f2db72022-05-25 12:28:09 -0700598 }
599};
600} // namespace nlohmann