Cleaned up Util functions
This commit is contained in:
@@ -5,6 +5,7 @@
|
|||||||
|
|
||||||
namespace LXC::Util
|
namespace LXC::Util
|
||||||
{
|
{
|
||||||
|
// Error returned when Util::ReadFile runs into errors //
|
||||||
struct FileReadError
|
struct FileReadError
|
||||||
{
|
{
|
||||||
// Different reasons why the error can occur //
|
// Different reasons why the error can occur //
|
||||||
@@ -38,20 +39,21 @@ namespace LXC::Util
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Util function to read a file as quick as possible with error handling //
|
||||||
inline ReturnVal<std::string, FileReadError> ReadFile(const std::filesystem::path& filepath)
|
inline ReturnVal<std::string, FileReadError> ReadFile(const std::filesystem::path& filepath)
|
||||||
{
|
{
|
||||||
// Checks the file exists //
|
// Checks the file exists //
|
||||||
if (!std::filesystem::exists(filepath))
|
if (!std::filesystem::exists(filepath))
|
||||||
return FunctionFail(FileReadError(std::filesystem::absolute(filepath), FileReadError::FileNotFound));
|
return FunctionFail<FileReadError>(std::filesystem::absolute(filepath), FileReadError::FileNotFound);
|
||||||
|
|
||||||
// Checks it is a regular file //
|
// Checks it is a regular file //
|
||||||
if (!std::filesystem::is_regular_file(filepath))
|
if (!std::filesystem::is_regular_file(filepath))
|
||||||
return FunctionFail(FileReadError(std::filesystem::absolute(filepath), FileReadError::NotAFile));
|
return FunctionFail<FileReadError>(std::filesystem::absolute(filepath), FileReadError::NotAFile);
|
||||||
|
|
||||||
// Checks it can open the file //
|
// Checks it can open the file //
|
||||||
std::ifstream file(filepath, std::ios::binary | std::ios::ate);
|
std::ifstream file(filepath, std::ios::binary | std::ios::ate);
|
||||||
if (!file)
|
if (!file)
|
||||||
return FunctionFail(FileReadError(std::filesystem::absolute(filepath), FileReadError::PermissionDenied));
|
return FunctionFail<FileReadError>(std::filesystem::absolute(filepath), FileReadError::PermissionDenied);
|
||||||
|
|
||||||
// Copies the file to the output string //
|
// Copies the file to the output string //
|
||||||
const std::streamsize len = file.tellg();
|
const std::streamsize len = file.tellg();
|
||||||
|
|||||||
177
Common/IO.h
177
Common/IO.h
@@ -6,18 +6,32 @@
|
|||||||
#include <cstdlib>
|
#include <cstdlib>
|
||||||
#include <ostream>
|
#include <ostream>
|
||||||
|
|
||||||
namespace LXC::Internal
|
namespace LXC::Util
|
||||||
{
|
{
|
||||||
// Returns a reference to the log file //
|
// Enum to translate to the Win32 code for the colors //
|
||||||
// To be used only by LXC components //
|
enum Color : WORD
|
||||||
inline std::ofstream& Log()
|
|
||||||
{
|
{
|
||||||
static std::ofstream sLog;
|
DEFAULT = 0x07,
|
||||||
return sLog;
|
|
||||||
}
|
BLACK = 0x00,
|
||||||
|
BLUE = 0x01,
|
||||||
|
GREEN = 0x02,
|
||||||
|
AQUA = 0x03,
|
||||||
|
RED = 0x04,
|
||||||
|
PURPLE = 0x05,
|
||||||
|
YELLOW = 0x06,
|
||||||
|
LIGHT_GRAY = 0x07,
|
||||||
|
LIGHT_BLUE = 0x09,
|
||||||
|
LIGHT_GREEN = 0x0a,
|
||||||
|
LIGHT_AQUA = 0x0b,
|
||||||
|
LIGHT_RED = 0x0c,
|
||||||
|
LIGHT_PURPLE = 0x0d,
|
||||||
|
LIGHT_YELLOW = 0x0e,
|
||||||
|
WHITE = 0x0f
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace LXC::Util
|
namespace LXC::Internal
|
||||||
{
|
{
|
||||||
// Checks if a type can be outputted to std::ostream //
|
// Checks if a type can be outputted to std::ostream //
|
||||||
template<typename T> concept Logable = requires(std::ostream& os, T t)
|
template<typename T> concept Logable = requires(std::ostream& os, T t)
|
||||||
@@ -36,57 +50,81 @@ namespace LXC::Util
|
|||||||
{ obj.LogStr() } -> std::same_as<std::string>;
|
{ obj.LogStr() } -> std::same_as<std::string>;
|
||||||
};
|
};
|
||||||
|
|
||||||
// Enum to translate to the Win32 code for the colors //
|
// Returns a reference to the log file //
|
||||||
enum Color : WORD
|
inline std::ofstream& Log()
|
||||||
{
|
{
|
||||||
BLACK = 0x00,
|
static std::ofstream sLog;
|
||||||
BLUE = 0x01,
|
return sLog;
|
||||||
GREEN = 0x02,
|
}
|
||||||
AQUA = 0x03,
|
|
||||||
RED = 0x04,
|
|
||||||
PURPLE = 0x05,
|
|
||||||
YELLOW = 0x06,
|
|
||||||
LIGHT_GRAY = 0x07,
|
|
||||||
LIGHT_BLUE = 0x09,
|
|
||||||
LIGHT_GREEN = 0x0a,
|
|
||||||
LIGHT_AQUA = 0x0b,
|
|
||||||
LIGHT_RED = 0x0c,
|
|
||||||
LIGHT_PURPLE = 0x0d,
|
|
||||||
LIGHT_YELLOW = 0x0e,
|
|
||||||
WHITE = 0x0f
|
|
||||||
};
|
|
||||||
|
|
||||||
|
// Base function for all derived Print() functions //
|
||||||
|
template<Util::Color col, bool newLine, typename... Args>
|
||||||
|
inline void WriteImpl(std::ostream& os, Args&&... args)
|
||||||
|
{
|
||||||
|
static HANDLE hConsole = GetStdHandle(STD_OUTPUT_HANDLE);
|
||||||
|
if constexpr(col != Util::Color::DEFAULT)
|
||||||
|
SetConsoleTextAttribute(hConsole, static_cast<WORD>(col));
|
||||||
|
|
||||||
|
(os << ... << std::forward<Args>(args));
|
||||||
|
|
||||||
|
if constexpr (newLine)
|
||||||
|
os << std::endl;
|
||||||
|
|
||||||
|
if constexpr (col != Util::Color::DEFAULT)
|
||||||
|
SetConsoleTextAttribute(hConsole, static_cast<WORD>(Util::Color::DEFAULT));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace LXC::Util
|
||||||
|
{
|
||||||
// Prints arguments to the console with the given color //
|
// Prints arguments to the console with the given color //
|
||||||
template<Color col, typename... Args>
|
template<Color col, typename... Args>
|
||||||
requires AllLogable<Args...>
|
requires Internal::AllLogable<Args...>
|
||||||
inline void PrintAs(Args... args)
|
inline void PrintAs(Args&&... args)
|
||||||
{
|
{
|
||||||
// Permenant handle to the console //
|
Internal::WriteImpl<col, false>(std::cout, std::forward<Args>(args)...);
|
||||||
static HANDLE hConsole = GetStdHandle(STD_OUTPUT_HANDLE);
|
|
||||||
|
|
||||||
// Prints it to the console and resets the color //
|
|
||||||
SetConsoleTextAttribute(hConsole, (WORD)col);
|
|
||||||
(std::cout << ... << args);
|
|
||||||
SetConsoleTextAttribute(hConsole, (WORD)Color::LIGHT_GRAY);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Prints arguments to the console //
|
// Prints arguments to the console //
|
||||||
template<typename... Args>
|
template<typename... Args>
|
||||||
requires AllLogable<Args...>
|
requires Internal::AllLogable<Args...>
|
||||||
inline void Print(Args... args)
|
inline void Print(Args&&... args)
|
||||||
{
|
{
|
||||||
// Fowards the arguments to the console //
|
Internal::WriteImpl<Util::Color::DEFAULT, false>(std::cout, std::forward<Args>(args)...);
|
||||||
(std::cout << ... << args);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Prints arguments to the console with a new-line character at the end //
|
// Prints arguments to the console with a new-line character at the end //
|
||||||
template<typename... Args>
|
template<typename... Args>
|
||||||
requires AllLogable<Args...>
|
requires Internal::AllLogable<Args...>
|
||||||
inline void PrintLn(Args... args)
|
inline void PrintLn(Args&&... args)
|
||||||
{
|
{
|
||||||
// Fowards the arguments to the console //
|
Internal::WriteImpl<Util::Color::DEFAULT, true>(std::cout, std::forward<Args>(args)...);
|
||||||
(std::cout << ... << args);
|
}
|
||||||
std::cout << std::endl;
|
|
||||||
|
// Logs all the arguments to the file (automatically flushes) //
|
||||||
|
template<typename... Args>
|
||||||
|
requires Internal::AllLogable<Args...> && (sizeof...(Args) > 1)
|
||||||
|
inline void Log(Args&&... args)
|
||||||
|
{
|
||||||
|
std::ofstream& log = Internal::Log();
|
||||||
|
if (log.is_open()) _UNLIKELY
|
||||||
|
Internal::WriteImpl<Util::Color::DEFAULT, true>(log, R"([LXC] ")", std::forward<Args>(args)..., '"');
|
||||||
|
}
|
||||||
|
|
||||||
|
// Logs a singular argument to the log, calls Log() if it can on the object //
|
||||||
|
template<typename T>
|
||||||
|
requires Internal::HasLogStrFunc<T> || Internal::Logable<T>
|
||||||
|
inline void Log(T arg)
|
||||||
|
{
|
||||||
|
std::ofstream& log = Internal::Log();
|
||||||
|
if (log.is_open()) _UNLIKELY
|
||||||
|
{
|
||||||
|
if constexpr (Internal::HasLogStrFunc<T>)
|
||||||
|
Internal::WriteImpl<Util::Color::DEFAULT, true>(log, "[LXC] ", '{', arg.LogStr(), '}');
|
||||||
|
|
||||||
|
else
|
||||||
|
Internal::WriteImpl<Util::Color::DEFAULT, true>(log, "[LXC] ", '"', arg, '"');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Intitalises the log with the given file name //
|
// Intitalises the log with the given file name //
|
||||||
@@ -103,55 +141,4 @@ namespace LXC::Util
|
|||||||
log.close();
|
log.close();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// Logs all the arguments to the file (automatically flushes) //
|
|
||||||
template<typename... Args>
|
|
||||||
requires AllLogable<Args...> && (sizeof...(Args) > 1)
|
|
||||||
inline void Log(Args... args)
|
|
||||||
{
|
|
||||||
std::ofstream& log = Internal::Log();
|
|
||||||
|
|
||||||
// Only attempts to write to an open log //
|
|
||||||
if (log.is_open()) _UNLIKELY
|
|
||||||
{
|
|
||||||
// Opening symbol to make the log look fancy //
|
|
||||||
log << "[LXC] \"";
|
|
||||||
|
|
||||||
// Fowards the arguments to the log and flushes //
|
|
||||||
(log << ... << args);
|
|
||||||
log << "\"\n";
|
|
||||||
log.flush();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Logs a singular argument to the log, calls Log() if it can on the object //
|
|
||||||
template<typename T>
|
|
||||||
requires HasLogStrFunc<T> || Logable<T>
|
|
||||||
inline void Log(T arg)
|
|
||||||
{
|
|
||||||
std::ofstream& log = Internal::Log();
|
|
||||||
|
|
||||||
// Only attempts to write to an open log //
|
|
||||||
if (log.is_open()) _UNLIKELY
|
|
||||||
{
|
|
||||||
// Opening symbol to make log look fancy //
|
|
||||||
log << "[LXC] ";
|
|
||||||
|
|
||||||
// Logic if it can call LogStr() //
|
|
||||||
if constexpr (HasLogStrFunc<T>)
|
|
||||||
{
|
|
||||||
// Prints the generated string //
|
|
||||||
log << '{' << arg.LogStr() << "}\n";
|
|
||||||
log.flush();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Default logic //
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// Prints the singular arg //
|
|
||||||
log << '"' << arg << "\"\n";
|
|
||||||
log.flush();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -24,10 +24,10 @@ namespace LXC::Util
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Custom version of std::unexpected //
|
// Custom version of std::unexpected //
|
||||||
template<typename ErrorType> struct FunctionFail
|
template<typename ErrorType> struct FunctionFail final
|
||||||
{
|
{
|
||||||
explicit FunctionFail(ErrorType _err)
|
// Basic constructor to copy the error across //
|
||||||
: error(_err)
|
explicit FunctionFail(ErrorType _err) : error(_err)
|
||||||
{
|
{
|
||||||
// Only checks for a debugger when compiled in Debug mode //
|
// Only checks for a debugger when compiled in Debug mode //
|
||||||
#ifdef _DEBUG
|
#ifdef _DEBUG
|
||||||
@@ -39,13 +39,28 @@ namespace LXC::Util
|
|||||||
#endif // _DEBUG
|
#endif // _DEBUG
|
||||||
}
|
}
|
||||||
|
|
||||||
ErrorType error;
|
// Constructs the FunctionFail with the error itself //
|
||||||
|
template<typename... Args> requires std::constructible_from<ErrorType, Args...>
|
||||||
|
explicit FunctionFail(Args&&... args)
|
||||||
|
: error(std::forward<Args>(args)...)
|
||||||
|
{
|
||||||
|
// Only checks for a debugger when compiled in Debug mode //
|
||||||
|
#ifdef _DEBUG
|
||||||
|
|
||||||
|
// Triggers a breakpoint when a debugger is attached as a function has failed //
|
||||||
|
if (IsDebuggerPresent())
|
||||||
|
DebugBreak();
|
||||||
|
|
||||||
|
#endif // _DEBUG
|
||||||
|
}
|
||||||
|
|
||||||
|
const ErrorType error;
|
||||||
};
|
};
|
||||||
|
|
||||||
// Custom version of std::expected //
|
// Custom version of std::expected //
|
||||||
template<typename ResultType, typename ErrorType>
|
template<typename ResultType, typename ErrorType>
|
||||||
requires (!std::same_as<ResultType, bool>) // ResultType being bool causes issues with operator overloads
|
requires (!std::same_as<ResultType, bool>) // ResultType being bool causes issues with operator overloads
|
||||||
class ReturnVal
|
class ReturnVal final
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
// Constructor for function sucess //
|
// Constructor for function sucess //
|
||||||
|
|||||||
Reference in New Issue
Block a user