8 #include "crow/logging.h"
9 #include "crow/returnable.h"
10 #include "crow/utility.h"
16 using context = json::wvalue;
18 template_t load(
const std::string& filename);
24 msg(
"crow::mustache error: " + msg)
26 virtual const char* what()
const throw()
39 returnable(
"text/html"), body_(std::move(body)) {}
43 std::string dump()
const override
66 Action(ActionType t,
size_t start,
size_t end,
size_t pos = 0):
67 start(
static_cast<int>(start)), end(
static_cast<int>(end)), pos(
static_cast<int>(pos)), t(t)
77 body_(std::move(body))
84 std::string tag_name(
const Action& action)
const
86 return body_.substr(action.start, action.end - action.start);
88 auto find_context(
const std::string& name,
const std::vector<const context*>& stack,
bool shouldUseOnlyFirstStackValue =
false)
const -> std::pair<bool, const context&>
92 return {
true, *stack.back()};
97 int dotPosition = name.find(
".");
98 if (dotPosition ==
static_cast<int>(name.npos))
100 for (
auto it = stack.rbegin(); it != stack.rend(); ++it)
102 if ((*it)->t() == json::type::Object)
104 if ((*it)->count(name))
105 return {
true, (**it)[name]};
111 std::vector<int> dotPositions;
112 dotPositions.push_back(-1);
113 while (dotPosition !=
static_cast<int>(name.npos))
115 dotPositions.push_back(dotPosition);
116 dotPosition = name.find(
".", dotPosition + 1);
118 dotPositions.push_back(name.size());
119 std::vector<std::string> names;
120 names.reserve(dotPositions.size() - 1);
121 for (
int i = 1; i < static_cast<int>(dotPositions.size()); i++)
122 names.emplace_back(name.substr(dotPositions[i - 1] + 1, dotPositions[i] - dotPositions[i - 1] - 1));
124 for (
auto it = stack.rbegin(); it != stack.rend(); ++it)
128 for (
auto jt = names.begin(); jt != names.end(); ++jt)
130 if (view->t() == json::type::Object &&
133 view = &(*view)[*jt];
137 if (shouldUseOnlyFirstStackValue)
139 return {
false, empty_str};
146 return {
true, *view};
150 return {
false, empty_str};
153 void escape(
const std::string& in, std::string& out)
const
155 out.reserve(out.size() + in.size());
156 for (
auto it = in.begin(); it != in.end(); ++it)
160 case '&': out +=
"&";
break;
161 case '<': out +=
"<";
break;
162 case '>': out +=
">";
break;
163 case '"': out +=
""";
break;
164 case '\'': out +=
"'";
break;
165 case '/': out +=
"/";
break;
166 case '`': out +=
"`";
break;
167 case '=': out +=
"=";
break;
168 default: out += *it;
break;
173 bool isTagInsideObjectBlock(
const int& current,
const std::vector<const context*>& stack)
const
176 for (
int i = current; i > 0; --i)
178 auto& action = actions_[i - 1];
180 if (action.t == ActionType::OpenBlock)
182 if (openedBlock == 0 && (*stack.rbegin())->t() == json::type::Object)
188 else if (action.t == ActionType::CloseBlock)
197 void render_internal(
int actionBegin,
int actionEnd, std::vector<const context*>& stack, std::string& out,
int indent)
const
199 int current = actionBegin;
202 out.insert(out.size(), indent,
' ');
204 while (current < actionEnd)
206 auto& fragment = fragments_[current];
207 auto& action = actions_[current];
208 render_fragment(fragment, indent, out);
211 case ActionType::Ignore:
214 case ActionType::Partial:
216 std::string partial_name = tag_name(action);
217 auto partial_templ = load(partial_name);
218 int partial_indent = action.pos;
219 partial_templ.render_internal(0, partial_templ.fragments_.size() - 1, stack, out, partial_indent ? indent + partial_indent : 0);
222 case ActionType::UnescapeTag:
223 case ActionType::Tag:
225 bool shouldUseOnlyFirstStackValue =
false;
226 if (isTagInsideObjectBlock(current, stack))
228 shouldUseOnlyFirstStackValue =
true;
230 auto optional_ctx = find_context(tag_name(action), stack, shouldUseOnlyFirstStackValue);
231 auto& ctx = optional_ctx.second;
234 case json::type::False:
235 case json::type::True:
236 case json::type::Number:
239 case json::type::String:
240 if (action.t == ActionType::Tag)
245 case json::type::Function:
247 std::string execute_result = ctx.execute();
248 while (execute_result.find(
"{{") != std::string::npos)
254 if (action.t == ActionType::Tag)
255 escape(execute_result, out);
257 out += execute_result;
261 throw std::runtime_error(
"not implemented tag type" + utility::lexical_cast<std::string>(
static_cast<int>(ctx.t())));
265 case ActionType::ElseBlock:
268 auto optional_ctx = find_context(tag_name(action), stack);
269 if (!optional_ctx.first)
271 stack.emplace_back(&nullContext);
275 auto& ctx = optional_ctx.second;
278 case json::type::List:
279 if (ctx.l && !ctx.l->empty())
280 current = action.pos;
282 stack.emplace_back(&nullContext);
284 case json::type::False:
285 case json::type::Null:
286 stack.emplace_back(&nullContext);
289 current = action.pos;
294 case ActionType::OpenBlock:
296 auto optional_ctx = find_context(tag_name(action), stack);
297 if (!optional_ctx.first)
299 current = action.pos;
303 auto& ctx = optional_ctx.second;
306 case json::type::List:
308 for (
auto it = ctx.l->begin(); it != ctx.l->end(); ++it)
310 stack.push_back(&*it);
311 render_internal(current + 1, action.pos, stack, out, indent);
314 current = action.pos;
316 case json::type::Number:
317 case json::type::String:
318 case json::type::Object:
319 case json::type::True:
320 stack.push_back(&ctx);
322 case json::type::False:
323 case json::type::Null:
324 current = action.pos;
327 throw std::runtime_error(
"{{#: not implemented context type: " + utility::lexical_cast<std::string>(
static_cast<int>(ctx.t())));
332 case ActionType::CloseBlock:
336 throw std::runtime_error(
"not implemented " + utility::lexical_cast<std::string>(
static_cast<int>(action.t)));
340 auto& fragment = fragments_[actionEnd];
341 render_fragment(fragment, indent, out);
343 void render_fragment(
const std::pair<int, int> fragment,
int indent, std::string& out)
const
347 for (
int i = fragment.first; i < fragment.second; i++)
350 if (body_[i] ==
'\n' && i + 1 !=
static_cast<int>(body_.size()))
351 out.insert(out.size(), indent,
' ');
355 out.insert(out.size(), body_, fragment.first, fragment.second - fragment.first);
363 std::vector<const context*> stack;
364 stack.emplace_back(&empty_ctx);
367 render_internal(0, fragments_.size() - 1, stack, ret, 0);
374 std::vector<const context*> stack;
375 stack.emplace_back(&ctx);
378 render_internal(0, fragments_.size() - 1, stack, ret, 0);
392 std::vector<const context*> stack;
393 stack.emplace_back(&empty_ctx);
396 render_internal(0, fragments_.size() - 1, stack, ret, 0);
403 std::vector<const context*> stack;
404 stack.emplace_back(&ctx);
407 render_internal(0, fragments_.size() - 1, stack, ret, 0);
414 std::string tag_open =
"{{";
415 std::string tag_close =
"}}";
417 std::vector<int> blockPositions;
422 size_t idx = body_.find(tag_open, current);
423 if (idx == body_.npos)
425 fragments_.emplace_back(
static_cast<int>(current),
static_cast<int>(body_.size()));
426 actions_.emplace_back(ActionType::Ignore, 0, 0);
429 fragments_.emplace_back(
static_cast<int>(current),
static_cast<int>(idx));
431 idx += tag_open.size();
432 size_t endIdx = body_.find(tag_close, idx);
435 throw invalid_template_exception(
"empty tag is not allowed");
437 if (endIdx == body_.npos)
440 throw invalid_template_exception(
"not matched opening tag");
442 current = endIdx + tag_close.size();
447 while (body_[idx] ==
' ')
449 while (body_[endIdx - 1] ==
' ')
451 blockPositions.emplace_back(
static_cast<int>(actions_.size()));
452 actions_.emplace_back(ActionType::OpenBlock, idx, endIdx);
456 while (body_[idx] ==
' ')
458 while (body_[endIdx - 1] ==
' ')
461 auto& matched = actions_[blockPositions.back()];
462 if (body_.compare(idx, endIdx - idx,
463 body_, matched.start, matched.end - matched.start) != 0)
465 throw invalid_template_exception(
"not matched {{# {{/ pair: " +
466 body_.substr(matched.start, matched.end - matched.start) +
", " +
467 body_.substr(idx, endIdx - idx));
469 matched.pos = actions_.size();
471 actions_.emplace_back(ActionType::CloseBlock, idx, endIdx, blockPositions.back());
472 blockPositions.pop_back();
476 while (body_[idx] ==
' ')
478 while (body_[endIdx - 1] ==
' ')
480 blockPositions.emplace_back(
static_cast<int>(actions_.size()));
481 actions_.emplace_back(ActionType::ElseBlock, idx, endIdx);
485 actions_.emplace_back(ActionType::Ignore, idx + 1, endIdx);
489 while (body_[idx] ==
' ')
491 while (body_[endIdx - 1] ==
' ')
493 actions_.emplace_back(ActionType::Partial, idx, endIdx);
496 if (tag_open !=
"{{" || tag_close !=
"}}")
497 throw invalid_template_exception(
"cannot use triple mustache when delimiter changed");
500 if (body_[endIdx + 2] !=
'}')
502 throw invalid_template_exception(
"{{{: }}} not matched");
504 while (body_[idx] ==
' ')
506 while (body_[endIdx - 1] ==
' ')
508 actions_.emplace_back(ActionType::UnescapeTag, idx, endIdx);
513 while (body_[idx] ==
' ')
515 while (body_[endIdx - 1] ==
' ')
517 actions_.emplace_back(ActionType::UnescapeTag, idx, endIdx);
522 actions_.emplace_back(ActionType::Ignore, idx, endIdx);
524 if (body_[endIdx] !=
'=')
525 throw invalid_template_exception(
"{{=: not matching = tag: " + body_.substr(idx, endIdx - idx));
527 while (body_[idx] ==
' ')
529 while (body_[endIdx] ==
' ')
533 bool succeeded =
false;
534 for (
size_t i = idx; i < endIdx; i++)
538 tag_open = body_.substr(idx, i - idx);
539 while (body_[i] ==
' ')
541 tag_close = body_.substr(i, endIdx - i);
542 if (tag_open.empty())
543 throw invalid_template_exception(
"{{=: empty open tag");
544 if (tag_close.empty())
545 throw invalid_template_exception(
"{{=: empty close tag");
547 if (tag_close.find(
" ") != tag_close.npos)
548 throw invalid_template_exception(
"{{=: invalid open/close tag: " + tag_open +
" " + tag_close);
554 throw invalid_template_exception(
"{{=: cannot find space between new open/close tags");
559 while (body_[idx] ==
' ')
561 while (body_[endIdx - 1] ==
' ')
563 actions_.emplace_back(ActionType::Tag, idx, endIdx);
569 for (
int i = actions_.size() - 2; i >= 0; i--)
571 if (actions_[i].t == ActionType::Tag || actions_[i].t == ActionType::UnescapeTag)
573 auto& fragment_before = fragments_[i];
574 auto& fragment_after = fragments_[i + 1];
575 bool is_last_action = i ==
static_cast<int>(actions_.size()) - 2;
576 bool all_space_before =
true;
578 for (j = fragment_before.second - 1; j >= fragment_before.first; j--)
582 all_space_before =
false;
586 if (all_space_before && i > 0)
588 if (!all_space_before && body_[j] !=
'\n')
590 bool all_space_after =
true;
591 for (k = fragment_after.first; k <
static_cast<int>(body_.size()) && k < fragment_after.second; k++)
595 all_space_after =
false;
599 if (all_space_after && !is_last_action)
601 if (!all_space_after &&
605 k + 1 <
static_cast<int>(body_.size()) &&
606 body_[k + 1] ==
'\n')))
608 if (actions_[i].t == ActionType::Partial)
610 actions_[i].pos = fragment_before.second - j - 1;
612 fragment_before.second = j + 1;
613 if (!all_space_after)
615 if (body_[k] ==
'\n')
619 fragment_after.first = k;
624 std::vector<std::pair<int, int>> fragments_;
625 std::vector<Action> actions_;
629 inline template_t compile(
const std::string& body)
631 return template_t(body);
635 inline std::string& get_template_base_directory_ref()
637 static std::string template_base_directory =
"templates";
638 return template_base_directory;
642 inline std::string& get_global_template_base_directory_ref()
644 static std::string template_base_directory =
"templates";
645 return template_base_directory;
649 inline std::string default_loader(
const std::string& filename)
651 std::string path = detail::get_template_base_directory_ref();
652 std::ifstream inf(utility::join_path(path, filename));
655 CROW_LOG_WARNING <<
"Template \"" << filename <<
"\" not found.";
658 return {std::istreambuf_iterator<char>(inf), std::istreambuf_iterator<char>()};
663 inline std::function<std::string(std::string)>& get_loader_ref()
665 static std::function<std::string(std::string)> loader = default_loader;
670 inline void set_base(
const std::string& path)
672 auto& base = detail::get_template_base_directory_ref();
674 if (base.back() !=
'\\' &&
681 inline void set_global_base(
const std::string& path)
683 auto& base = detail::get_global_template_base_directory_ref();
685 if (base.back() !=
'\\' &&
692 inline void set_loader(std::function<std::string(std::string)> loader)
694 detail::get_loader_ref() = std::move(loader);
697 inline std::string load_text(
const std::string& filename)
699 std::string filename_sanitized(filename);
700 utility::sanitize_filename(filename_sanitized);
701 return detail::get_loader_ref()(filename_sanitized);
704 inline std::string load_text_unsafe(
const std::string& filename)
706 return detail::get_loader_ref()(filename);
709 inline template_t load(
const std::string& filename)
711 std::string filename_sanitized(filename);
712 utility::sanitize_filename(filename_sanitized);
713 return compile(detail::get_loader_ref()(filename_sanitized));
716 inline template_t load_unsafe(
const std::string& filename)
718 return compile(detail::get_loader_ref()(filename));
JSON write value.
Definition: json.h:1289
Definition: mustache.h:21
A mustache template object.
Definition: mustache.h:74
std::string render_string() const
Output a returnable template from this mustache template.
Definition: mustache.h:389
rendered_template render() const
Output a returnable template from this mustache template.
Definition: mustache.h:360
rendered_template render(const context &ctx) const
Apply the values from the context provided and output a returnable template from this mustache templa...
Definition: mustache.h:372
rendered_template render(const context &&ctx) const
Apply the values from the context provided and output a returnable template from this mustache templa...
Definition: mustache.h:383
std::string render_string(const context &ctx) const
Apply the values from the context provided and output a returnable template from this mustache templa...
Definition: mustache.h:401
The main namespace of the library. In this namespace is defined the most important classes and functi...
Definition: mustache.h:61
Definition: mustache.h:34
An abstract class that allows any other class to be returned by a handler.
Definition: returnable.h:9