solverforge_solver/heuristic/move/
ruin.rs1use 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
23pub struct RuinMove<S, V> {
66 entity_indices: SmallVec<[usize; 8]>,
68 getter: fn(&S, usize, usize) -> Option<V>,
69 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 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 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 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 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}