use crate::construction::heuristics::UnassignmentInfo;
use crate::models::common::{Cost, Location};
use crate::models::problem::*;
use crate::models::solution::{Registry, Route};
use crate::models::*;
use rosomaxa::evolution::TelemetryMetrics;
use rosomaxa::prelude::*;
use std::fmt::{Debug, Formatter};
use std::sync::Arc;
pub struct Problem {
pub fleet: Arc<Fleet>,
pub jobs: Arc<Jobs>,
pub locks: Vec<Arc<Lock>>,
pub goal: Arc<GoalContext>,
pub activity: Arc<dyn ActivityCost>,
pub transport: Arc<dyn TransportCost>,
pub extras: Arc<Extras>,
}
impl Debug for Problem {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
f.debug_struct(short_type_name::<Self>())
.field("fleet", &self.fleet)
.field("jobs", &self.jobs.size())
.field("locks", &self.locks.len())
.field("goal", self.goal.as_ref())
.finish_non_exhaustive()
}
}
pub struct Solution {
pub cost: Cost,
pub registry: Registry,
pub routes: Vec<Route>,
pub unassigned: Vec<(Job, UnassignmentInfo)>,
pub telemetry: Option<TelemetryMetrics>,
}
pub enum LockOrder {
Any,
Sequence,
Strict,
}
#[derive(Clone)]
pub enum LockPosition {
Any,
Departure,
Arrival,
Fixed,
}
pub struct LockDetail {
pub order: LockOrder,
pub position: LockPosition,
pub jobs: Vec<Job>,
}
pub struct Lock {
pub condition_fn: Arc<dyn Fn(&Actor) -> bool + Sync + Send>,
pub details: Vec<LockDetail>,
pub is_lazy: bool,
}
impl LockDetail {
pub fn new(order: LockOrder, position: LockPosition, jobs: Vec<Job>) -> Self {
Self { order, position, jobs }
}
}
impl Lock {
pub fn new(condition: Arc<dyn Fn(&Actor) -> bool + Sync + Send>, details: Vec<LockDetail>, is_lazy: bool) -> Self {
Self { condition_fn: condition, details, is_lazy }
}
}
pub type FleetGroupKeyFn = dyn Fn(&Actor) -> usize + Send + Sync;
#[derive(Default)]
pub struct ProblemBuilder {
jobs: Vec<Job>,
vehicles: Vec<Vehicle>,
#[allow(clippy::type_complexity)]
group_key_fn: Option<Box<dyn Fn(&[Arc<Actor>]) -> Box<FleetGroupKeyFn>>>,
goal: Option<Arc<GoalContext>>,
activity: Option<Arc<dyn ActivityCost>>,
transport: Option<Arc<dyn TransportCost>>,
extras: Option<Arc<Extras>>,
logger: Option<InfoLogger>,
}
impl ProblemBuilder {
pub fn add_job(mut self, job: Job) -> Self {
self.jobs.push(job);
self
}
pub fn add_jobs(mut self, jobs: impl Iterator<Item = Job>) -> Self {
self.jobs.extend(jobs);
self
}
pub fn add_vehicle(mut self, vehicle: Vehicle) -> Self {
self.vehicles.push(vehicle);
self
}
pub fn add_vehicles(mut self, vehicles: impl Iterator<Item = Vehicle>) -> Self {
self.vehicles.extend(vehicles);
self
}
pub fn with_vehicle_similarity(
mut self,
group_key_fn: impl Fn(&[Arc<Actor>]) -> Box<FleetGroupKeyFn> + 'static,
) -> Self {
self.group_key_fn = Some(Box::new(group_key_fn));
self
}
pub fn with_goal(mut self, goal: GoalContext) -> Self {
self.goal = Some(Arc::new(goal));
self
}
pub fn with_transport_cost(mut self, transport: Arc<dyn TransportCost>) -> Self {
self.transport = Some(transport);
self
}
pub fn with_activity_cost(mut self, activity: Arc<dyn ActivityCost>) -> Self {
self.activity = Some(activity);
self
}
pub fn with_extras(mut self, extras: Extras) -> Self {
self.extras = Some(Arc::new(extras));
self
}
pub fn with_logger(mut self, logger: InfoLogger) -> Self {
self.logger = Some(logger);
self
}
pub fn build(mut self) -> GenericResult<Problem> {
if self.jobs.is_empty() {
return Err("empty list of jobs: specify at least one job".into());
}
if self.vehicles.is_empty() {
return Err("empty list of vehicles: specify at least one vehicle".into());
}
let transport = self.transport.take().ok_or_else(|| {
GenericError::from("no information about routing data: use 'with_transport_cost' method to specify it")
})?;
let activity = self.activity.take().unwrap_or_else(|| Arc::new(SimpleActivityCost::default()));
let goal = self
.goal
.take()
.ok_or_else(|| GenericError::from("unknown goal of optimization: use 'with_goal' method to set it"))?;
let extras = self.extras.take().unwrap_or_else(|| Arc::new(Extras::default()));
let driver = Arc::new(Driver::empty());
let vehicles = self.vehicles.into_iter().map(Arc::new).collect();
let group_key = self.group_key_fn.take().unwrap_or_else(|| Box::new(|_| Box::new(|a| a.vehicle.profile.index)));
let fleet = Arc::new(Fleet::new(vec![driver], vehicles, group_key));
let logger = self.logger.unwrap_or_else(|| Arc::new(|msg| println!("{}", msg)));
let jobs = Arc::new(Jobs::new(fleet.as_ref(), self.jobs, transport.as_ref(), &logger)?);
Ok(Problem { fleet, jobs, locks: vec![], goal, activity, transport, extras })
}
}
impl Solution {
pub fn get_locations(&self) -> impl Iterator<Item = impl Iterator<Item = Location> + '_> + '_ {
self.routes.iter().map(|route| route.tour.all_activities().map(|activity| activity.place.location))
}
}