use super::*; use core::fmt::{Debug, Display}; use core::result::Result; use std::error::Error; use std::io; ////////////////////////////////////////////////////////////////// // Non-fallible network results conversions pub trait NetworkResultExt { fn into_network_result(self) -> NetworkResult; } impl NetworkResultExt for Result { fn into_network_result(self) -> NetworkResult { self.ok() .map(|v| NetworkResult::::Value(v)) .unwrap_or(NetworkResult::::Timeout) } } pub trait IoNetworkResultExt { fn into_network_result(self) -> io::Result>; } impl IoNetworkResultExt for io::Result { fn into_network_result(self) -> io::Result> { match self { Ok(v) => Ok(NetworkResult::Value(v)), #[cfg(feature = "io_error_more")] Err(e) => match e.kind() { io::ErrorKind::TimedOut => Ok(NetworkResult::Timeout), io::ErrorKind::ConnectionAborted | io::ErrorKind::ConnectionRefused | io::ErrorKind::ConnectionReset | io::ErrorKind::HostUnreachable | io::ErrorKind::NetworkUnreachable => Ok(NetworkResult::NoConnection(e)), io::ErrorKind::AddrNotAvailable => Ok(NetworkResult::AlreadyExists(e)), _ => Err(e), }, #[cfg(not(feature = "io_error_more"))] Err(e) => { #[cfg(not(target_arch = "wasm32"))] if let Some(os_err) = e.raw_os_error() { if os_err == libc::EHOSTUNREACH || os_err == libc::ENETUNREACH { return Ok(NetworkResult::NoConnection(e)); } } match e.kind() { io::ErrorKind::TimedOut => Ok(NetworkResult::Timeout), io::ErrorKind::ConnectionAborted | io::ErrorKind::ConnectionRefused | io::ErrorKind::ConnectionReset => Ok(NetworkResult::NoConnection(e)), io::ErrorKind::AddrNotAvailable => Ok(NetworkResult::AlreadyExists(e)), _ => Err(e), } } } } } pub trait NetworkResultResultExt { fn into_result_network_result(self) -> Result, E>; } impl NetworkResultResultExt for NetworkResult> { fn into_result_network_result(self) -> Result, E> { match self { NetworkResult::Timeout => Ok(NetworkResult::::Timeout), NetworkResult::NoConnection(e) => Ok(NetworkResult::::NoConnection(e)), NetworkResult::AlreadyExists(e) => Ok(NetworkResult::::AlreadyExists(e)), NetworkResult::InvalidMessage(s) => Ok(NetworkResult::::InvalidMessage(s)), NetworkResult::Value(Ok(v)) => Ok(NetworkResult::::Value(v)), NetworkResult::Value(Err(e)) => Err(e), } } } pub trait FoldedNetworkResultExt { fn folded(self) -> io::Result>; } impl FoldedNetworkResultExt for io::Result> { fn folded(self) -> io::Result> { match self { Ok(TimeoutOr::Timeout) => Ok(NetworkResult::Timeout), Ok(TimeoutOr::Value(v)) => Ok(NetworkResult::Value(v)), #[cfg(feature = "io_error_more")] Err(e) => match e.kind() { io::ErrorKind::TimedOut => Ok(NetworkResult::Timeout), io::ErrorKind::ConnectionAborted | io::ErrorKind::ConnectionRefused | io::ErrorKind::ConnectionReset | io::ErrorKind::HostUnreachable | io::ErrorKind::NetworkUnreachable => Ok(NetworkResult::NoConnection(e)), io::ErrorKind::AddrNotAvailable => Ok(NetworkResult::AlreadyExists(e)), _ => Err(e), }, #[cfg(not(feature = "io_error_more"))] Err(e) => { #[cfg(not(target_arch = "wasm32"))] if let Some(os_err) = e.raw_os_error() { if os_err == libc::EHOSTUNREACH || os_err == libc::ENETUNREACH { return Ok(NetworkResult::NoConnection(e)); } } match e.kind() { io::ErrorKind::TimedOut => Ok(NetworkResult::Timeout), io::ErrorKind::ConnectionAborted | io::ErrorKind::ConnectionRefused | io::ErrorKind::ConnectionReset => Ok(NetworkResult::NoConnection(e)), io::ErrorKind::AddrNotAvailable => Ok(NetworkResult::AlreadyExists(e)), _ => Err(e), } } } } } impl FoldedNetworkResultExt for io::Result> { fn folded(self) -> io::Result> { match self { Ok(v) => Ok(v), #[cfg(feature = "io_error_more")] Err(e) => match e.kind() { io::ErrorKind::TimedOut => Ok(NetworkResult::Timeout), io::ErrorKind::ConnectionAborted | io::ErrorKind::ConnectionRefused | io::ErrorKind::ConnectionReset | io::ErrorKind::HostUnreachable | io::ErrorKind::NetworkUnreachable => Ok(NetworkResult::NoConnection(e)), io::ErrorKind::AddrNotAvailable => Ok(NetworkResult::AlreadyExists(e)), _ => Err(e), }, #[cfg(not(feature = "io_error_more"))] Err(e) => { #[cfg(not(target_arch = "wasm32"))] if let Some(os_err) = e.raw_os_error() { if os_err == libc::EHOSTUNREACH || os_err == libc::ENETUNREACH { return Ok(NetworkResult::NoConnection(e)); } } match e.kind() { io::ErrorKind::TimedOut => Ok(NetworkResult::Timeout), io::ErrorKind::ConnectionAborted | io::ErrorKind::ConnectionRefused | io::ErrorKind::ConnectionReset => Ok(NetworkResult::NoConnection(e)), io::ErrorKind::AddrNotAvailable => Ok(NetworkResult::AlreadyExists(e)), _ => Err(e), } } } } } ////////////////////////////////////////////////////////////////// // Non-fallible network result #[must_use] pub enum NetworkResult { Timeout, NoConnection(io::Error), AlreadyExists(io::Error), InvalidMessage(String), Value(T), } impl NetworkResult { pub fn timeout() -> Self { Self::Timeout } pub fn no_connection(e: io::Error) -> Self { Self::NoConnection(e) } pub fn no_connection_other(s: S) -> Self { Self::NoConnection(io::Error::new(io::ErrorKind::Other, s.to_string())) } pub fn invalid_message(s: S) -> Self { Self::InvalidMessage(s.to_string()) } pub fn already_exists(e: io::Error) -> Self { Self::AlreadyExists(e) } pub fn value(value: T) -> Self { Self::Value(value) } pub fn is_timeout(&self) -> bool { matches!(self, Self::Timeout) } pub fn is_no_connection(&self) -> bool { matches!(self, Self::NoConnection(_)) } pub fn is_already_exists(&self) -> bool { matches!(self, Self::AlreadyExists(_)) } pub fn is_value(&self) -> bool { matches!(self, Self::Value(_)) } pub fn map X>(self, f: F) -> NetworkResult { match self { Self::Timeout => NetworkResult::::Timeout, Self::NoConnection(e) => NetworkResult::::NoConnection(e), Self::AlreadyExists(e) => NetworkResult::::AlreadyExists(e), Self::InvalidMessage(s) => NetworkResult::::InvalidMessage(s), Self::Value(v) => NetworkResult::::Value(f(v)), } } pub fn into_result(self) -> Result { match self { Self::Timeout => Err(io::Error::new(io::ErrorKind::TimedOut, "Timed out")), Self::NoConnection(e) => Err(e), Self::AlreadyExists(e) => Err(e), Self::InvalidMessage(s) => Err(io::Error::new( io::ErrorKind::InvalidData, format!("Invalid message: {}", s), )), Self::Value(v) => Ok(v), } } } impl From> for Option { fn from(t: NetworkResult) -> Self { match t { NetworkResult::Value(v) => Some(v), _ => None, } } } // impl Clone for NetworkResult { // fn clone(&self) -> Self { // match self { // Self::Timeout => Self::Timeout, // Self::NoConnection(e) => Self::NoConnection(e.clone()), // Self::InvalidMessage(s) => Self::InvalidMessage(s.clone()), // Self::Value(t) => Self::Value(t.clone()), // } // } // } impl Debug for NetworkResult { fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { match self { Self::Timeout => write!(f, "Timeout"), Self::NoConnection(e) => f.debug_tuple("NoConnection").field(e).finish(), Self::AlreadyExists(e) => f.debug_tuple("AlreadyExists").field(e).finish(), Self::InvalidMessage(s) => f.debug_tuple("InvalidMessage").field(s).finish(), Self::Value(v) => f.debug_tuple("Value").field(v).finish(), } } } impl Display for NetworkResult { fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { match self { Self::Timeout => write!(f, "Timeout"), Self::NoConnection(e) => write!(f, "NoConnection({})", e.kind()), Self::AlreadyExists(e) => write!(f, "AlreadyExists({})", e.kind()), Self::InvalidMessage(s) => write!(f, "InvalidMessage({})", s), Self::Value(_) => write!(f, "Value"), } } } impl Error for NetworkResult {} ////////////////////////////////////////////////////////////////// // Non-fallible network result macros #[macro_export] macro_rules! network_result_try { ($r: expr) => { match $r { NetworkResult::Timeout => return Ok(NetworkResult::Timeout), NetworkResult::NoConnection(e) => return Ok(NetworkResult::NoConnection(e)), NetworkResult::AlreadyExists(e) => return Ok(NetworkResult::AlreadyExists(e)), NetworkResult::InvalidMessage(s) => return Ok(NetworkResult::InvalidMessage(s)), NetworkResult::Value(v) => v, } }; ($r:expr => $f:tt) => { match $r { NetworkResult::Timeout => { $f; return Ok(NetworkResult::Timeout); } NetworkResult::NoConnection(e) => { $f; return Ok(NetworkResult::NoConnection(e)); } NetworkResult::AlreadyExists(e) => { $f; return Ok(NetworkResult::AlreadyExists(e)); } NetworkResult::InvalidMessage(s) => { $f; return Ok(NetworkResult::InvalidMessage(s)); } NetworkResult::Value(v) => v, } }; } #[macro_export] macro_rules! log_network_result { ($text:expr) => { cfg_if::cfg_if! { if #[cfg(debug_assertions)] { info!(target: "network_result", "{}", $text) } else { debug!(target: "network_result", "{}", $text) } } }; ($fmt:literal, $($arg:expr),+) => { cfg_if::cfg_if! { if #[cfg(debug_assertions)] { info!(target: "network_result", $fmt, $($arg),+); } else { debug!(target: "network_result", $fmt, $($arg),+); } } }; } #[macro_export] macro_rules! network_result_value_or_log { ($level: ident $r: expr => $f:tt) => { match $r { NetworkResult::Timeout => { log_network_result!( "{} at {}@{}:{}", "Timeout".cyan(), file!(), line!(), column!() ); $f } NetworkResult::NoConnection(e) => { log_network_result!( "{}({}) at {}@{}:{}", "No connection".cyan(), e.to_string(), file!(), line!(), column!() ); $f } NetworkResult::AlreadyExists(e) => { log_network_result!( "{}({}) at {}@{}:{}", "Already exists".cyan(), e.to_string(), file!(), line!(), column!() ); $f } NetworkResult::InvalidMessage(s) => { log_network_result!( "{}({}) at {}@{}:{}", "Invalid message".cyan(), s, file!(), line!(), column!() ); $f } NetworkResult::Value(v) => v, } }; }