#pragma once /* * Copyright (C) 2024 Brett Terpstra * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef BLT_LOGGING_LOGGING_H #define BLT_LOGGING_LOGGING_H #include #include #include #include #include namespace blt::logging { struct logger_t { explicit logger_t() = default; template std::string log(std::string fmt, Args&&... args) { compile(std::move(fmt)); auto sequence = std::make_integer_sequence{}; while (auto pair = consume_until_fmt()) { auto [begin, end] = *pair; if (end - begin > 0) { auto format_data = handle_fmt(m_fmt.substr(begin + 1, begin - end - 1)); auto [arg_pos, fmt_type] = format_data; if (arg_pos == -1) arg_pos = static_cast(m_arg_pos++); if (fmt_type) { if (fmt_type == fmt_type_t::GENERAL) { apply_func([this](auto&& value) { if (static_cast(value) > 0xFFFFFFFFFul) exponential(); else fixed(); m_stream << std::forward(value); }, arg_pos, sequence, std::forward(args)...); } else if (fmt_type == fmt_type_t::CHAR) { } else if (fmt_type == fmt_type_t::BINARY) { } } else { apply_func([this](auto&& value) { m_stream << std::forward(value); }, arg_pos, sequence, std::forward(args)...); } } else apply_func([this](auto&& value) { m_stream << std::forward(value); }, m_arg_pos++, sequence, std::forward(args)...); } finish(); return to_string(); } std::string to_string(); private: template void apply_func(const Func& func, const size_t arg, std::integer_sequence, Args&&... args) { ((handle_func(func, arg, std::forward(args))), ...); } template void handle_func(const Func& func, const size_t arg, T&& t) { if (index == arg) func(std::forward(t)); } [[nodiscard]] std::pair> handle_fmt(std::string_view fmt); void exponential(); void fixed(); void compile(std::string fmt); std::optional> consume_until_fmt(); void finish(); std::string m_fmt; std::stringstream m_stream; fmt_parser_t m_parser; size_t m_last_fmt_pos = 0; size_t m_arg_pos = 0; }; void print(const std::string& fmt); void newline(); logger_t& get_global_logger(); template void print(std::string fmt, Args&&... args) { auto& logger = get_global_logger(); print(logger.log(std::move(fmt), std::forward(args)...)); } template void println(std::string fmt, Args&&... args) { print(std::move(fmt), std::forward(args)...); newline(); } } #endif // BLT_LOGGING_LOGGING_H