pumpkin_solver/branching/variable_selection/
first_fail.rsuse log::warn;
use crate::branching::Direction;
use crate::branching::InOrderTieBreaker;
use crate::branching::SelectionContext;
use crate::branching::TieBreaker;
use crate::branching::VariableSelector;
use crate::engine::variables::DomainId;
use crate::pumpkin_assert_eq_simple;
pub struct FirstFail<Var, TieBreaking> {
variables: Vec<Var>,
tie_breaker: TieBreaking,
}
impl<Var, TieBreaking> std::fmt::Debug for FirstFail<Var, TieBreaking> {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.debug_struct("FirstFail").finish()
}
}
impl<Var: Clone> FirstFail<Var, InOrderTieBreaker<Var, i32>> {
pub fn new(variables: &[Var]) -> Self {
if variables.is_empty() {
warn!("The FirstFail variable selector was not provided with any variables");
}
Self {
variables: variables.to_vec(),
tie_breaker: InOrderTieBreaker::new(Direction::Minimum),
}
}
}
impl<Var: Clone + 'static, TieBreaking: TieBreaker<Var, i32>> FirstFail<Var, TieBreaking> {
pub fn with_tie_breaker(variables: &[Var], tie_breaker: TieBreaking) -> Self {
pumpkin_assert_eq_simple!(
tie_breaker.get_direction(),
Direction::Minimum,
"The provided tie-breaker to FirstFail attempts to find the Maximum value
instead of the Minimum value, please ensure that you have passed the correct tie-breaker");
if variables.is_empty() {
warn!("The FirstFail variable selector was not provided with any variables");
return FirstFail {
variables: vec![],
tie_breaker,
};
}
Self {
variables: variables.to_vec(),
tie_breaker,
}
}
}
impl<TieBreaking> VariableSelector<DomainId> for FirstFail<DomainId, TieBreaking>
where
TieBreaking: TieBreaker<DomainId, i32>,
{
fn select_variable(&mut self, context: &SelectionContext) -> Option<DomainId> {
self.variables
.iter()
.filter(|variable| !context.is_integer_fixed(**variable))
.for_each(|variable| {
self.tie_breaker
.consider(*variable, context.get_size_of_domain(*variable));
});
self.tie_breaker.select()
}
}
#[cfg(test)]
mod tests {
use crate::basic_types::tests::TestRandom;
use crate::branching::FirstFail;
use crate::branching::SelectionContext;
use crate::branching::VariableSelector;
#[test]
fn test_correctly_selected() {
let (mut assignments_integer, assignments_propositional) =
SelectionContext::create_for_testing(2, 0, Some(vec![(0, 10), (5, 20)]));
let mut test_rng = TestRandom::default();
let integer_variables = assignments_integer.get_domains().collect::<Vec<_>>();
let mut strategy = FirstFail::new(&integer_variables);
{
let context = SelectionContext::new(
&assignments_integer,
&assignments_propositional,
&mut test_rng,
);
let selected = strategy.select_variable(&context);
assert!(selected.is_some());
assert_eq!(selected.unwrap(), integer_variables[0]);
}
let _ = assignments_integer.tighten_lower_bound(integer_variables[1], 15, None);
let context = SelectionContext::new(
&assignments_integer,
&assignments_propositional,
&mut test_rng,
);
let selected = strategy.select_variable(&context);
assert!(selected.is_some());
assert_eq!(selected.unwrap(), integer_variables[1]);
}
#[test]
fn fixed_variables_are_not_selected() {
let (assignments_integer, assignments_propositional) =
SelectionContext::create_for_testing(2, 0, Some(vec![(10, 10), (20, 20)]));
let mut test_rng = TestRandom::default();
let context = SelectionContext::new(
&assignments_integer,
&assignments_propositional,
&mut test_rng,
);
let integer_variables = context.get_domains().collect::<Vec<_>>();
let mut strategy = FirstFail::new(&integer_variables);
let selected = strategy.select_variable(&context);
assert!(selected.is_none());
}
}