Crow  0.3
A C++ microframework for the web
mustache.h
1#pragma once
2#include <string>
3#include <vector>
4#include <fstream>
5#include <iterator>
6#include <functional>
7#include "crow/json.h"
8#include "crow/logging.h"
9#include "crow/utility.h"
10
11namespace crow
12{
13 namespace mustache
14 {
15 using context = json::wvalue;
16
17 template_t load(const std::string& filename);
18
19 class invalid_template_exception : public std::exception
20 {
21 public:
22 invalid_template_exception(const std::string& msg)
23 : msg("crow::mustache error: " + msg)
24 {
25 }
26 virtual const char* what() const throw()
27 {
28 return msg.c_str();
29 }
30 std::string msg;
31 };
32
33 enum class ActionType
34 {
35 Ignore,
36 Tag,
37 UnescapeTag,
38 OpenBlock,
39 CloseBlock,
40 ElseBlock,
41 Partial,
42 };
43
44 struct Action
45 {
46 int start;
47 int end;
48 int pos;
49 ActionType t;
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)
52 {}
53 };
54
55 /// A mustache template object.
57 {
58 public:
59 template_t(std::string body)
60 : body_(std::move(body))
61 {
62 // {{ {{# {{/ {{^ {{! {{> {{=
63 parse();
64 }
65
66 private:
67 std::string tag_name(const Action& action)
68 {
69 return body_.substr(action.start, action.end - action.start);
70 }
71 auto find_context(const std::string& name, const std::vector<context*>& stack, bool shouldUseOnlyFirstStackValue = false)->std::pair<bool, context&>
72 {
73 if (name == ".")
74 {
75 return {true, *stack.back()};
76 }
77 static json::wvalue empty_str;
78 empty_str = "";
79
80 int dotPosition = name.find(".");
81 if (dotPosition == static_cast<int>(name.npos))
82 {
83 for(auto it = stack.rbegin(); it != stack.rend(); ++it)
84 {
85 if ((*it)->t() == json::type::Object)
86 {
87 if ((*it)->count(name))
88 return {true, (**it)[name]};
89 }
90 }
91 }
92 else
93 {
94 std::vector<int> dotPositions;
95 dotPositions.push_back(-1);
96 while(dotPosition != static_cast<int>(name.npos))
97 {
98 dotPositions.push_back(dotPosition);
99 dotPosition = name.find(".", dotPosition+1);
100 }
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));
106
107 for(auto it = stack.rbegin(); it != stack.rend(); ++it)
108 {
109 context* view = *it;
110 bool found = true;
111 for(auto jt = names.begin(); jt != names.end(); ++jt)
112 {
113 if (view->t() == json::type::Object &&
114 view->count(*jt))
115 {
116 view = &(*view)[*jt];
117 }
118 else
119 {
120 if (shouldUseOnlyFirstStackValue) {
121 return {false, empty_str};
122 }
123 found = false;
124 break;
125 }
126 }
127 if (found)
128 return {true, *view};
129 }
130
131 }
132
133 return {false, empty_str};
134 }
135
136 void escape(const std::string& in, std::string& out)
137 {
138 out.reserve(out.size() + in.size());
139 for(auto it = in.begin(); it != in.end(); ++it)
140 {
141 switch(*it)
142 {
143 case '&': out += "&amp;"; break;
144 case '<': out += "&lt;"; break;
145 case '>': out += "&gt;"; break;
146 case '"': out += "&quot;"; break;
147 case '\'': out += "&#39;"; break;
148 case '/': out += "&#x2F;"; break;
149 case '`': out += "&#x60"; break;
150 case '=': out += "&#x3D"; break;
151 default: out += *it; break;
152 }
153 }
154 }
155
156 bool isTagInsideObjectBlock(const int& current, const std::vector<context*>& stack)
157 {
158 int openedBlock = 0;
159 int totalBlocksBefore = 0;
160 for (int i = current; i > 0; --i) {
161 ++totalBlocksBefore;
162 auto& action = actions_[i - 1];
163
164 if (action.t == ActionType::OpenBlock) {
165 if (openedBlock == 0 && (*stack.rbegin())->t() == json::type::Object) {
166 return true;
167 }
168 --openedBlock;
169 } else if (action.t == ActionType::CloseBlock) {
170 ++openedBlock;
171 }
172 }
173
174 return false;
175 }
176
177 void render_internal(int actionBegin, int actionEnd, std::vector<context*>& stack, std::string& out, int indent)
178 {
179 int current = actionBegin;
180
181 if (indent)
182 out.insert(out.size(), indent, ' ');
183
184 while(current < actionEnd)
185 {
186 auto& fragment = fragments_[current];
187 auto& action = actions_[current];
188 render_fragment(fragment, indent, out);
189 switch(action.t)
190 {
191 case ActionType::Ignore:
192 // do nothing
193 break;
194 case ActionType::Partial:
195 {
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);
200 }
201 break;
202 case ActionType::UnescapeTag:
203 case ActionType::Tag:
204 {
205 bool shouldUseOnlyFirstStackValue = false;
206 if (isTagInsideObjectBlock(current, stack)) {
207 shouldUseOnlyFirstStackValue = true;
208 }
209 auto optional_ctx = find_context(tag_name(action), stack, shouldUseOnlyFirstStackValue);
210 auto& ctx = optional_ctx.second;
211 switch(ctx.t())
212 {
213 case json::type::Number:
214 out += ctx.dump();
215 break;
216 case json::type::String:
217 if (action.t == ActionType::Tag)
218 escape(ctx.s, out);
219 else
220 out += ctx.s;
221 break;
222 default:
223 throw std::runtime_error("not implemented tag type" + boost::lexical_cast<std::string>(static_cast<int>(ctx.t())));
224 }
225 }
226 break;
227 case ActionType::ElseBlock:
228 {
229 static context nullContext;
230 auto optional_ctx = find_context(tag_name(action), stack);
231 if (!optional_ctx.first)
232 {
233 stack.emplace_back(&nullContext);
234 break;
235 }
236
237 auto& ctx = optional_ctx.second;
238 switch(ctx.t())
239 {
240 case json::type::List:
241 if (ctx.l && !ctx.l->empty())
242 current = action.pos;
243 else
244 stack.emplace_back(&nullContext);
245 break;
246 case json::type::False:
247 case json::type::Null:
248 stack.emplace_back(&nullContext);
249 break;
250 default:
251 current = action.pos;
252 break;
253 }
254 break;
255 }
256 case ActionType::OpenBlock:
257 {
258 auto optional_ctx = find_context(tag_name(action), stack);
259 if (!optional_ctx.first)
260 {
261 current = action.pos;
262 break;
263 }
264
265 auto& ctx = optional_ctx.second;
266 switch(ctx.t())
267 {
268 case json::type::List:
269 if (ctx.l)
270 for(auto it = ctx.l->begin(); it != ctx.l->end(); ++it)
271 {
272 stack.push_back(&*it);
273 render_internal(current+1, action.pos, stack, out, indent);
274 stack.pop_back();
275 }
276 current = action.pos;
277 break;
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);
283 break;
284 case json::type::False:
285 case json::type::Null:
286 current = action.pos;
287 break;
288 default:
289 throw std::runtime_error("{{#: not implemented context type: " + boost::lexical_cast<std::string>(static_cast<int>(ctx.t())));
290 break;
291 }
292 break;
293 }
294 case ActionType::CloseBlock:
295 stack.pop_back();
296 break;
297 default:
298 throw std::runtime_error("not implemented " + boost::lexical_cast<std::string>(static_cast<int>(action.t)));
299 }
300 current++;
301 }
302 auto& fragment = fragments_[actionEnd];
303 render_fragment(fragment, indent, out);
304 }
305 void render_fragment(const std::pair<int, int> fragment, int indent, std::string& out)
306 {
307 if (indent)
308 {
309 for(int i = fragment.first; i < fragment.second; i ++)
310 {
311 out += body_[i];
312 if (body_[i] == '\n' && i+1 != static_cast<int>(body_.size()))
313 out.insert(out.size(), indent, ' ');
314 }
315 }
316 else
317 out.insert(out.size(), body_, fragment.first, fragment.second-fragment.first);
318 }
319 public:
320 std::string render()
321 {
322 context empty_ctx;
323 std::vector<context*> stack;
324 stack.emplace_back(&empty_ctx);
325
326 std::string ret;
327 render_internal(0, fragments_.size()-1, stack, ret, 0);
328 return ret;
329 }
330 std::string render(context& ctx)
331 {
332 std::vector<context*> stack;
333 stack.emplace_back(&ctx);
334
335 std::string ret;
336 render_internal(0, fragments_.size()-1, stack, ret, 0);
337 return ret;
338 }
339
340 private:
341
342 void parse()
343 {
344 std::string tag_open = "{{";
345 std::string tag_close = "}}";
346
347 std::vector<int> blockPositions;
348
349 size_t current = 0;
350 while(1)
351 {
352 size_t idx = body_.find(tag_open, current);
353 if (idx == body_.npos)
354 {
355 fragments_.emplace_back(static_cast<int>(current), static_cast<int>(body_.size()));
356 actions_.emplace_back(ActionType::Ignore, 0, 0);
357 break;
358 }
359 fragments_.emplace_back(static_cast<int>(current), static_cast<int>(idx));
360
361 idx += tag_open.size();
362 size_t endIdx = body_.find(tag_close, idx);
363 if (endIdx == idx)
364 {
365 throw invalid_template_exception("empty tag is not allowed");
366 }
367 if (endIdx == body_.npos)
368 {
369 // error, no matching tag
370 throw invalid_template_exception("not matched opening tag");
371 }
372 current = endIdx + tag_close.size();
373 switch(body_[idx])
374 {
375 case '#':
376 idx++;
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);
381 break;
382 case '/':
383 idx++;
384 while(body_[idx] == ' ') idx++;
385 while(body_[endIdx-1] == ' ') endIdx--;
386 {
387 auto& matched = actions_[blockPositions.back()];
388 if (body_.compare(idx, endIdx-idx,
389 body_, matched.start, matched.end - matched.start) != 0)
390 {
391 throw invalid_template_exception("not matched {{# {{/ pair: " +
392 body_.substr(matched.start, matched.end - matched.start) + ", " +
393 body_.substr(idx, endIdx-idx));
394 }
395 matched.pos = actions_.size();
396 }
397 actions_.emplace_back(ActionType::CloseBlock, idx, endIdx, blockPositions.back());
398 blockPositions.pop_back();
399 break;
400 case '^':
401 idx++;
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);
406 break;
407 case '!':
408 // do nothing action
409 actions_.emplace_back(ActionType::Ignore, idx+1, endIdx);
410 break;
411 case '>': // partial
412 idx++;
413 while(body_[idx] == ' ') idx++;
414 while(body_[endIdx-1] == ' ') endIdx--;
415 actions_.emplace_back(ActionType::Partial, idx, endIdx);
416 break;
417 case '{':
418 if (tag_open != "{{" || tag_close != "}}")
419 throw invalid_template_exception("cannot use triple mustache when delimiter changed");
420
421 idx ++;
422 if (body_[endIdx+2] != '}')
423 {
424 throw invalid_template_exception("{{{: }}} not matched");
425 }
426 while(body_[idx] == ' ') idx++;
427 while(body_[endIdx-1] == ' ') endIdx--;
428 actions_.emplace_back(ActionType::UnescapeTag, idx, endIdx);
429 current++;
430 break;
431 case '&':
432 idx ++;
433 while(body_[idx] == ' ') idx++;
434 while(body_[endIdx-1] == ' ') endIdx--;
435 actions_.emplace_back(ActionType::UnescapeTag, idx, endIdx);
436 break;
437 case '=':
438 // tag itself is no-op
439 idx ++;
440 actions_.emplace_back(ActionType::Ignore, idx, endIdx);
441 endIdx --;
442 if (body_[endIdx] != '=')
443 throw invalid_template_exception("{{=: not matching = tag: "+body_.substr(idx, endIdx-idx));
444 endIdx --;
445 while(body_[idx] == ' ') idx++;
446 while(body_[endIdx] == ' ') endIdx--;
447 endIdx++;
448 {
449 bool succeeded = false;
450 for(size_t i = idx; i < endIdx; i++)
451 {
452 if (body_[i] == ' ')
453 {
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())
458 throw invalid_template_exception("{{=: empty open tag");
459 if (tag_close.empty())
460 throw invalid_template_exception("{{=: empty close tag");
461
462 if (tag_close.find(" ") != tag_close.npos)
463 throw invalid_template_exception("{{=: invalid open/close tag: "+tag_open+" " + tag_close);
464 succeeded = true;
465 break;
466 }
467 }
468 if (!succeeded)
469 throw invalid_template_exception("{{=: cannot find space between new open/close tags");
470 }
471 break;
472 default:
473 // normal tag case;
474 while(body_[idx] == ' ') idx++;
475 while(body_[endIdx-1] == ' ') endIdx--;
476 actions_.emplace_back(ActionType::Tag, idx, endIdx);
477 break;
478 }
479 }
480
481 // removing standalones
482 for(int i = actions_.size()-2; i >= 0; i --)
483 {
484 if (actions_[i].t == ActionType::Tag || actions_[i].t == ActionType::UnescapeTag)
485 continue;
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;
490 int j, k;
491 for(j = fragment_before.second-1;j >= fragment_before.first;j--)
492 {
493 if (body_[j] != ' ')
494 {
495 all_space_before = false;
496 break;
497 }
498 }
499 if (all_space_before && i > 0)
500 continue;
501 if (!all_space_before && body_[j] != '\n')
502 continue;
503 bool all_space_after = true;
504 for(k = fragment_after.first; k < static_cast<int>(body_.size()) && k < fragment_after.second; k ++)
505 {
506 if (body_[k] != ' ')
507 {
508 all_space_after = false;
509 break;
510 }
511 }
512 if (all_space_after && !is_last_action)
513 continue;
514 if (!all_space_after &&
515 !(
516 body_[k] == '\n'
517 ||
518 (body_[k] == '\r' &&
519 k + 1 < static_cast<int>(body_.size()) &&
520 body_[k+1] == '\n')))
521 continue;
522 if (actions_[i].t == ActionType::Partial)
523 {
524 actions_[i].pos = fragment_before.second - j - 1;
525 }
526 fragment_before.second = j+1;
527 if (!all_space_after)
528 {
529 if (body_[k] == '\n')
530 k++;
531 else
532 k += 2;
533 fragment_after.first = k;
534 }
535 }
536 }
537
538 std::vector<std::pair<int,int>> fragments_;
539 std::vector<Action> actions_;
540 std::string body_;
541 };
542
543 inline template_t compile(const std::string& body)
544 {
545 return template_t(body);
546 }
547 namespace detail
548 {
549 inline std::string& get_template_base_directory_ref()
550 {
551 static std::string template_base_directory = "templates";
552 return template_base_directory;
553 }
554 }
555
556 inline std::string default_loader(const std::string& filename)
557 {
558 std::string path = detail::get_template_base_directory_ref();
559 if (!(path.back() == '/' || path.back() == '\\'))
560 path += '/';
561 path += filename;
562 std::ifstream inf(path);
563 if (!inf)
564 {
565 CROW_LOG_WARNING << "Template \"" << filename << "\" not found.";
566 return {};
567 }
568 return {std::istreambuf_iterator<char>(inf), std::istreambuf_iterator<char>()};
569 }
570
571 namespace detail
572 {
573 inline std::function<std::string (std::string)>& get_loader_ref()
574 {
575 static std::function<std::string (std::string)> loader = default_loader;
576 return loader;
577 }
578 }
579
580 inline void set_base(const std::string& path)
581 {
582 auto& base = detail::get_template_base_directory_ref();
583 base = path;
584 if (base.back() != '\\' &&
585 base.back() != '/')
586 {
587 base += '/';
588 }
589 }
590
591 inline void set_loader(std::function<std::string(std::string)> loader)
592 {
593 detail::get_loader_ref() = std::move(loader);
594 }
595
596 inline std::string load_text(const std::string& filename)
597 {
598 return detail::get_loader_ref()(filename);
599 }
600
601 inline template_t load(const std::string& filename)
602 {
603 std::string filename_sanitized(filename);
604 utility::sanitize_filename(filename_sanitized);
605 return compile(detail::get_loader_ref()(filename_sanitized));
606 }
607 }
608}
JSON write value.
Definition: json.h:1225
A mustache template object.
Definition: mustache.h:57
Definition: mustache.h:45