use std::collections::HashMap;
use std::ffi::OsString;
use std::fs::File;
use std::io::{BufRead, BufReader};
use std::path::{Path, PathBuf};
use crate::lp_format::*;
use crate::solvers::{
Solution, SolverProgram, SolverWithSolutionParsing, Status, WithMaxSeconds, WithMipGap,
WithNbThreads,
};
#[derive(Debug, Clone)]
pub struct CbcSolver {
name: String,
command_name: String,
temp_solution_file: Option<PathBuf>,
threads: Option<u32>,
seconds: Option<u32>,
mipgap: Option<f32>,
}
impl Default for CbcSolver {
fn default() -> Self {
Self::new()
}
}
impl CbcSolver {
pub fn new() -> CbcSolver {
CbcSolver {
name: "Cbc".to_string(),
command_name: "cbc".to_string(),
temp_solution_file: None,
threads: None,
seconds: None,
mipgap: None,
}
}
pub fn command_name(&self, command_name: String) -> CbcSolver {
CbcSolver {
name: self.name.clone(),
command_name,
temp_solution_file: self.temp_solution_file.clone(),
threads: self.threads,
seconds: self.seconds,
mipgap: self.mipgap,
}
}
pub fn with_temp_solution_file(&self, temp_solution_file: String) -> CbcSolver {
CbcSolver {
name: self.name.clone(),
command_name: self.command_name.clone(),
temp_solution_file: Some(temp_solution_file.into()),
threads: self.threads,
seconds: self.seconds,
mipgap: self.mipgap,
}
}
}
impl SolverWithSolutionParsing for CbcSolver {
fn read_specific_solution<'a, P: LpProblem<'a>>(
&self,
f: &File,
problem: Option<&'a P>,
) -> Result<Solution, String> {
let mut vars_value: HashMap<String, _> = HashMap::new();
if let Some(p) = problem {
for var in p.variables() {
vars_value.insert(var.name().to_string(), 0.0);
}
}
let mut file = BufReader::new(f);
let mut buffer = String::new();
let _ = file.read_line(&mut buffer);
let mut buffer_split = buffer.split_whitespace();
let status = if let Some(status) = buffer_split.next() {
match status {
"Optimal" => {
if let Some(substatus) = buffer_split.next() {
match substatus {
"(within" => Status::MipGap,
_ => Status::Optimal,
}
} else {
Status::Optimal
}
}
"Infeasible" | "Integer" => Status::Infeasible,
"Unbounded" => Status::Unbounded,
"Stopped" => Status::SubOptimal,
_ => Status::NotSolved,
}
} else {
return Err("Incorrect solution format".to_string());
};
for line in file.lines() {
let l = line.unwrap();
let mut result_line: Vec<_> = l.split_whitespace().collect();
if result_line[0] == "**" {
result_line.remove(0);
};
if result_line.len() == 4 {
match result_line[2].parse::<f32>() {
Ok(n) => {
vars_value.insert(result_line[1].to_string(), n);
}
Err(e) => return Err(e.to_string()),
}
} else {
return Err("Incorrect solution format".to_string());
}
}
Ok(Solution::new(status, vars_value))
}
}
impl WithMaxSeconds<CbcSolver> for CbcSolver {
fn max_seconds(&self) -> Option<u32> {
self.seconds
}
fn with_max_seconds(&self, seconds: u32) -> CbcSolver {
CbcSolver {
seconds: Some(seconds),
..(*self).clone()
}
}
}
impl WithMipGap<CbcSolver> for CbcSolver {
fn mip_gap(&self) -> Option<f32> {
self.mipgap
}
fn with_mip_gap(&self, mipgap: f32) -> Result<CbcSolver, String> {
if mipgap.is_sign_positive() && mipgap.is_finite() {
Ok(CbcSolver {
mipgap: Some(mipgap),
..(*self).clone()
})
} else {
Err("Invalid MIP gap: must be positive and finite".to_string())
}
}
}
impl WithNbThreads<CbcSolver> for CbcSolver {
fn nb_threads(&self) -> Option<u32> {
self.threads
}
fn with_nb_threads(&self, threads: u32) -> CbcSolver {
CbcSolver {
threads: Some(threads),
..(*self).clone()
}
}
}
impl SolverProgram for CbcSolver {
fn command_name(&self) -> &str {
&self.command_name
}
fn arguments(&self, lp_file: &Path, solution_file: &Path) -> Vec<OsString> {
let mut args = vec![lp_file.as_os_str().to_owned()];
if let Some(mipgap) = self.mip_gap() {
args.push("ratiogap".into());
args.push(mipgap.to_string().into());
}
for (name, value) in [
("seconds", self.max_seconds()),
("threads", self.nb_threads()),
]
.iter()
{
if let Some(val) = value {
args.push(name.into());
args.push(val.to_string().into());
}
}
args.extend_from_slice(&["solve".into(), "solution".into(), solution_file.into()]);
args
}
fn preferred_temp_solution_file(&self) -> Option<&Path> {
self.temp_solution_file.as_deref()
}
}
#[cfg(test)]
mod tests {
use crate::solvers::{CbcSolver, SolverProgram, WithMaxSeconds, WithMipGap, WithNbThreads};
use std::ffi::OsString;
use std::path::Path;
#[test]
fn cli_args_default() {
let solver = CbcSolver::new();
let args = solver.arguments(Path::new("test.lp"), Path::new("test.sol"));
let expected: Vec<OsString> = vec![
"test.lp".into(),
"solve".into(),
"solution".into(),
"test.sol".into(),
];
assert_eq!(args, expected);
}
#[test]
fn cli_args_seconds() {
let solver = CbcSolver::new().with_max_seconds(10);
let args = solver.arguments(Path::new("test.lp"), Path::new("test.sol"));
let expected: Vec<OsString> = vec![
"test.lp".into(),
"seconds".into(),
"10".into(),
"solve".into(),
"solution".into(),
"test.sol".into(),
];
assert_eq!(args, expected);
}
#[test]
fn cli_args_mipgap() {
let solver = CbcSolver::new()
.with_mip_gap(0.05)
.expect("mipgap should be valid");
let args = solver.arguments(Path::new("test.lp"), Path::new("test.sol"));
let expected: Vec<OsString> = vec![
"test.lp".into(),
"ratiogap".into(),
"0.05".to_string().into(),
"solve".into(),
"solution".into(),
"test.sol".into(),
];
assert_eq!(args, expected);
}
#[test]
fn cli_args_mipgap_negative() {
let solver = CbcSolver::new().with_mip_gap(-0.05);
assert!(solver.is_err());
}
#[test]
fn cli_args_mipgap_infinite() {
let solver = CbcSolver::new().with_mip_gap(f32::INFINITY);
assert!(solver.is_err());
}
#[test]
fn cli_args_threads() {
let solver = CbcSolver::new().with_nb_threads(3);
let args = solver.arguments(Path::new("test.lp"), Path::new("test.sol"));
let expected: Vec<OsString> = vec![
"test.lp".into(),
"threads".into(),
"3".into(),
"solve".into(),
"solution".into(),
"test.sol".into(),
];
assert_eq!(args, expected);
}
#[test]
fn cli_args_multiple() {
let solver = CbcSolver::new()
.with_nb_threads(3)
.with_max_seconds(10)
.with_mip_gap(0.05)
.expect("mipgap should be valid");
let args = solver.arguments(Path::new("test.lp"), Path::new("test.sol"));
let expected: Vec<OsString> = vec![
"test.lp".into(),
"ratiogap".into(),
"0.05".into(),
"seconds".into(),
"10".into(),
"threads".into(),
"3".into(),
"solve".into(),
"solution".into(),
"test.sol".into(),
];
assert_eq!(args, expected);
}
}