Browse Source

add mstch

Mistivia 2 months ago
parent
commit
8689a7c78c
100 changed files with 1681 additions and 0 deletions
  1. 28 0
      lib/mstch/CMakeLists.txt
  2. 22 0
      lib/mstch/LICENSE
  3. 281 0
      lib/mstch/README.md
  4. 6 0
      lib/mstch/benchmark/CMakeLists.txt
  5. 26 0
      lib/mstch/benchmark/benchmark_main.cpp
  6. 1 0
      lib/mstch/cmake/mstch-config.cmake
  7. 113 0
      lib/mstch/include/mstch/mstch.hpp
  8. 64 0
      lib/mstch/src/CMakeLists.txt
  9. 20 0
      lib/mstch/src/mstch.cpp
  10. 72 0
      lib/mstch/src/render_context.cpp
  11. 51 0
      lib/mstch/src/render_context.hpp
  12. 34 0
      lib/mstch/src/state/in_section.cpp
  13. 24 0
      lib/mstch/src/state/in_section.hpp
  14. 32 0
      lib/mstch/src/state/outside_section.cpp
  15. 12 0
      lib/mstch/src/state/outside_section.hpp
  16. 17 0
      lib/mstch/src/state/render_state.hpp
  17. 104 0
      lib/mstch/src/template_type.cpp
  18. 30 0
      lib/mstch/src/template_type.hpp
  19. 42 0
      lib/mstch/src/token.cpp
  20. 39 0
      lib/mstch/src/token.hpp
  21. 44 0
      lib/mstch/src/utils.cpp
  22. 23 0
      lib/mstch/src/utils.hpp
  23. 35 0
      lib/mstch/src/visitor/get_token.hpp
  24. 31 0
      lib/mstch/src/visitor/has_token.hpp
  25. 41 0
      lib/mstch/src/visitor/is_node_empty.hpp
  26. 56 0
      lib/mstch/src/visitor/render_node.hpp
  27. 57 0
      lib/mstch/src/visitor/render_section.hpp
  28. 59 0
      lib/mstch/test/CMakeLists.txt
  29. 3 0
      lib/mstch/test/data/ampersand_escape.hpp
  30. 1 0
      lib/mstch/test/data/ampersand_escape.mustache
  31. 1 0
      lib/mstch/test/data/ampersand_escape.txt
  32. 4 0
      lib/mstch/test/data/apostrophe.hpp
  33. 1 0
      lib/mstch/test/data/apostrophe.mustache
  34. 1 0
      lib/mstch/test/data/apostrophe.txt
  35. 3 0
      lib/mstch/test/data/array_of_strings.hpp
  36. 1 0
      lib/mstch/test/data/array_of_strings.mustache
  37. 1 0
      lib/mstch/test/data/array_of_strings.txt
  38. 3 0
      lib/mstch/test/data/backslashes.hpp
  39. 7 0
      lib/mstch/test/data/backslashes.mustache
  40. 7 0
      lib/mstch/test/data/backslashes.txt
  41. 3 0
      lib/mstch/test/data/bug_11_eating_whitespace.hpp
  42. 1 0
      lib/mstch/test/data/bug_11_eating_whitespace.mustache
  43. 1 0
      lib/mstch/test/data/bug_11_eating_whitespace.txt
  44. 3 0
      lib/mstch/test/data/bug_length_property.hpp
  45. 1 0
      lib/mstch/test/data/bug_length_property.mustache
  46. 1 0
      lib/mstch/test/data/bug_length_property.txt
  47. 4 0
      lib/mstch/test/data/changing_delimiters.hpp
  48. 1 0
      lib/mstch/test/data/changing_delimiters.mustache
  49. 1 0
      lib/mstch/test/data/changing_delimiters.txt
  50. 3 0
      lib/mstch/test/data/comments.hpp
  51. 1 0
      lib/mstch/test/data/comments.mustache
  52. 1 0
      lib/mstch/test/data/comments.txt
  53. 69 0
      lib/mstch/test/data/complex.hpp
  54. 16 0
      lib/mstch/test/data/complex.mustache
  55. 6 0
      lib/mstch/test/data/complex.txt
  56. 8 0
      lib/mstch/test/data/context_lookup.hpp
  57. 1 0
      lib/mstch/test/data/context_lookup.mustache
  58. 1 0
      lib/mstch/test/data/context_lookup.txt
  59. 6 0
      lib/mstch/test/data/delimiters.hpp
  60. 7 0
      lib/mstch/test/data/delimiters.mustache
  61. 5 0
      lib/mstch/test/data/delimiters.txt
  62. 4 0
      lib/mstch/test/data/disappearing_whitespace.hpp
  63. 1 0
      lib/mstch/test/data/disappearing_whitespace.mustache
  64. 1 0
      lib/mstch/test/data/disappearing_whitespace.txt
  65. 34 0
      lib/mstch/test/data/dot_notation.hpp
  66. 9 0
      lib/mstch/test/data/dot_notation.mustache
  67. 9 0
      lib/mstch/test/data/dot_notation.txt
  68. 5 0
      lib/mstch/test/data/double_render.hpp
  69. 1 0
      lib/mstch/test/data/double_render.mustache
  70. 1 0
      lib/mstch/test/data/double_render.txt
  71. 3 0
      lib/mstch/test/data/empty_list.hpp
  72. 4 0
      lib/mstch/test/data/empty_list.mustache
  73. 1 0
      lib/mstch/test/data/empty_list.txt
  74. 1 0
      lib/mstch/test/data/empty_sections.hpp
  75. 1 0
      lib/mstch/test/data/empty_sections.mustache
  76. 1 0
      lib/mstch/test/data/empty_sections.txt
  77. 6 0
      lib/mstch/test/data/empty_string.hpp
  78. 1 0
      lib/mstch/test/data/empty_string.mustache
  79. 1 0
      lib/mstch/test/data/empty_string.txt
  80. 1 0
      lib/mstch/test/data/empty_template.hpp
  81. 1 0
      lib/mstch/test/data/empty_template.mustache
  82. 1 0
      lib/mstch/test/data/empty_template.txt
  83. 3 0
      lib/mstch/test/data/error_eof_in_section.hpp
  84. 1 0
      lib/mstch/test/data/error_eof_in_section.mustache
  85. 1 0
      lib/mstch/test/data/error_eof_in_section.txt
  86. 1 0
      lib/mstch/test/data/error_eof_in_tag.hpp
  87. 1 0
      lib/mstch/test/data/error_eof_in_tag.mustache
  88. 1 0
      lib/mstch/test/data/error_eof_in_tag.txt
  89. 3 0
      lib/mstch/test/data/error_not_found.hpp
  90. 1 0
      lib/mstch/test/data/error_not_found.mustache
  91. 0 0
      lib/mstch/test/data/error_not_found.txt
  92. 4 0
      lib/mstch/test/data/escaped.hpp
  93. 2 0
      lib/mstch/test/data/escaped.mustache
  94. 2 0
      lib/mstch/test/data/escaped.txt
  95. 6 0
      lib/mstch/test/data/falsy.hpp
  96. 8 0
      lib/mstch/test/data/falsy.mustache
  97. 8 0
      lib/mstch/test/data/falsy.txt
  98. 8 0
      lib/mstch/test/data/falsy_array.hpp
  99. 3 0
      lib/mstch/test/data/falsy_array.mustache
  100. 4 0
      lib/mstch/test/data/falsy_array.txt

+ 28 - 0
lib/mstch/CMakeLists.txt

@@ -0,0 +1,28 @@
+cmake_minimum_required(VERSION 3.0.2)
+project(mstch)
+
+option(WITH_UNIT_TESTS "enable building unit test executable" OFF)
+option(WITH_BENCHMARK "enable building benchmark executable" OFF)
+
+set(CMAKE_INCLUDE_CURRENT_DIR ON)
+set(CMAKE_INCLUDE_CURRENT_DIR_IN_INTERFACE ON)
+set(CMAKE_BUILD_TYPE Release)
+
+set(mstch_VERSION 1.0.1)
+
+if(NOT MSVC)
+  set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++11 -Wall -Wextra -O3")
+endif()
+
+add_subdirectory(src)
+
+if(WITH_UNIT_TESTS)
+  enable_testing()
+  add_subdirectory(vendor/headerize)
+  add_subdirectory(test)
+endif()
+
+if(WITH_BENCHMARK)
+  add_subdirectory(vendor/benchmark)
+  add_subdirectory(benchmark)
+endif()

+ 22 - 0
lib/mstch/LICENSE

@@ -0,0 +1,22 @@
+The MIT License (MIT)
+
+Copyright (c) 2015 Daniel Sipka
+
+Permission is hereby granted, free of charge, to any person obtaining a copy
+of this software and associated documentation files (the "Software"), to deal
+in the Software without restriction, including without limitation the rights
+to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+copies of the Software, and to permit persons to whom the Software is
+furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in all
+copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
+SOFTWARE.
+

+ 281 - 0
lib/mstch/README.md

