8#include "crow/logging.h"
9#include "crow/utility.h"
15 using context = json::wvalue;
17 template_t load(
const std::string& filename);
23 : msg(
"crow::mustache error: " + msg)
26 virtual const char* what()
const throw()
50 Action(ActionType t,
size_t start,
size_t end,
size_t pos = 0)
51 : start(
static_cast<int>(start)), end(
static_cast<int>(end)), pos(
static_cast<int>(pos)), t(t)
60 : body_(std::move(body))
67 std::string tag_name(
const Action& action)
69 return body_.substr(action.start, action.end - action.start);
71 auto find_context(
const std::string& name,
const std::vector<context*>& stack,
bool shouldUseOnlyFirstStackValue =
false)->std::pair<bool,
context&>
75 return {
true, *stack.back()};
80 int dotPosition = name.find(
".");
81 if (dotPosition ==
static_cast<int>(name.npos))
83 for(
auto it = stack.rbegin(); it != stack.rend(); ++it)
85 if ((*it)->t() == json::type::Object)
87 if ((*it)->count(name))
88 return {
true, (**it)[name]};
94 std::vector<int> dotPositions;
95 dotPositions.push_back(-1);
96 while(dotPosition !=
static_cast<int>(name.npos))
98 dotPositions.push_back(dotPosition);
99 dotPosition = name.find(
".", dotPosition+1);
101 dotPositions.push_back(name.size());
102 std::vector<std::string> names;
103 names.reserve(dotPositions.size()-1);
104 for(
int i = 1; i < static_cast<int>(dotPositions.size()); i ++)
105 names.emplace_back(name.substr(dotPositions[i-1]+1, dotPositions[i]-dotPositions[i-1]-1));
107 for(
auto it = stack.rbegin(); it != stack.rend(); ++it)
111 for(
auto jt = names.begin(); jt != names.end(); ++jt)
113 if (view->t() == json::type::Object &&
116 view = &(*view)[*jt];
120 if (shouldUseOnlyFirstStackValue) {
121 return {
false, empty_str};
128 return {
true, *view};
133 return {
false, empty_str};
136 void escape(
const std::string& in, std::string& out)
138 out.reserve(out.size() + in.size());
139 for(
auto it = in.begin(); it != in.end(); ++it)
143 case '&': out +=
"&";
break;
144 case '<': out +=
"<";
break;
145 case '>': out +=
">";
break;
146 case '"': out +=
""";
break;
147 case '\'': out +=
"'";
break;
148 case '/': out +=
"/";
break;
149 case '`': out +=
"`";
break;
150 case '=': out +=
"=";
break;
151 default: out += *it;
break;
156 bool isTagInsideObjectBlock(
const int& current,
const std::vector<context*>& stack)
159 int totalBlocksBefore = 0;
160 for (
int i = current; i > 0; --i) {
162 auto& action = actions_[i - 1];
164 if (action.t == ActionType::OpenBlock) {
165 if (openedBlock == 0 && (*stack.rbegin())->t() == json::type::Object) {
169 }
else if (action.t == ActionType::CloseBlock) {
177 void render_internal(
int actionBegin,
int actionEnd, std::vector<context*>& stack, std::string& out,
int indent)
179 int current = actionBegin;
182 out.insert(out.size(), indent,
' ');
184 while(current < actionEnd)
186 auto& fragment = fragments_[current];
187 auto& action = actions_[current];
188 render_fragment(fragment, indent, out);
191 case ActionType::Ignore:
194 case ActionType::Partial:
196 std::string partial_name = tag_name(action);
197 auto partial_templ = load(partial_name);
198 int partial_indent = action.pos;
199 partial_templ.render_internal(0, partial_templ.fragments_.size()-1, stack, out, partial_indent?indent+partial_indent:0);
202 case ActionType::UnescapeTag:
203 case ActionType::Tag:
205 bool shouldUseOnlyFirstStackValue =
false;
206 if (isTagInsideObjectBlock(current, stack)) {
207 shouldUseOnlyFirstStackValue =
true;
209 auto optional_ctx = find_context(tag_name(action), stack, shouldUseOnlyFirstStackValue);
210 auto& ctx = optional_ctx.second;
213 case json::type::Number:
216 case json::type::String:
217 if (action.t == ActionType::Tag)
223 throw std::runtime_error(
"not implemented tag type" + boost::lexical_cast<std::string>(
static_cast<int>(ctx.t())));
227 case ActionType::ElseBlock:
230 auto optional_ctx = find_context(tag_name(action), stack);
231 if (!optional_ctx.first)
233 stack.emplace_back(&nullContext);
237 auto& ctx = optional_ctx.second;
240 case json::type::List:
241 if (ctx.l && !ctx.l->empty())
242 current = action.pos;
244 stack.emplace_back(&nullContext);
246 case json::type::False:
247 case json::type::Null:
248 stack.emplace_back(&nullContext);
251 current = action.pos;
256 case ActionType::OpenBlock:
258 auto optional_ctx = find_context(tag_name(action), stack);
259 if (!optional_ctx.first)
261 current = action.pos;
265 auto& ctx = optional_ctx.second;
268 case json::type::List:
270 for(
auto it = ctx.l->begin(); it != ctx.l->end(); ++it)
272 stack.push_back(&*it);
273 render_internal(current+1, action.pos, stack, out, indent);
276 current = action.pos;
278 case json::type::Number:
279 case json::type::String:
280 case json::type::Object:
281 case json::type::True:
282 stack.push_back(&ctx);
284 case json::type::False:
285 case json::type::Null:
286 current = action.pos;
289 throw std::runtime_error(
"{{#: not implemented context type: " + boost::lexical_cast<std::string>(
static_cast<int>(ctx.t())));
294 case ActionType::CloseBlock:
298 throw std::runtime_error(
"not implemented " + boost::lexical_cast<std::string>(
static_cast<int>(action.t)));
302 auto& fragment = fragments_[actionEnd];
303 render_fragment(fragment, indent, out);
305 void render_fragment(
const std::pair<int, int> fragment,
int indent, std::string& out)
309 for(
int i = fragment.first; i < fragment.second; i ++)
312 if (body_[i] ==
'\n' && i+1 !=
static_cast<int>(body_.size()))
313 out.insert(out.size(), indent,
' ');
317 out.insert(out.size(), body_, fragment.first, fragment.second-fragment.first);
323 std::vector<context*> stack;
324 stack.emplace_back(&empty_ctx);
327 render_internal(0, fragments_.size()-1, stack, ret, 0);
330 std::string render(
context& ctx)
332 std::vector<context*> stack;
333 stack.emplace_back(&ctx);
336 render_internal(0, fragments_.size()-1, stack, ret, 0);
344 std::string tag_open =
"{{";
345 std::string tag_close =
"}}";
347 std::vector<int> blockPositions;
352 size_t idx = body_.find(tag_open, current);
353 if (idx == body_.npos)
355 fragments_.emplace_back(
static_cast<int>(current),
static_cast<int>(body_.size()));
356 actions_.emplace_back(ActionType::Ignore, 0, 0);
359 fragments_.emplace_back(
static_cast<int>(current),
static_cast<int>(idx));
361 idx += tag_open.size();
362 size_t endIdx = body_.find(tag_close, idx);
367 if (endIdx == body_.npos)
372 current = endIdx + tag_close.size();
377 while(body_[idx] ==
' ') idx++;
378 while(body_[endIdx-1] ==
' ') endIdx--;
379 blockPositions.emplace_back(
static_cast<int>(actions_.size()));
380 actions_.emplace_back(ActionType::OpenBlock, idx, endIdx);
384 while(body_[idx] ==
' ') idx++;
385 while(body_[endIdx-1] ==
' ') endIdx--;
387 auto& matched = actions_[blockPositions.back()];
388 if (body_.compare(idx, endIdx-idx,
389 body_, matched.start, matched.end - matched.start) != 0)
392 body_.substr(matched.start, matched.end - matched.start) +
", " +
393 body_.substr(idx, endIdx-idx));
395 matched.pos = actions_.size();
397 actions_.emplace_back(ActionType::CloseBlock, idx, endIdx, blockPositions.back());
398 blockPositions.pop_back();
402 while(body_[idx] ==
' ') idx++;
403 while(body_[endIdx-1] ==
' ') endIdx--;
404 blockPositions.emplace_back(
static_cast<int>(actions_.size()));
405 actions_.emplace_back(ActionType::ElseBlock, idx, endIdx);
409 actions_.emplace_back(ActionType::Ignore, idx+1, endIdx);
413 while(body_[idx] ==
' ') idx++;
414 while(body_[endIdx-1] ==
' ') endIdx--;
415 actions_.emplace_back(ActionType::Partial, idx, endIdx);
418 if (tag_open !=
"{{" || tag_close !=
"}}")
422 if (body_[endIdx+2] !=
'}')
426 while(body_[idx] ==
' ') idx++;
427 while(body_[endIdx-1] ==
' ') endIdx--;
428 actions_.emplace_back(ActionType::UnescapeTag, idx, endIdx);
433 while(body_[idx] ==
' ') idx++;
434 while(body_[endIdx-1] ==
' ') endIdx--;
435 actions_.emplace_back(ActionType::UnescapeTag, idx, endIdx);
440 actions_.emplace_back(ActionType::Ignore, idx, endIdx);
442 if (body_[endIdx] !=
'=')
445 while(body_[idx] ==
' ') idx++;
446 while(body_[endIdx] ==
' ') endIdx--;
449 bool succeeded =
false;
450 for(
size_t i = idx; i < endIdx; i++)
454 tag_open = body_.substr(idx, i-idx);
455 while(body_[i] ==
' ') i++;
456 tag_close = body_.substr(i, endIdx-i);
457 if (tag_open.empty())
459 if (tag_close.empty())
462 if (tag_close.find(
" ") != tag_close.npos)
474 while(body_[idx] ==
' ') idx++;
475 while(body_[endIdx-1] ==
' ') endIdx--;
476 actions_.emplace_back(ActionType::Tag, idx, endIdx);
482 for(
int i = actions_.size()-2; i >= 0; i --)
484 if (actions_[i].t == ActionType::Tag || actions_[i].t == ActionType::UnescapeTag)
486 auto& fragment_before = fragments_[i];
487 auto& fragment_after = fragments_[i+1];
488 bool is_last_action = i ==
static_cast<int>(actions_.size())-2;
489 bool all_space_before =
true;
491 for(j = fragment_before.second-1;j >= fragment_before.first;j--)
495 all_space_before =
false;
499 if (all_space_before && i > 0)
501 if (!all_space_before && body_[j] !=
'\n')
503 bool all_space_after =
true;
504 for(k = fragment_after.first; k <
static_cast<int>(body_.size()) && k < fragment_after.second; k ++)
508 all_space_after =
false;
512 if (all_space_after && !is_last_action)
514 if (!all_space_after &&
519 k + 1 <
static_cast<int>(body_.size()) &&
520 body_[k+1] ==
'\n')))
522 if (actions_[i].t == ActionType::Partial)
524 actions_[i].pos = fragment_before.second - j - 1;
526 fragment_before.second = j+1;
527 if (!all_space_after)
529 if (body_[k] ==
'\n')
533 fragment_after.first = k;
538 std::vector<std::pair<int,int>> fragments_;
539 std::vector<Action> actions_;
543 inline template_t compile(
const std::string& body)
549 inline std::string& get_template_base_directory_ref()
551 static std::string template_base_directory =
"templates";
552 return template_base_directory;
556 inline std::string default_loader(
const std::string& filename)
558 std::string path = detail::get_template_base_directory_ref();
559 if (!(path.back() ==
'/' || path.back() ==
'\\'))
562 std::ifstream inf(path);
565 CROW_LOG_WARNING <<
"Template \"" << filename <<
"\" not found.";
568 return {std::istreambuf_iterator<char>(inf), std::istreambuf_iterator<char>()};
573 inline std::function<std::string (std::string)>& get_loader_ref()
575 static std::function<std::string (std::string)> loader = default_loader;
580 inline void set_base(
const std::string& path)
582 auto& base = detail::get_template_base_directory_ref();
584 if (base.back() !=
'\\' &&
591 inline void set_loader(std::function<std::string(std::string)> loader)
593 detail::get_loader_ref() = std::move(loader);
596 inline std::string load_text(
const std::string& filename)
598 return detail::get_loader_ref()(filename);
601 inline template_t load(
const std::string& filename)
603 std::string filename_sanitized(filename);
604 utility::sanitize_filename(filename_sanitized);
605 return compile(detail::get_loader_ref()(filename_sanitized));
JSON write value.
Definition: json.h:1225
Definition: mustache.h:20
A mustache template object.
Definition: mustache.h:57
Definition: mustache.h:45