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
|
// Copyright 2022 Huawei Cloud Computing Technology Co., Ltd.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
#ifndef INCLUDED_SRC_BUILDTOOL_BUILD_ENGINE_BASE_MAPS_EXPRESSION_FUNCTION_HPP
#define INCLUDED_SRC_BUILDTOOL_BUILD_ENGINE_BASE_MAPS_EXPRESSION_FUNCTION_HPP
#include <memory>
#include <string>
#include <unordered_map>
#include <vector>
#include "fmt/core.h"
#include "gsl-lite/gsl-lite.hpp"
#include "src/buildtool/build_engine/expression/configuration.hpp"
#include "src/buildtool/build_engine/expression/evaluator.hpp"
#include "src/buildtool/build_engine/expression/expression.hpp"
#include "src/buildtool/logging/logger.hpp"
namespace BuildMaps::Base {
class ExpressionFunction {
public:
using Ptr = std::shared_ptr<ExpressionFunction>;
using imports_t = std::unordered_map<std::string, gsl::not_null<Ptr>>;
ExpressionFunction(std::vector<std::string> vars,
imports_t imports,
ExpressionPtr expr) noexcept
: vars_{std::move(vars)},
imports_{std::move(imports)},
expr_{std::move(expr)} {}
[[nodiscard]] auto Evaluate(
Configuration const& env,
FunctionMapPtr const& functions,
std::function<void(std::string const&)> const& logger =
[](std::string const& error) noexcept -> void {
Logger::Log(LogLevel::Error, error);
},
std::function<void(void)> const& note_user_context =
[]() noexcept -> void {}) const noexcept -> ExpressionPtr {
try { // try-catch to silence clang-tidy's bugprone-exception-escape,
// only imports_caller can throw but it is not called here.
auto imports_caller = [this, &functions](
SubExprEvaluator&& /*eval*/,
ExpressionPtr const& expr,
Configuration const& env) {
auto name_expr = expr["name"];
auto const& name = name_expr->String();
auto it = imports_.find(name);
if (it != imports_.end()) {
std::stringstream ss{};
bool user_context = false;
auto result = it->second->Evaluate(
env,
functions,
[&ss](auto const& msg) { ss << msg; },
[&user_context]() { user_context = true; }
);
if (result) {
return result;
}
if (user_context) {
throw Evaluator::EvaluationError(ss.str(), true, true);
}
throw Evaluator::EvaluationError(
fmt::format(
"This call to {} failed in the following way:\n{}",
name_expr->ToString(),
ss.str()),
true);
}
throw Evaluator::EvaluationError(
fmt::format("Unknown expression '{}'.", name));
};
auto newenv = env.Prune(vars_);
return expr_.Evaluate(
newenv,
FunctionMap::MakePtr(
functions, "CALL_EXPRESSION", imports_caller),
logger,
note_user_context);
} catch (...) {
gsl_EnsuresAudit(false); // ensure that the try-block never throws
return ExpressionPtr{nullptr};
}
}
inline static Ptr const kEmptyTransition =
std::make_shared<ExpressionFunction>(
std::vector<std::string>{},
ExpressionFunction::imports_t{},
Expression::FromJson(R"([{"type": "empty_map"}])"_json));
private:
std::vector<std::string> vars_{};
imports_t imports_{};
ExpressionPtr expr_{};
};
using ExpressionFunctionPtr = ExpressionFunction::Ptr;
} // namespace BuildMaps::Base
#endif // INCLUDED_SRC_BUILDTOOL_BUILD_ENGINE_BASE_MAPS_EXPRESSION_FUNCTION_HPP
|