OASIS
Open Algebra Software
Loading...
Searching...
No Matches
MatchCast.hpp
Go to the documentation of this file.
1//
2// Created by Matthew McCall on 10/22/24.
3//
4
5// Hear ye, valiant coder! Within this hallowed script of Oasis' MatchCast,
6// lies an unfathomable confluence of templated magic, scarcely understood only by even the
7// most exalted compilers and the divine overseers. This arcane construct,
8// when tampered with, may envelop thee in utter confusion, leaving thee solitary in thy pursuits.
9// Of course, this sagely advice surely does not emanate from a mere language model. Rather,
10// it is the timeless counsel of battle-hardened developers who have faced such arcane complexity.
11// Should regret cloud thy mind and thou desires to undo thine alterations, perform the following
12// sacred rite to restore the code to its pristine state:
13//
14// git checkout -- <path_to_this_file>
15//
16
17#ifndef OASIS_MATCHCAST_HPP
18#define OASIS_MATCHCAST_HPP
19
20#include <boost/callable_traits/args.hpp>
21#include <boost/mpl/for_each.hpp>
22#include <boost/mpl/push_back.hpp>
23#include <boost/mpl/vector.hpp>
24
25#include <gsl/gsl-lite.hpp>
26
27#include <concepts>
28#include <functional>
29#include <tuple>
30
31namespace Oasis {
32template <typename Lambda>
34
35template <typename CheckF, typename TransformerF, typename ArgumentT>
36concept TransformerAcceptsCheckArg = requires(TransformerF f, const lambda_argument_type<CheckF>& t) {
39
40template <typename ArgumentT, typename Cases>
42public:
43 template <typename Check, typename Transformer>
45 consteval auto Case(
46 Check,
48 {
49 return {};
50 }
51
52 template <typename VisitorPtrT>
54 auto Execute(const ArgumentT& arg, VisitorPtrT visitor) const -> std::expected<std::unique_ptr<ArgumentT>, std::string_view>
55 {
57 boost::mpl::for_each<Cases>(
58 [&]<typename Check, typename Transformer>(std::pair<Check, Transformer> checkAndTransformer) {
59 using CaseType = lambda_argument_type<Check>;
60 if (!result.has_value() || (result.has_value() && *result))
61 return;
62
63 auto [check, transformer] = checkAndTransformer;
64 if (std::unique_ptr<CaseType> castResult = RecursiveCast<CaseType>(arg); castResult && check(*castResult))
65 result = transformer(*castResult, visitor).transform([](gsl::not_null<std::unique_ptr<ArgumentT>>&& transformResult) { return std::move(transformResult); });
66 });
67 return result;
68 }
69};
70
71template <typename ArgumentT>
73}
74
75#endif // OASIS_MATCHCAST_HPP
Definition MatchCast.hpp:41
auto Execute(const ArgumentT &arg, VisitorPtrT visitor) const -> std::expected< std::unique_ptr< ArgumentT >, std::string_view >
Definition MatchCast.hpp:54
consteval auto Case(Check, Transformer) const -> MatchCastImpl< ArgumentT, typename boost::mpl::push_back< Cases, std::pair< Check, Transformer > >::type >
Definition MatchCast.hpp:45
Definition Concepts.hpp:63
Definition MatchCast.hpp:36
T is_same_v
Definition Add.hpp:11