use std::{fmt::Write, result::Result as StdResult};
use rquickjs::{Ctx, Exception, Result};
pub trait ResultExt<T> {
fn or_throw_msg(self, ctx: &Ctx, msg: &str) -> Result<T>;
fn or_throw_range(self, ctx: &Ctx, msg: Option<&str>) -> Result<T>;
fn or_throw_type(self, ctx: &Ctx, msg: Option<&str>) -> Result<T>;
fn or_throw(self, ctx: &Ctx) -> Result<T>;
}
pub trait OptionExt<T> {
fn and_then_ok<U, E, F>(self, f: F) -> StdResult<Option<U>, E>
where
F: FnOnce(T) -> StdResult<Option<U>, E>;
fn unwrap_or_else_ok<E, F>(self, f: F) -> StdResult<T, E>
where
F: FnOnce() -> StdResult<T, E>;
}
impl<T, E: std::fmt::Display> ResultExt<T> for StdResult<T, E> {
fn or_throw_msg(self, ctx: &Ctx, msg: &str) -> Result<T> {
self.map_err(|e| {
let mut message = String::with_capacity(100);
message.push_str(msg);
message.push_str(". ");
write!(message, "{}", e).unwrap();
Exception::throw_message(ctx, &message)
})
}
fn or_throw_range(self, ctx: &Ctx, msg: Option<&str>) -> Result<T> {
self.map_err(|e| {
let mut message = String::with_capacity(100);
if let Some(msg) = msg {
message.push_str(msg);
message.push_str(". ");
}
write!(message, "{}", e).unwrap();
Exception::throw_range(ctx, &message)
})
}
fn or_throw_type(self, ctx: &Ctx, msg: Option<&str>) -> Result<T> {
self.map_err(|e| {
let mut message = String::with_capacity(100);
if let Some(msg) = msg {
message.push_str(msg);
message.push_str(". ");
}
write!(message, "{}", e).unwrap();
Exception::throw_type(ctx, &message)
})
}
fn or_throw(self, ctx: &Ctx) -> Result<T> {
self.map_err(|err| Exception::throw_message(ctx, &err.to_string()))
}
}
impl<T> ResultExt<T> for Option<T> {
fn or_throw_msg(self, ctx: &Ctx, msg: &str) -> Result<T> {
self.ok_or_else(|| Exception::throw_message(ctx, msg))
}
fn or_throw_range(self, ctx: &Ctx, msg: Option<&str>) -> Result<T> {
self.ok_or_else(|| Exception::throw_range(ctx, msg.unwrap_or("")))
}
fn or_throw_type(self, ctx: &Ctx, msg: Option<&str>) -> Result<T> {
self.ok_or_else(|| Exception::throw_type(ctx, msg.unwrap_or("")))
}
fn or_throw(self, ctx: &Ctx) -> Result<T> {
self.ok_or_else(|| Exception::throw_message(ctx, "Value is not present"))
}
}
impl<T> OptionExt<T> for Option<T> {
fn and_then_ok<U, E, F>(self, f: F) -> StdResult<Option<U>, E>
where
F: FnOnce(T) -> StdResult<Option<U>, E>,
{
match self {
Some(v) => f(v),
None => Ok(None),
}
}
fn unwrap_or_else_ok<E, F>(self, f: F) -> StdResult<T, E>
where
F: FnOnce() -> StdResult<T, E>,
{
match self {
Some(v) => Ok(v),
None => f(),
}
}
}