Skip to main content

solverforge_solver/heuristic/move/
ruin.rs

1/* RuinMove - unassigns a subset of entities for Large Neighborhood Search.
2
3This move "ruins" (unassigns) selected entities, allowing a construction
4heuristic to reassign them. This is the fundamental building block for
5Large Neighborhood Search (LNS) algorithms.
6
7# Zero-Erasure Design
8
9Uses concrete function pointers for variable access. No `dyn Any`, no downcasting.
10*/
11
12use std::fmt::Debug;
13
14use smallvec::SmallVec;
15use solverforge_core::domain::PlanningSolution;
16use solverforge_scoring::Director;
17
18use super::metadata::{
19    encode_option_debug, encode_usize, hash_str, MoveTabuScope, ScopedEntityTabuToken,
20};
21use super::{Move, MoveTabuSignature};
22
23/// A move that unassigns multiple entities for Large Neighborhood Search.
24///
25/// This move sets the planning variable to `None` for a set of entities,
26/// creating "gaps" that a construction heuristic can fill. Combined with
27/// construction, this enables exploring distant regions of the search space.
28///
29/// # Type Parameters
30/// * `S` - The planning solution type
31/// * `V` - The variable value type
32///
33/// # Example
34///
35/// ```
36/// use solverforge_solver::heuristic::r#move::RuinMove;
37/// use solverforge_core::domain::PlanningSolution;
38/// use solverforge_core::score::SoftScore;
39///
40/// #[derive(Clone, Debug)]
41/// struct Task { assigned_to: Option<i32>, score: Option<SoftScore> }
42/// #[derive(Clone, Debug)]
43/// struct Schedule { tasks: Vec<Task>, score: Option<SoftScore> }
44///
45/// impl PlanningSolution for Schedule {
46///     type Score = SoftScore;
47///     fn score(&self) -> Option<Self::Score> { self.score }
48///     fn set_score(&mut self, score: Option<Self::Score>) { self.score = score; }
49/// }
50///
51/// fn get_task(s: &Schedule, idx: usize, _var: usize) -> Option<i32> {
52///     s.tasks.get(idx).and_then(|t| t.assigned_to)
53/// }
54/// fn set_task(s: &mut Schedule, idx: usize, _var: usize, v: Option<i32>) {
55///     if let Some(t) = s.tasks.get_mut(idx) { t.assigned_to = v; }
56/// }
57///
58/// // Ruin entities 0, 2, and 4
59/// let m = RuinMove::<Schedule, i32>::new(
60///     &[0, 2, 4],
61///     get_task, set_task,
62///     0, "assigned_to", 0,
63/// );
64/// ```
65pub struct RuinMove<S, V> {
66    // Indices of entities to unassign
67    entity_indices: SmallVec<[usize; 8]>,
68    getter: fn(&S, usize, usize) -> Option<V>,
69    // Set value for an entity
70    setter: fn(&mut S, usize, usize, Option<V>),
71    variable_index: usize,
72    variable_name: &'static str,
73    descriptor_index: usize,
74}
75
76impl<S, V> Clone for RuinMove<S, V> {
77    fn clone(&self) -> Self {
78        Self {
79            entity_indices: self.entity_indices.clone(),
80            getter: self.getter,
81            setter: self.setter,
82            variable_index: self.variable_index,
83            variable_name: self.variable_name,
84            descriptor_index: self.descriptor_index,
85        }
86    }
87}
88
89impl<S, V: Debug> Debug for RuinMove<S, V> {
90    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
91        f.debug_struct("RuinMove")
92            .field("entities", &self.entity_indices.as_slice())
93            .field("variable_index", &self.variable_index)
94            .field("variable_name", &self.variable_name)
95            .finish()
96    }
97}
98
99impl<S, V> RuinMove<S, V> {
100    /// Creates a new ruin move with concrete function pointers.
101    ///
102    /// # Arguments
103    /// * `entity_indices` - Indices of entities to unassign
104    /// * `getter` - Function to get current value
105    /// * `setter` - Function to set value
106    /// * `variable_name` - Name of the planning variable
107    /// * `descriptor_index` - Entity descriptor index
108    pub fn new(
109        entity_indices: &[usize],
110        getter: fn(&S, usize, usize) -> Option<V>,
111        setter: fn(&mut S, usize, usize, Option<V>),
112        variable_index: usize,
113        variable_name: &'static str,
114        descriptor_index: usize,
115    ) -> Self {
116        Self {
117            entity_indices: SmallVec::from_slice(entity_indices),
118            getter,
119            setter,
120            variable_index,
121            variable_name,
122            descriptor_index,
123        }
124    }
125
126    pub fn entity_indices_slice(&self) -> &[usize] {
127        &self.entity_indices
128    }
129
130    pub fn ruin_count(&self) -> usize {
131        self.entity_indices.len()
132    }
133}
134
135impl<S, V> Move<S> for RuinMove<S, V>
136where
137    S: PlanningSolution,
138    V: Clone + Send + Sync + Debug + 'static,
139{
140    type Undo = SmallVec<[(usize, Option<V>); 8]>;
141
142    fn is_doable<D: Director<S>>(&self, score_director: &D) -> bool {
143        // At least one entity must be currently assigned
144        let solution = score_director.working_solution();
145        self.entity_indices
146            .iter()
147            .any(|&idx| (self.getter)(solution, idx, self.variable_index).is_some())
148    }
149
150    fn do_move<D: Director<S>>(&self, score_director: &mut D) -> Self::Undo {
151        let getter = self.getter;
152        let setter = self.setter;
153        let descriptor = self.descriptor_index;
154        let variable_index = self.variable_index;
155
156        // Collect old values for undo
157        let old_values: SmallVec<[(usize, Option<V>); 8]> = self
158            .entity_indices
159            .iter()
160            .map(|&idx| {
161                let old = getter(score_director.working_solution(), idx, variable_index);
162                (idx, old)
163            })
164            .collect();
165
166        // Unassign all entities
167        for &idx in &self.entity_indices {
168            score_director.before_variable_changed(descriptor, idx);
169            setter(
170                score_director.working_solution_mut(),
171                idx,
172                variable_index,
173                None,
174            );
175            score_director.after_variable_changed(descriptor, idx);
176        }
177
178        old_values
179    }
180
181    fn undo_move<D: Director<S>>(&self, score_director: &mut D, undo: Self::Undo) {
182        for (idx, _) in &undo {
183            score_director.before_variable_changed(self.descriptor_index, *idx);
184        }
185        for (idx, old_value) in undo {
186            (self.setter)(
187                score_director.working_solution_mut(),
188                idx,
189                self.variable_index,
190                old_value,
191            );
192        }
193        for &idx in &self.entity_indices {
194            score_director.after_variable_changed(self.descriptor_index, idx);
195        }
196    }
197
198    fn descriptor_index(&self) -> usize {
199        self.descriptor_index
200    }
201
202    fn entity_indices(&self) -> &[usize] {
203        &self.entity_indices
204    }
205
206    fn variable_name(&self) -> &str {
207        self.variable_name
208    }
209
210    fn tabu_signature<D: Director<S>>(&self, score_director: &D) -> MoveTabuSignature {
211        let scope = MoveTabuScope::new(self.descriptor_index, self.variable_name);
212        let entity_ids: SmallVec<[u64; 2]> = self
213            .entity_indices
214            .iter()
215            .map(|&idx| encode_usize(idx))
216            .collect();
217        let entity_tokens: SmallVec<[ScopedEntityTabuToken; 2]> = entity_ids
218            .iter()
219            .copied()
220            .map(|entity_id| scope.entity_token(entity_id))
221            .collect();
222        let mut value_ids: SmallVec<[u64; 2]> = SmallVec::new();
223        for &idx in &self.entity_indices {
224            let value = (self.getter)(score_director.working_solution(), idx, self.variable_index);
225            value_ids.push(encode_option_debug(value.as_ref()));
226        }
227        let variable_id = hash_str(self.variable_name);
228        let mut move_id = SmallVec::<[u64; 8]>::from_slice(&[
229            encode_usize(self.descriptor_index),
230            variable_id,
231            encode_usize(self.entity_indices.len()),
232        ]);
233        move_id.extend(entity_ids.iter().copied());
234        move_id.extend(value_ids.iter().copied());
235
236        MoveTabuSignature::new(scope, move_id.clone(), move_id).with_entity_tokens(entity_tokens)
237    }
238}