show location for render errors

This commit is contained in:
pantor
2020-06-27 17:05:56 +02:00
parent 0398edd419
commit d25a4de54f
14 changed files with 334 additions and 279 deletions
+1 -17
View File
@@ -213,23 +213,7 @@ public:
explicit Lexer(const LexerConfig &config) : config(config) {}
SourceLocation current_position() const {
// Get line and offset position (starts at 1:1)
auto sliced = string_view::slice(m_in, 0, tok_start);
std::size_t last_newline = sliced.rfind("\n");
if (last_newline == nonstd::string_view::npos) {
return {1, sliced.length() + 1};
}
// Count newlines
size_t count_lines = 0;
size_t search_start = 0;
while (search_start < sliced.size()) {
search_start = sliced.find("\n", search_start + 1);
count_lines += 1;
}
return {count_lines + 1, sliced.length() - last_newline + 1};
return get_source_location(m_in, tok_start);
}
void start(nonstd::string_view input) {
+2 -1
View File
@@ -119,9 +119,10 @@ struct Node {
json value;
std::string str;
nonstd::string_view view;
explicit Node(Op op, unsigned int args = 0) : op(op), args(args), flags(0) {}
explicit Node(Op op, nonstd::string_view str, unsigned int flags) : op(op), args(0), flags(flags), str(str) {}
explicit Node(Op op, nonstd::string_view str, unsigned int flags) : op(op), args(0), flags(flags), str(str), view(str) {}
explicit Node(Op op, json &&value, unsigned int flags) : op(op), args(0), flags(flags), value(std::move(value)) {}
};
+5 -4
View File
@@ -89,7 +89,9 @@ class Parser {
std::vector<IfData> if_stack;
std::vector<size_t> loop_stack;
void throw_parser_error(const std::string &message) { throw ParserError(message, lexer.current_position()); }
void throw_parser_error(const std::string &message) {
throw ParserError(message, lexer.current_position());
}
void get_next_token() {
if (have_peek_tok) {
@@ -270,9 +272,8 @@ public:
} else {
// normal literal (json read)
tmpl.nodes.emplace_back(Node::Op::Push, tok.text,
config.notation == ElementNotation::Pointer ? Node::Flag::ValueLookupPointer
: Node::Flag::ValueLookupDot);
auto flag = config.notation == ElementNotation::Pointer ? Node::Flag::ValueLookupPointer : Node::Flag::ValueLookupDot;
tmpl.nodes.emplace_back(Node::Op::Push, tok.text, flag);
get_next_token();
return true;
}
+121 -111
View File
@@ -33,13 +33,13 @@ inline nonstd::string_view convert_dot_to_json_pointer(nonstd::string_view dot,
* \brief Class for rendering a Template with data.
*/
class Renderer {
std::vector<const json *> &get_args(const Node &bc) {
std::vector<const json *> &get_args(const Node &node) {
m_tmp_args.clear();
bool has_imm = ((bc.flags & Node::Flag::ValueMask) != Node::Flag::ValuePop);
bool has_imm = ((node.flags & Node::Flag::ValueMask) != Node::Flag::ValuePop);
// get args from stack
unsigned int pop_args = bc.args;
unsigned int pop_args = node.args;
if (has_imm) {
pop_args -= 1;
}
@@ -50,15 +50,15 @@ class Renderer {
// get immediate arg
if (has_imm) {
m_tmp_args.push_back(get_imm(bc));
m_tmp_args.push_back(get_imm(node));
}
return m_tmp_args;
}
void pop_args(const Node &bc) {
unsigned int pop_args = bc.args;
if ((bc.flags & Node::Flag::ValueMask) != Node::Flag::ValuePop) {
void pop_args(const Node &node) {
unsigned int pop_args = node.args;
if ((node.flags & Node::Flag::ValueMask) != Node::Flag::ValuePop) {
pop_args -= 1;
}
for (unsigned int i = 0; i < pop_args; ++i) {
@@ -66,20 +66,20 @@ class Renderer {
}
}
const json *get_imm(const Node &bc) {
const json *get_imm(const Node &node) {
std::string ptr_buffer;
nonstd::string_view ptr;
switch (bc.flags & Node::Flag::ValueMask) {
switch (node.flags & Node::Flag::ValueMask) {
case Node::Flag::ValuePop:
return nullptr;
case Node::Flag::ValueImmediate:
return &bc.value;
return &node.value;
case Node::Flag::ValueLookupDot:
ptr = convert_dot_to_json_pointer(bc.str, ptr_buffer);
ptr = convert_dot_to_json_pointer(node.str, ptr_buffer);
break;
case Node::Flag::ValueLookupPointer:
ptr_buffer += '/';
ptr_buffer += bc.str;
ptr_buffer += node.str; // static_cast<std::string>(node.view);
ptr = ptr_buffer;
break;
}
@@ -94,13 +94,13 @@ class Renderer {
return &m_data->at(json_ptr);
} catch (std::exception &) {
// try to evaluate as a no-argument callback
if (auto callback = function_storage.find_callback(bc.str, 0)) {
if (auto callback = function_storage.find_callback(node.str, 0)) {
std::vector<const json *> arguments {};
m_tmp_val = callback(arguments);
return &m_tmp_val;
}
throw RenderError("variable '" + static_cast<std::string>(bc.str) + "' not found");
throw_renderer_error("variable '" + static_cast<std::string>(node.str) + "' not found", node);
return nullptr;
}
}
@@ -137,6 +137,12 @@ class Renderer {
loop_data["is_last"] = (level.index == level.size - 1);
}
void throw_renderer_error(const std::string &message, const Node& node) {
size_t pos = node.view.data() - current_template->content.c_str();
SourceLocation loc = get_source_location(current_template->content, pos);
throw RenderError(message, loc);
}
struct LoopLevel {
enum class Type { Map, Array };
@@ -161,6 +167,7 @@ class Renderer {
const TemplateStorage &template_storage;
const FunctionStorage &function_storage;
const Template *current_template;
std::vector<json> m_stack;
std::vector<LoopLevel> m_loop_stack;
json *m_loop_data;
@@ -178,58 +185,59 @@ public:
}
void render_to(std::ostream &os, const Template &tmpl, const json &data, json *loop_data = nullptr) {
current_template = &tmpl;
m_data = &data;
m_loop_data = loop_data;
for (size_t i = 0; i < tmpl.nodes.size(); ++i) {
const auto &bc = tmpl.nodes[i];
const auto &node = tmpl.nodes[i];
switch (bc.op) {
switch (node.op) {
case Node::Op::Nop: {
break;
}
case Node::Op::PrintText: {
os << bc.str;
os << node.str;
break;
}
case Node::Op::PrintValue: {
const json &val = *get_args(bc)[0];
const json &val = *get_args(node)[0];
if (val.is_string()) {
os << val.get_ref<const std::string &>();
} else {
os << val.dump();
}
pop_args(bc);
pop_args(node);
break;
}
case Node::Op::Push: {
m_stack.emplace_back(*get_imm(bc));
m_stack.emplace_back(*get_imm(node));
break;
}
case Node::Op::Upper: {
auto result = get_args(bc)[0]->get<std::string>();
auto result = get_args(node)[0]->get<std::string>();
std::transform(result.begin(), result.end(), result.begin(), ::toupper);
pop_args(bc);
pop_args(node);
m_stack.emplace_back(std::move(result));
break;
}
case Node::Op::Lower: {
auto result = get_args(bc)[0]->get<std::string>();
auto result = get_args(node)[0]->get<std::string>();
std::transform(result.begin(), result.end(), result.begin(), ::tolower);
pop_args(bc);
pop_args(node);
m_stack.emplace_back(std::move(result));
break;
}
case Node::Op::Range: {
int number = get_args(bc)[0]->get<int>();
int number = get_args(node)[0]->get<int>();
std::vector<int> result(number);
std::iota(std::begin(result), std::end(result), 0);
pop_args(bc);
pop_args(node);
m_stack.emplace_back(std::move(result));
break;
}
case Node::Op::Length: {
const json &val = *get_args(bc)[0];
const json &val = *get_args(node)[0];
size_t result;
if (val.is_string()) {
@@ -238,213 +246,213 @@ public:
result = val.size();
}
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::Sort: {
auto result = get_args(bc)[0]->get<std::vector<json>>();
auto result = get_args(node)[0]->get<std::vector<json>>();
std::sort(result.begin(), result.end());
pop_args(bc);
pop_args(node);
m_stack.emplace_back(std::move(result));
break;
}
case Node::Op::At: {
auto args = get_args(bc);
auto args = get_args(node);
auto result = args[0]->at(args[1]->get<int>());
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::First: {
auto result = get_args(bc)[0]->front();
pop_args(bc);
auto result = get_args(node)[0]->front();
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::Last: {
auto result = get_args(bc)[0]->back();
pop_args(bc);
auto result = get_args(node)[0]->back();
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::Round: {
auto args = get_args(bc);
auto args = get_args(node);
double number = args[0]->get<double>();
int precision = args[1]->get<int>();
pop_args(bc);
pop_args(node);
m_stack.emplace_back(std::round(number * std::pow(10.0, precision)) / std::pow(10.0, precision));
break;
}
case Node::Op::DivisibleBy: {
auto args = get_args(bc);
auto args = get_args(node);
int number = args[0]->get<int>();
int divisor = args[1]->get<int>();
pop_args(bc);
pop_args(node);
m_stack.emplace_back((divisor != 0) && (number % divisor == 0));
break;
}
case Node::Op::Odd: {
int number = get_args(bc)[0]->get<int>();
pop_args(bc);
int number = get_args(node)[0]->get<int>();
pop_args(node);
m_stack.emplace_back(number % 2 != 0);
break;
}
case Node::Op::Even: {
int number = get_args(bc)[0]->get<int>();
pop_args(bc);
int number = get_args(node)[0]->get<int>();
pop_args(node);
m_stack.emplace_back(number % 2 == 0);
break;
}
case Node::Op::Max: {
auto args = get_args(bc);
auto args = get_args(node);
auto result = *std::max_element(args[0]->begin(), args[0]->end());
pop_args(bc);
pop_args(node);
m_stack.emplace_back(std::move(result));
break;
}
case Node::Op::Min: {
auto args = get_args(bc);
auto args = get_args(node);
auto result = *std::min_element(args[0]->begin(), args[0]->end());
pop_args(bc);
pop_args(node);
m_stack.emplace_back(std::move(result));
break;
}
case Node::Op::Not: {
bool result = !truthy(*get_args(bc)[0]);
pop_args(bc);
bool result = !truthy(*get_args(node)[0]);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::And: {
auto args = get_args(bc);
auto args = get_args(node);
bool result = truthy(*args[0]) && truthy(*args[1]);
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::Or: {
auto args = get_args(bc);
auto args = get_args(node);
bool result = truthy(*args[0]) || truthy(*args[1]);
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::In: {
auto args = get_args(bc);
auto args = get_args(node);
bool result = std::find(args[1]->begin(), args[1]->end(), *args[0]) != args[1]->end();
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::Equal: {
auto args = get_args(bc);
auto args = get_args(node);
bool result = (*args[0] == *args[1]);
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::Greater: {
auto args = get_args(bc);
auto args = get_args(node);
bool result = (*args[0] > *args[1]);
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::Less: {
auto args = get_args(bc);
auto args = get_args(node);
bool result = (*args[0] < *args[1]);
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::GreaterEqual: {
auto args = get_args(bc);
auto args = get_args(node);
bool result = (*args[0] >= *args[1]);
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::LessEqual: {
auto args = get_args(bc);
auto args = get_args(node);
bool result = (*args[0] <= *args[1]);
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::Different: {
auto args = get_args(bc);
auto args = get_args(node);
bool result = (*args[0] != *args[1]);
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::Float: {
double result = std::stod(get_args(bc)[0]->get_ref<const std::string &>());
pop_args(bc);
double result = std::stod(get_args(node)[0]->get_ref<const std::string &>());
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::Int: {
int result = std::stoi(get_args(bc)[0]->get_ref<const std::string &>());
pop_args(bc);
int result = std::stoi(get_args(node)[0]->get_ref<const std::string &>());
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::Exists: {
auto &&name = get_args(bc)[0]->get_ref<const std::string &>();
auto &&name = get_args(node)[0]->get_ref<const std::string &>();
bool result = (data.find(name) != data.end());
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::ExistsInObject: {
auto args = get_args(bc);
auto args = get_args(node);
auto &&name = args[1]->get_ref<const std::string &>();
bool result = (args[0]->find(name) != args[0]->end());
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::IsBoolean: {
bool result = get_args(bc)[0]->is_boolean();
pop_args(bc);
bool result = get_args(node)[0]->is_boolean();
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::IsNumber: {
bool result = get_args(bc)[0]->is_number();
pop_args(bc);
bool result = get_args(node)[0]->is_number();
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::IsInteger: {
bool result = get_args(bc)[0]->is_number_integer();
pop_args(bc);
bool result = get_args(node)[0]->is_number_integer();
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::IsFloat: {
bool result = get_args(bc)[0]->is_number_float();
pop_args(bc);
bool result = get_args(node)[0]->is_number_float();
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::IsObject: {
bool result = get_args(bc)[0]->is_object();
pop_args(bc);
bool result = get_args(node)[0]->is_object();
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::IsArray: {
bool result = get_args(bc)[0]->is_array();
pop_args(bc);
bool result = get_args(node)[0]->is_array();
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::IsString: {
bool result = get_args(bc)[0]->is_string();
pop_args(bc);
bool result = get_args(node)[0]->is_string();
pop_args(node);
m_stack.emplace_back(result);
break;
}
@@ -454,7 +462,7 @@ public:
// the parse phase so the second argument is pushed on the stack and
// the first argument is in the immediate
try {
const json *imm = get_imm(bc);
const json *imm = get_imm(node);
// if no exception was raised, replace the stack value with it
m_stack.back() = *imm;
} catch (std::exception &) {
@@ -462,29 +470,31 @@ public:
}
break;
}
case Node::Op::Include:
Renderer(template_storage, function_storage)
.render_to(os, template_storage.find(get_imm(bc)->get_ref<const std::string &>())->second, *m_data,
m_loop_data);
case Node::Op::Include: {
auto sub_renderer = Renderer(template_storage, function_storage);
auto include_name = get_imm(node)->get_ref<const std::string &>();
auto included_template = template_storage.find(include_name)->second;
sub_renderer.render_to(os, included_template, *m_data, m_loop_data);
break;
}
case Node::Op::Callback: {
auto callback = function_storage.find_callback(bc.str, bc.args);
auto callback = function_storage.find_callback(node.str, node.args);
if (!callback) {
throw RenderError("function '" + static_cast<std::string>(bc.str) + "' (" +
std::to_string(static_cast<unsigned int>(bc.args)) + ") not found");
throw_renderer_error("function '" + static_cast<std::string>(node.str) + "' (" +
std::to_string(static_cast<unsigned int>(node.args)) + ") not found", node);
}
json result = callback(get_args(bc));
pop_args(bc);
json result = callback(get_args(node));
pop_args(node);
m_stack.emplace_back(std::move(result));
break;
}
case Node::Op::Jump: {
i = bc.args - 1; // -1 due to ++i in loop
i = node.args - 1; // -1 due to ++i in loop
break;
}
case Node::Op::ConditionalJump: {
if (!truthy(m_stack.back())) {
i = bc.args - 1; // -1 due to ++i in loop
i = node.args - 1; // -1 due to ++i in loop
}
m_stack.pop_back();
break;
@@ -493,13 +503,13 @@ public:
// jump past loop body if empty
if (m_stack.back().empty()) {
m_stack.pop_back();
i = bc.args; // ++i in loop will take it past EndLoop
i = node.args; // ++i in loop will take it past EndLoop
break;
}
m_loop_stack.emplace_back();
LoopLevel &level = m_loop_stack.back();
level.value_name = bc.str;
level.value_name = node.str;
level.values = std::move(m_stack.back());
if (m_loop_data) {
level.data = *m_loop_data;
@@ -507,14 +517,14 @@ public:
level.index = 0;
m_stack.pop_back();
if (bc.value.is_string()) {
if (node.value.is_string()) {
// map iterator
if (!level.values.is_object()) {
m_loop_stack.pop_back();
throw RenderError("for key, value requires object");
throw_renderer_error("for key, value requires object", node);
}
level.loop_type = LoopLevel::Type::Map;
level.key_name = bc.value.get_ref<const std::string &>();
level.key_name = node.value.get_ref<const std::string &>();
// sort by key
for (auto it = level.values.begin(), end = level.values.end(); it != end; ++it) {
@@ -529,7 +539,7 @@ public:
} else {
if (!level.values.is_array()) {
m_loop_stack.pop_back();
throw RenderError("type must be array");
throw_renderer_error("type must be array", node);
}
// list iterator
@@ -551,7 +561,7 @@ public:
}
case Node::Op::EndLoop: {
if (m_loop_stack.empty()) {
throw RenderError("unexpected state in renderer");
throw_renderer_error("unexpected state in renderer", node);
}
LoopLevel &level = m_loop_stack.back();
@@ -578,11 +588,11 @@ public:
update_loop_data();
// jump back to start of loop
i = bc.args - 1; // -1 due to ++i in loop
i = node.args - 1; // -1 due to ++i in loop
break;
}
default: {
throw RenderError("unknown operation in renderer: " + std::to_string(static_cast<unsigned int>(bc.op)));
throw_renderer_error("unknown operation in renderer: " + std::to_string(static_cast<unsigned int>(node.op)), node);
}
}
}
+23
View File
@@ -44,6 +44,29 @@ inline bool starts_with(nonstd::string_view view, nonstd::string_view prefix) {
}
} // namespace string_view
inline SourceLocation get_source_location(nonstd::string_view content, size_t pos) {
// Get line and offset position (starts at 1:1)
auto sliced = string_view::slice(content, 0, pos);
std::size_t last_newline = sliced.rfind("\n");
if (last_newline == nonstd::string_view::npos) {
return {1, sliced.length() + 1};
}
// Count newlines
size_t count_lines = 0;
size_t search_start = 0;
while (search_start <= sliced.size()) {
search_start = sliced.find("\n", search_start) + 1;
if (search_start <= 0) {
break;
}
count_lines += 1;
}
return {count_lines + 1, sliced.length() - last_newline};
}
} // namespace inja
#endif // INCLUDE_INJA_UTILS_HPP_
+152 -133
View File
@@ -1625,9 +1625,10 @@ struct Node {
json value;
std::string str;
nonstd::string_view view;
explicit Node(Op op, unsigned int args = 0) : op(op), args(args), flags(0) {}
explicit Node(Op op, nonstd::string_view str, unsigned int flags) : op(op), args(0), flags(flags), str(str) {}
explicit Node(Op op, nonstd::string_view str, unsigned int flags) : op(op), args(0), flags(flags), str(str), view(str) {}
explicit Node(Op op, json &&value, unsigned int flags) : op(op), args(0), flags(flags), value(std::move(value)) {}
};
@@ -1916,6 +1917,29 @@ inline bool starts_with(nonstd::string_view view, nonstd::string_view prefix) {
}
} // namespace string_view
inline SourceLocation get_source_location(nonstd::string_view content, size_t pos) {
// Get line and offset position (starts at 1:1)
auto sliced = string_view::slice(content, 0, pos);
std::size_t last_newline = sliced.rfind("\n");
if (last_newline == nonstd::string_view::npos) {
return {1, sliced.length() + 1};
}
// Count newlines
size_t count_lines = 0;
size_t search_start = 0;
while (search_start <= sliced.size()) {
search_start = sliced.find("\n", search_start) + 1;
if (search_start <= 0) {
break;
}
count_lines += 1;
}
return {count_lines + 1, sliced.length() - last_newline};
}
} // namespace inja
#endif // INCLUDE_INJA_UTILS_HPP_
@@ -2124,23 +2148,7 @@ public:
explicit Lexer(const LexerConfig &config) : config(config) {}
SourceLocation current_position() const {
// Get line and offset position (starts at 1:1)
auto sliced = string_view::slice(m_in, 0, tok_start);
std::size_t last_newline = sliced.rfind("\n");
if (last_newline == nonstd::string_view::npos) {
return {1, sliced.length() + 1};
}
// Count newlines
size_t count_lines = 0;
size_t search_start = 0;
while (search_start < sliced.size()) {
search_start = sliced.find("\n", search_start + 1);
count_lines += 1;
}
return {count_lines + 1, sliced.length() - last_newline + 1};
return get_source_location(m_in, tok_start);
}
void start(nonstd::string_view input) {
@@ -2359,7 +2367,9 @@ class Parser {
std::vector<IfData> if_stack;
std::vector<size_t> loop_stack;
void throw_parser_error(const std::string &message) { throw ParserError(message, lexer.current_position()); }
void throw_parser_error(const std::string &message) {
throw ParserError(message, lexer.current_position());
}
void get_next_token() {
if (have_peek_tok) {
@@ -2540,9 +2550,8 @@ public:
} else {
// normal literal (json read)
tmpl.nodes.emplace_back(Node::Op::Push, tok.text,
config.notation == ElementNotation::Pointer ? Node::Flag::ValueLookupPointer
: Node::Flag::ValueLookupDot);
auto flag = config.notation == ElementNotation::Pointer ? Node::Flag::ValueLookupPointer : Node::Flag::ValueLookupDot;
tmpl.nodes.emplace_back(Node::Op::Push, tok.text, flag);
get_next_token();
return true;
}
@@ -2920,13 +2929,13 @@ inline nonstd::string_view convert_dot_to_json_pointer(nonstd::string_view dot,
* \brief Class for rendering a Template with data.
*/
class Renderer {
std::vector<const json *> &get_args(const Node &bc) {
std::vector<const json *> &get_args(const Node &node) {
m_tmp_args.clear();
bool has_imm = ((bc.flags & Node::Flag::ValueMask) != Node::Flag::ValuePop);
bool has_imm = ((node.flags & Node::Flag::ValueMask) != Node::Flag::ValuePop);
// get args from stack
unsigned int pop_args = bc.args;
unsigned int pop_args = node.args;
if (has_imm) {
pop_args -= 1;
}
@@ -2937,15 +2946,15 @@ class Renderer {
// get immediate arg
if (has_imm) {
m_tmp_args.push_back(get_imm(bc));
m_tmp_args.push_back(get_imm(node));
}
return m_tmp_args;
}
void pop_args(const Node &bc) {
unsigned int pop_args = bc.args;
if ((bc.flags & Node::Flag::ValueMask) != Node::Flag::ValuePop) {
void pop_args(const Node &node) {
unsigned int pop_args = node.args;
if ((node.flags & Node::Flag::ValueMask) != Node::Flag::ValuePop) {
pop_args -= 1;
}
for (unsigned int i = 0; i < pop_args; ++i) {
@@ -2953,20 +2962,20 @@ class Renderer {
}
}
const json *get_imm(const Node &bc) {
const json *get_imm(const Node &node) {
std::string ptr_buffer;
nonstd::string_view ptr;
switch (bc.flags & Node::Flag::ValueMask) {
switch (node.flags & Node::Flag::ValueMask) {
case Node::Flag::ValuePop:
return nullptr;
case Node::Flag::ValueImmediate:
return &bc.value;
return &node.value;
case Node::Flag::ValueLookupDot:
ptr = convert_dot_to_json_pointer(bc.str, ptr_buffer);
ptr = convert_dot_to_json_pointer(node.str, ptr_buffer);
break;
case Node::Flag::ValueLookupPointer:
ptr_buffer += '/';
ptr_buffer += bc.str;
ptr_buffer += node.str; // static_cast<std::string>(node.view);
ptr = ptr_buffer;
break;
}
@@ -2981,13 +2990,13 @@ class Renderer {
return &m_data->at(json_ptr);
} catch (std::exception &) {
// try to evaluate as a no-argument callback
if (auto callback = function_storage.find_callback(bc.str, 0)) {
if (auto callback = function_storage.find_callback(node.str, 0)) {
std::vector<const json *> arguments {};
m_tmp_val = callback(arguments);
return &m_tmp_val;
}
throw RenderError("variable '" + static_cast<std::string>(bc.str) + "' not found");
throw_renderer_error("variable '" + static_cast<std::string>(node.str) + "' not found", node);
return nullptr;
}
}
@@ -3024,6 +3033,12 @@ class Renderer {
loop_data["is_last"] = (level.index == level.size - 1);
}
void throw_renderer_error(const std::string &message, const Node& node) {
size_t pos = node.view.data() - current_template->content.c_str();
SourceLocation loc = get_source_location(current_template->content, pos);
throw RenderError(message, loc);
}
struct LoopLevel {
enum class Type { Map, Array };
@@ -3048,6 +3063,7 @@ class Renderer {
const TemplateStorage &template_storage;
const FunctionStorage &function_storage;
const Template *current_template;
std::vector<json> m_stack;
std::vector<LoopLevel> m_loop_stack;
json *m_loop_data;
@@ -3065,58 +3081,59 @@ public:
}
void render_to(std::ostream &os, const Template &tmpl, const json &data, json *loop_data = nullptr) {
current_template = &tmpl;
m_data = &data;
m_loop_data = loop_data;
for (size_t i = 0; i < tmpl.nodes.size(); ++i) {
const auto &bc = tmpl.nodes[i];
const auto &node = tmpl.nodes[i];
switch (bc.op) {
switch (node.op) {
case Node::Op::Nop: {
break;
}
case Node::Op::PrintText: {
os << bc.str;
os << node.str;
break;
}
case Node::Op::PrintValue: {
const json &val = *get_args(bc)[0];
const json &val = *get_args(node)[0];
if (val.is_string()) {
os << val.get_ref<const std::string &>();
} else {
os << val.dump();
}
pop_args(bc);
pop_args(node);
break;
}
case Node::Op::Push: {
m_stack.emplace_back(*get_imm(bc));
m_stack.emplace_back(*get_imm(node));
break;
}
case Node::Op::Upper: {
auto result = get_args(bc)[0]->get<std::string>();
auto result = get_args(node)[0]->get<std::string>();
std::transform(result.begin(), result.end(), result.begin(), ::toupper);
pop_args(bc);
pop_args(node);
m_stack.emplace_back(std::move(result));
break;
}
case Node::Op::Lower: {
auto result = get_args(bc)[0]->get<std::string>();
auto result = get_args(node)[0]->get<std::string>();
std::transform(result.begin(), result.end(), result.begin(), ::tolower);
pop_args(bc);
pop_args(node);
m_stack.emplace_back(std::move(result));
break;
}
case Node::Op::Range: {
int number = get_args(bc)[0]->get<int>();
int number = get_args(node)[0]->get<int>();
std::vector<int> result(number);
std::iota(std::begin(result), std::end(result), 0);
pop_args(bc);
pop_args(node);
m_stack.emplace_back(std::move(result));
break;
}
case Node::Op::Length: {
const json &val = *get_args(bc)[0];
const json &val = *get_args(node)[0];
size_t result;
if (val.is_string()) {
@@ -3125,213 +3142,213 @@ public:
result = val.size();
}
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::Sort: {
auto result = get_args(bc)[0]->get<std::vector<json>>();
auto result = get_args(node)[0]->get<std::vector<json>>();
std::sort(result.begin(), result.end());
pop_args(bc);
pop_args(node);
m_stack.emplace_back(std::move(result));
break;
}
case Node::Op::At: {
auto args = get_args(bc);
auto args = get_args(node);
auto result = args[0]->at(args[1]->get<int>());
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::First: {
auto result = get_args(bc)[0]->front();
pop_args(bc);
auto result = get_args(node)[0]->front();
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::Last: {
auto result = get_args(bc)[0]->back();
pop_args(bc);
auto result = get_args(node)[0]->back();
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::Round: {
auto args = get_args(bc);
auto args = get_args(node);
double number = args[0]->get<double>();
int precision = args[1]->get<int>();
pop_args(bc);
pop_args(node);
m_stack.emplace_back(std::round(number * std::pow(10.0, precision)) / std::pow(10.0, precision));
break;
}
case Node::Op::DivisibleBy: {
auto args = get_args(bc);
auto args = get_args(node);
int number = args[0]->get<int>();
int divisor = args[1]->get<int>();
pop_args(bc);
pop_args(node);
m_stack.emplace_back((divisor != 0) && (number % divisor == 0));
break;
}
case Node::Op::Odd: {
int number = get_args(bc)[0]->get<int>();
pop_args(bc);
int number = get_args(node)[0]->get<int>();
pop_args(node);
m_stack.emplace_back(number % 2 != 0);
break;
}
case Node::Op::Even: {
int number = get_args(bc)[0]->get<int>();
pop_args(bc);
int number = get_args(node)[0]->get<int>();
pop_args(node);
m_stack.emplace_back(number % 2 == 0);
break;
}
case Node::Op::Max: {
auto args = get_args(bc);
auto args = get_args(node);
auto result = *std::max_element(args[0]->begin(), args[0]->end());
pop_args(bc);
pop_args(node);
m_stack.emplace_back(std::move(result));
break;
}
case Node::Op::Min: {
auto args = get_args(bc);
auto args = get_args(node);
auto result = *std::min_element(args[0]->begin(), args[0]->end());
pop_args(bc);
pop_args(node);
m_stack.emplace_back(std::move(result));
break;
}
case Node::Op::Not: {
bool result = !truthy(*get_args(bc)[0]);
pop_args(bc);
bool result = !truthy(*get_args(node)[0]);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::And: {
auto args = get_args(bc);
auto args = get_args(node);
bool result = truthy(*args[0]) && truthy(*args[1]);
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::Or: {
auto args = get_args(bc);
auto args = get_args(node);
bool result = truthy(*args[0]) || truthy(*args[1]);
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::In: {
auto args = get_args(bc);
auto args = get_args(node);
bool result = std::find(args[1]->begin(), args[1]->end(), *args[0]) != args[1]->end();
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::Equal: {
auto args = get_args(bc);
auto args = get_args(node);
bool result = (*args[0] == *args[1]);
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::Greater: {
auto args = get_args(bc);
auto args = get_args(node);
bool result = (*args[0] > *args[1]);
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::Less: {
auto args = get_args(bc);
auto args = get_args(node);
bool result = (*args[0] < *args[1]);
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::GreaterEqual: {
auto args = get_args(bc);
auto args = get_args(node);
bool result = (*args[0] >= *args[1]);
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::LessEqual: {
auto args = get_args(bc);
auto args = get_args(node);
bool result = (*args[0] <= *args[1]);
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::Different: {
auto args = get_args(bc);
auto args = get_args(node);
bool result = (*args[0] != *args[1]);
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::Float: {
double result = std::stod(get_args(bc)[0]->get_ref<const std::string &>());
pop_args(bc);
double result = std::stod(get_args(node)[0]->get_ref<const std::string &>());
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::Int: {
int result = std::stoi(get_args(bc)[0]->get_ref<const std::string &>());
pop_args(bc);
int result = std::stoi(get_args(node)[0]->get_ref<const std::string &>());
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::Exists: {
auto &&name = get_args(bc)[0]->get_ref<const std::string &>();
auto &&name = get_args(node)[0]->get_ref<const std::string &>();
bool result = (data.find(name) != data.end());
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::ExistsInObject: {
auto args = get_args(bc);
auto args = get_args(node);
auto &&name = args[1]->get_ref<const std::string &>();
bool result = (args[0]->find(name) != args[0]->end());
pop_args(bc);
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::IsBoolean: {
bool result = get_args(bc)[0]->is_boolean();
pop_args(bc);
bool result = get_args(node)[0]->is_boolean();
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::IsNumber: {
bool result = get_args(bc)[0]->is_number();
pop_args(bc);
bool result = get_args(node)[0]->is_number();
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::IsInteger: {
bool result = get_args(bc)[0]->is_number_integer();
pop_args(bc);
bool result = get_args(node)[0]->is_number_integer();
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::IsFloat: {
bool result = get_args(bc)[0]->is_number_float();
pop_args(bc);
bool result = get_args(node)[0]->is_number_float();
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::IsObject: {
bool result = get_args(bc)[0]->is_object();
pop_args(bc);
bool result = get_args(node)[0]->is_object();
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::IsArray: {
bool result = get_args(bc)[0]->is_array();
pop_args(bc);
bool result = get_args(node)[0]->is_array();
pop_args(node);
m_stack.emplace_back(result);
break;
}
case Node::Op::IsString: {
bool result = get_args(bc)[0]->is_string();
pop_args(bc);
bool result = get_args(node)[0]->is_string();
pop_args(node);
m_stack.emplace_back(result);
break;
}
@@ -3341,7 +3358,7 @@ public:
// the parse phase so the second argument is pushed on the stack and
// the first argument is in the immediate
try {
const json *imm = get_imm(bc);
const json *imm = get_imm(node);
// if no exception was raised, replace the stack value with it
m_stack.back() = *imm;
} catch (std::exception &) {
@@ -3349,29 +3366,31 @@ public:
}
break;
}
case Node::Op::Include:
Renderer(template_storage, function_storage)
.render_to(os, template_storage.find(get_imm(bc)->get_ref<const std::string &>())->second, *m_data,
m_loop_data);
case Node::Op::Include: {
auto sub_renderer = Renderer(template_storage, function_storage);
auto include_name = get_imm(node)->get_ref<const std::string &>();
auto included_template = template_storage.find(include_name)->second;
sub_renderer.render_to(os, included_template, *m_data, m_loop_data);
break;
}
case Node::Op::Callback: {
auto callback = function_storage.find_callback(bc.str, bc.args);
auto callback = function_storage.find_callback(node.str, node.args);
if (!callback) {
throw RenderError("function '" + static_cast<std::string>(bc.str) + "' (" +
std::to_string(static_cast<unsigned int>(bc.args)) + ") not found");
throw_renderer_error("function '" + static_cast<std::string>(node.str) + "' (" +
std::to_string(static_cast<unsigned int>(node.args)) + ") not found", node);
}
json result = callback(get_args(bc));
pop_args(bc);
json result = callback(get_args(node));
pop_args(node);
m_stack.emplace_back(std::move(result));
break;
}
case Node::Op::Jump: {
i = bc.args - 1; // -1 due to ++i in loop
i = node.args - 1; // -1 due to ++i in loop
break;
}
case Node::Op::ConditionalJump: {
if (!truthy(m_stack.back())) {
i = bc.args - 1; // -1 due to ++i in loop
i = node.args - 1; // -1 due to ++i in loop
}
m_stack.pop_back();
break;
@@ -3380,13 +3399,13 @@ public:
// jump past loop body if empty
if (m_stack.back().empty()) {
m_stack.pop_back();
i = bc.args; // ++i in loop will take it past EndLoop
i = node.args; // ++i in loop will take it past EndLoop
break;
}
m_loop_stack.emplace_back();
LoopLevel &level = m_loop_stack.back();
level.value_name = bc.str;
level.value_name = node.str;
level.values = std::move(m_stack.back());
if (m_loop_data) {
level.data = *m_loop_data;
@@ -3394,14 +3413,14 @@ public:
level.index = 0;
m_stack.pop_back();
if (bc.value.is_string()) {
if (node.value.is_string()) {
// map iterator
if (!level.values.is_object()) {
m_loop_stack.pop_back();
throw RenderError("for key, value requires object");
throw_renderer_error("for key, value requires object", node);
}
level.loop_type = LoopLevel::Type::Map;
level.key_name = bc.value.get_ref<const std::string &>();
level.key_name = node.value.get_ref<const std::string &>();
// sort by key
for (auto it = level.values.begin(), end = level.values.end(); it != end; ++it) {
@@ -3416,7 +3435,7 @@ public:
} else {
if (!level.values.is_array()) {
m_loop_stack.pop_back();
throw RenderError("type must be array");
throw_renderer_error("type must be array", node);
}
// list iterator
@@ -3438,7 +3457,7 @@ public:
}
case Node::Op::EndLoop: {
if (m_loop_stack.empty()) {
throw RenderError("unexpected state in renderer");
throw_renderer_error("unexpected state in renderer", node);
}
LoopLevel &level = m_loop_stack.back();
@@ -3465,11 +3484,11 @@ public:
update_loop_data();
// jump back to start of loop
i = bc.args - 1; // -1 due to ++i in loop
i = node.args - 1; // -1 due to ++i in loop
break;
}
default: {
throw RenderError("unknown operation in renderer: " + std::to_string(static_cast<unsigned int>(bc.op)));
throw_renderer_error("unknown operation in renderer: " + std::to_string(static_cast<unsigned int>(node.op)), node);
}
}
}
+2 -2
View File
@@ -255,9 +255,9 @@ std::string string_template {
"cupiditate nostrum iure. Voluptatem accusamus vel corporis. \n Debitis {{ name }} sunt est debitis distinctio ut. "
"Provident corrupti nihil velit aut tempora corporis corrupti exercitationem. Praesentium cumque ex est itaque."};
BENCHMARK(InjaBenchmarkerSmallData, render, 10, 100) { env.render(string_template, smallData); }
BENCHMARK(InjaBenchmarkerSmallData, render, 5, 50) { env.render(string_template, smallData); }
BENCHMARK(InjaBenchmarkerLargeData, render, 10, 100) { env.render(string_template, largeData); }
BENCHMARK(InjaBenchmarkerLargeData, render, 5, 25) { env.render(string_template, largeData); }
int main() {
hayai::ConsoleOutputter consoleOutputter;
+2 -1
View File
@@ -7,5 +7,6 @@
],
"views": 123,
"title": "Inja works.",
"content": "Inja is the best and fastest template engine for C++. Period."
"content": "Inja is the best and fastest template engine for C++. Period.",
"footer-text": "This is the footer."
}
+6
View File
@@ -0,0 +1,6 @@
<footer>
<div>
<h3>About</h3>
<p>{{ footer-text }}</p>
</div>
</footer>
+3
View File
@@ -0,0 +1,3 @@
<head>
<title>{{ title }}</title>
</head>
+7
View File
@@ -15,5 +15,12 @@
<li>test</li>
<li>templates</li>
</ul>
<footer>
<div>
<h3>About</h3>
<p>This is the footer.</p>
</div>
</footer>
</body>
</html>
+3 -3
View File
@@ -1,8 +1,6 @@
<!DOCTYPE html>
<html>
<head>
<title>{{ title }}</title>
</head>
{% include "header.txt" %}
<body>
<h1>{{ title }}</h1>
<small>Written by {{ author }}</small>
@@ -16,5 +14,7 @@
<li>{{ tag }}</li>
## endfor
</ul>
{% include "footer.txt" %}
</body>
</html>
+1 -1
View File
@@ -44,7 +44,7 @@ TEST_CASE("complete-files") {
for (std::string test_name : {"error-unknown"}) {
SUBCASE(test_name.c_str()) {
CHECK_THROWS_WITH(env.render_file_with_json_file(test_name + "/template.txt", test_name + "/data.json"),
"[inja.exception.parser_error] (at 2:11) expected 'in', got 'ins'");
"[inja.exception.parser_error] (at 2:10) expected 'in', got 'ins'");
}
}
}
+6 -6
View File
@@ -45,7 +45,7 @@ TEST_CASE("types") {
CHECK(env.render("Hello {{ brother.daughter0.name }}!", data) == "Hello Maria!");
CHECK(env.render("{{ \"{{ no_value }}\" }}", data) == "{{ no_value }}");
CHECK_THROWS_WITH(env.render("{{unknown}}", data), "[inja.exception.render_error] variable 'unknown' not found");
CHECK_THROWS_WITH(env.render("{{unknown}}", data), "[inja.exception.render_error] (at 1:3) variable 'unknown' not found");
}
SUBCASE("comments") {
@@ -74,7 +74,7 @@ TEST_CASE("types") {
CHECK_THROWS_WITH(env.render("{% for name ins names %}a{% endfor %}", data),
"[inja.exception.parser_error] (at 1:13) expected 'in', got 'ins'");
CHECK_THROWS_WITH(env.render("{% for name in empty_loop %}a{% endfor %}", data),
"[inja.exception.render_error] variable 'empty_loop' not found");
"[inja.exception.render_error] (at 1:16) variable 'empty_loop' not found");
// CHECK_THROWS_WITH( env.render("{% for name in relatives %}{{ name }}{% endfor %}", data),
// "[inja.exception.json_error] [json.exception.type_error.302] type must be array, but is object" );
}
@@ -281,7 +281,7 @@ TEST_CASE("functions") {
CHECK(env.render("{{ default(surname, \"nobody\") }}", data) == "nobody");
CHECK(env.render("{{ default(surname, \"{{ surname }}\") }}", data) == "{{ surname }}");
CHECK_THROWS_WITH(env.render("{{ default(surname, lastname) }}", data),
"[inja.exception.render_error] variable 'lastname' not found");
"[inja.exception.render_error] (at 1:21) variable 'lastname' not found");
}
SUBCASE("exists") {
@@ -297,9 +297,9 @@ TEST_CASE("functions") {
CHECK(env.render("{{ existsIn(brother, property) }}", data) == "true");
CHECK(env.render("{{ existsIn(brother, name) }}", data) == "false");
CHECK_THROWS_WITH(env.render("{{ existsIn(sister, \"lastname\") }}", data),
"[inja.exception.render_error] variable 'sister' not found");
"[inja.exception.render_error] (at 1:13) variable 'sister' not found");
CHECK_THROWS_WITH(env.render("{{ existsIn(brother, sister) }}", data),
"[inja.exception.render_error] variable 'sister' not found");
"[inja.exception.render_error] (at 1:22) variable 'sister' not found");
}
SUBCASE("isType") {
@@ -440,7 +440,7 @@ TEST_CASE("other-syntax") {
CHECK(env.render("Hello {{ brother/daughter0/name }}!", data) == "Hello Maria!");
CHECK_THROWS_WITH(env.render("{{unknown/name}}", data),
"[inja.exception.render_error] variable 'unknown/name' not found");
"[inja.exception.render_error] (at 1:3) variable 'unknown/name' not found");
}
SUBCASE("other expression syntax") {