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
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
#[cfg(test)]
#[path = "../../../tests/unit/construction/constraints/conditional_test.rs"]
mod conditional_test;
use crate::construction::constraints::{ConstraintModule, ConstraintVariant};
use crate::construction::heuristics::{RouteContext, SolutionContext};
use crate::models::problem::Job;
use hashbrown::HashSet;
use std::slice::Iter;
pub trait JobContextTransition {
fn remove_from_required(&self, ctx: &SolutionContext, job: &Job) -> bool;
fn promote_to_required(&self, ctx: &SolutionContext, job: &Job) -> bool;
fn remove_from_locked(&self, ctx: &SolutionContext, job: &Job) -> bool;
fn promote_to_locked(&self, ctx: &SolutionContext, job: &Job) -> bool;
}
pub struct ConcreteJobContextTransition<FRemoveRequired, FPromoteRequired, FRemoveLocked, FPromoteLocked>
where
FRemoveRequired: Fn(&SolutionContext, &Job) -> bool,
FPromoteRequired: Fn(&SolutionContext, &Job) -> bool,
FRemoveLocked: Fn(&SolutionContext, &Job) -> bool,
FPromoteLocked: Fn(&SolutionContext, &Job) -> bool,
{
pub remove_required: FRemoveRequired,
pub promote_required: FPromoteRequired,
pub remove_locked: FRemoveLocked,
pub promote_locked: FPromoteLocked,
}
impl<FRemoveRequired, FPromoteRequired, FRemoveLocked, FPromoteLocked> JobContextTransition
for ConcreteJobContextTransition<FRemoveRequired, FPromoteRequired, FRemoveLocked, FPromoteLocked>
where
FRemoveRequired: Fn(&SolutionContext, &Job) -> bool,
FPromoteRequired: Fn(&SolutionContext, &Job) -> bool,
FRemoveLocked: Fn(&SolutionContext, &Job) -> bool,
FPromoteLocked: Fn(&SolutionContext, &Job) -> bool,
{
fn remove_from_required(&self, ctx: &SolutionContext, job: &Job) -> bool {
(self.remove_required)(ctx, job)
}
fn promote_to_required(&self, ctx: &SolutionContext, job: &Job) -> bool {
(self.promote_required)(ctx, job)
}
fn remove_from_locked(&self, ctx: &SolutionContext, job: &Job) -> bool {
(self.remove_locked)(ctx, job)
}
fn promote_to_locked(&self, ctx: &SolutionContext, job: &Job) -> bool {
(self.promote_locked)(ctx, job)
}
}
pub struct ConditionalJobModule {
context_transition: Box<dyn JobContextTransition + Send + Sync>,
state_keys: Vec<i32>,
constraints: Vec<ConstraintVariant>,
}
impl ConditionalJobModule {
pub fn new(context_transition: Box<dyn JobContextTransition + Send + Sync>) -> Self {
Self { context_transition, state_keys: vec![], constraints: vec![] }
}
}
impl ConstraintModule for ConditionalJobModule {
fn accept_insertion(&self, solution_ctx: &mut SolutionContext, _route_index: usize, _job: &Job) {
self.accept_solution_state(solution_ctx);
}
fn accept_route_state(&self, _ctx: &mut RouteContext) {}
fn accept_solution_state(&self, ctx: &mut SolutionContext) {
let ignored: HashSet<Job> =
ctx.required.iter().filter(|job| self.context_transition.remove_from_required(ctx, job)).cloned().collect();
ctx.required.retain(|job| !ignored.contains(job));
let required: HashSet<Job> =
ctx.ignored.iter().filter(|job| self.context_transition.promote_to_required(ctx, job)).cloned().collect();
ctx.ignored.retain(|job| !required.contains(job));
ctx.required.extend(required);
ctx.ignored.extend(ignored);
let not_locked: HashSet<Job> =
ctx.locked.iter().filter(|job| self.context_transition.remove_from_locked(ctx, job)).cloned().collect();
ctx.locked.retain(|job| !not_locked.contains(job));
let locked: HashSet<Job> = ctx
.required
.iter()
.chain(ctx.ignored.iter())
.filter(|job| self.context_transition.promote_to_locked(ctx, job))
.cloned()
.collect();
ctx.locked.extend(locked);
}
fn state_keys(&self) -> Iter<i32> {
self.state_keys.iter()
}
fn get_constraints(&self) -> Iter<ConstraintVariant> {
self.constraints.iter()
}
}