summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--doc/concepts/expressions.org5
-rw-r--r--src/buildtool/build_engine/expression/evaluator.cpp15
-rw-r--r--test/buildtool/build_engine/expression/expression.test.cpp18
3 files changed, 38 insertions, 0 deletions
diff --git a/doc/concepts/expressions.org b/doc/concepts/expressions.org
index 50737413..895b5e0a 100644
--- a/doc/concepts/expressions.org
+++ b/doc/concepts/expressions.org
@@ -211,6 +211,11 @@ those) argument(s) to obtain the final result.
first non-negative integers. For example, ~{"type": "range",
"$1": "3"}~ evaluates to ~["0", "1", "2"]~.
+- ~"enumerate"~ The argument has to be a list. The result is a map
+ containing one entry for each element of the list; the key is the
+ decimal representation of the position in the list (starting from
+ ~0~) and the value is the element.
+
- ~"++"~ The argument has to be a list of lists. The result is the
concatenation of those lists.
diff --git a/src/buildtool/build_engine/expression/evaluator.cpp b/src/buildtool/build_engine/expression/evaluator.cpp
index cfd0be31..76ad3076 100644
--- a/src/buildtool/build_engine/expression/evaluator.cpp
+++ b/src/buildtool/build_engine/expression/evaluator.cpp
@@ -139,6 +139,20 @@ auto Values(ExpressionPtr const& d) -> ExpressionPtr {
return ExpressionPtr{d->Map().Values()};
}
+auto Enumerate(ExpressionPtr const& expr) -> ExpressionPtr {
+ if (not expr->IsList()) {
+ throw Evaluator::EvaluationError{fmt::format(
+ "enumerate expects list but instead got: {}.", expr->ToString())};
+ }
+ auto result = Expression::map_t::underlying_map_t{};
+ size_t count = 0;
+ for (auto const& entry : expr->List()) {
+ result[fmt::format("{}", count)] = entry;
+ count++;
+ }
+ return ExpressionPtr{Expression::map_t{result}};
+}
+
auto NubRight(ExpressionPtr const& expr) -> ExpressionPtr {
if (not expr->IsList()) {
throw Evaluator::EvaluationError{fmt::format(
@@ -849,6 +863,7 @@ auto const kBuiltInFunctions =
{"json_encode", JsonEncodeExpr},
{"escape_chars", EscapeCharsExpr},
{"keys", UnaryExpr(Keys)},
+ {"enumerate", UnaryExpr(Enumerate)},
{"values", UnaryExpr(Values)},
{"lookup", LookupExpr},
{"empty_map", EmptyMapExpr},
diff --git a/test/buildtool/build_engine/expression/expression.test.cpp b/test/buildtool/build_engine/expression/expression.test.cpp
index cbc8c7bf..11ff7a43 100644
--- a/test/buildtool/build_engine/expression/expression.test.cpp
+++ b/test/buildtool/build_engine/expression/expression.test.cpp
@@ -890,6 +890,24 @@ TEST_CASE("Expression Evaluation", "[expression]") { // NOLINT
CHECK(result == Expression::FromJson(R"("esXcXape me XX")"_json));
}
+ SECTION("enumerate expression") {
+ auto expr = Expression::FromJson(R"(
+ { "type": "enumerate"
+ , "$1": ["foo", "bar", "baz"]
+ }
+ )"_json);
+ REQUIRE(expr);
+
+ auto result = expr.Evaluate(env, fcts);
+ REQUIRE(result);
+ CHECK(result == Expression::FromJson(R"(
+ { "0": "foo"
+ , "1": "bar"
+ , "2": "baz"
+ }
+ )"_json));
+ }
+
SECTION("keys expression") {
auto expr = Expression::FromJson(R"(
{ "type": "keys"