blob: 4f5cea74fded562f3454e6c1c35aecbeba417068 [file] [log] [blame]
Ed Tanous7045c8d2017-04-03 10:04:37 -07001#pragma once
Ed Tanous1abe55e2018-09-05 08:30:59 -07002
Ed Tanouseb1c47d2022-02-09 11:47:27 -08003#include <bmcweb_config.h>
Ed Tanous51dae672018-09-05 16:07:32 -07004#include <openssl/crypto.h>
5
Ed Tanousc867a832022-03-10 14:17:00 -08006#include <boost/callable_traits.hpp>
Ed Tanous079360a2022-06-29 10:05:19 -07007#include <boost/url/parse.hpp>
Ed Tanouseae855c2021-10-26 11:26:02 -07008#include <boost/url/url.hpp>
Ed Tanous079360a2022-06-29 10:05:19 -07009#include <boost/url/url_view.hpp>
Ed Tanous71f2db72022-05-25 12:28:09 -070010#include <nlohmann/json.hpp>
Nan Zhou1d8782e2021-11-29 22:23:18 -080011
Ed Tanous9ea15c32022-01-04 14:18:22 -080012#include <array>
Ed Tanous74849be2021-02-05 09:47:47 -080013#include <chrono>
Ed Tanousc715ec22022-03-10 15:38:01 -080014#include <cstddef>
Ed Tanous7045c8d2017-04-03 10:04:37 -070015#include <cstdint>
Ed Tanous9ea15c32022-01-04 14:18:22 -080016#include <ctime>
Ed Tanous7045c8d2017-04-03 10:04:37 -070017#include <functional>
Ed Tanous9896eae2022-07-23 15:07:33 -070018#include <iomanip>
Ed Tanous9ea15c32022-01-04 14:18:22 -080019#include <limits>
Ed Tanous7045c8d2017-04-03 10:04:37 -070020#include <stdexcept>
21#include <string>
Ed Tanous9ea15c32022-01-04 14:18:22 -080022#include <string_view>
Ed Tanous7045c8d2017-04-03 10:04:37 -070023#include <tuple>
Ed Tanous9ea15c32022-01-04 14:18:22 -080024#include <type_traits>
25#include <utility>
Szymon Dompkeca1600c2022-03-03 14:42:52 +010026#include <variant>
Ed Tanous7045c8d2017-04-03 10:04:37 -070027
Ed Tanous1abe55e2018-09-05 08:30:59 -070028namespace crow
29{
30namespace black_magic
31{
Ed Tanous7045c8d2017-04-03 10:04:37 -070032
Ed Tanousc715ec22022-03-10 15:38:01 -080033enum class TypeCode : uint8_t
34{
35 Unspecified = 0,
36 Integer = 1,
37 UnsignedInteger = 2,
38 Float = 3,
39 String = 4,
40 Path = 5,
41 Max = 6,
42};
43
44// Remove when we have c++23
45template <typename E>
46constexpr typename std::underlying_type<E>::type toUnderlying(E e) noexcept
47{
48 return static_cast<typename std::underlying_type<E>::type>(e);
49}
50
Gunnar Mills1214b7e2020-06-04 10:11:30 -050051template <typename T>
Ed Tanousc715ec22022-03-10 15:38:01 -080052constexpr TypeCode getParameterTag()
Ed Tanous1abe55e2018-09-05 08:30:59 -070053{
Ed Tanous69509012019-10-24 16:53:05 -070054 if constexpr (std::is_same_v<int, T>)
55 {
Ed Tanousc715ec22022-03-10 15:38:01 -080056 return TypeCode::Integer;
Ed Tanous1abe55e2018-09-05 08:30:59 -070057 }
Ed Tanous69509012019-10-24 16:53:05 -070058 if constexpr (std::is_same_v<char, T>)
59 {
Ed Tanousc715ec22022-03-10 15:38:01 -080060 return TypeCode::Integer;
Ed Tanous69509012019-10-24 16:53:05 -070061 }
62 if constexpr (std::is_same_v<short, T>)
63 {
Ed Tanousc715ec22022-03-10 15:38:01 -080064 return TypeCode::Integer;
Ed Tanous69509012019-10-24 16:53:05 -070065 }
66 if constexpr (std::is_same_v<long, T>)
67 {
Ed Tanousc715ec22022-03-10 15:38:01 -080068 return TypeCode::Integer;
Ed Tanous69509012019-10-24 16:53:05 -070069 }
70 if constexpr (std::is_same_v<long long, T>)
71 {
Ed Tanousc715ec22022-03-10 15:38:01 -080072 return TypeCode::Integer;
Ed Tanous69509012019-10-24 16:53:05 -070073 }
74 if constexpr (std::is_same_v<unsigned int, T>)
75 {
Ed Tanousc715ec22022-03-10 15:38:01 -080076 return TypeCode::UnsignedInteger;
Ed Tanous69509012019-10-24 16:53:05 -070077 }
78 if constexpr (std::is_same_v<unsigned char, T>)
79 {
Ed Tanousc715ec22022-03-10 15:38:01 -080080 return TypeCode::UnsignedInteger;
Ed Tanous69509012019-10-24 16:53:05 -070081 }
82 if constexpr (std::is_same_v<unsigned short, T>)
83 {
Ed Tanousc715ec22022-03-10 15:38:01 -080084 return TypeCode::UnsignedInteger;
Ed Tanous69509012019-10-24 16:53:05 -070085 }
86 if constexpr (std::is_same_v<unsigned long, T>)
87 {
Ed Tanousc715ec22022-03-10 15:38:01 -080088 return TypeCode::UnsignedInteger;
Ed Tanous69509012019-10-24 16:53:05 -070089 }
90 if constexpr (std::is_same_v<unsigned long long, T>)
91 {
Ed Tanousc715ec22022-03-10 15:38:01 -080092 return TypeCode::UnsignedInteger;
Ed Tanous69509012019-10-24 16:53:05 -070093 }
94 if constexpr (std::is_same_v<double, T>)
95 {
Ed Tanousc715ec22022-03-10 15:38:01 -080096 return TypeCode::Float;
Ed Tanous69509012019-10-24 16:53:05 -070097 }
98 if constexpr (std::is_same_v<std::string, T>)
99 {
Ed Tanousc715ec22022-03-10 15:38:01 -0800100 return TypeCode::String;
Ed Tanous69509012019-10-24 16:53:05 -0700101 }
Ed Tanousc715ec22022-03-10 15:38:01 -0800102 return TypeCode::Unspecified;
Ed Tanous69509012019-10-24 16:53:05 -0700103}
104
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500105template <typename... Args>
Ed Tanous988403c2020-08-24 11:29:49 -0700106struct computeParameterTagFromArgsList;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700107
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500108template <>
Ed Tanous988403c2020-08-24 11:29:49 -0700109struct computeParameterTagFromArgsList<>
Ed Tanous1abe55e2018-09-05 08:30:59 -0700110{
Ed Tanous69509012019-10-24 16:53:05 -0700111 static constexpr int value = 0;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700112};
113
114template <typename Arg, typename... Args>
Ed Tanous988403c2020-08-24 11:29:49 -0700115struct computeParameterTagFromArgsList<Arg, Args...>
Ed Tanous1abe55e2018-09-05 08:30:59 -0700116{
Ed Tanous69509012019-10-24 16:53:05 -0700117 static constexpr int subValue =
Ed Tanous988403c2020-08-24 11:29:49 -0700118 computeParameterTagFromArgsList<Args...>::value;
Ed Tanous69509012019-10-24 16:53:05 -0700119 static constexpr int value =
Ed Tanousc715ec22022-03-10 15:38:01 -0800120 getParameterTag<typename std::decay<Arg>::type>() !=
121 TypeCode::Unspecified
122 ? static_cast<unsigned long>(subValue *
123 toUnderlying(TypeCode::Max)) +
124 static_cast<uint64_t>(
125 getParameterTag<typename std::decay<Arg>::type>())
Ed Tanous1abe55e2018-09-05 08:30:59 -0700126 : subValue;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700127};
128
Ed Tanous988403c2020-08-24 11:29:49 -0700129inline bool isParameterTagCompatible(uint64_t a, uint64_t b)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700130{
Ed Tanous1c30e502022-03-08 18:02:24 -0800131 while (true)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700132 {
Ed Tanousef641b62022-06-28 19:53:24 -0700133 if (a == 0 && b == 0)
Ed Tanous1c30e502022-03-08 18:02:24 -0800134 {
Ed Tanousef641b62022-06-28 19:53:24 -0700135 // Both tags were equivalent, parameters are compatible
136 return true;
Ed Tanous1c30e502022-03-08 18:02:24 -0800137 }
Ed Tanousef641b62022-06-28 19:53:24 -0700138 if (a == 0 || b == 0)
Ed Tanous1c30e502022-03-08 18:02:24 -0800139 {
Ed Tanousef641b62022-06-28 19:53:24 -0700140 // one of the tags had more parameters than the other
141 return false;
Ed Tanous1c30e502022-03-08 18:02:24 -0800142 }
Ed Tanousc715ec22022-03-10 15:38:01 -0800143 TypeCode sa = static_cast<TypeCode>(a % toUnderlying(TypeCode::Max));
144 TypeCode sb = static_cast<TypeCode>(b % toUnderlying(TypeCode::Max));
145
146 if (sa == TypeCode::Path)
Ed Tanous1c30e502022-03-08 18:02:24 -0800147 {
Ed Tanousc715ec22022-03-10 15:38:01 -0800148 sa = TypeCode::String;
Ed Tanous1c30e502022-03-08 18:02:24 -0800149 }
Ed Tanousc715ec22022-03-10 15:38:01 -0800150 if (sb == TypeCode::Path)
Ed Tanous1c30e502022-03-08 18:02:24 -0800151 {
Ed Tanousc715ec22022-03-10 15:38:01 -0800152 sb = TypeCode::String;
Ed Tanous1c30e502022-03-08 18:02:24 -0800153 }
154 if (sa != sb)
155 {
156 return false;
157 }
Ed Tanousc715ec22022-03-10 15:38:01 -0800158 a /= toUnderlying(TypeCode::Max);
159 b /= toUnderlying(TypeCode::Max);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700160 }
Ed Tanous1c30e502022-03-08 18:02:24 -0800161 return false;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700162}
163
Ed Tanous1c30e502022-03-08 18:02:24 -0800164constexpr inline uint64_t getParameterTag(std::string_view url)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700165{
Ed Tanous1c30e502022-03-08 18:02:24 -0800166 uint64_t tagValue = 0;
167 size_t urlSegmentIndex = std::string_view::npos;
Ed Tanousb00dcc22021-02-23 12:52:50 -0800168
Ed Tanous1c30e502022-03-08 18:02:24 -0800169 size_t paramIndex = 0;
170
171 for (size_t urlIndex = 0; urlIndex < url.size(); urlIndex++)
172 {
173 char character = url[urlIndex];
174 if (character == '<')
175 {
176 if (urlSegmentIndex != std::string_view::npos)
177 {
178 return 0;
179 }
180 urlSegmentIndex = urlIndex;
181 }
182 if (character == '>')
183 {
184 if (urlSegmentIndex == std::string_view::npos)
185 {
186 return 0;
187 }
188 std::string_view tag =
189 url.substr(urlSegmentIndex, urlIndex + 1 - urlSegmentIndex);
190
191 // Note, this is a really lame way to do std::pow(6, paramIndex)
192 // std::pow doesn't work in constexpr in clang.
193 // Ideally in the future we'd move this to use a power of 2 packing
194 // (probably 8 instead of 6) so that these just become bit shifts
195 uint64_t insertIndex = 1;
196 for (size_t unused = 0; unused < paramIndex; unused++)
197 {
198 insertIndex *= 6;
199 }
200
201 if (tag == "<int>")
202 {
Ed Tanousc715ec22022-03-10 15:38:01 -0800203 tagValue += insertIndex * toUnderlying(TypeCode::Integer);
Ed Tanous1c30e502022-03-08 18:02:24 -0800204 }
205 if (tag == "<uint>")
206 {
Ed Tanousc715ec22022-03-10 15:38:01 -0800207 tagValue +=
208 insertIndex * toUnderlying(TypeCode::UnsignedInteger);
Ed Tanous1c30e502022-03-08 18:02:24 -0800209 }
210 if (tag == "<float>" || tag == "<double>")
211 {
Ed Tanousc715ec22022-03-10 15:38:01 -0800212 tagValue += insertIndex * toUnderlying(TypeCode::Float);
Ed Tanous1c30e502022-03-08 18:02:24 -0800213 }
214 if (tag == "<str>" || tag == "<string>")
215 {
Ed Tanousc715ec22022-03-10 15:38:01 -0800216 tagValue += insertIndex * toUnderlying(TypeCode::String);
Ed Tanous1c30e502022-03-08 18:02:24 -0800217 }
218 if (tag == "<path>")
219 {
Ed Tanousc715ec22022-03-10 15:38:01 -0800220 tagValue += insertIndex * toUnderlying(TypeCode::Path);
Ed Tanous1c30e502022-03-08 18:02:24 -0800221 }
222 paramIndex++;
223 urlSegmentIndex = std::string_view::npos;
224 }
225 }
226 if (urlSegmentIndex != std::string_view::npos)
Ed Tanous988403c2020-08-24 11:29:49 -0700227 {
228 return 0;
229 }
Ed Tanous1c30e502022-03-08 18:02:24 -0800230 return tagValue;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700231}
Ed Tanous7045c8d2017-04-03 10:04:37 -0700232
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500233template <typename... T>
234struct S
Ed Tanous1abe55e2018-09-05 08:30:59 -0700235{
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500236 template <typename U>
237 using push = S<U, T...>;
238 template <typename U>
239 using push_back = S<T..., U>;
240 template <template <typename... Args> class U>
241 using rebind = U<T...>;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700242};
Ed Tanous988403c2020-08-24 11:29:49 -0700243
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500244template <typename F, typename Set>
245struct CallHelper;
Ed Tanous988403c2020-08-24 11:29:49 -0700246
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500247template <typename F, typename... Args>
248struct CallHelper<F, S<Args...>>
Ed Tanous1abe55e2018-09-05 08:30:59 -0700249{
250 template <typename F1, typename... Args1,
251 typename = decltype(std::declval<F1>()(std::declval<Args1>()...))>
Ed Tanous2c70f802020-09-28 14:29:23 -0700252 static char test(int);
Ed Tanous7045c8d2017-04-03 10:04:37 -0700253
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500254 template <typename...>
Ed Tanous2c70f802020-09-28 14:29:23 -0700255 static int test(...);
Ed Tanous7045c8d2017-04-03 10:04:37 -0700256
Ed Tanous2c70f802020-09-28 14:29:23 -0700257 static constexpr bool value = sizeof(test<F, Args...>(0)) == sizeof(char);
Ed Tanous7045c8d2017-04-03 10:04:37 -0700258};
259
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500260template <uint64_t N>
261struct SingleTagToType
262{};
Ed Tanous7045c8d2017-04-03 10:04:37 -0700263
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500264template <>
265struct SingleTagToType<1>
Ed Tanous1abe55e2018-09-05 08:30:59 -0700266{
267 using type = int64_t;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700268};
269
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500270template <>
271struct SingleTagToType<2>
Ed Tanous1abe55e2018-09-05 08:30:59 -0700272{
273 using type = uint64_t;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700274};
275
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500276template <>
277struct SingleTagToType<3>
Ed Tanous1abe55e2018-09-05 08:30:59 -0700278{
279 using type = double;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700280};
281
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500282template <>
283struct SingleTagToType<4>
Ed Tanous1abe55e2018-09-05 08:30:59 -0700284{
285 using type = std::string;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700286};
287
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500288template <>
289struct SingleTagToType<5>
Ed Tanous1abe55e2018-09-05 08:30:59 -0700290{
291 using type = std::string;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700292};
293
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500294template <uint64_t Tag>
295struct Arguments
Ed Tanous1abe55e2018-09-05 08:30:59 -0700296{
297 using subarguments = typename Arguments<Tag / 6>::type;
298 using type = typename subarguments::template push<
299 typename SingleTagToType<Tag % 6>::type>;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700300};
301
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500302template <>
303struct Arguments<0>
Ed Tanous1abe55e2018-09-05 08:30:59 -0700304{
305 using type = S<>;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700306};
307
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500308template <typename T>
Ed Tanous988403c2020-08-24 11:29:49 -0700309struct Promote
Ed Tanous1abe55e2018-09-05 08:30:59 -0700310{
311 using type = T;
312};
313
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500314template <typename T>
Ed Tanous988403c2020-08-24 11:29:49 -0700315using PromoteT = typename Promote<T>::type;
316
317template <>
318struct Promote<char>
319{
320 using type = int64_t;
321};
322template <>
323struct Promote<short>
324{
325 using type = int64_t;
326};
327template <>
328struct Promote<int>
329{
330 using type = int64_t;
331};
332template <>
333struct Promote<long>
334{
335 using type = int64_t;
336};
337template <>
338struct Promote<long long>
339{
340 using type = int64_t;
341};
342template <>
343struct Promote<unsigned char>
344{
345 using type = uint64_t;
346};
347template <>
348struct Promote<unsigned short>
349{
350 using type = uint64_t;
351};
352template <>
353struct Promote<unsigned int>
354{
355 using type = uint64_t;
356};
357template <>
358struct Promote<unsigned long>
359{
360 using type = uint64_t;
361};
362template <>
363struct Promote<unsigned long long>
364{
365 using type = uint64_t;
366};
Ed Tanous7045c8d2017-04-03 10:04:37 -0700367
Ed Tanous1abe55e2018-09-05 08:30:59 -0700368} // namespace black_magic
Ed Tanous7045c8d2017-04-03 10:04:37 -0700369
Ed Tanous1abe55e2018-09-05 08:30:59 -0700370namespace utility
371{
Ed Tanous7045c8d2017-04-03 10:04:37 -0700372
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500373template <typename T>
Ed Tanousc867a832022-03-10 14:17:00 -0800374struct FunctionTraits
Ed Tanous1abe55e2018-09-05 08:30:59 -0700375{
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500376 template <size_t i>
Ed Tanousc867a832022-03-10 14:17:00 -0800377 using arg = std::tuple_element_t<i, boost::callable_traits::args_t<T>>;
Ed Tanous7045c8d2017-04-03 10:04:37 -0700378};
379
Adriana Kobylakd830ff52021-01-27 14:15:27 -0600380inline std::string base64encode(const std::string_view data)
381{
382 const std::array<char, 64> key = {
383 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M',
384 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z',
385 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm',
386 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z',
387 '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '+', '/'};
388
389 size_t size = data.size();
390 std::string ret;
391 ret.resize((size + 2) / 3 * 4);
392 auto it = ret.begin();
393
394 size_t i = 0;
395 while (i < size)
396 {
Ed Tanous543f4402022-01-06 13:12:53 -0800397 size_t keyIndex = 0;
Adriana Kobylakd830ff52021-01-27 14:15:27 -0600398
399 keyIndex = static_cast<size_t>(data[i] & 0xFC) >> 2;
400 *it++ = key[keyIndex];
401
402 if (i + 1 < size)
403 {
404 keyIndex = static_cast<size_t>(data[i] & 0x03) << 4;
405 keyIndex += static_cast<size_t>(data[i + 1] & 0xF0) >> 4;
406 *it++ = key[keyIndex];
407
408 if (i + 2 < size)
409 {
410 keyIndex = static_cast<size_t>(data[i + 1] & 0x0F) << 2;
411 keyIndex += static_cast<size_t>(data[i + 2] & 0xC0) >> 6;
412 *it++ = key[keyIndex];
413
414 keyIndex = static_cast<size_t>(data[i + 2] & 0x3F);
415 *it++ = key[keyIndex];
416 }
417 else
418 {
419 keyIndex = static_cast<size_t>(data[i + 1] & 0x0F) << 2;
420 *it++ = key[keyIndex];
421 *it++ = '=';
422 }
423 }
424 else
425 {
426 keyIndex = static_cast<size_t>(data[i] & 0x03) << 4;
427 *it++ = key[keyIndex];
428 *it++ = '=';
429 *it++ = '=';
430 }
431
432 i += 3;
433 }
434
435 return ret;
436}
437
Borawski.Lukasz9d8fd302018-01-05 14:56:09 +0100438// TODO this is temporary and should be deleted once base64 is refactored out of
439// crow
Ed Tanous39e77502019-03-04 17:35:53 -0800440inline bool base64Decode(const std::string_view input, std::string& output)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700441{
Ed Tanous271584a2019-07-09 16:24:22 -0700442 static const char nop = static_cast<char>(-1);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700443 // See note on encoding_data[] in above function
Jonathan Doman5beaf842020-08-14 11:23:33 -0700444 static const std::array<char, 256> decodingData = {
Ed Tanous1abe55e2018-09-05 08:30:59 -0700445 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
446 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
447 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
448 nop, 62, nop, nop, nop, 63, 52, 53, 54, 55, 56, 57, 58, 59,
449 60, 61, nop, nop, nop, nop, nop, nop, nop, 0, 1, 2, 3, 4,
450 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18,
451 19, 20, 21, 22, 23, 24, 25, nop, nop, nop, nop, nop, nop, 26,
452 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40,
453 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, nop, nop, nop,
454 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
455 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
456 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
457 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
458 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
459 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
460 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
461 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
462 nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop, nop,
463 nop, nop, nop, nop};
Borawski.Lukasz9d8fd302018-01-05 14:56:09 +0100464
Ed Tanous1abe55e2018-09-05 08:30:59 -0700465 size_t inputLength = input.size();
Borawski.Lukasz9d8fd302018-01-05 14:56:09 +0100466
Ed Tanous1abe55e2018-09-05 08:30:59 -0700467 // allocate space for output string
468 output.clear();
469 output.reserve(((inputLength + 2) / 3) * 4);
Borawski.Lukasz9d8fd302018-01-05 14:56:09 +0100470
Jonathan Doman5beaf842020-08-14 11:23:33 -0700471 auto getCodeValue = [](char c) {
472 auto code = static_cast<unsigned char>(c);
473 // Ensure we cannot index outside the bounds of the decoding array
474 static_assert(std::numeric_limits<decltype(code)>::max() <
475 decodingData.size());
476 return decodingData[code];
477 };
478
Ed Tanous1abe55e2018-09-05 08:30:59 -0700479 // for each 4-bytes sequence from the input, extract 4 6-bits sequences by
Gunnar Millscaa3ce32020-07-08 14:46:53 -0500480 // dropping first two bits
Ed Tanous1abe55e2018-09-05 08:30:59 -0700481 // and regenerate into 3 8-bits sequences
James Feist5a806642020-07-31 16:40:33 +0000482
Ed Tanous1abe55e2018-09-05 08:30:59 -0700483 for (size_t i = 0; i < inputLength; i++)
484 {
Ed Tanous543f4402022-01-06 13:12:53 -0800485 char base64code0 = 0;
486 char base64code1 = 0;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700487 char base64code2 = 0; // initialized to 0 to suppress warnings
Ed Tanous543f4402022-01-06 13:12:53 -0800488 char base64code3 = 0;
Borawski.Lukasz9d8fd302018-01-05 14:56:09 +0100489
Jonathan Doman5beaf842020-08-14 11:23:33 -0700490 base64code0 = getCodeValue(input[i]);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700491 if (base64code0 == nop)
492 { // non base64 character
493 return false;
494 }
495 if (!(++i < inputLength))
496 { // we need at least two input bytes for first
497 // byte output
498 return false;
499 }
Jonathan Doman5beaf842020-08-14 11:23:33 -0700500 base64code1 = getCodeValue(input[i]);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700501 if (base64code1 == nop)
502 { // non base64 character
503 return false;
504 }
505 output +=
506 static_cast<char>((base64code0 << 2) | ((base64code1 >> 4) & 0x3));
Borawski.Lukasz9d8fd302018-01-05 14:56:09 +0100507
Ed Tanous1abe55e2018-09-05 08:30:59 -0700508 if (++i < inputLength)
509 {
510 char c = input[i];
511 if (c == '=')
512 { // padding , end of input
513 return (base64code1 & 0x0f) == 0;
514 }
Jonathan Doman5beaf842020-08-14 11:23:33 -0700515 base64code2 = getCodeValue(input[i]);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700516 if (base64code2 == nop)
517 { // non base64 character
518 return false;
519 }
520 output += static_cast<char>(((base64code1 << 4) & 0xf0) |
521 ((base64code2 >> 2) & 0x0f));
522 }
523
524 if (++i < inputLength)
525 {
526 char c = input[i];
527 if (c == '=')
528 { // padding , end of input
529 return (base64code2 & 0x03) == 0;
530 }
Jonathan Doman5beaf842020-08-14 11:23:33 -0700531 base64code3 = getCodeValue(input[i]);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700532 if (base64code3 == nop)
533 { // non base64 character
534 return false;
535 }
536 output +=
537 static_cast<char>((((base64code2 << 6) & 0xc0) | base64code3));
538 }
Borawski.Lukasz9d8fd302018-01-05 14:56:09 +0100539 }
540
Ed Tanous1abe55e2018-09-05 08:30:59 -0700541 return true;
Borawski.Lukasz9d8fd302018-01-05 14:56:09 +0100542}
543
Ed Tanous51dae672018-09-05 16:07:32 -0700544inline bool constantTimeStringCompare(const std::string_view a,
545 const std::string_view b)
546{
547 // Important note, this function is ONLY constant time if the two input
548 // sizes are the same
549 if (a.size() != b.size())
550 {
551 return false;
552 }
553 return CRYPTO_memcmp(a.data(), b.data(), a.size()) == 0;
554}
555
556struct ConstantTimeCompare
557{
558 bool operator()(const std::string_view a, const std::string_view b) const
559 {
560 return constantTimeStringCompare(a, b);
561 }
562};
563
Ed Tanouseae855c2021-10-26 11:26:02 -0700564namespace details
565{
566inline boost::urls::url
Willy Tuc6bcedc2022-09-27 05:36:59 +0000567 appendUrlPieces(boost::urls::url& url,
568 const std::initializer_list<std::string_view> args)
Ed Tanouseae855c2021-10-26 11:26:02 -0700569{
Ed Tanouseae855c2021-10-26 11:26:02 -0700570 for (const std::string_view& arg : args)
571 {
572 url.segments().push_back(arg);
573 }
574 return url;
575}
Willy Tuc6bcedc2022-09-27 05:36:59 +0000576
577inline boost::urls::url
578 urlFromPiecesDetail(const std::initializer_list<std::string_view> args)
579{
580 boost::urls::url url("/");
581 appendUrlPieces(url, args);
582 return url;
583}
Ed Tanouseae855c2021-10-26 11:26:02 -0700584} // namespace details
585
Ed Tanous7f8d8fa2022-08-19 07:00:38 -0700586class OrMorePaths
587{};
588
Ed Tanouseae855c2021-10-26 11:26:02 -0700589template <typename... AV>
590inline boost::urls::url urlFromPieces(const AV... args)
591{
592 return details::urlFromPiecesDetail({args...});
593}
594
Willy Tuc6bcedc2022-09-27 05:36:59 +0000595template <typename... AV>
596inline void appendUrlPieces(boost::urls::url& url, const AV... args)
597{
598 details::appendUrlPieces(url, {args...});
599}
600
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100601namespace details
602{
603
604// std::reference_wrapper<std::string> - extracts segment to variable
605// std::string_view - checks if segment is equal to variable
Ed Tanous7f8d8fa2022-08-19 07:00:38 -0700606using UrlSegment = std::variant<std::reference_wrapper<std::string>,
607 std::string_view, OrMorePaths>;
608
609enum class UrlParseResult
610{
611 Continue,
612 Fail,
613 Done,
614};
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100615
616class UrlSegmentMatcherVisitor
617{
618 public:
Ed Tanous7f8d8fa2022-08-19 07:00:38 -0700619 UrlParseResult operator()(std::string& output)
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100620 {
Ed Tanous079360a2022-06-29 10:05:19 -0700621 output = segment;
Ed Tanous7f8d8fa2022-08-19 07:00:38 -0700622 return UrlParseResult::Continue;
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100623 }
624
Ed Tanous7f8d8fa2022-08-19 07:00:38 -0700625 UrlParseResult operator()(std::string_view expected)
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100626 {
Ed Tanous079360a2022-06-29 10:05:19 -0700627 if (segment == expected)
Ed Tanous7f8d8fa2022-08-19 07:00:38 -0700628 {
629 return UrlParseResult::Continue;
630 }
631 return UrlParseResult::Fail;
632 }
633
634 UrlParseResult operator()(OrMorePaths /*unused*/)
635 {
636 return UrlParseResult::Done;
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100637 }
638
Ed Tanous079360a2022-06-29 10:05:19 -0700639 explicit UrlSegmentMatcherVisitor(std::string_view segmentIn) :
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100640 segment(segmentIn)
641 {}
642
643 private:
Ed Tanous079360a2022-06-29 10:05:19 -0700644 std::string_view segment;
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100645};
646
647inline bool readUrlSegments(const boost::urls::url_view& urlView,
648 std::initializer_list<UrlSegment>&& segments)
649{
650 const boost::urls::segments_view& urlSegments = urlView.segments();
651
Ed Tanous7f8d8fa2022-08-19 07:00:38 -0700652 if (!urlSegments.is_absolute())
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100653 {
654 return false;
655 }
656
657 boost::urls::segments_view::iterator it = urlSegments.begin();
658 boost::urls::segments_view::iterator end = urlSegments.end();
659
660 for (const auto& segment : segments)
661 {
Ed Tanous7f8d8fa2022-08-19 07:00:38 -0700662 if (it == end)
663 {
664 // If the request ends with an "any" path, this was successful
665 return std::holds_alternative<OrMorePaths>(segment);
666 }
667 UrlParseResult res = std::visit(UrlSegmentMatcherVisitor(*it), segment);
668 if (res == UrlParseResult::Done)
669 {
670 return true;
671 }
672 if (res == UrlParseResult::Fail)
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100673 {
674 return false;
675 }
676 it++;
677 }
Carson Labrado4c30e222022-06-24 22:16:00 +0000678
679 // There will be an empty segment at the end if the URI ends with a "/"
680 // e.g. /redfish/v1/Chassis/
681 if ((it != end) && urlSegments.back().empty())
682 {
683 it++;
684 }
Ed Tanous7f8d8fa2022-08-19 07:00:38 -0700685 return it == end;
Szymon Dompkeca1600c2022-03-03 14:42:52 +0100686}
687
688} // namespace details
689
690template <typename... Args>
691inline bool readUrlSegments(const boost::urls::url_view& urlView,
692 Args&&... args)
693{
694 return details::readUrlSegments(urlView, {std::forward<Args>(args)...});
695}
696
Carson Labrado1c0bb5c2022-05-18 00:12:52 +0000697inline boost::urls::url replaceUrlSegment(const boost::urls::url_view& urlView,
698 const uint replaceLoc,
699 const std::string_view newSegment)
700{
701 const boost::urls::segments_view& urlSegments = urlView.segments();
702 boost::urls::url url("/");
703
704 if (!urlSegments.is_absolute())
705 {
706 return url;
707 }
708
709 boost::urls::segments_view::iterator it = urlSegments.begin();
710 boost::urls::segments_view::iterator end = urlSegments.end();
711
712 for (uint idx = 0; it != end; it++, idx++)
713 {
714 if (idx == replaceLoc)
715 {
716 url.segments().push_back(newSegment);
717 }
718 else
719 {
720 url.segments().push_back(*it);
721 }
722 }
723
724 return url;
725}
726
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800727inline std::string setProtocolDefaults(const boost::urls::url_view& url)
728{
729 if (url.scheme() == "https")
730 {
731 return "https";
732 }
733 if (url.scheme() == "http")
734 {
735 if (bmcwebInsecureEnableHttpPushStyleEventing)
736 {
737 return "http";
738 }
739 return "";
740 }
741 return "";
742}
743
744inline uint16_t setPortDefaults(const boost::urls::url_view& url)
745{
746 uint16_t port = url.port_number();
747 if (port != 0)
748 {
749 // user picked a port already.
750 return port;
751 }
752
753 // If the user hasn't explicitly stated a port, pick one explicitly for them
754 // based on the protocol defaults
755 if (url.scheme() == "http")
756 {
757 return 80;
758 }
759 if (url.scheme() == "https")
760 {
761 return 443;
762 }
763 return 0;
764}
765
Ed Tanous11baefe2022-02-09 12:14:12 -0800766inline bool validateAndSplitUrl(std::string_view destUrl, std::string& urlProto,
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800767 std::string& host, uint16_t& port,
Ed Tanous11baefe2022-02-09 12:14:12 -0800768 std::string& path)
769{
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800770 boost::urls::result<boost::urls::url_view> url =
Ed Tanous079360a2022-06-29 10:05:19 -0700771 boost::urls::parse_uri(destUrl);
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800772 if (!url)
773 {
774 return false;
775 }
776 urlProto = setProtocolDefaults(url.value());
777 if (urlProto.empty())
Ed Tanous11baefe2022-02-09 12:14:12 -0800778 {
779 return false;
780 }
781
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800782 port = setPortDefaults(url.value());
Ed Tanous11baefe2022-02-09 12:14:12 -0800783
Ed Tanous079360a2022-06-29 10:05:19 -0700784 host = url->encoded_host();
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800785
Ed Tanous079360a2022-06-29 10:05:19 -0700786 path = url->encoded_path();
Ed Tanous11baefe2022-02-09 12:14:12 -0800787 if (path.empty())
788 {
789 path = "/";
790 }
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800791 if (url->has_fragment())
792 {
793 path += '#';
Ed Tanous079360a2022-06-29 10:05:19 -0700794 path += url->encoded_fragment();
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800795 }
796
797 if (url->has_query())
798 {
799 path += '?';
Ed Tanous079360a2022-06-29 10:05:19 -0700800 path += url->encoded_query();
Ed Tanouseb1c47d2022-02-09 11:47:27 -0800801 }
802
Ed Tanous11baefe2022-02-09 12:14:12 -0800803 return true;
804}
805
Ed Tanous1abe55e2018-09-05 08:30:59 -0700806} // namespace utility
807} // namespace crow
Ed Tanous71f2db72022-05-25 12:28:09 -0700808
809namespace nlohmann
810{
811template <>
812struct adl_serializer<boost::urls::url>
813{
814 // nlohmann requires a specific casing to look these up in adl
815 // NOLINTNEXTLINE(readability-identifier-naming)
816 static void to_json(json& j, const boost::urls::url& url)
817 {
Ed Tanous079360a2022-06-29 10:05:19 -0700818 j = url.buffer();
Ed Tanous71f2db72022-05-25 12:28:09 -0700819 }
820};
821
822template <>
823struct adl_serializer<boost::urls::url_view>
824{
825 // NOLINTNEXTLINE(readability-identifier-naming)
826 static void to_json(json& j, const boost::urls::url_view& url)
827 {
Ed Tanous079360a2022-06-29 10:05:19 -0700828 j = url.buffer();
Ed Tanous71f2db72022-05-25 12:28:09 -0700829 }
830};
831} // namespace nlohmann