#include "openvic-dataloader/v2script/Parser.hpp" #include #include #include #include #include #include #include "SimpleGrammar.hpp" #include "detail/DetectUtf8.hpp" #include "detail/Errors.hpp" #include "detail/LexyReportError.hpp" #include "detail/NullBuff.hpp" #include "detail/OStreamOutputIterator.hpp" #include "detail/Warnings.hpp" #include "v2script/DecisionGrammar.hpp" #include "v2script/EventGrammar.hpp" #include #include #include #include #include #include #include #include #include using namespace ovdl::v2script; /// BufferHandler /// class Parser::BufferHandler { public: bool is_valid() const { return _buffer.size() != 0; } std::optional load_buffer(const char* data, std::size_t size) { _buffer = lexy::buffer(data, size); return std::nullopt; } std::optional load_buffer(const char* start, const char* end) { _buffer = lexy::buffer(start, end); return std::nullopt; } std::optional load_file(const char* path) { auto file = lexy::read_file(path); if (!file) { return errors::make_no_file_error(path); } _buffer = file.buffer(); return std::nullopt; } constexpr bool is_exclusive_utf8() const { return detail::is_utf8_no_ascii(_buffer); } template std::optional> parse(const ErrorCallback& callback) { auto result = lexy::parse(_buffer, callback); if (!result) { return result.errors(); } // This is mighty frustrating _root = std::unique_ptr(result.value()); return std::nullopt; } std::unique_ptr& get_root() { return _root; } private: lexy::buffer _buffer; std::unique_ptr _root; }; /// BufferHandler /// Parser::Parser() : _buffer_handler(std::make_unique()), _error_stream(detail::cnull) { set_error_log_to_stderr(); } Parser::Parser(Parser&&) = default; Parser& Parser::operator=(Parser&& value) = default; Parser::~Parser() = default; Parser Parser::from_buffer(const char* data, std::size_t size) { Parser result; return std::move(result.load_from_buffer(data, size)); } Parser Parser::from_buffer(const char* start, const char* end) { Parser result; return std::move(result.load_from_buffer(start, end)); } Parser Parser::from_file(const char* path) { Parser result; return std::move(result.load_from_file(path)); } /// /// @brief Executes a function on _buffer_handler that is expected to load a buffer /// /// Expected Use: /// @code {.cpp} /// _run_load_func(&BufferHandler::, ); /// @endcode /// /// @tparam Args /// @param func /// @param args /// template inline void Parser::_run_load_func(std::optional (BufferHandler::*func)(Args...), Args... args) { _warnings.clear(); _errors.clear(); _has_fatal_error = false; if (auto error = (_buffer_handler.get()->*func)(args...); error) { _has_fatal_error = error.value().type == ParseError::Type::Fatal; _errors.push_back(error.value()); _error_stream.get() << "Error: " << _errors.back().message << '\n'; } } Parser& Parser::load_from_buffer(const char* data, std::size_t size) { _run_load_func(&BufferHandler::load_buffer, data, size); return *this; } Parser& Parser::load_from_buffer(const char* start, const char* end) { _run_load_func(&BufferHandler::load_buffer, start, end); return *this; } Parser& Parser::load_from_file(const char* path) { _file_path = path; _run_load_func(&BufferHandler::load_file, path); return *this; } void Parser::set_error_log_to_null() { set_error_log_to(detail::cnull); } void Parser::set_error_log_to_stderr() { set_error_log_to(std::cerr); } void Parser::set_error_log_to_stdout() { set_error_log_to(std::cout); } void Parser::set_error_log_to(std::basic_ostream& stream) { _error_stream = stream; } bool Parser::simple_parse() { if (!_buffer_handler->is_valid()) { return false; } if (_buffer_handler->is_exclusive_utf8()) { _warnings.push_back(warnings::make_utf8_warning(_file_path)); } auto errors = _buffer_handler->parse(ovdl::detail::ReporError.path(_file_path).to(detail::OStreamOutputItertaor { _error_stream })); if (errors) { _errors.reserve(errors->size()); for (auto& err : errors.value()) { _has_fatal_error |= err.type == ParseError::Type::Fatal; _errors.push_back(err); } return false; } _file_node.reset(static_cast(_buffer_handler->get_root().release())); return true; } bool Parser::event_parse() { if (!_buffer_handler->is_valid()) { return false; } if (_buffer_handler->is_exclusive_utf8()) { _warnings.push_back(warnings::make_utf8_warning(_file_path)); } auto errors = _buffer_handler->parse(ovdl::detail::ReporError.path(_file_path).to(detail::OStreamOutputItertaor { _error_stream })); if (errors) { _errors.reserve(errors->size()); for (auto& err : errors.value()) { _has_fatal_error |= err.type == ParseError::Type::Fatal; _errors.push_back(err); } return false; } _file_node.reset(static_cast(_buffer_handler->get_root().release())); return true; } bool Parser::decision_parse() { if (!_buffer_handler->is_valid()) { return false; } if (_buffer_handler->is_exclusive_utf8()) { _warnings.push_back(warnings::make_utf8_warning(_file_path)); } auto errors = _buffer_handler->parse(ovdl::detail::ReporError.path(_file_path).to(detail::OStreamOutputItertaor { _error_stream })); if (errors) { _errors.reserve(errors->size()); for (auto& err : errors.value()) { _has_fatal_error |= err.type == ParseError::Type::Fatal; _errors.push_back(err); } return false; } _file_node.reset(static_cast(_buffer_handler->get_root().release())); return true; } bool Parser::has_error() const { return !_errors.empty(); } bool Parser::has_fatal_error() const { return _has_fatal_error; } bool Parser::has_warning() const { return !_warnings.empty(); } const std::vector& Parser::get_errors() const { return _errors; } const std::vector& Parser::get_warnings() const { return _warnings; } const FileNode* Parser::get_file_node() const { return _file_node.get(); }