314 lines
6.6 KiB
C++
Raw Normal View History

2020-04-05 14:16:16 -05:00
// ftpd is a server implementation based on the following:
// - RFC 959 (https://tools.ietf.org/html/rfc959)
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
//
// Copyright (C) 2024 Michael Theall
2020-04-05 14:16:16 -05:00
//
// 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 <https://www.gnu.org/licenses/>.
#include "log.h"
2020-04-09 21:21:25 -05:00
#include "platform.h"
#if !defined(__WIIU__) && !defined(CLASSIC)
2020-04-05 14:16:16 -05:00
#include "imgui.h"
#endif
2020-04-05 14:16:16 -05:00
#include <mutex>
2022-10-03 21:52:19 -05:00
#include <ranges>
2020-04-09 21:21:25 -05:00
#include <vector>
2020-04-05 14:16:16 -05:00
2023-11-19 13:55:58 +01:00
#ifdef __WIIU__
#include <coreinit/debug.h>
#endif
2020-04-05 14:16:16 -05:00
namespace
{
2022-03-31 11:53:47 -05:00
#ifdef __3DS__
2020-04-06 00:36:03 -05:00
/// \brief Maximum number of log messages to keep
2020-04-05 14:16:16 -05:00
constexpr auto MAX_LOGS = 250;
#else
2020-04-06 00:36:03 -05:00
/// \brief Maximum number of log messages to keep
constexpr auto MAX_LOGS = 100;
2020-04-05 14:16:16 -05:00
#endif
2020-04-06 00:36:03 -05:00
2020-04-17 15:32:39 -05:00
#ifdef CLASSIC
bool s_logUpdated = true;
#endif
2020-04-06 00:36:03 -05:00
/// \brief Message prefix
2020-04-05 14:16:16 -05:00
static char const *const s_prefix[] = {
2024-04-27 12:25:32 +02:00
[DEBUGLOG] = "[DEBUG]",
2020-04-09 21:21:25 -05:00
[INFO] = "[INFO]",
[ERROR] = "[ERROR]",
[COMMAND] = "[COMMAND]",
[RESPONSE] = "[RESPONSE]",
2020-04-05 14:16:16 -05:00
};
2020-04-09 21:21:25 -05:00
/// \brief Log message
struct Message
{
/// \brief Parameterized constructor
/// \param level_ Log level
/// \param message_ Log message
Message (LogLevel const level_, std::string message_)
: level (level_), message (std::move (message_))
{
}
/// \brief Log level
LogLevel level;
/// \brief Log message
std::string message;
};
2020-04-05 14:16:16 -05:00
2020-04-09 21:21:25 -05:00
/// \brief Log messages
std::vector<Message> s_messages;
#ifndef __NDS__
2020-04-09 21:21:25 -05:00
/// \brief Log lock
platform::Mutex s_lock;
2020-04-17 15:32:39 -05:00
#endif
2020-04-09 21:21:25 -05:00
}
2020-04-05 14:16:16 -05:00
2020-04-09 21:21:25 -05:00
void drawLog ()
2020-04-05 14:16:16 -05:00
{
#ifndef __NDS__
2022-10-03 21:52:19 -05:00
auto const lock = std::scoped_lock (s_lock);
2020-04-17 15:32:39 -05:00
#endif
#ifdef CLASSIC
if (!s_logUpdated)
return;
2020-04-05 14:16:16 -05:00
2020-04-17 15:32:39 -05:00
s_logUpdated = false;
#endif
auto const maxLogs =
#if defined(CLASSIC) && !defined(__WIIU__)
2020-04-17 15:32:39 -05:00
g_logConsole.windowHeight;
#else
MAX_LOGS;
#endif
if (s_messages.size () > static_cast<unsigned> (maxLogs))
2020-04-05 14:16:16 -05:00
{
2020-04-09 21:21:25 -05:00
auto const begin = std::begin (s_messages);
2020-04-17 15:32:39 -05:00
auto const end = std::next (begin, s_messages.size () - maxLogs);
2020-04-09 21:21:25 -05:00
s_messages.erase (begin, end);
2020-04-05 14:16:16 -05:00
}
2020-04-17 15:32:39 -05:00
#ifdef CLASSIC
char const *const s_colors[] = {
2024-04-27 12:25:32 +02:00
[DEBUGLOG] = "\x1b[33;1m", // yellow
2020-04-17 15:32:39 -05:00
[INFO] = "\x1b[37;1m", // white
[ERROR] = "\x1b[31;1m", // red
[COMMAND] = "\x1b[32;1m", // green
[RESPONSE] = "\x1b[36;1m", // cyan
};
2023-11-19 13:55:58 +01:00
#ifdef __WIIU__
for (auto &cur : s_messages)
{
OSReport ("ftpiiu plugin: %s %s\x1b[0m", s_colors[cur.level], cur.message.c_str ());
2023-11-19 13:55:58 +01:00
}
#else
2020-04-17 15:32:39 -05:00
auto it = std::begin (s_messages);
if (s_messages.size () > static_cast<unsigned> (g_logConsole.windowHeight))
it = std::next (it, s_messages.size () - g_logConsole.windowHeight);
consoleSelect (&g_logConsole);
while (it != std::end (s_messages))
{
std::fputs (s_colors[it->level], stdout);
std::fputs (it->message.c_str (), stdout);
++it;
}
std::fflush (stdout);
2023-11-19 13:55:58 +01:00
#endif
2020-04-17 15:32:39 -05:00
s_messages.clear ();
#else
2020-04-10 00:46:46 -05:00
ImVec4 const s_colors[] = {
[DEBUG] = ImVec4 (1.0f, 1.0f, 0.4f, 1.0f), // yellow
[INFO] = ImGui::GetStyleColorVec4 (ImGuiCol_Text), // normal
[ERROR] = ImVec4 (1.0f, 0.4f, 0.4f, 1.0f), // red
[COMMAND] = ImVec4 (0.4f, 1.0f, 0.4f, 1.0f), // green
[RESPONSE] = ImVec4 (0.4f, 1.0f, 1.0f, 1.0f), // cyan
2020-04-05 14:16:16 -05:00
};
2020-04-09 21:21:25 -05:00
for (auto const &message : s_messages)
2020-04-05 14:16:16 -05:00
{
ImGui::PushStyleColor (ImGuiCol_Text, s_colors[message.level]);
ImGui::TextUnformatted (s_prefix[message.level]);
ImGui::SameLine ();
ImGui::TextUnformatted (message.message.c_str ());
ImGui::PopStyleColor ();
}
2020-04-06 00:36:03 -05:00
// auto-scroll if scroll bar is at end
2020-04-05 14:16:16 -05:00
if (ImGui::GetScrollY () >= ImGui::GetScrollMaxY ())
ImGui::SetScrollHereY (1.0f);
2020-04-17 15:32:39 -05:00
#endif
2020-04-05 14:16:16 -05:00
}
2022-10-03 21:52:19 -05:00
#ifndef CLASSIC
std::string getLog ()
{
#ifndef __NDS__
2022-10-03 21:52:19 -05:00
auto const lock = std::scoped_lock (s_lock);
#endif
if (s_messages.empty ())
return {};
std::vector<Message const *> stack;
stack.reserve (s_messages.size ());
std::size_t size = 0;
for (auto const &msg : s_messages | std::views::reverse)
{
if (size + msg.message.size () > 1024 * 1024)
break;
size += msg.message.size ();
stack.emplace_back (&msg);
}
std::string log;
log.reserve (size);
for (auto const &msg : stack | std::views::reverse)
log += msg->message;
return log;
}
#endif
2020-04-09 21:21:25 -05:00
void debug (char const *const fmt_, ...)
2020-04-05 14:16:16 -05:00
{
#ifndef NDEBUG
va_list ap;
va_start (ap, fmt_);
2024-04-27 12:25:32 +02:00
addLog (DEBUGLOG, fmt_, ap);
2020-04-05 14:16:16 -05:00
va_end (ap);
#else
(void)fmt_;
2020-04-05 14:16:16 -05:00
#endif
}
2020-04-09 21:21:25 -05:00
void info (char const *const fmt_, ...)
2020-04-05 14:16:16 -05:00
{
va_list ap;
va_start (ap, fmt_);
2020-04-09 21:21:25 -05:00
addLog (INFO, fmt_, ap);
2020-04-05 14:16:16 -05:00
va_end (ap);
}
2020-04-09 21:21:25 -05:00
void error (char const *const fmt_, ...)
2020-04-05 14:16:16 -05:00
{
va_list ap;
va_start (ap, fmt_);
2020-04-09 21:21:25 -05:00
addLog (ERROR, fmt_, ap);
2020-04-05 14:16:16 -05:00
va_end (ap);
}
2020-04-09 21:21:25 -05:00
void command (char const *const fmt_, ...)
2020-04-05 14:16:16 -05:00
{
va_list ap;
va_start (ap, fmt_);
2020-04-09 21:21:25 -05:00
addLog (COMMAND, fmt_, ap);
2020-04-05 14:16:16 -05:00
va_end (ap);
}
2020-04-09 21:21:25 -05:00
void response (char const *const fmt_, ...)
2020-04-05 14:16:16 -05:00
{
va_list ap;
va_start (ap, fmt_);
2020-04-09 21:21:25 -05:00
addLog (RESPONSE, fmt_, ap);
2020-04-05 14:16:16 -05:00
va_end (ap);
}
2020-04-09 21:21:25 -05:00
void addLog (LogLevel const level_, char const *const fmt_, va_list ap_)
2020-04-05 14:16:16 -05:00
{
#ifdef __WIIU__
// the plugin is currently never calling drawLogs
return;
#endif
2020-04-05 14:16:16 -05:00
#ifdef NDEBUG
2024-05-07 06:40:45 +02:00
if (level_ == DEBUGLOG)
2020-04-05 14:16:16 -05:00
return;
#endif
#ifndef __NDS__
2023-11-19 13:55:58 +01:00
auto const lock = std::scoped_lock (s_lock);
#endif
#if !defined(NDS) && !defined(__WIIU__)
2020-04-17 15:32:39 -05:00
thread_local
#endif
2023-11-19 13:55:58 +01:00
#if !defined(__WIIU__)
static
2023-11-19 13:55:58 +01:00
#endif
char buffer[1024];
2020-04-05 14:16:16 -05:00
std::vsnprintf (buffer, sizeof (buffer), fmt_, ap_);
#ifndef NDEBUG
2020-04-17 15:32:39 -05:00
// std::fprintf (stderr, "%s", s_prefix[level_]);
// std::fputs (buffer, stderr);
2020-04-05 14:16:16 -05:00
#endif
2020-04-09 21:21:25 -05:00
s_messages.emplace_back (level_, buffer);
2020-04-17 15:32:39 -05:00
#ifdef CLASSIC
s_logUpdated = true;
#endif
2020-04-05 14:16:16 -05:00
}
2020-04-09 21:21:25 -05:00
void addLog (LogLevel const level_, std::string_view const message_)
2020-04-05 14:16:16 -05:00
{
#ifdef __WIIU__
// the plugin is currently never calling drawLogs
return;
#endif
2020-04-05 14:16:16 -05:00
#ifdef NDEBUG
2024-05-07 06:40:45 +02:00
if (level_ == DEBUGLOG)
2020-04-05 14:16:16 -05:00
return;
#endif
auto msg = std::string (message_);
for (auto &c : msg)
{
2020-04-06 00:36:03 -05:00
// replace nul-characters with ? to avoid truncation
2020-04-05 14:16:16 -05:00
if (c == '\0')
c = '?';
}
#ifndef __NDS__
2022-10-03 21:52:19 -05:00
auto const lock = std::scoped_lock (s_lock);
2020-04-17 15:32:39 -05:00
#endif
2020-04-05 14:16:16 -05:00
#ifndef NDEBUG
2020-04-17 15:32:39 -05:00
// std::fprintf (stderr, "%s", s_prefix[level_]);
// std::fwrite (msg.data (), 1, msg.size (), stderr);
2020-04-05 14:16:16 -05:00
#endif
2020-04-09 21:21:25 -05:00
s_messages.emplace_back (level_, msg);
2020-04-17 15:32:39 -05:00
#ifdef CLASSIC
s_logUpdated = true;
#endif
2020-04-05 14:16:16 -05:00
}