Struct cfg_expr::expr::Expression
source · pub struct Expression { /* private fields */ }
Expand description
A parsed cfg()
expression that can evaluated
Implementations§
source§impl Expression
impl Expression
sourcepub fn parse(original: &str) -> Result<Self, ParseError>
pub fn parse(original: &str) -> Result<Self, ParseError>
Given a cfg() expression (the cfg( and ) are optional), attempts to parse it into a form where it can be evaluated
assert!(cfg_expr::Expression::parse(r#"cfg(all(unix, target_arch = "x86_64"))"#).is_ok());
source§impl Expression
impl Expression
sourcepub fn predicates(&self) -> impl Iterator<Item = Predicate<'_>>
pub fn predicates(&self) -> impl Iterator<Item = Predicate<'_>>
An iterator over each predicate in the expression
sourcepub fn eval<EP, T>(&self, eval_predicate: EP) -> Twhere
EP: FnMut(&Predicate<'_>) -> T,
T: Logic + Debug,
pub fn eval<EP, T>(&self, eval_predicate: EP) -> Twhere
EP: FnMut(&Predicate<'_>) -> T,
T: Logic + Debug,
Evaluates the expression, using the provided closure to determine the value of each predicate, which are then combined into a final result depending on the functions not(), all(), or any() in the expression.
eval_predicate
typically returns bool
, but may return any type that implements
the Logic
trait.
Examples
use cfg_expr::{targets::*, Expression, Predicate};
let linux_musl = get_builtin_target_by_triple("x86_64-unknown-linux-musl").unwrap();
let expr = Expression::parse(r#"all(not(windows), target_env = "musl", any(target_arch = "x86", target_arch = "x86_64"))"#).unwrap();
assert!(expr.eval(|pred| {
match pred {
Predicate::Target(tp) => tp.matches(linux_musl),
_ => false,
}
}));
Returning Option<bool>
, where None
indicates the result is unknown:
use cfg_expr::{targets::*, Expression, Predicate};
let expr = Expression::parse(r#"any(target_feature = "sse2", target_env = "musl")"#).unwrap();
let linux_gnu = get_builtin_target_by_triple("x86_64-unknown-linux-gnu").unwrap();
let linux_musl = get_builtin_target_by_triple("x86_64-unknown-linux-musl").unwrap();
fn eval(expr: &Expression, target: &TargetInfo) -> Option<bool> {
expr.eval(|pred| {
match pred {
Predicate::Target(tp) => Some(tp.matches(target)),
Predicate::TargetFeature(_) => None,
_ => panic!("unexpected predicate"),
}
})
}
// Whether the target feature is present is unknown, so the whole expression evaluates to
// None (unknown).
assert_eq!(eval(&expr, linux_gnu), None);
// Whether the target feature is present is irrelevant for musl, since the any() always
// evaluates to true.
assert_eq!(eval(&expr, linux_musl), Some(true));
sourcepub fn original(&self) -> &str
pub fn original(&self) -> &str
The original string which has been parsed to produce this Expression
.
use cfg_expr::Expression;
assert_eq!(
Expression::parse("any()").unwrap().original(),
"any()"
);
Trait Implementations§
source§impl Clone for Expression
impl Clone for Expression
source§fn clone(&self) -> Expression
fn clone(&self) -> Expression
1.0.0 · source§fn clone_from(&mut self, source: &Self)
fn clone_from(&mut self, source: &Self)
source
. Read moresource§impl Debug for Expression
impl Debug for Expression
source§impl PartialEq<Expression> for Expression
impl PartialEq<Expression> for Expression
PartialEq
will do a syntactical comparaison, so will just check if both
expressions have been parsed from the same string, not if they are semantically
equivalent.
use cfg_expr::Expression;
assert_eq!(
Expression::parse("any()").unwrap(),
Expression::parse("any()").unwrap()
);
assert_ne!(
Expression::parse("any()").unwrap(),
Expression::parse("unix").unwrap()
);