// // Created by brett on 21/08/23. // #ifndef BLT_MUSTACHE_H #define BLT_MUSTACHE_H #include <string> #include <exception> #include <blt/std/hashmap.h> namespace blt { class mustache_syntax_error : public std::runtime_error { public: explicit mustache_syntax_error(): std::runtime_error("mustache syntax is invalid!") {} explicit mustache_syntax_error(const std::string& str): std::runtime_error(str) {} }; class mustache_lexer { private: std::string str; size_t index = 0; public: explicit mustache_lexer(std::string str): str(std::move(str)) {} inline bool hasNext() { if (index >= str.size()) return false; return true; } inline char peek() { return str[index]; } inline bool hasTemplatePrefix() { if (index + 1 >= str.size()) return false; return str[index] == '{' && str[index + 1] == '{'; } inline bool hasTemplateSuffix() { if (index + 1 >= str.size()) return false; return str[index] == '}' && str[index + 1] == '}'; } inline void consumeTemplatePrefix() { index += 2; } inline void consumeTemplateSuffix() { index += 2; } inline std::string consumeToken() { std::string token; while (!hasTemplateSuffix()) { if (!hasNext()) throw mustache_syntax_error("Error processing token. Mustache template incomplete."); token += consume(); } return token; } inline char consume() { return str[index++]; } }; class mustache { private: mustache_lexer lexer; std::string assemble(){ std::string buffer; while (lexer.hasNext()){ if (lexer.hasTemplatePrefix()){ lexer.consumeTemplatePrefix(); if (!lexer.hasNext()) throw mustache_syntax_error("template incomplete, found '{{' missing '}}'"); auto c = lexer.peek(); switch (c) { case '%': break; case '$': break; case '#': break; case '@': break; case '/': break; default: break; } } else buffer += lexer.consume(); } return buffer; } public: explicit mustache(std::string str): lexer(std::move(str)) {} static std::string compile(std::string input) { mustache compiler(std::move(input)); return compiler.assemble(); } }; } #endif //BLT_MUSTACHE_H