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
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
//! An example component used in benchmarking.
//! It was originally defined [here](https://github.com/HotDrink/hotdrink/blob/3f9bc25751079c69f8277918521a75dd1163bee4/test/models/ladder-10.js).

use super::factory::ComponentFactory;
use crate::{
    model::{Component, Constraint, Method},
    planner::{MethodFailure, MethodSpec},
};
use std::{collections::HashMap, sync::Arc};

fn avg<T: Default>(_: Vec<T>) -> Result<Vec<T>, MethodFailure> {
    // Ok(vec![(args[0] + args[1]) / 2.0])
    Ok(vec![T::default()])
}

fn rev<T: Default>(_: Vec<T>) -> Result<Vec<T>, MethodFailure> {
    // Ok(vec![2.0 * args[0] - args[1]])
    Ok(vec![T::default()])
}

/// Generate a component with the "ladder"-shape.
pub fn ladder<T>(n_variables: usize) -> Component<T>
where
    T: Clone + Default + 'static,
{
    let values = vec![T::default(); n_variables];

    let mut constraints = Vec::new();
    for i in (0..n_variables.saturating_sub(3)).step_by(2) {
        let a0 = i;
        let b0 = i + 1;
        let a1 = i + 2;
        let b1 = i + 3;
        let lower = Constraint::new_with_name(
            format!("c{}", i),
            vec![
                Method::new("lower1".to_string(), vec![a0, a1], vec![b0], Arc::new(avg)),
                Method::new("lower2".to_string(), vec![b0, a0], vec![a1], Arc::new(rev)),
                Method::new("lower3".to_string(), vec![b0, a1], vec![a0], Arc::new(rev)),
            ],
        );
        constraints.push(lower);
        let upper = Constraint::new_with_name(
            format!("c{}", i + 1),
            vec![
                Method::new("upper1".to_string(), vec![b0, b1], vec![a1], Arc::new(avg)),
                Method::new("upper2".to_string(), vec![a1, b0], vec![b1], Arc::new(rev)),
                Method::new("upper3".to_string(), vec![a1, b1], vec![b0], Arc::new(rev)),
            ],
        );
        constraints.push(upper);
    }
    let mut name_to_index = HashMap::new();
    for i in 0..n_variables {
        name_to_index.insert(format!("var{}", i), i);
    }
    Component::new_with_map(
        Ladder::name().to_string(),
        name_to_index,
        values,
        constraints,
    )
}

/// A component factory for creating ladder-like components.
#[derive(Copy, Clone, Debug, Default, PartialEq, Eq)]
pub struct Ladder;

impl ComponentFactory for Ladder {
    fn name() -> &'static str {
        "ladder"
    }
    fn build<T>(n_constraints: usize) -> Component<T>
    where
        T: Clone + std::fmt::Debug + Default + 'static,
    {
        ladder(n_constraints + 2)
    }
}

#[cfg(test)]
mod tests {
    use super::{ladder, Ladder};
    use crate::{
        examples::components::factory::ComponentFactory, model::Component, planner::ComponentSpec,
    };

    #[test]
    fn constructs_without_error() {
        for i in 0..20 {
            let mut ladder = ladder::<()>(i);
            let result = ladder.update();
            assert_eq!(result, Ok(()));
        }
    }

    #[test]
    fn right_number_of_constraints() {
        for nc in (2..20).step_by(2) {
            let comp: Component<()> = Ladder::build(nc);
            assert_eq!(comp.constraints().len(), nc);
        }
    }
}