odbc_api/handles/sql_result.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83
use odbc_sys::SqlReturn;
/// Result of an ODBC function call. Variants hold the same meaning as the constants associated with
/// [`SqlReturn`]. This type may hold results, but it is still the responsibility of the user to
/// fetch and handle the diagnostics in case of an Error.
#[derive(Debug, PartialEq, Eq, Clone, Copy)]
pub enum SqlResult<T> {
/// The function has been executed successfully.
Success(T),
/// The function has been executed successfully. There have been warnings.
SuccessWithInfo(T),
/// Meaning depends on the function emitting `NoData`.
NoData,
/// Emmitted by execute in case delayed parameters have been bound and their input values are
/// now required.
NeedData,
/// The function was started asynchronously and is still executing.
StillExecuting,
/// The function returned an error state. Check diagnostics.
Error {
/// Name of the ODBC Api call which caused the error. This might help interpreting
/// associated ODBC diagnostics if the error is bubbled all the way up to the
/// end users output, but the context is lost.
function: &'static str,
},
}
impl SqlResult<()> {
/// Append a return value a successful to Result
pub fn on_success<F, T>(self, f: F) -> SqlResult<T>
where
F: FnOnce() -> T,
{
self.map(|()| f())
}
}
impl<T> SqlResult<T> {
/// `True` if variant is [`SqlResult::Error`].
pub fn is_err(&self) -> bool {
matches!(self, SqlResult::Error { .. })
}
/// Applies `f` to any value wrapped in `Success` or `SuccessWithInfo`.
pub fn map<U, F>(self, f: F) -> SqlResult<U>
where
F: FnOnce(T) -> U,
{
match self {
SqlResult::Success(v) => SqlResult::Success(f(v)),
SqlResult::SuccessWithInfo(v) => SqlResult::SuccessWithInfo(f(v)),
SqlResult::Error { function } => SqlResult::Error { function },
SqlResult::StillExecuting => SqlResult::StillExecuting,
SqlResult::NoData => SqlResult::NoData,
SqlResult::NeedData => SqlResult::NeedData,
}
}
pub fn unwrap(self) -> T {
match self {
SqlResult::Success(v) | SqlResult::SuccessWithInfo(v) => v,
_ => panic!("Invalid unwrapping of SqlResult"),
}
}
}
pub trait ExtSqlReturn {
fn into_sql_result(self, function_name: &'static str) -> SqlResult<()>;
}
impl ExtSqlReturn for SqlReturn {
fn into_sql_result(self, function: &'static str) -> SqlResult<()> {
match self {
SqlReturn::SUCCESS => SqlResult::Success(()),
SqlReturn::SUCCESS_WITH_INFO => SqlResult::SuccessWithInfo(()),
SqlReturn::ERROR => SqlResult::Error { function },
SqlReturn::NO_DATA => SqlResult::NoData,
SqlReturn::NEED_DATA => SqlResult::NeedData,
SqlReturn::STILL_EXECUTING => SqlResult::StillExecuting,
r => panic!("Unexpected return value '{r:?}' for ODBC function '{function}'"),
}
}
}