Echo Writes Code

errors.cpp

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
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
#include "crucible/parsing/errors.inl"

#include "crucible/core/format.inl"

namespace crucible::parsing::errors {
  namespace {
    struct ParsingErrorDescribeVisitor final {
      auto operator()(AmbiguousMatch const &) const -> std::string {
        return "Ambiguous match";
      }

      auto operator()(UnexpectedInput const &) const -> std::string {
        return "Unexpected input";
      }

      auto operator()(UnexpectedEndOfInput const &) const -> std::string {
        return "Unexpected end of input";
      }

      auto operator()(ExpectedExactBytes const &e) const -> std::string {
        std::string result { "Expected exact sequence of bytes: " };

        auto current { e.bytes.begin() };
        auto const end { e.bytes.end() };

        if (current != end) {
          result += core::format::format_unsigned_hexadecimal(std::to_integer<std::uint8_t>(*current));
          ++current;
        }

        while (current != end) {
          result += " ";
          result += core::format::format_unsigned_hexadecimal(std::to_integer<std::uint8_t>(*current));
          ++current;
        }

        return result;
      }

      auto operator()(InsufficientInput const &e) const -> std::string {
        return "Expected " + std::to_string(e.count) + " more bytes of input";
      }
    };

    struct ParsingErrorFormatVisitor final {
      auto operator()(AmbiguousMatch const &) const -> std::string {
        return "AmbiguousMatch {}";
      }

      auto operator()(UnexpectedInput const &) const -> std::string {
        return "UnexpectedInput {}";
      }

      auto operator()(UnexpectedEndOfInput const &) const -> std::string {
        return "UnexpectedEndOfInput {}";
      }

      auto operator()(ExpectedExactBytes const &e) const -> std::string {
        return "ExpectedExactBytes { " + core::format::format_anything(e.bytes) + " }";
      }

      auto operator()(InsufficientInput const &e) const -> std::string {
        return "InsufficientInput { " + core::format::format_anything(e.count) + " }";
      }
    };

    struct ParsingErrorSourceVisitor final {
      auto operator()(AmbiguousMatch const &) const -> std::optional<std::reference_wrapper<core::errors::AbstractError const>> {
        return std::nullopt;
      }

      auto operator()(UnexpectedInput const &) const -> std::optional<std::reference_wrapper<core::errors::AbstractError const>> {
        return std::nullopt;
      }

      auto operator()(UnexpectedEndOfInput const &) const -> std::optional<std::reference_wrapper<core::errors::AbstractError const>> {
        return std::nullopt;
      }

      auto operator()(ExpectedExactBytes const &) const -> std::optional<std::reference_wrapper<core::errors::AbstractError const>> {
        return std::nullopt;
      }

      auto operator()(InsufficientInput const &) const -> std::optional<std::reference_wrapper<core::errors::AbstractError const>> {
        return std::nullopt;
      }
    };
  }

  ParsingError::ParsingError(AmbiguousMatch const &variant) :
    my_variant { variant } {
  }

  ParsingError::ParsingError(UnexpectedInput const &variant) :
    my_variant { variant } {
  }

  ParsingError::ParsingError(UnexpectedEndOfInput const &variant) :
    my_variant { variant } {
  }

  ParsingError::ParsingError(ExpectedExactBytes const &variant) :
    my_variant { variant } {
  }

  ParsingError::ParsingError(InsufficientInput const &variant) :
    my_variant { variant } {
  }

  ParsingError::~ParsingError() = default;

  auto ParsingError::describe() const -> std::string {
    static ParsingErrorDescribeVisitor visitor {};
    return std::visit(visitor, my_variant);
  }

  auto ParsingError::format() const -> std::string {
    static ParsingErrorFormatVisitor visitor {};
    return "ParsingError { " + std::visit(visitor, my_variant) + " }";
  }

  auto ParsingError::source() const & -> std::optional<std::reference_wrapper<core::errors::AbstractError const>> {
    static ParsingErrorSourceVisitor visitor {};
    return std::visit(visitor, my_variant);
  }
}