File: OneOf.hpp

package info (click to toggle)
reflect-cpp 0.18.0%2Bds-3
  • links: PTS, VCS
  • area: main
  • in suites: trixie
  • size: 12,524 kB
  • sloc: cpp: 44,484; python: 131; makefile: 30; sh: 3
file content (69 lines) | stat: -rw-r--r-- 2,193 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
#ifndef RFL_ONEOF_HPP_
#define RFL_ONEOF_HPP_

#include <sstream>
#include <string>
#include <utility>
#include <vector>

#include "Result.hpp"
#include "parsing/schema/ValidationType.hpp"

namespace rfl {

/// Requires that all of the contraints C and Cs be true.
template <class C, class... Cs>
struct OneOf {
  template <class T>
  static rfl::Result<T> validate(const T& _value) noexcept {
    return validate_impl<T, C, Cs...>(_value, {});
  }

  template <class T>
  static parsing::schema::ValidationType to_schema() {
    using ValidationType = parsing::schema::ValidationType;
    const auto types = std::vector<ValidationType>(
        {C::template to_schema<T>(), Cs::template to_schema<T>()...});
    return ValidationType{ValidationType::OneOf{.types_ = types}};
  }

 private:
  static std::string make_error_message(const std::vector<Error>& _errors) {
    std::stringstream stream;
    stream << "Expected exactly 1 out of " << sizeof...(Cs) + 1
           << " validations to pass, but " << sizeof...(Cs) + 1 - _errors.size()
           << " of them did. The following errors were generated: ";
    for (size_t i = 0; i < _errors.size(); ++i) {
      stream << "\n" << i + 1 << ") " << _errors.at(i).what();
    }
    return stream.str();
  }

  template <class T, class Head, class... Tail>
  static rfl::Result<T> validate_impl(const T& _value,
                                      std::vector<Error> _errors) {
    return Head::validate(_value)
        .and_then([&](auto&& _res) -> rfl::Result<T> {
          if constexpr (sizeof...(Tail) == 0) {
            if (_errors.size() == sizeof...(Cs)) {
              return _value;
            }
            return error(make_error_message(_errors));
          } else {
            return validate_impl<T, Tail...>(_value, std::move(_errors));
          }
        })
        .or_else([&](auto&& _err) -> rfl::Result<T> {
          _errors.emplace_back(std::move(_err));
          if constexpr (sizeof...(Tail) == 0) {
            return error(make_error_message(_errors));
          } else {
            return validate_impl<T, Tail...>(_value, std::move(_errors));
          }
        });
  }
};

}  // namespace rfl

#endif