mirror of
https://github.com/pantor/inja.git
synced 2026-06-25 03:54:17 +00:00
split parser and renderer
This commit is contained in:
@@ -12,7 +12,7 @@ TEST_CASE("Files handling") {
|
||||
data["name"] = "Jeff";
|
||||
|
||||
SECTION("Files should be loaded") {
|
||||
CHECK( env.load_file("data/simple.txt") == "Hello {{ name }}." );
|
||||
CHECK( env.load_global_file("data/simple.txt") == "Hello {{ name }}." );
|
||||
}
|
||||
|
||||
SECTION("Files should be rendered") {
|
||||
@@ -29,7 +29,7 @@ TEST_CASE("Complete files") {
|
||||
|
||||
for (std::string test_name : {"simple-file", "nested", "nested-line"}) {
|
||||
SECTION(test_name) {
|
||||
CHECK( env.render_template_with_json_file(test_name + "/template.txt", test_name + "/data.json") == env.load_file(test_name + "/result.txt") );
|
||||
CHECK( env.render_template_with_json_file(test_name + "/template.txt", test_name + "/data.json") == env.load_global_file(test_name + "/result.txt") );
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
+11
-158
@@ -4,10 +4,11 @@
|
||||
|
||||
|
||||
using json = nlohmann::json;
|
||||
using Type = inja::Parser::Type;
|
||||
using Type = inja::Parsed::Type;
|
||||
|
||||
|
||||
TEST_CASE("Parse structure") {
|
||||
|
||||
/* TEST_CASE("Parse structure") {
|
||||
inja::Parser parser = inja::Parser();
|
||||
|
||||
SECTION("Basic string") {
|
||||
@@ -24,7 +25,7 @@ TEST_CASE("Parse structure") {
|
||||
|
||||
SECTION("Variable") {
|
||||
std::string test = "{{ name }}";
|
||||
json result = {{{"type", Type::Variable}, {"command", "name"}}};
|
||||
json result = {{{"type", Type::Expression}, {"command", "name"}}};
|
||||
CHECK( parser.parse(test) == result );
|
||||
}
|
||||
|
||||
@@ -32,7 +33,7 @@ TEST_CASE("Parse structure") {
|
||||
std::string test = "Hello {{ name }}!";
|
||||
json result = {
|
||||
{{"type", Type::String}, {"text", "Hello "}},
|
||||
{{"type", Type::Variable}, {"command", "name"}},
|
||||
{{"type", Type::Expression}, {"command", "name"}},
|
||||
{{"type", Type::String}, {"text", "!"}}
|
||||
};
|
||||
CHECK( parser.parse(test) == result );
|
||||
@@ -42,9 +43,9 @@ TEST_CASE("Parse structure") {
|
||||
std::string test = "Hello {{ name }}! I come from {{ city }}.";
|
||||
json result = {
|
||||
{{"type", Type::String}, {"text", "Hello "}},
|
||||
{{"type", Type::Variable}, {"command", "name"}},
|
||||
{{"type", Type::Expression}, {"command", "name"}},
|
||||
{{"type", Type::String}, {"text", "! I come from "}},
|
||||
{{"type", Type::Variable}, {"command", "city"}},
|
||||
{{"type", Type::Expression}, {"command", "city"}},
|
||||
{{"type", Type::String}, {"text", "."}}
|
||||
};
|
||||
CHECK( parser.parse(test) == result );
|
||||
@@ -54,7 +55,7 @@ TEST_CASE("Parse structure") {
|
||||
std::string test = "open {% for e in list %}lorem{% endfor %} closing";
|
||||
json result = {
|
||||
{{"type", Type::String}, {"text", "open "}},
|
||||
{{"type", Type::Loop}, {"command", "for e in list"}, {"children", {
|
||||
{{"type", Type::Loop}, {"item", "e"}, {"list", "list"}, {"children", {
|
||||
{{"type", Type::String}, {"text", "lorem"}}
|
||||
}}},
|
||||
{{"type", Type::String}, {"text", " closing"}}
|
||||
@@ -65,8 +66,8 @@ TEST_CASE("Parse structure") {
|
||||
SECTION("Nested loops") {
|
||||
std::string test = "{% for e in list %}{% for b in list2 %}lorem{% endfor %}{% endfor %}";
|
||||
json result = {
|
||||
{{"type", Type::Loop}, {"command", "for e in list"}, {"children", {
|
||||
{{"type", Type::Loop}, {"command", "for b in list2"}, {"children", {
|
||||
{{"type", Type::Loop}, {"item", "e"}, {"list", "list"}, {"children", {
|
||||
{{"type", Type::Loop}, {"item", "b"}, {"list", "list2"}, {"children", {
|
||||
{{"type", Type::String}, {"text", "lorem"}}
|
||||
}}}
|
||||
}}}
|
||||
@@ -127,152 +128,4 @@ lorem ipsum
|
||||
};
|
||||
CHECK( parser.parse(test) == result );
|
||||
}
|
||||
}
|
||||
|
||||
TEST_CASE("Parse variables") {
|
||||
inja::Environment env = inja::Environment();
|
||||
|
||||
|
||||
json data;
|
||||
data["name"] = "Peter";
|
||||
data["city"] = "Washington D.C.";
|
||||
data["age"] = 29;
|
||||
data["names"] = {"Jeff", "Seb"};
|
||||
data["brother"]["name"] = "Chris";
|
||||
data["brother"]["daughters"] = {"Maria", "Helen"};
|
||||
data["brother"]["daughter0"] = { { "name", "Maria" } };
|
||||
|
||||
SECTION("Variables from values") {
|
||||
CHECK( env.eval_variable("42", data) == 42 );
|
||||
CHECK( env.eval_variable("3.1415", data) == 3.1415 );
|
||||
CHECK( env.eval_variable("\"hello\"", data) == "hello" );
|
||||
CHECK( env.eval_variable("true", data) == true );
|
||||
CHECK( env.eval_variable("[5, 6, 8]", data) == std::vector<int>({5, 6, 8}) );
|
||||
}
|
||||
|
||||
SECTION("Variables from JSON data, dot notation") {
|
||||
env.setElementNotation(inja::ElementNotation::Dot);
|
||||
|
||||
CHECK( env.eval_variable("name", data) == "Peter" );
|
||||
CHECK( env.eval_variable("age", data) == 29 );
|
||||
CHECK( env.eval_variable("names.1", data) == "Seb" );
|
||||
CHECK( env.eval_variable("brother.name", data) == "Chris" );
|
||||
CHECK( env.eval_variable("brother.daughters.0", data) == "Maria" );
|
||||
|
||||
CHECK_THROWS_WITH( env.eval_variable("noelement", data), "JSON pointer found no element." );
|
||||
CHECK_THROWS_WITH( env.eval_variable("&4s-", data), "JSON pointer found no element." );
|
||||
}
|
||||
|
||||
SECTION("Variables from JSON data, pointer notation") {
|
||||
env.setElementNotation(inja::ElementNotation::Pointer);
|
||||
|
||||
CHECK( env.eval_variable("names/1", data) == "Seb" );
|
||||
CHECK( env.eval_variable("brother/name", data) == "Chris" );
|
||||
CHECK( env.eval_variable("brother/daughters/0", data) == "Maria" );
|
||||
CHECK( env.eval_variable("/age", data) == 29 );
|
||||
}
|
||||
}
|
||||
|
||||
TEST_CASE("Parse conditions") {
|
||||
inja::Environment env = inja::Environment();
|
||||
|
||||
json data;
|
||||
data["age"] = 29;
|
||||
data["brother"] = "Peter";
|
||||
data["father"] = "Peter";
|
||||
data["guests"] = {"Jeff", "Seb"};
|
||||
|
||||
SECTION("Elements") {
|
||||
CHECK( env.eval_condition("age", data) );
|
||||
CHECK( env.eval_condition("guests", data) );
|
||||
CHECK_FALSE( env.eval_condition("size", data) );
|
||||
CHECK_FALSE( env.eval_condition("false", data) );
|
||||
}
|
||||
|
||||
SECTION("Operators") {
|
||||
CHECK( env.eval_condition("not size", data) );
|
||||
CHECK_FALSE( env.eval_condition("not true", data) );
|
||||
CHECK( env.eval_condition("true and true", data) );
|
||||
CHECK( env.eval_condition("true or false", data) );
|
||||
CHECK_FALSE( env.eval_condition("true and not true", data) );
|
||||
}
|
||||
|
||||
SECTION("Numbers") {
|
||||
CHECK( env.eval_condition("age == 29", data) );
|
||||
CHECK( env.eval_condition("age >= 29", data) );
|
||||
CHECK( env.eval_condition("age <= 29", data) );
|
||||
CHECK( env.eval_condition("age < 100", data) );
|
||||
CHECK_FALSE( env.eval_condition("age > 29", data) );
|
||||
CHECK_FALSE( env.eval_condition("age != 29", data) );
|
||||
CHECK_FALSE( env.eval_condition("age < 28", data) );
|
||||
CHECK_FALSE( env.eval_condition("age < -100.0", data) );
|
||||
}
|
||||
|
||||
SECTION("Strings") {
|
||||
CHECK( env.eval_condition("brother == father", data) );
|
||||
CHECK( env.eval_condition("brother == \"Peter\"", data) );
|
||||
CHECK_FALSE( env.eval_condition("not brother == father", data) );
|
||||
}
|
||||
|
||||
SECTION("Lists") {
|
||||
CHECK( env.eval_condition("\"Jeff\" in guests", data) );
|
||||
CHECK_FALSE( env.eval_condition("brother in guests", data) );
|
||||
}
|
||||
}
|
||||
|
||||
TEST_CASE("Parse functions") {
|
||||
inja::Environment env = inja::Environment();
|
||||
|
||||
json data;
|
||||
data["name"] = "Peter";
|
||||
data["city"] = "New York";
|
||||
data["names"] = {"Jeff", "Seb", "Peter", "Tom"};
|
||||
data["temperature"] = 25.6789;
|
||||
|
||||
SECTION("Upper") {
|
||||
CHECK( env.eval_variable("upper(name)", data) == "PETER" );
|
||||
CHECK( env.eval_variable("upper(city)", data) == "NEW YORK" );
|
||||
CHECK_THROWS_WITH( env.eval_variable("upper(5)", data), "[json.exception.type_error.302] type must be string, but is number" );
|
||||
}
|
||||
|
||||
SECTION("Lower") {
|
||||
CHECK( env.eval_variable("lower(name)", data) == "peter" );
|
||||
CHECK( env.eval_variable("lower(city)", data) == "new york" );
|
||||
CHECK_THROWS_WITH( env.eval_variable("lower(5.45)", data), "[json.exception.type_error.302] type must be string, but is number" );
|
||||
}
|
||||
|
||||
SECTION("Range") {
|
||||
CHECK( env.eval_variable("range(4)", data) == std::vector<int>({0, 1, 2, 3}) );
|
||||
CHECK_THROWS_WITH( env.eval_variable("range(name)", data), "[json.exception.type_error.302] type must be number, but is string" );
|
||||
}
|
||||
|
||||
SECTION("Length") {
|
||||
CHECK( env.eval_variable("length(names)", data) == 4 );
|
||||
CHECK_THROWS_WITH( env.eval_variable("length(5)", data), "[json.exception.type_error.302] type must be array, but is number" );
|
||||
}
|
||||
|
||||
SECTION("Round") {
|
||||
CHECK( env.eval_variable("round(4, 0)", data) == 4 );
|
||||
CHECK( env.eval_variable("round(temperature, 2)", data) == 25.68 );
|
||||
CHECK_THROWS_WITH( env.eval_variable("round(name, 2)", data), "[json.exception.type_error.302] type must be number, but is string" );
|
||||
}
|
||||
|
||||
SECTION("DivisibleBy") {
|
||||
CHECK( env.eval_variable("divisibleBy(50, 5)", data) == true );
|
||||
CHECK( env.eval_variable("divisibleBy(12, 3)", data) == true );
|
||||
CHECK( env.eval_variable("divisibleBy(11, 3)", data) == false );
|
||||
CHECK_THROWS_WITH( env.eval_variable("divisibleBy(name, 2)", data), "[json.exception.type_error.302] type must be number, but is string" );
|
||||
}
|
||||
|
||||
SECTION("Odd") {
|
||||
CHECK( env.eval_variable("odd(11)", data) == true );
|
||||
CHECK( env.eval_variable("odd(12)", data) == false );
|
||||
CHECK_THROWS_WITH( env.eval_variable("odd(name)", data), "[json.exception.type_error.302] type must be number, but is string" );
|
||||
}
|
||||
|
||||
SECTION("Even") {
|
||||
CHECK( env.eval_variable("even(11)", data) == false );
|
||||
CHECK( env.eval_variable("even(12)", data) == true );
|
||||
CHECK_THROWS_WITH( env.eval_variable("even(name)", data), "[json.exception.type_error.302] type must be number, but is string" );
|
||||
}
|
||||
}
|
||||
} */
|
||||
|
||||
@@ -20,7 +20,7 @@ TEST_CASE("Renderer") {
|
||||
|
||||
SECTION("Basic") {
|
||||
CHECK( env.render("Hello World!", data) == "Hello World!" );
|
||||
CHECK( env.render("", data, "../") == "" );
|
||||
CHECK( env.render("", data) == "" );
|
||||
}
|
||||
|
||||
SECTION("Variables") {
|
||||
@@ -57,6 +57,63 @@ TEST_CASE("Renderer") {
|
||||
}
|
||||
}
|
||||
|
||||
TEST_CASE("Render functions") {
|
||||
inja::Environment env = inja::Environment();
|
||||
|
||||
json data;
|
||||
data["name"] = "Peter";
|
||||
data["city"] = "New York";
|
||||
data["names"] = {"Jeff", "Seb", "Peter", "Tom"};
|
||||
data["temperature"] = 25.6789;
|
||||
|
||||
SECTION("Upper") {
|
||||
CHECK( env.render("{{ upper(name) }}", data) == "PETER" );
|
||||
CHECK( env.render("{{ upper(city) }}", data) == "NEW YORK" );
|
||||
CHECK_THROWS_WITH( env.render("{{ upper(5) }}", data), "[json.exception.type_error.302] type must be string, but is number" );
|
||||
}
|
||||
|
||||
SECTION("Lower") {
|
||||
CHECK( env.render("{{ lower(name) }}", data) == "peter" );
|
||||
CHECK( env.render("{{ lower(city) }}", data) == "new york" );
|
||||
CHECK_THROWS_WITH( env.render("{{ lower(5.45) }}", data), "[json.exception.type_error.302] type must be string, but is number" );
|
||||
}
|
||||
|
||||
SECTION("Range") {
|
||||
// CHECK( env.render("range(4)", data) == std::vector<int>({0, 1, 2, 3}) );
|
||||
CHECK_THROWS_WITH( env.render("{{ range(name) }}", data), "[json.exception.type_error.302] type must be number, but is string" );
|
||||
}
|
||||
|
||||
SECTION("Length") {
|
||||
CHECK( env.render("{{ length(names) }}", data) == "4" );
|
||||
CHECK_THROWS_WITH( env.render("{{ length(5) }}", data), "[json.exception.type_error.302] type must be array, but is number" );
|
||||
}
|
||||
|
||||
SECTION("Round") {
|
||||
CHECK( env.render("{{ round(4, 0) }}", data) == "4.0" );
|
||||
CHECK( env.render("{{ round(temperature, 2) }}", data) == "25.68" );
|
||||
CHECK_THROWS_WITH( env.render("{{ round(name, 2) }}", data), "[json.exception.type_error.302] type must be number, but is string" );
|
||||
}
|
||||
|
||||
SECTION("DivisibleBy") {
|
||||
CHECK( env.render("{{ divisibleBy(50, 5) }}", data) == "true" );
|
||||
CHECK( env.render("{{ divisibleBy(12, 3) }}", data) == "true" );
|
||||
CHECK( env.render("{{ divisibleBy(11, 3) }}", data) == "false" );
|
||||
CHECK_THROWS_WITH( env.render("{{ divisibleBy(name, 2) }}", data), "[json.exception.type_error.302] type must be number, but is string" );
|
||||
}
|
||||
|
||||
SECTION("Odd") {
|
||||
CHECK( env.render("{{ odd(11) }}", data) == "true" );
|
||||
CHECK( env.render("{{ odd(12) }}", data) == "false" );
|
||||
CHECK_THROWS_WITH( env.render("{{ odd(name) }}", data), "[json.exception.type_error.302] type must be number, but is string" );
|
||||
}
|
||||
|
||||
SECTION("Even") {
|
||||
CHECK( env.render("{{ even(11) }}", data) == "false" );
|
||||
CHECK( env.render("{{ even(12) }}", data) == "true" );
|
||||
CHECK_THROWS_WITH( env.render("{{ even(name) }}", data), "[json.exception.type_error.302] type must be number, but is string" );
|
||||
}
|
||||
}
|
||||
|
||||
TEST_CASE("Renderer other syntax") {
|
||||
json data;
|
||||
data["name"] = "Peter";
|
||||
|
||||
Reference in New Issue
Block a user