aboutsummaryrefslogtreecommitdiffstats
path: root/nihil.ucl/integer.ccm
diff options
context:
space:
mode:
authorLexi Winter <lexi@le-fay.org>2025-06-27 12:08:58 +0100
committerLexi Winter <lexi@le-fay.org>2025-06-27 12:08:58 +0100
commit001c9917ace09f7b1c80d96eb067e1d37e86c546 (patch)
tree89e360961b9659a8c6b897c5412b7d6834b8eed9 /nihil.ucl/integer.ccm
parent90aa957ca9b7c217af7569009d1675e0f3ff8e9b (diff)
downloadnihil-001c9917ace09f7b1c80d96eb067e1d37e86c546.tar.gz
nihil-001c9917ace09f7b1c80d96eb067e1d37e86c546.tar.bz2
improve error handling
Diffstat (limited to 'nihil.ucl/integer.ccm')
-rw-r--r--nihil.ucl/integer.ccm89
1 files changed, 78 insertions, 11 deletions
diff --git a/nihil.ucl/integer.ccm b/nihil.ucl/integer.ccm
index 7a87df3..d5ac72a 100644
--- a/nihil.ucl/integer.ccm
+++ b/nihil.ucl/integer.ccm
@@ -7,11 +7,15 @@ module;
#include <compare>
#include <cstdint>
#include <cstdlib>
+#include <expected>
+#include <format>
+#include <utility>
#include <ucl.h>
export module nihil.ucl:integer;
+import nihil;
import :object;
import :type;
@@ -21,29 +25,92 @@ export struct integer final : object {
using contained_type = std::int64_t;
inline static constexpr object_type ucl_type = object_type::integer;
- // Create a new integer from a UCL object.
- integer(ref_t, ::ucl_object_t const *uobj);
- integer(noref_t, ::ucl_object_t *uobj);
-
- // Create a new default-initialised integer.
+ /*
+ * Create an integer holding the value 0. Throws std::system_error
+ * on failure.
+ */
integer();
- // Create a new integer from a value.
+ /*
+ * Create an integer holding a specific value. Throws std::system_error
+ * on failure.
+ */
explicit integer(contained_type value);
+ /*
+ * Create a new integer from a UCL object. Throws type_mismatch
+ * on failure.
+ */
+ integer(ref_t, ::ucl_object_t const *uobj);
+ integer(noref_t, ::ucl_object_t *uobj);
+
// Return the value of this object.
- auto value(this integer const &self) -> contained_type;
+ [[nodiscard]] auto value(this integer const &self) -> contained_type;
};
/*
+ * Integer constructors. These return an error instead of throwing.
+ */
+
+export [[nodiscard]] auto
+make_integer(integer::contained_type = 0) -> std::expected<integer, error>;
+
+/*
* Comparison operators.
*/
-export auto operator== (integer const &a, integer const &b) -> bool;
-export auto operator== (integer const &a, integer::contained_type b) -> bool;
-export auto operator<=> (integer const &a, integer const &b)
+export [[nodiscard]] auto operator== (integer const &a,
+ integer const &b) -> bool;
+
+export [[nodiscard]] auto operator== (integer const &a,
+ integer::contained_type b) -> bool;
+
+export [[nodiscard]] auto operator<=> (integer const &a,
+ integer const &b)
-> std::strong_ordering;
-export auto operator<=> (integer const &a, integer::contained_type b)
+
+export [[nodiscard]] auto operator<=> (integer const &a,
+ integer::contained_type b)
-> std::strong_ordering;
+/*
+ * Literal operator.
+ */
+inline namespace literals {
+export constexpr auto operator""_ucl (unsigned long long i) -> integer
+{
+ if (std::cmp_greater(i, std::numeric_limits<std::int64_t>::max()))
+ throw std::out_of_range("literal out of range");
+
+ return integer(i);
+}
+} // namespace nihil::ucl::literals
+
} // namespace nihil::ucl
+
+namespace nihil { inline namespace literals {
+ export using namespace ::nihil::ucl::literals;
+}} // namespace nihil::literals
+
+/*
+ * std::formatter for an integer. This provides the same format operations
+ * as std::formatter<std::int64_t>.
+ */
+export template<>
+struct std::formatter<nihil::ucl::integer, char>
+{
+ std::formatter<std::int64_t> base_formatter;
+
+ template<class ParseContext>
+ constexpr ParseContext::iterator parse(ParseContext& ctx)
+ {
+ return base_formatter.parse(ctx);
+ }
+
+ template<class FmtContext>
+ FmtContext::iterator format(nihil::ucl::integer const &o,
+ FmtContext& ctx) const
+ {
+ return base_formatter.format(o.value(), ctx);
+ }
+};