draconisplusplus/include/util/result.h

109 lines
2.8 KiB
C
Raw Normal View History

2024-06-07 04:23:11 -04:00
#pragma once
#include <stdexcept>
#include <string>
#include <utility>
#include <variant>
2024-06-09 20:14:51 -04:00
#include "util/numtypes.h"
2024-06-07 04:23:11 -04:00
// Define an error type
class Error {
public:
explicit Error(std::string message) : m_Message(std::move(message)) {}
2024-06-09 20:14:51 -04:00
[[nodiscard]] fn message() const -> const std::string& { return m_Message; }
2024-06-07 04:23:11 -04:00
private:
std::string m_Message;
};
// Primary template for Result with a default type of void
template <typename T = void>
class Result;
// Specialization for Result<void>
template <>
class Result<void> {
public:
// Constructors for success and error
Result() : m_Result(std::monostate {}) {}
Result(const Error& error) : m_Result(error) {}
Result(Error&& error) : m_Result(std::move(error)) {}
// Check if the result is an error
2024-06-09 20:14:51 -04:00
[[nodiscard]] fn isOk() const -> bool {
2024-06-07 04:23:11 -04:00
return std::holds_alternative<std::monostate>(m_Result);
}
2024-06-09 20:14:51 -04:00
[[nodiscard]] fn isErr() const -> bool {
2024-06-07 04:23:11 -04:00
return std::holds_alternative<Error>(m_Result);
}
// Throw an exception if it is an error
void value() const {
if (isErr()) {
throw std::logic_error("Attempted to access value of an error Result");
}
}
// Get the error or throw an exception if it is a value
2024-06-09 20:14:51 -04:00
[[nodiscard]] fn error() const -> const Error& {
2024-06-07 04:23:11 -04:00
if (isOk()) {
throw std::logic_error("Attempted to access error of an ok Result");
}
return std::get<Error>(m_Result);
}
private:
std::variant<std::monostate, Error> m_Result;
};
// Primary template for Result
template <typename T>
class Result {
public:
// Constructors for success and error
Result(const T& value) : m_Result(value) {}
Result(T&& value) : m_Result(std::move(value)) {}
Result(const Error& error) : m_Result(error) {}
Result(Error&& error) : m_Result(std::move(error)) {}
// Check if the result is an error
2024-06-09 20:14:51 -04:00
[[nodiscard]] fn isOk() const -> bool {
2024-06-07 04:23:11 -04:00
return std::holds_alternative<T>(m_Result);
}
2024-06-09 20:14:51 -04:00
[[nodiscard]] fn isErr() const -> bool {
2024-06-07 04:23:11 -04:00
return std::holds_alternative<Error>(m_Result);
}
// Get the value or throw an exception if it is an error
2024-06-09 20:14:51 -04:00
fn value() const -> const T& {
2024-06-07 04:23:11 -04:00
if (isErr()) {
throw std::logic_error("Attempted to access value of an error Result");
}
return std::get<T>(m_Result);
}
// Get the error or throw an exception if it is a value
2024-06-09 20:14:51 -04:00
[[nodiscard]] fn error() const -> const Error& {
2024-06-07 04:23:11 -04:00
if (isOk()) {
throw std::logic_error("Attempted to access error of an ok Result");
}
return std::get<Error>(m_Result);
}
// Optional: Get the value or provide a default
2024-06-09 20:14:51 -04:00
fn valueOr(const T& defaultValue) const -> T {
2024-06-07 04:23:11 -04:00
return isOk() ? std::get<T>(m_Result) : defaultValue;
}
private:
std::variant<T, Error> m_Result;
};
template <typename T>
2024-06-09 20:14:51 -04:00
fn Ok(T&& value) {
2024-06-07 04:23:11 -04:00
return Result<std::decay_t<T>>(std::forward<T>(value));
}
2024-06-09 20:14:51 -04:00
inline fn Ok() -> Result<void> { return {}; }