xgboost-rs 0.1.3

Bindings to dmlc/xgboost.
Documentation
//! Functionality related to errors and error handling.

use std::error::Error;
use std::ffi::CStr;
use std::fmt::{self, Display};

/// Convenience return type for most operations which can return an `XGBError`.
pub type XGBResult<T> = std::result::Result<T, XGBError>;

/// Wrap errors returned by the `XGBoost` library.
#[derive(Debug, Eq, PartialEq)]
pub struct XGBError {
    desc: String,
}

impl XGBError {
    pub(crate) fn new<S: Into<String>>(desc: S) -> Self {
        XGBError { desc: desc.into() }
    }

    /// Check the return value from an `XGBoost` FFI call, and return the last error message on error.
    ///
    /// Return values of 0 are treated as success, returns values of -1 are treated as errors.
    ///
    /// Meaning of any other return values are undefined, and will cause a panic.
    pub(crate) fn check_return_value(ret_val: i32) -> XGBResult<()> {
        match ret_val {
            0 => Ok(()),
            -1 => Err(XGBError::from_xgboost()),
            _ => panic!("unexpected return value '{}', expected 0 or -1", ret_val),
        }
    }

    /// Get the last error message from `XGBoost`.
    fn from_xgboost() -> Self {
        let c_str = unsafe { CStr::from_ptr(xgboost_rs_sys::XGBGetLastError()) };
        let str_slice = c_str.to_str().unwrap();
        XGBError {
            desc: str_slice.to_owned(),
        }
    }
}

impl Error for XGBError {}

impl Display for XGBError {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        write!(f, "XGBoost error: {}", &self.desc)
    }
}

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn return_value_handling() {
        let result = XGBError::check_return_value(0);
        assert_eq!(result, Ok(()));

        let result = XGBError::check_return_value(-1);
        assert_eq!(
            result,
            Err(XGBError {
                desc: String::new()
            })
        );
    }
}