use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
pub enum LoadState {
NotLoaded,
Partial(std::collections::HashSet<String>),
FullyLoaded,
}
impl Default for LoadState {
fn default() -> Self {
LoadState::NotLoaded
}
}
impl LoadState {
pub fn is_loaded(&self, field_or_relation: &str) -> bool {
match self {
LoadState::NotLoaded => false,
LoadState::FullyLoaded => true,
LoadState::Partial(set) => set.contains(field_or_relation),
}
}
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub enum EvalResult<T> {
Value(T),
Null,
NotLoaded {
failed_node: String,
attempted_path: String,
},
}
impl<T> EvalResult<T> {
pub fn and_then<U, F: FnOnce(T) -> EvalResult<U>>(self, field_name: &str, f: F) -> EvalResult<U> {
match self {
EvalResult::Value(val) => match f(val) {
EvalResult::NotLoaded { failed_node, attempted_path } => {
let new_path = if attempted_path == field_name {
attempted_path
} else if attempted_path.is_empty() {
field_name.to_string()
} else {
format!("{}.{}", field_name, attempted_path)
};
EvalResult::NotLoaded {
failed_node,
attempted_path: new_path
}
},
other => other,
},
EvalResult::Null => EvalResult::Null,
EvalResult::NotLoaded { failed_node, attempted_path } => {
let new_path = if attempted_path.is_empty() {
field_name.to_string()
} else {
format!("{}.{}", attempted_path, field_name)
};
EvalResult::NotLoaded {
failed_node,
attempted_path: new_path
}
},
}
}
pub fn map<U, F: FnOnce(T) -> U>(self, f: F) -> EvalResult<U> {
match self {
EvalResult::Value(val) => EvalResult::Value(f(val)),
EvalResult::Null => EvalResult::Null,
EvalResult::NotLoaded { failed_node, attempted_path } => EvalResult::NotLoaded { failed_node, attempted_path },
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use std::collections::HashSet;
struct Company {
pub name: Option<String>,
pub __load_state: LoadState,
}
impl Company {
fn eval_name(&self) -> EvalResult<&str> {
if !self.__load_state.is_loaded("name") {
EvalResult::NotLoaded { failed_node: "name".to_string(), attempted_path: "name".to_string() }
} else {
match &self.name {
Some(n) => EvalResult::Value(n.as_str()),
None => EvalResult::Null,
}
}
}
}
struct Platform {
pub company: Option<Box<Company>>,
pub __load_state: LoadState,
}
impl Platform {
fn eval_company(&self) -> EvalResult<&Company> {
if !self.__load_state.is_loaded("company") {
EvalResult::NotLoaded { failed_node: "company".to_string(), attempted_path: "company".to_string() }
} else {
match &self.company {
Some(c) => EvalResult::Value(c.as_ref()),
None => EvalResult::Null,
}
}
}
}
struct User {
pub platform: Option<Box<Platform>>,
pub __load_state: LoadState,
}
impl User {
fn eval_platform(&self) -> EvalResult<&Platform> {
if !self.__load_state.is_loaded("platform") {
EvalResult::NotLoaded { failed_node: "platform".to_string(), attempted_path: "platform".to_string() }
} else {
match &self.platform {
Some(p) => EvalResult::Value(p.as_ref()),
None => EvalResult::Null,
}
}
}
}
#[test]
fn test_eval_tracking_chain_perfect_path() {
let company = Company {
name: None,
__load_state: LoadState::NotLoaded,
};
let platform = Platform {
company: Some(Box::new(company)),
__load_state: LoadState::FullyLoaded,
};
let user = User {
platform: Some(Box::new(platform)),
__load_state: LoadState::FullyLoaded,
};
let result = user.eval_platform()
.and_then("platform", |p| p.eval_company().and_then("company", |c| c.eval_name()));
match &result {
EvalResult::NotLoaded { missing_path } => {
assert_eq!(missing_path, "platform.company.name");
println!("\n\n>>> 【系统捕获到未加载异常】 <<<\n{:#?}\n\n", result);
}
_ => panic!("Expected NotLoaded but got {:?}", result),
}
}
#[test]
fn test_eval_tracking_chain_middle_break() {
let platform = Platform {
company: None, __load_state: LoadState::NotLoaded, };
let user = User {
platform: Some(Box::new(platform)),
__load_state: LoadState::FullyLoaded,
};
let result = user.eval_platform()
.and_then("platform", |p| p.eval_company().and_then("company", |c| c.eval_name()));
match result {
EvalResult::NotLoaded { missing_path } => {
assert_eq!(missing_path, "platform.company");
println!("Success! Intercepted middle missing path: {}", missing_path);
}
_ => panic!("Expected NotLoaded"),
}
}
#[test]
fn test_eval_tracking_chain_normal_null() {
let company = Company {
name: None, __load_state: LoadState::FullyLoaded,
};
let platform = Platform {
company: Some(Box::new(company)),
__load_state: LoadState::FullyLoaded,
};
let user = User {
platform: Some(Box::new(platform)),
__load_state: LoadState::FullyLoaded,
};
let result = user.eval_platform()
.and_then("platform", |p| p.eval_company().and_then("company", |c| c.eval_name()));
match result {
EvalResult::Null => {
println!("Success! Legitimately empty (Null), not an error.");
}
_ => panic!("Expected Null"),
}
}
}