@@ -0,0 +1,281 @@
+# mstch - {{mustache}} templates in C++11 
+
+![mstch logo](http://i.imgur.com/MRyStO5.png)
+
+mstch is a complete implementation of [{{mustache}}](http://mustache.github.io/) 
+templates using modern C++. It's compliant with [specifications](https://github.com/mustache/spec)
+v1.1.3, including the lambda module.
+
+[![Try it online](https://img.shields.io/badge/try%20it-online-blue.svg)](http://melpon.org/wandbox/permlink/EqyOe7IBRYPGVk5f)
+[![GitHub version](https://badge.fury.io/gh/no1msd%2Fmstch.svg)](http://badge.fury.io/gh/no1msd%2Fmstch)
+[![Build Status](https://travis-ci.org/no1msd/mstch.svg?branch=master)](https://travis-ci.org/no1msd/mstch)
+[![Build status](https://ci.appveyor.com/api/projects/status/d6mxp0uba5646x16?svg=true)](https://ci.appveyor.com/project/no1msd/mstch)
+
+## Supported features
+
+mstch supports the complete feature set described in the `mustache(5)` [manpage](http://mustache.github.com/mustache.5.html):
+
+ - JSON-like data structure using [Boost.Variant](http://www.boost.org/libs/variant)
+ - variables, sections, inverted sections
+ - partials
+ - changing the delimiter
+ - C++11 lambdas
+ - C++ objects as view models
+
+## Basic usage
+
+```c++
+#include <iostream>
+#include <mstch/mstch.hpp>
+
+int main() {
+  std::string view{"{{#names}}Hi {{name}}!\n{{/names}}"};
+  mstch::map context{
+    {"names", mstch::array{
+      mstch::map{{"name", std::string{"Chris"}}},
+      mstch::map{{"name", std::string{"Mark"}}},
+      mstch::map{{"name", std::string{"Scott"}}},
+    }}
+  };
+  
+  std::cout << mstch::render(view, context) << std::endl;
+  
+  return 0;
+}
+
+```
+
+The output of this example will be:
+
+```html
+Hi Chris!
+Hi Mark!
+Hi Scott!
+```
+
+### Data structure
+
+The types in the example above, `mstch::array` and `mstch::map` are  actually 
+aliases for standard types:
+
+```c++
+using map = std::map<const std::string, node>;
+using array = std::vector<node>;
+```
+
+`mstch::node` is a `boost::variant` that can hold a `std::string`, `int`, 
+`double`, `bool`, `mstch::lambda` or a `std::shared_ptr<mstch::object>` 
+(see below), also a map or an array recursively. Essentially it works just like 
+a JSON object.
+
+Note that when using a `std::string` as value you must explicitly specify the 
+type, since a `const char*` literal like `"foobar"` would be implicitly 
+converted to `bool`. Alternatively you can use [C++14 string_literals](http://en.cppreference.com/w/cpp/string/basic_string/operator%22%22s)
+if your compiler supports it.
+
+## Advanced usage
+
+### Partials
+
+Partials can be passed in a `std::map` as the third parameter of the 
+`mstch::render` function:
+
+```c++
+std::string view{"{{#names}}{{> user}}{{/names}}"};
+std::string user_view{"<strong>{{name}}\n</strong>"};
+mstch::map context{
+  {"names", mstch::array{
+    mstch::map{{"name", std::string{"Chris"}}},
+    mstch::map{{"name", std::string{"Mark"}}},
+    mstch::map{{"name", std::string{"Scott"}}},
+  }}
+};
+  
+std::cout << mstch::render(view, context, {{"user", user_view}}) << std::endl;
+```
+
+Output:
+
+```html
+<strong>Chris</strong>
+<strong>Mark</strong>
+<strong>Scott</strong>
+```
+
+### Lambdas
+
+C++11 lambda expressions can be used to add logic to your templates. Like a
+`const char*` literal, lambdas can be implicitly converted to `bool`, so they
+must be wrapped in a `mstch::lambda` object when used in a `mstch::node`. The 
+lambda expression passed to `mstch::lambda` must itself return a `mstch::node`.
+The returned node will be rendered to a string, then it will be parsed as a
+template.
+
+The lambda expression accepts either no parameters:
+
+```c++
+std::string view{"Hello {{lambda}}!"};
+mstch::map context{
+  {"lambda", mstch::lambda{[]() -> mstch::node {
+    return std::string{"World"};
+  }}}
+};
+
+std::cout << mstch::render(view, context) << std::endl;
+```
+
+Output:
+
+```html
+Hello World!
+```
+
+Or it accepts a `const std::string&` that gets the unrendered literal block:
+
+```c++
+std::string view{"{{#bold}}{{yay}} :){{/bold}}"};
+mstch::map context{
+  {"yay", std::string{"Yay!"}},
+  {"bold", mstch::lambda{[](const std::string& text) -> mstch::node {
+    return "<b>" + text + "</b>";
+  }}}
+};
+
+std::cout << mstch::render(view, context) << std::endl;
+```
+
+Output:
+
+```html
+<b>Yay! :)</b>
+```
+
+### Objects
+
+Custom objects can also be used as context for rendering templates. The class 
+must inherit from `mstch::object`, and register it's exported methods with
+`register_methods`. Exported methods must have the return type of `mstch::node`.
+Objects must be created as a `std::shared_ptr`.
+
+```c++
+class example: public mstch::object {
+ public:
+  example(): m_value(1) {
+    register_methods(this, {
+      {"count", &example::count},
+      {"names", &example::names}  
+    });
+  }
+  
+  mstch::node count() {
+    return m_value++;
+  }
+  
+  mstch::node names() {
+    return mstch::array{
+      std::string{"Chris"}, std::string{"Mark"}, std::string{"Scott"}};
+  }
+  
+ private:
+  int m_value;
+};
+
+std::string view{"{{#names}}<b>{{count}}</b>: {{.}}\n{{/names}}"};
+const auto context = std::make_shared<example>();
+
+std::cout << mstch::render(view, context) << std::endl;
+```
+
+Output:
+
+```html
+<b>1</b>: Chris
+<b>2</b>: Mark
+<b>3</b>: Scott
+```
+
+### Custom escape function
+
+By default, mstch uses HTML escaping on the output, as per specification. This
+is not useful if your output is not HTML, so mstch provides a way to supply
+your own escape implementation. Just assign any callable object to the static
+`mstch::config::escape`, which is an initially empty
+`std::function<std::string(const std::string&)>`. 
+
+For example you can turn off escaping entirely with a lambda:
+
+```c++
+mstch::config::escape = [](const std::string& str) -> std::string {
+  return str;
+};
+```
+
+## Requirements
+
+ - A C++ compiler with decent C++11 support. Currently tested with:
+   - GCC 4.7, 4.8, 4.9, 5.1
+   - clang 3.5, 3.6, 3.7 (both libstdc++ and libc++ are supported)
+   - MSVC 2013, 2015
+ - Boost 1.54+ for [Boost.Variant](http://www.boost.org/libs/variant)
+ - CMake 3.0+ for building
+
+## Using mstch in your project
+
+If you are using CMake, the easiest way to include mstch in your project is to 
+copy the whole directory to your source tree, and use `add_subdirectory` in your 
+CMakeLists.txt. This will set a variable named `mstch_INCLUDE_DIR` that contains 
+its include path, and add a static library target named `mstch`. For example:
+
+```cmake
+add_subdirectory(external/mstch)
+include_directories(${mstch_INCLUDE_DIR})
+target_link_libraries(your_project mstch)
+```
+
+If you prefer to install the library globally, you can simply do the following 
+from the root of the source tree:
+
+```bash
+ $ mkdir build
+ $ cd build
+ $ cmake ..
+ $ make
+ $ make install
+```
+
+The install command may require root privileges. This will also install CMake 
+config files, so you can use use `find_package` in your CMakeLists.txt:
+
+```cmake
+find_package(mstch)
+target_link_libraries(your_project mstch::mstch)
+```
+
+## Running the unit tests
+
+Unit tests are using the [Catch](https://github.com/philsquared/Catch) framework
+and [rapidjson](http://rapidjson.org/) to parse the
+[Mustache specifications](https://github.com/mustache/spec), all of which are 
+included in the repository as git submodules. Various 
+[Boost](http://www.boost.org/) libraries are also required to build them.
+
+Don't forget to initialize submodules:
+
+```bash
+ $ git submodule init
+ $ git submodule update
+```
+
+To build and run the unit tests:
+
+```bash
+ $ mkdir build
+ $ cd build
+ $ cmake -DWITH_UNIT_TESTS=ON ..
+ $ make
+ $ make test
+```
+
+## License
+
+mstch is licensed under the [MIT license](https://github.com/no1msd/mstch/blob/master/LICENSE).

+ 6 - 0
lib/mstch/benchmark/CMakeLists.txt

@@ -0,0 +1,6 @@
+include_directories(
+  ${CMAKE_SOURCE_DIR}/include
+  ${CMAKE_SOURCE_DIR}/vendor/benchmark/include)
+
+add_executable(mstch_benchmark benchmark_main.cpp)
+target_link_libraries(mstch_benchmark mstch benchmark)

+ 26 - 0
lib/mstch/benchmark/benchmark_main.cpp

@@ -0,0 +1,26 @@
+#include <benchmark/benchmark.h>
+
+#include "mstch/mstch.hpp"
+
+static void basic_usage(benchmark::State& state) {
+    std::string comment_tmp{
+        "<div class=\"comments\"><h3>{{header}}</h3><ul>"
+        "{{#comments}}<li class=\"comment\"><h5>{{name}}</h5>"
+        "<p>{{body}}</p></li>{{/comments}}</ul></div>"};
+
+    auto comment_view = mstch::map{
+        {"header", std::string{"My Post Comments"}},
+            {"comments", mstch::array{
+                mstch::map{{"name", std::string{"Joe"}}, {"body", std::string{"Thanks for this post!"}}},
+                mstch::map{{"name", std::string{"Sam"}}, {"body", std::string{"Thanks for this post!"}}},
+                mstch::map{{"name", std::string{"Heather"}}, {"body", std::string{"Thanks for this post!"}}},
+                mstch::map{{"name", std::string{"Kathy"}}, {"body", std::string{"Thanks for this post!"}}},
+                mstch::map{{"name", std::string{"George"}}, {"body", std::string{"Thanks for this post!"}}}}}};
+
+    while (state.KeepRunning())
+        mstch::render(comment_tmp, comment_view);
+}
+
+BENCHMARK(basic_usage);
+
+BENCHMARK_MAIN();

+ 1 - 0
lib/mstch/cmake/mstch-config.cmake

@@ -0,0 +1 @@
+include("${CMAKE_CURRENT_LIST_DIR}/mstch-targets.cmake")

+ 113 - 0
lib/mstch/include/mstch/mstch.hpp

@@ -0,0 +1,113 @@
+#pragma once
+
+#include <vector>
+#include <map>
+#include <string>
+#include <memory>
+#include <functional>
+
+#include <boost/variant.hpp>
+
+namespace mstch {
+
+struct config {
+  static std::function<std::string(const std::string&)> escape;
+};
+
+namespace internal {
+
+template<class N>
+class object_t {
+ public:
+  const N& at(const std::string& name) const {
+    cache[name] = (methods.at(name))();
+    return cache[name];
+  }
+
+  bool has(const std::string name) const {
+    return methods.count(name) != 0;
+  }
+
+ protected:
+  template<class S>
+  void register_methods(S* s, std::map<std::string,N(S::*)()> methods) {
+    for(auto& item: methods)
+      this->methods.insert({item.first, std::bind(item.second, s)});
+  }
+
+ private:
+  std::map<std::string, std::function<N()>> methods;
+  mutable std::map<std::string, N> cache;
+};
+
+template<class T, class N>
+class is_fun {
+ private:
+  using not_fun = char;
+  using fun_without_args = char[2];
+  using fun_with_args = char[3];
+  template <typename U, U> struct really_has;
+  template <typename C> static fun_without_args& test(
+      really_has<N(C::*)() const, &C::operator()>*);
+  template <typename C> static fun_with_args& test(
+      really_has<N(C::*)(const std::string&) const,
+      &C::operator()>*);
+  template <typename> static not_fun& test(...);
+
+ public:
+  static bool const no_args = sizeof(test<T>(0)) == sizeof(fun_without_args);
+  static bool const has_args = sizeof(test<T>(0)) == sizeof(fun_with_args);
+};
+
+template<class N>
+using node_renderer = std::function<std::string(const N& n)>;
+
+template<class N>
+class lambda_t {
+ public:
+  template<class F>
+  lambda_t(F f, typename std::enable_if<is_fun<F, N>::no_args>::type* = 0):
+      fun([f](node_renderer<N> renderer, const std::string&) {
+        return renderer(f());
+      })
+  {
+  }
+
+  template<class F>
+  lambda_t(F f, typename std::enable_if<is_fun<F, N>::has_args>::type* = 0):
+      fun([f](node_renderer<N> renderer, const std::string& text) {
+        return renderer(f(text));
+      })
+  {
+  }
+
+  std::string operator()(node_renderer<N> renderer,
+      const std::string& text = "") const
+  {
+    return fun(renderer, text);
+  }
+
+ private:
+  std::function<std::string(node_renderer<N> renderer, const std::string&)> fun;
+};
+
+}
+
+using node = boost::make_recursive_variant<
+    std::nullptr_t, std::string, int, double, bool,
+    internal::lambda_t<boost::recursive_variant_>,
+    std::shared_ptr<internal::object_t<boost::recursive_variant_>>,
+    std::map<const std::string, boost::recursive_variant_>,
+    std::vector<boost::recursive_variant_>>::type;
+using object = internal::object_t<node>;
+using lambda = internal::lambda_t<node>;
+using map = std::map<const std::string, node>;
+using array = std::vector<node>;
+
+std::string render(
+    const std::string& tmplt,
+    const node& root,
+    const std::map<std::string,std::string>& partials =
+        std::map<std::string,std::string>());
+
+}

+ 64 - 0
lib/mstch/src/CMakeLists.txt

@@ -0,0 +1,64 @@
+find_package(Boost 1.54 REQUIRED)
+
+set(mstch_INCLUDE_DIR
+    ${PROJECT_SOURCE_DIR}/include CACHE STRING "mstch include directory")
+
+include_directories(
+    ${mstch_INCLUDE_DIR}
+    ${Boost_INCLUDE_DIR})
+
+set(SRC
+    state/in_section.cpp
+    state/outside_section.cpp
+    state/render_state.hpp
+    visitor/get_token.hpp
+    visitor/has_token.hpp
+    visitor/is_node_empty.hpp
+    visitor/render_node.hpp
+    visitor/render_section.hpp
+    mstch.cpp
+    render_context.cpp
+    template_type.cpp
+    token.cpp
+    utils.cpp)
+
+add_library(mstch STATIC ${SRC})
+
+set_property(TARGET mstch PROPERTY VERSION ${mstch_VERSION})
+
+install(
+    TARGETS mstch EXPORT mstchTargets
+    LIBRARY DESTINATION lib
+    ARCHIVE DESTINATION lib)
+
+install(
+    FILES "${PROJECT_SOURCE_DIR}/include/mstch/mstch.hpp"
+    DESTINATION include/mstch
+    COMPONENT Devel)
+
+include(CMakePackageConfigHelpers)
+write_basic_package_version_file(
+    "${CMAKE_CURRENT_BINARY_DIR}/mstch/mstch-config-version.cmake"
+    VERSION ${mstch_VERSION}
+    COMPATIBILITY AnyNewerVersion)
+
+export(
+    EXPORT mstchTargets
+    FILE "${CMAKE_CURRENT_BINARY_DIR}/mstch/mstch-targets.cmake"
+    NAMESPACE mstch::)
+
+configure_file(
+    "${PROJECT_SOURCE_DIR}/cmake/mstch-config.cmake"
+    "${CMAKE_CURRENT_BINARY_DIR}/mstch/mstch-config.cmake")
+
+install(
+    EXPORT mstchTargets
+    FILE mstch-targets.cmake
+    NAMESPACE mstch::
+    DESTINATION lib/cmake/mstch)
+
+install(FILES
+    "${PROJECT_SOURCE_DIR}/cmake/mstch-config.cmake"
+    "${CMAKE_CURRENT_BINARY_DIR}/mstch/mstch-config-version.cmake"
+    DESTINATION lib/cmake/mstch
+    COMPONENT Devel)

+ 20 - 0
lib/mstch/src/mstch.cpp

@@ -0,0 +1,20 @@
+#include <iostream>
+
+#include "mstch/mstch.hpp"
+#include "render_context.hpp"
+
+using namespace mstch;
+
+std::function<std::string(const std::string&)> mstch::config::escape;
+
+std::string mstch::render(
+    const std::string& tmplt,
+    const node& root,
+    const std::map<std::string,std::string>& partials)
+{
+  std::map<std::string, template_type> partial_templates;
+  for (auto& partial: partials)
+    partial_templates.insert({partial.first, {partial.second}});
+
+  return render_context(root, partial_templates).render(tmplt);
+}

+ 72 - 0
lib/mstch/src/render_context.cpp

@@ -0,0 +1,72 @@
+#include "render_context.hpp"
+#include "state/outside_section.hpp"
+#include "visitor/get_token.hpp"
+
+using namespace mstch;
+
+const mstch::node render_context::null_node;
+
+render_context::push::push(render_context& context, const mstch::node& node):
+    m_context(context)
+{
+  context.m_nodes.emplace_front(node);
+  context.m_node_ptrs.emplace_front(&node);
+  context.m_state.push(std::unique_ptr<render_state>(new outside_section));
+}
+
+render_context::push::~push() {
+  m_context.m_nodes.pop_front();
+  m_context.m_node_ptrs.pop_front();
+  m_context.m_state.pop();
+}
+
+std::string render_context::push::render(const template_type& templt) {
+  return m_context.render(templt);
+}
+
+render_context::render_context(
+    const mstch::node& node,
+    const std::map<std::string, template_type>& partials):
+    m_partials(partials), m_nodes(1, node), m_node_ptrs(1, &node)
+{
+  m_state.push(std::unique_ptr<render_state>(new outside_section));
+}
+
+const mstch::node& render_context::find_node(
+    const std::string& token,
+    std::list<node const*> current_nodes)
+{
+  if (token != "." && token.find('.') != std::string::npos)
+    return find_node(token.substr(token.rfind('.') + 1),
+        {&find_node(token.substr(0, token.rfind('.')), current_nodes)});
+  else
+    for (auto& node: current_nodes)
+      if (visit(has_token(token), *node))
+        return visit(get_token(token, *node), *node);
+  return null_node;
+}
+
+const mstch::node& render_context::get_node(const std::string& token) {
+    return find_node(token, m_node_ptrs);
+}
+
+std::string render_context::render(
+    const template_type& templt, const std::string& prefix)
+{
+  std::string output;
+  bool prev_eol = true;
+  for (auto& token: templt) {
+    if (prev_eol && prefix.length() != 0)
+      output += m_state.top()->render(*this, {prefix});
+    output += m_state.top()->render(*this, token);
+    prev_eol = token.eol();
+  }
+  return output;
+}
+
+std::string render_context::render_partial(
+    const std::string& partial_name, const std::string& prefix)
+{
+  return m_partials.count(partial_name) ?
+      render(m_partials.at(partial_name), prefix) : "";
+}

+ 51 - 0
lib/mstch/src/render_context.hpp

@@ -0,0 +1,51 @@
+#pragma once
+
+#include <deque>
+#include <list>
+#include <sstream>
+#include <string>
+#include <stack>
+
+#include "mstch/mstch.hpp"
+#include "state/render_state.hpp"
+#include "template_type.hpp"
+
+namespace mstch {
+
+class render_context {
+ public:
+  class push {
+   public:
+    push(render_context& context, const mstch::node& node = {});
+    ~push();
+    std::string render(const template_type& templt);
+   private:
+    render_context& m_context;
+  };
+
+  render_context(
+      const mstch::node& node,
+      const std::map<std::string, template_type>& partials);
+  const mstch::node& get_node(const std::string& token);
+  std::string render(
+      const template_type& templt, const std::string& prefix = "");
+  std::string render_partial(
+      const std::string& partial_name, const std::string& prefix);
+  template<class T, class... Args>
+  void set_state(Args&& ... args) {
+    m_state.top() = std::unique_ptr<render_state>(
+        new T(std::forward<Args>(args)...));
+  }
+
+ private:
+  static const mstch::node null_node;
+  const mstch::node& find_node(
+      const std::string& token,
+      std::list<node const*> current_nodes);
+  std::map<std::string, template_type> m_partials;
+  std::deque<mstch::node> m_nodes;
+  std::list<const mstch::node*> m_node_ptrs;
+  std::stack<std::unique_ptr<render_state>> m_state;
+};
+
+}

+ 34 - 0
lib/mstch/src/state/in_section.cpp

@@ -0,0 +1,34 @@
+#include "in_section.hpp"
+#include "outside_section.hpp"
+#include "visitor/is_node_empty.hpp"
+#include "visitor/render_section.hpp"
+
+using namespace mstch;
+
+in_section::in_section(type type, const token& start_token):
+    m_type(type), m_start_token(start_token), m_skipped_openings(0)
+{
+}
+
+std::string in_section::render(render_context& ctx, const token& token) {
+  if (token.token_type() == token::type::section_close)
+    if (token.name() == m_start_token.name() && m_skipped_openings == 0) {
+      auto& node = ctx.get_node(m_start_token.name());
+      std::string out;
+
+      if (m_type == type::normal && !visit(is_node_empty(), node))
+        out = visit(render_section(ctx, m_section, m_start_token.delims()), node);
+      else if (m_type == type::inverted && visit(is_node_empty(), node))
+        out = render_context::push(ctx).render(m_section);
+
+      ctx.set_state<outside_section>();
+      return out;
+    } else
+      m_skipped_openings--;
+  else if (token.token_type() == token::type::inverted_section_open ||
+      token.token_type() == token::type::section_open)
+    m_skipped_openings++;
+
+  m_section << token;
+  return "";
+}

+ 24 - 0
lib/mstch/src/state/in_section.hpp

@@ -0,0 +1,24 @@
+#pragma once
+
+#include <sstream>
+#include <vector>
+
+#include "render_state.hpp"
+#include "template_type.hpp"
+
+namespace mstch {
+
+class in_section: public render_state {
+ public:
+  enum class type { inverted, normal };
+  in_section(type type, const token& start_token);
+  std::string render(render_context& context, const token& token) override;
+
+ private:
+  const type m_type;
+  const token& m_start_token;
+  template_type m_section;
+  int m_skipped_openings;
+};
+
+}

+ 32 - 0
lib/mstch/src/state/outside_section.cpp

@@ -0,0 +1,32 @@
+#include "outside_section.hpp"
+
+#include "visitor/render_node.hpp"
+#include "in_section.hpp"
+#include "render_context.hpp"
+
+using namespace mstch;
+
+std::string outside_section::render(
+    render_context& ctx, const token& token)
+{
+  using flag = render_node::flag;
+  switch (token.token_type()) {
+    case token::type::section_open:
+      ctx.set_state<in_section>(in_section::type::normal, token);
+      break;
+    case token::type::inverted_section_open:
+      ctx.set_state<in_section>(in_section::type::inverted, token);
+      break;
+    case token::type::variable:
+      return visit(render_node(ctx, flag::escape_html), ctx.get_node(token.name()));
+    case token::type::unescaped_variable:
+      return visit(render_node(ctx, flag::none), ctx.get_node(token.name()));
+    case token::type::text:
+      return token.raw();
+    case token::type::partial:
+      return ctx.render_partial(token.name(), token.partial_prefix());
+    default:
+      break;
+  }
+  return "";
+}

+ 12 - 0
lib/mstch/src/state/outside_section.hpp

@@ -0,0 +1,12 @@
+#pragma once
+
+#include "render_state.hpp"
+
+namespace mstch {
+
+class outside_section: public render_state {
+ public:
+  std::string render(render_context& context, const token& token) override;
+};
+
+}

+ 17 - 0
lib/mstch/src/state/render_state.hpp

@@ -0,0 +1,17 @@
+#pragma once
+
+#include <memory>
+
+#include "token.hpp"
+
+namespace mstch {
+
+class render_context;
+
+class render_state {
+ public:
+  virtual ~render_state() {}
+  virtual std::string render(render_context& context, const token& token) = 0;
+};
+
+}

+ 104 - 0
lib/mstch/src/template_type.cpp

@@ -0,0 +1,104 @@
+#include "template_type.hpp"
+
+using namespace mstch;
+
+template_type::template_type(const std::string& str, const delim_type& delims):
+    m_open(delims.first), m_close(delims.second)
+{
+  tokenize(str);
+  strip_whitespace();
+}
+
+template_type::template_type(const std::string& str):
+    m_open("{{"), m_close("}}")
+{
+  tokenize(str);
+  strip_whitespace();
+}
+
+void template_type::process_text(citer begin, citer end) {
+  if (begin == end)
+    return;
+  auto start = begin;
+  for (auto it = begin; it != end; ++it)
+    if (*it == '\n' || it == end - 1) {
+      m_tokens.push_back({{start, it + 1}});
+      start = it + 1;
+    }
+}
+
+void template_type::tokenize(const std::string& tmp) {
+  citer beg = tmp.begin();
+  auto npos = std::string::npos;
+
+  for (std::size_t cur_pos = 0; cur_pos < tmp.size();) {
+    auto open_pos = tmp.find(m_open, cur_pos);
+    auto close_pos = tmp.find(
+        m_close, open_pos == npos ? open_pos : open_pos + 1);
+
+    if (close_pos != npos && open_pos != npos) {
+      if (*(beg + open_pos + m_open.size()) == '{' &&
+          *(beg + close_pos + m_close.size()) == '}')
+        ++close_pos;
+
+      process_text(beg + cur_pos, beg + open_pos);
+      cur_pos = close_pos + m_close.size();
+      m_tokens.push_back({{beg + open_pos, beg + close_pos + m_close.size()},
+          m_open.size(), m_close.size()});
+
+      if (cur_pos == tmp.size()) {
+        m_tokens.push_back({{""}});
+        m_tokens.back().eol(true);
+      }
+
+      if (*(beg + open_pos + m_open.size()) == '=' &&
+          *(beg + close_pos - 1) == '=')
+      {
+        auto tok_beg = beg + open_pos + m_open.size() + 1;
+        auto tok_end = beg + close_pos - 1;
+        auto front_skip = first_not_ws(tok_beg, tok_end);
+        auto back_skip = first_not_ws(reverse(tok_end), reverse(tok_beg));
+        m_open = {front_skip, beg + tmp.find(' ', front_skip - beg)};
+        m_close = {beg + tmp.rfind(' ', back_skip - beg) + 1, back_skip + 1};
+      }
+    } else {
+      process_text(beg + cur_pos, tmp.end());
+      cur_pos = close_pos;
+    }
+  }
+}
+
+void template_type::strip_whitespace() {
+  auto line_begin = m_tokens.begin();
+  bool has_tag = false, non_space = false;
+
+  for (auto it = m_tokens.begin(); it != m_tokens.end(); ++it) {
+    auto type = (*it).token_type();
+    if (type != token::type::text && type != token::type::variable &&
+        type != token::type::unescaped_variable)
+      has_tag = true;
+    else if (!(*it).ws_only())
+      non_space = true;
+
+    if ((*it).eol()) {
+      if (has_tag && !non_space) {
+        store_prefixes(line_begin);
+
+        auto c = line_begin;
+        for (bool end = false; !end; (*c).ws_only() ? c = m_tokens.erase(c) : ++c)
+          if ((end = (*c).eol()))
+            it = c - 1;
+      }
+
+      non_space = has_tag = false;
+      line_begin = it + 1;
+    }
+  }
+}
+
+void template_type::store_prefixes(std::vector<token>::iterator beg) {
+  for (auto cur = beg; !(*cur).eol(); ++cur)
+    if ((*cur).token_type() == token::type::partial &&
+        cur != beg && (*(cur - 1)).ws_only())
+      (*cur).partial_prefix((*(cur - 1)).raw());
+}

+ 30 - 0
lib/mstch/src/template_type.hpp

@@ -0,0 +1,30 @@
+#pragma once
+
+#include <string>
+#include <vector>
+
+#include "token.hpp"
+#include "utils.hpp"
+
+namespace mstch {
+
+class template_type {
+ public:
+  template_type() = default;
+  template_type(const std::string& str);
+  template_type(const std::string& str, const delim_type& delims);
+  std::vector<token>::const_iterator begin() const { return m_tokens.begin(); }
+  std::vector<token>::const_iterator end() const { return m_tokens.end(); }
+  void operator<<(const token& token) { m_tokens.push_back(token); }
+
+ private:
+  std::vector<token> m_tokens;
+  std::string m_open;
+  std::string m_close;
+  void strip_whitespace();
+  void process_text(citer beg, citer end);
+  void tokenize(const std::string& tmp);
+  void store_prefixes(std::vector<token>::iterator beg);
+};
+
+}

+ 42 - 0
lib/mstch/src/token.cpp

@@ -0,0 +1,42 @@
+#include "token.hpp"
+#include "utils.hpp"
+
+using namespace mstch;
+
+token::type token::token_info(char c) {
+  switch (c) {
+    case '>': return type::partial;
+    case '^': return type::inverted_section_open;
+    case '/': return type::section_close;
+    case '&': return type::unescaped_variable;
+    case '#': return type::section_open;
+    case '!': return type::comment;
+    default: return type::variable;
+  }
+}
+
+token::token(const std::string& str, std::size_t left, std::size_t right):
+    m_raw(str), m_eol(false), m_ws_only(false)
+{
+  if (left != 0 && right != 0) {
+    if (str[left] == '=' && str[str.size() - right - 1] == '=') {
+      m_type = type::delimiter_change;
+    } else if (str[left] == '{' && str[str.size() - right - 1] == '}') {
+      m_type = type::unescaped_variable;
+      m_name = {first_not_ws(str.begin() + left + 1, str.end() - right),
+          first_not_ws(str.rbegin() + 1 + right, str.rend() - left) + 1};
+    } else {
+      auto c = first_not_ws(str.begin() + left, str.end() - right);
+      m_type = token_info(*c);
+      if (m_type != type::variable)
+        c = first_not_ws(c + 1, str.end() - right);
+      m_name = {c, first_not_ws(str.rbegin() + right, str.rend() - left) + 1};
+      m_delims = {{str.begin(), str.begin() + left},
+          {str.end() - right, str.end()}};
+    }
+  } else {
+    m_type = type::text;
+    m_eol = (str.size() > 0 && str[str.size() - 1] == '\n');
+    m_ws_only = (str.find_first_not_of(" \r\n\t") == std::string::npos);
+  }
+}

+ 39 - 0
lib/mstch/src/token.hpp

@@ -0,0 +1,39 @@
+#pragma once
+
+#include <string>
+
+namespace mstch {
+
+using delim_type = std::pair<std::string, std::string>;
+
+class token {
+ public:
+  enum class type {
+    text, variable, section_open, section_close, inverted_section_open,
+    unescaped_variable, comment, partial, delimiter_change
+  };
+  token(const std::string& str, std::size_t left = 0, std::size_t right = 0);
+  type token_type() const { return m_type; };
+  const std::string& raw() const { return m_raw; };
+  const std::string& name() const { return m_name; };
+  const std::string& partial_prefix() const { return m_partial_prefix; };
+  const delim_type& delims() const { return m_delims; };
+  void partial_prefix(const std::string& p_partial_prefix) {
+    m_partial_prefix = p_partial_prefix;
+  };
+  bool eol() const { return m_eol; }
+  void eol(bool eol) { m_eol = eol; }
+  bool ws_only() const { return m_ws_only; }
+
+ private:
+  type m_type;
+  std::string m_name;
+  std::string m_raw;
+  std::string m_partial_prefix;
+  delim_type m_delims;
+  bool m_eol;
+  bool m_ws_only;
+  type token_info(char c);
+};
+
+}

+ 44 - 0
lib/mstch/src/utils.cpp

@@ -0,0 +1,44 @@
+#include "utils.hpp"
+#include "mstch/mstch.hpp"
+
+mstch::citer mstch::first_not_ws(mstch::citer begin, mstch::citer end) {
+  for (auto it = begin; it != end; ++it)
+    if (*it != ' ') return it;
+  return end;
+}
+
+mstch::citer mstch::first_not_ws(mstch::criter begin, mstch::criter end) {
+  for (auto rit = begin; rit != end; ++rit)
+    if (*rit != ' ') return --(rit.base());
+  return --(end.base());
+}
+
+mstch::criter mstch::reverse(mstch::citer it) {
+  return std::reverse_iterator<mstch::citer>(it);
+}
+
+std::string mstch::html_escape(const std::string& str) {
+  if (mstch::config::escape)
+    return mstch::config::escape(str);
+  
+  std::string out;
+  citer start = str.begin();
+
+  auto add_escape = [&out, &start](const std::string& escaped, citer& it) {
+    out += std::string{start, it} + escaped;
+    start = it + 1;
+  };
+
+  for (auto it = str.begin(); it != str.end(); ++it)
+    switch (*it) {
+      case '&': add_escape("&amp;", it); break;
+      case '\'': add_escape("&#39;", it); break;
+      case '"': add_escape("&quot;", it); break;
+      case '<': add_escape("&lt;", it); break;
+      case '>': add_escape("&gt;", it); break;
+      case '/': add_escape("&#x2F;", it); break;
+      default: break;
+    }
+
+  return out + std::string{start, str.end()};
+}

+ 23 - 0
lib/mstch/src/utils.hpp

@@ -0,0 +1,23 @@
+#pragma once
+
+#include <string>
+#include <boost/variant/apply_visitor.hpp>
+
+namespace mstch {
+
+using citer = std::string::const_iterator;
+using criter = std::string::const_reverse_iterator;
+
+citer first_not_ws(citer begin, citer end);
+citer first_not_ws(criter begin, criter end);
+std::string html_escape(const std::string& str);
+criter reverse(citer it);
+
+template<class... Args>
+auto visit(Args&&... args) -> decltype(boost::apply_visitor(
+    std::forward<Args>(args)...))
+{
+  return boost::apply_visitor(std::forward<Args>(args)...);
+}
+
+}

+ 35 - 0
lib/mstch/src/visitor/get_token.hpp

@@ -0,0 +1,35 @@
+#pragma once
+
+#include <boost/variant/static_visitor.hpp>
+
+#include "mstch/mstch.hpp"
+#include "has_token.hpp"
+
+namespace mstch {
+
+class get_token: public boost::static_visitor<const mstch::node&> {
+ public:
+  get_token(const std::string& token, const mstch::node& node):
+      m_token(token), m_node(node)
+  {
+  }
+
+  template<class T>
+  const mstch::node& operator()(const T&) const {
+    return m_node;
+  }
+
+  const mstch::node& operator()(const map& map) const {
+    return map.at(m_token);
+  }
+
+  const mstch::node& operator()(const std::shared_ptr<object>& object) const {
+    return object->at(m_token);
+  }
+
+ private:
+  const std::string& m_token;
+  const mstch::node& m_node;
+};
+
+}

+ 31 - 0
lib/mstch/src/visitor/has_token.hpp

@@ -0,0 +1,31 @@
+#pragma once
+
+#include <boost/variant/static_visitor.hpp>
+
+#include "mstch/mstch.hpp"
+
+namespace mstch {
+
+class has_token: public boost::static_visitor<bool> {
+ public:
+  has_token(const std::string& token): m_token(token) {
+  }
+
+  template<class T>
+  bool operator()(const T&) const {
+    return m_token == ".";
+  }
+
+  bool operator()(const map& map) const {
+    return map.count(m_token) == 1;
+  }
+
+  bool operator()(const std::shared_ptr<object>& object) const {
+    return object->has(m_token);
+  }
+
+ private:
+  const std::string& m_token;
+};
+
+}

+ 41 - 0
lib/mstch/src/visitor/is_node_empty.hpp

@@ -0,0 +1,41 @@
+#pragma once
+
+#include <boost/variant/static_visitor.hpp>
+
+#include "mstch/mstch.hpp"
+
+namespace mstch {
+
+class is_node_empty: public boost::static_visitor<bool> {
+ public:
+  template<class T>
+  bool operator()(const T&) const {
+    return false;
+  }
+
+  bool operator()(const std::nullptr_t&) const {
+    return true;
+  }
+
+  bool operator()(const int& value) const {
+    return value == 0;
+  }
+
+  bool operator()(const double& value) const {
+    return value == 0;
+  }
+
+  bool operator()(const bool& value) const {
+    return !value;
+  }
+
+  bool operator()(const std::string& value) const {
+    return value == "";
+  }
+
+  bool operator()(const array& array) const {
+    return array.size() == 0;
+  }
+};
+
+}

+ 56 - 0
lib/mstch/src/visitor/render_node.hpp

@@ -0,0 +1,56 @@
+#pragma once
+
+#include <sstream>
+#include <boost/variant/static_visitor.hpp>
+
+#include "render_context.hpp"
+#include "mstch/mstch.hpp"
+#include "utils.hpp"
+
+namespace mstch {
+
+class render_node: public boost::static_visitor<std::string> {
+ public:
+  enum class flag { none, escape_html };
+  render_node(render_context& ctx, flag p_flag = flag::none):
+      m_ctx(ctx), m_flag(p_flag)
+  {
+  }
+
+  template<class T>
+  std::string operator()(const T&) const {
+    return "";
+  }
+
+  std::string operator()(const int& value) const {
+    return std::to_string(value);
+  }
+
+  std::string operator()(const double& value) const {
+    std::stringstream ss;
+    ss << value;
+    return ss.str();
+  }
+
+  std::string operator()(const bool& value) const {
+    return value ? "true" : "false";
+  }
+
+  std::string operator()(const lambda& value) const {
+    template_type interpreted{value([this](const mstch::node& n) {
+      return visit(render_node(m_ctx), n);
+    })};
+    auto rendered = render_context::push(m_ctx).render(interpreted);
+    return (m_flag == flag::escape_html) ? html_escape(rendered) : rendered;
+  }
+
+  std::string operator()(const std::string& value) const {
+    return (m_flag == flag::escape_html) ? html_escape(value) : value;
+  }
+
+ private:
+  render_context& m_ctx;
+  flag m_flag;
+};
+
+}

+ 57 - 0
lib/mstch/src/visitor/render_section.hpp

@@ -0,0 +1,57 @@
+#pragma once
+
+#include <boost/variant/static_visitor.hpp>
+
+#include "render_context.hpp"
+#include "mstch/mstch.hpp"
+#include "utils.hpp"
+#include "render_node.hpp"
+
+namespace mstch {
+
+class render_section: public boost::static_visitor<std::string> {
+ public:
+  enum class flag { none, keep_array };
+  render_section(
+      render_context& ctx,
+      const template_type& section,
+      const delim_type& delims,
+      flag p_flag = flag::none):
+      m_ctx(ctx), m_section(section), m_delims(delims), m_flag(p_flag)
+  {
+  }
+
+  template<class T>
+  std::string operator()(const T& t) const {
+    return render_context::push(m_ctx, t).render(m_section);
+  }
+
+  std::string operator()(const lambda& fun) const {
+    std::string section_str;
+    for (auto& token: m_section)
+      section_str += token.raw();
+    template_type interpreted{fun([this](const mstch::node& n) {
+      return visit(render_node(m_ctx), n);
+    }, section_str), m_delims};
+    return render_context::push(m_ctx).render(interpreted);
+  }
+
+  std::string operator()(const array& array) const {
+    std::string out;
+    if (m_flag == flag::keep_array)
+      return render_context::push(m_ctx, array).render(m_section);
+    else
+      for (auto& item: array)
+        out += visit(render_section(
+            m_ctx, m_section, m_delims, flag::keep_array), item);
+    return out;
+  }
+
+ private:
+  render_context& m_ctx;
+  const template_type& m_section;
+  const delim_type& m_delims;
+  flag m_flag;
+};
+
+}

+ 59 - 0
lib/mstch/test/CMakeLists.txt

@@ -0,0 +1,59 @@
+find_package(Boost 1.54 REQUIRED)
+
+include_directories(
+  ${CMAKE_SOURCE_DIR}/include
+  ${CMAKE_SOURCE_DIR}/vendor/Catch/single_include
+  ${CMAKE_SOURCE_DIR}/vendor/rapidjson/include
+  ${Boost_INCLUDE_DIR})
+
+file(GLOB data_files RELATIVE
+  "${CMAKE_SOURCE_DIR}/test/data"
+  "${CMAKE_SOURCE_DIR}/test/data/*.hpp")
+
+foreach(data_file ${data_files})
+  string(REGEX REPLACE "\\.hpp" "" test_name "${data_file}")
+  list(APPEND tests "${test_name}")
+endforeach(data_file)
+
+file(GLOB string_files RELATIVE
+  "${CMAKE_SOURCE_DIR}/test/data"
+  "${CMAKE_SOURCE_DIR}/test/data/*.mustache"
+  "${CMAKE_SOURCE_DIR}/test/data/*.txt"
+  "${CMAKE_SOURCE_DIR}/test/data/*.partial")
+
+foreach(string_file ${string_files})
+  list(APPEND genargs "-i${string_file}")
+endforeach(string_file)
+
+add_custom_command(
+  OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/test_data.hpp
+  COMMAND headerize --output ${CMAKE_CURRENT_BINARY_DIR}/test_data.hpp --namespace mstchtest ${genargs}
+  WORKING_DIRECTORY ${CMAKE_SOURCE_DIR}/test/data/)
+set_source_files_properties(${CMAKE_CURRENT_BINARY_DIR}/test_data.hpp PROPERTIES GENERATED TRUE)
+add_custom_target(test_data_hpp DEPENDS ${CMAKE_CURRENT_BINARY_DIR}/test_data.hpp)
+
+file(GLOB specs_files RELATIVE
+  "${CMAKE_SOURCE_DIR}/vendor/spec/specs"
+  "${CMAKE_SOURCE_DIR}/vendor/spec/specs/*.json")
+
+foreach(specs_file ${specs_files})
+  list(APPEND specsargs "-i${specs_file}")
+  string(REGEX REPLACE "\\.json" "" test_name "${specs_file}")
+  string(REGEX REPLACE "~" "" test_name "${test_name}")
+  list(APPEND tests "specs_${test_name}")
+endforeach(specs_file)
+
+add_custom_command(
+  OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/specs_data.hpp
+  COMMAND headerize --output ${CMAKE_CURRENT_BINARY_DIR}/specs_data.hpp --namespace mstchtest ${specsargs}
+  WORKING_DIRECTORY ${CMAKE_SOURCE_DIR}/vendor/spec/specs/)
+set_source_files_properties(${CMAKE_CURRENT_BINARY_DIR}/specs_data.hpp PROPERTIES GENERATED TRUE)
+add_custom_target(specs_data_hpp DEPENDS ${CMAKE_CURRENT_BINARY_DIR}/specs_data.hpp)
+
+add_executable(mstch_test test_main.cpp)
+target_link_libraries(mstch_test mstch)
+add_dependencies(mstch_test test_data_hpp specs_data_hpp)
+
+foreach(test ${tests})
+  add_test(NAME ${test} COMMAND mstch_test ${test})
+endforeach(test)

+ 3 - 0
lib/mstch/test/data/ampersand_escape.hpp

@@ -0,0 +1,3 @@
+const auto ampersand_escape_data = mstch::map{
+  {"message", std::string{"Some <code>"}}
+};

+ 1 - 0
lib/mstch/test/data/ampersand_escape.mustache

@@ -0,0 +1 @@
+{{&message}}

+ 1 - 0
lib/mstch/test/data/ampersand_escape.txt

@@ -0,0 +1 @@
+Some <code>

+ 4 - 0
lib/mstch/test/data/apostrophe.hpp

@@ -0,0 +1,4 @@
+const auto apostrophe_data = mstch::map{
+  {"apos", std::string{"'"}},
+  {"control", std::string{"X"}}
+};

+ 1 - 0
lib/mstch/test/data/apostrophe.mustache

@@ -0,0 +1 @@
+{{apos}}{{control}}

+ 1 - 0
lib/mstch/test/data/apostrophe.txt

@@ -0,0 +1 @@
+&#39;X

+ 3 - 0
lib/mstch/test/data/array_of_strings.hpp

@@ -0,0 +1,3 @@
+const auto array_of_strings_data = mstch::map{
+  {"array_of_strings", mstch::array{std::string{"hello"}, std::string{"world"}}}
+};

+ 1 - 0
lib/mstch/test/data/array_of_strings.mustache

@@ -0,0 +1 @@
+{{#array_of_strings}}{{.}} {{/array_of_strings}}

+ 1 - 0
lib/mstch/test/data/array_of_strings.txt

@@ -0,0 +1 @@
+hello world 

+ 3 - 0
lib/mstch/test/data/backslashes.hpp

@@ -0,0 +1,3 @@
+const auto backslashes_data = mstch::map{
+  {"value", std::string{"\\abc"}}
+};

+ 7 - 0
lib/mstch/test/data/backslashes.mustache

@@ -0,0 +1,7 @@
+* {{value}}
+* {{{value}}}
+* {{&value}}
+<script>
+foo = { bar: 'abc\"xyz\"' };
+foo = { bar: 'x\'y' };
+</script>

+ 7 - 0
lib/mstch/test/data/backslashes.txt

@@ -0,0 +1,7 @@
+* \abc
+* \abc
+* \abc
+<script>
+foo = { bar: 'abc\"xyz\"' };
+foo = { bar: 'x\'y' };
+</script>

+ 3 - 0
lib/mstch/test/data/bug_11_eating_whitespace.hpp

@@ -0,0 +1,3 @@
+const auto bug_11_eating_whitespace_data = mstch::map{
+  {"tag", std::string{"yo"}}
+};

+ 1 - 0
lib/mstch/test/data/bug_11_eating_whitespace.mustache

@@ -0,0 +1 @@
+{{tag}} foo

+ 1 - 0
lib/mstch/test/data/bug_11_eating_whitespace.txt

@@ -0,0 +1 @@
+yo foo

+ 3 - 0
lib/mstch/test/data/bug_length_property.hpp

@@ -0,0 +1,3 @@
+const auto bug_length_property_data = mstch::map{
+    {"length", std::string{"hello"}}
+};

+ 1 - 0
lib/mstch/test/data/bug_length_property.mustache

@@ -0,0 +1 @@
+{{#length}}The length variable is: {{length}}{{/length}}

+ 1 - 0
lib/mstch/test/data/bug_length_property.txt

@@ -0,0 +1 @@
+The length variable is: hello

+ 4 - 0
lib/mstch/test/data/changing_delimiters.hpp

@@ -0,0 +1,4 @@
+const mstch::node changing_delimiters_data = mstch::map{
+  {"foo", std::string{"foooooooooooooo"}},
+  {"bar", std::string{"<b>bar!</b>"}}
+};

+ 1 - 0
lib/mstch/test/data/changing_delimiters.mustache

@@ -0,0 +1 @@
+{{=<% %>=}}<% foo %> {{foo}} <%{bar}%> {{{bar}}}

+ 1 - 0
lib/mstch/test/data/changing_delimiters.txt

@@ -0,0 +1 @@
+foooooooooooooo {{foo}} <b>bar!</b> {{{bar}}}

+ 3 - 0
lib/mstch/test/data/comments.hpp

@@ -0,0 +1,3 @@
+const mstch::node comments_data = mstch::map{
+    {"title", mstch::lambda{[]()->mstch::node{return std::string{"A Comedy of Errors"};}}}
+};

+ 1 - 0
lib/mstch/test/data/comments.mustache

@@ -0,0 +1 @@
+<h1>{{title}}{{! just something interesting... or not... }}</h1>

+ 1 - 0
lib/mstch/test/data/comments.txt

@@ -0,0 +1 @@
+<h1>A Comedy of Errors</h1>

+ 69 - 0
lib/mstch/test/data/complex.hpp

@@ -0,0 +1,69 @@
+class complex_item: public mstch::object {
+private:
+  std::string m_name;
+  bool m_current;
+  std::string m_url;
+public:
+  complex_item(const std::string& name, bool current, const std::string& url):
+      m_name(name), m_current(current), m_url(url)
+  {
+    register_methods(this, std::map<std::string,mstch::node(complex_item::*)()>{
+      {"name", &complex_item::name}, {"current", &complex_item::current},
+      {"url", &complex_item::url},   {"link", &complex_item::link}
+    });
+  }
+
+  mstch::node current() {
+    return m_current;
+  }
+
+  mstch::node url() {
+    return m_url;
+  }
+
+  mstch::node name() {
+    return m_name;
+  }
+
+  mstch::node link() {
+    return !m_current;
+  }
+};
+
+class complex: public mstch::object {
+private:
+  std::string m_header;
+  mstch::array m_item;
+public:
+  complex():
+      m_header("Colors"),
+      m_item(mstch::array{
+        std::make_shared<complex_item>("red", true, "#Red"),
+        std::make_shared<complex_item>("green", false, "#Green"),
+        std::make_shared<complex_item>("blue", false, "#Blue")
+      })
+  {
+    register_methods(this, std::map<std::string,mstch::node(complex::*)()>{
+      {"header", &complex::header}, {"item", &complex::item},
+      {"list", &complex::list}, {"empty", &complex::empty}
+    });
+  }
+
+  mstch::node header() {
+    return m_header;
+  }
+
+  mstch::node item() {
+    return m_item;
+  }
+
+  mstch::node list() {
+    return m_item.size() != 0;
+  }
+
+  mstch::node empty() {
+    return m_item.size() == 0;
+  }
+};
+
+const auto complex_data = std::make_shared<complex>();

+ 16 - 0
lib/mstch/test/data/complex.mustache

@@ -0,0 +1,16 @@
+<h1>{{header}}</h1>
+{{#list}}
+  <ul>
+  {{#item}}
+  {{#current}}
+  <li><strong>{{name}}</strong></li>
+  {{/current}}
+  {{#link}}
+  <li><a href="{{url}}">{{name}}</a></li>
+  {{/link}}
+  {{/item}}
+  </ul>
+{{/list}}
+{{#empty}}
+  <p>The list is empty.</p>
+{{/empty}}

+ 6 - 0
lib/mstch/test/data/complex.txt

@@ -0,0 +1,6 @@
+<h1>Colors</h1>
+  <ul>
+  <li><strong>red</strong></li>
+  <li><a href="#Green">green</a></li>
+  <li><a href="#Blue">blue</a></li>
+  </ul>

+ 8 - 0
lib/mstch/test/data/context_lookup.hpp

@@ -0,0 +1,8 @@
+const auto context_lookup_data = mstch::map{
+  {"outer", mstch::map{
+    {"id", 1},
+    {"second", mstch::map{
+      {"nothing", 2}
+    }}
+  }}
+};

+ 1 - 0
lib/mstch/test/data/context_lookup.mustache

@@ -0,0 +1 @@
+{{#outer}}{{#second}}{{id}}{{/second}}{{/outer}}

+ 1 - 0
lib/mstch/test/data/context_lookup.txt

@@ -0,0 +1 @@
+1

+ 6 - 0
lib/mstch/test/data/delimiters.hpp

@@ -0,0 +1,6 @@
+const mstch::node delimiters_data = mstch::map{
+  {"first", std::string{"It worked the first time."}},
+  {"second", std::string{"And it worked the second time."}},
+  {"third", std::string{"Then, surprisingly, it worked the third time."}},
+  {"fourth", std::string{"Fourth time also fine!."}}
+};

+ 7 - 0
lib/mstch/test/data/delimiters.mustache

@@ -0,0 +1,7 @@
+{{=<% %>=}}*
+<% first %>
+* <% second %>
+<%=| |=%>
+* | third |
+|={{ }}=|
+* {{ fourth }}

+ 5 - 0
lib/mstch/test/data/delimiters.txt

@@ -0,0 +1,5 @@
+*
+It worked the first time.
+* And it worked the second time.
+* Then, surprisingly, it worked the third time.
+* Fourth time also fine!.

+ 4 - 0
lib/mstch/test/data/disappearing_whitespace.hpp

@@ -0,0 +1,4 @@
+const auto disappearing_whitespace_data = mstch::map{
+  {"bedrooms", true},
+  {"total", 1}
+};

+ 1 - 0
lib/mstch/test/data/disappearing_whitespace.mustache

@@ -0,0 +1 @@
+{{#bedrooms}}{{total}}{{/bedrooms}} BED

+ 1 - 0
lib/mstch/test/data/disappearing_whitespace.txt

@@ -0,0 +1 @@
+1 BED

+ 34 - 0
lib/mstch/test/data/dot_notation.hpp

@@ -0,0 +1,34 @@
+class dot_notation_price: public mstch::object {
+private:
+  int m_value;
+  mstch::map m_currency;
+public:
+  dot_notation_price():
+    m_value(200), m_currency(mstch::map{{"symbol", std::string{"$"}}, {"name", std::string{"USD"}}})
+  {
+    register_methods(this, std::map<std::string,mstch::node(dot_notation_price::*)()>{
+      {"value", &dot_notation_price::value},
+      {"vat", &dot_notation_price::vat},
+      {"currency", &dot_notation_price::currency}});
+  }
+
+  mstch::node value() {
+    return m_value;
+  }
+
+  mstch::node vat() {
+    return m_value * 0.2;
+  }
+
+  mstch::node currency() {
+    return m_currency;
+  }
+};
+
+const auto dot_notation_data = mstch::map{
+  {"name", std::string{"A Book"}},
+  {"authors", mstch::array{std::string{"John Power"}, std::string{"Jamie Walsh"}}},
+  {"price", std::make_shared<dot_notation_price>()},
+  {"availability", mstch::map{{"status", true}, {"text", std::string{"In Stock"}}}},
+  {"truthy", mstch::map{{"zero", 0}, {"notTrue", false}}}
+};

+ 9 - 0
lib/mstch/test/data/dot_notation.mustache

@@ -0,0 +1,9 @@
+<!-- exciting part -->
+<h1>{{name}}</h1>
+<p>Authors: <ul>{{#authors}}<li>{{.}}</li>{{/authors}}</ul></p>
+<p>Price: {{{price.currency.symbol}}}{{price.value}} {{#price.currency}}{{name}} <b>{{availability.text}}</b>{{/price.currency}}</p>
+<p>VAT: {{{price.currency.symbol}}}{{#price}}{{vat}}{{/price}}</p>
+<!-- boring part -->
+<h2>Test truthy false values:</h2>
+<p>Zero: {{truthy.zero}}</p>
+<p>False: {{truthy.notTrue}}</p>

+ 9 - 0
lib/mstch/test/data/dot_notation.txt

@@ -0,0 +1,9 @@
+<!-- exciting part -->
+<h1>A Book</h1>
+<p>Authors: <ul><li>John Power</li><li>Jamie Walsh</li></ul></p>
+<p>Price: $200 USD <b>In Stock</b></p>
+<p>VAT: $40</p>
+<!-- boring part -->
+<h2>Test truthy false values:</h2>
+<p>Zero: 0</p>
+<p>False: false</p>

+ 5 - 0
lib/mstch/test/data/double_render.hpp

@@ -0,0 +1,5 @@
+const auto double_render_data = mstch::map{
+  {"foo", true},
+  {"bar", std::string{"{{win}}"}},
+  {"win", std::string{"FAIL"}}
+};

+ 1 - 0
lib/mstch/test/data/double_render.mustache

@@ -0,0 +1 @@
+{{#foo}}{{bar}}{{/foo}}

+ 1 - 0
lib/mstch/test/data/double_render.txt

@@ -0,0 +1 @@
+{{win}}

+ 3 - 0
lib/mstch/test/data/empty_list.hpp

@@ -0,0 +1,3 @@
+const auto empty_list_data = mstch::map{
+  {"jobs", mstch::array{}}
+};

+ 4 - 0
lib/mstch/test/data/empty_list.mustache

@@ -0,0 +1,4 @@
+These are the jobs:
+{{#jobs}}
+{{.}}
+{{/jobs}}

+ 1 - 0
lib/mstch/test/data/empty_list.txt

@@ -0,0 +1 @@
+These are the jobs:

+ 1 - 0
lib/mstch/test/data/empty_sections.hpp

@@ -0,0 +1 @@
+const auto empty_sections_data = mstch::map{};

+ 1 - 0
lib/mstch/test/data/empty_sections.mustache

@@ -0,0 +1 @@
+{{#foo}}{{/foo}}foo{{#bar}}{{/bar}}

+ 1 - 0
lib/mstch/test/data/empty_sections.txt

@@ -0,0 +1 @@
+foo

+ 6 - 0
lib/mstch/test/data/empty_string.hpp

@@ -0,0 +1,6 @@
+const auto empty_string_data = mstch::map{
+  {"description", std::string{"That is all!"}},
+  {"child", mstch::map{
+    {"description", std::string{""}}
+  }}
+};

+ 1 - 0
lib/mstch/test/data/empty_string.mustache

@@ -0,0 +1 @@
+{{description}}{{#child}}{{description}}{{/child}}

+ 1 - 0
lib/mstch/test/data/empty_string.txt

@@ -0,0 +1 @@
+That is all!

+ 1 - 0
lib/mstch/test/data/empty_template.hpp

@@ -0,0 +1 @@
+const auto empty_template_data = mstch::map{};

+ 1 - 0
lib/mstch/test/data/empty_template.mustache

@@ -0,0 +1 @@
+<html><head></head><body><h1>Test</h1></body></html>

+ 1 - 0
lib/mstch/test/data/empty_template.txt

@@ -0,0 +1 @@
+<html><head></head><body><h1>Test</h1></body></html>

+ 3 - 0
lib/mstch/test/data/error_eof_in_section.hpp

@@ -0,0 +1,3 @@
+const auto error_eof_in_section_data = mstch::map{
+    {"hello", mstch::array{std::string{"a"}, std::string{"b"}}}
+};

+ 1 - 0
lib/mstch/test/data/error_eof_in_section.mustache

@@ -0,0 +1 @@
+yay{{#hello}}{{.}}

+ 1 - 0
lib/mstch/test/data/error_eof_in_section.txt

@@ -0,0 +1 @@
+yay

+ 1 - 0
lib/mstch/test/data/error_eof_in_tag.hpp

@@ -0,0 +1 @@
+const auto error_eof_in_tag_data = mstch::map{{"hello", std::string{"world"}}};

+ 1 - 0
lib/mstch/test/data/error_eof_in_tag.mustache

@@ -0,0 +1 @@
+{{hello{{hello}}{{hello

+ 1 - 0
lib/mstch/test/data/error_eof_in_tag.txt

@@ -0,0 +1 @@
+{{hello

+ 3 - 0
lib/mstch/test/data/error_not_found.hpp

@@ -0,0 +1,3 @@
+const auto error_not_found_data = mstch::map{
+  {"bar", 2}
+};

+ 1 - 0
lib/mstch/test/data/error_not_found.mustache

@@ -0,0 +1 @@
+{{foo}}

+ 0 - 0
lib/mstch/test/data/error_not_found.txt


+ 4 - 0
lib/mstch/test/data/escaped.hpp

@@ -0,0 +1,4 @@
+const mstch::node escaped_data = mstch::map{
+    {"title", mstch::lambda{[]()->mstch::node{ return std::string{"Bear > Shark"}; }}},
+    {"entities", mstch::lambda{[]()->mstch::node{ return std::string{"&quot; \"'<>/"}; }}}
+};

+ 2 - 0
lib/mstch/test/data/escaped.mustache

@@ -0,0 +1,2 @@
+<h1>{{title}}</h1>
+And even {{entities}}, but not {{{entities}}}.

+ 2 - 0
lib/mstch/test/data/escaped.txt

@@ -0,0 +1,2 @@
+<h1>Bear &gt; Shark</h1>
+And even &amp;quot; &quot;&#39;&lt;&gt;&#x2F;, but not &quot; "'<>/.

+ 6 - 0
lib/mstch/test/data/falsy.hpp

@@ -0,0 +1,6 @@
+const auto falsy_data = mstch::map{
+  {"emptyString", std::string{""}},
+  {"emptyArray", mstch::array{}},
+  {"zero", 0},
+  {"null", mstch::node{}}
+};

+ 8 - 0
lib/mstch/test/data/falsy.mustache

@@ -0,0 +1,8 @@
+{{#emptyString}}empty string{{/emptyString}}
+{{^emptyString}}inverted empty string{{/emptyString}}
+{{#emptyArray}}empty array{{/emptyArray}}
+{{^emptyArray}}inverted empty array{{/emptyArray}}
+{{#zero}}zero{{/zero}}
+{{^zero}}inverted zero{{/zero}}
+{{#null}}null{{/null}}
+{{^null}}inverted null{{/null}}

+ 8 - 0
lib/mstch/test/data/falsy.txt

@@ -0,0 +1,8 @@
+
+inverted empty string
+
+inverted empty array
+
+inverted zero
+
+inverted null

+ 8 - 0
lib/mstch/test/data/falsy_array.hpp

@@ -0,0 +1,8 @@
+const auto falsy_array_data = mstch::map{
+  {"list", mstch::array{
+  	mstch::array{std::string{""}, std::string{"emptyString"}},
+  	mstch::array{mstch::array{}, std::string{"emptyArray"}},
+  	mstch::array{0, std::string{"zero"}},
+  	mstch::array{mstch::node{}, std::string{"null"}}}
+	}
+};

+ 3 - 0
lib/mstch/test/data/falsy_array.mustache

@@ -0,0 +1,3 @@
+{{#list}}
+{{#.}}{{#.}}{{.}}{{/.}}{{^.}}inverted {{/.}}{{/.}}
+{{/list}}

+ 4 - 0
lib/mstch/test/data/falsy_array.txt

@@ -0,0 +1,4 @@
+inverted emptyString
+inverted emptyArray
+inverted zero
+inverted null

Some files were not shown because too many files changed in this diff