use std::collections::HashMap;
use std::ffi::OsString;
use std::fs::File;
use std::io::{BufRead, BufReader, Error};
use std::path::{Path, PathBuf};
use crate::lp_format::*;
use crate::solvers::{Solution, SolverProgram, SolverWithSolutionParsing, Status};
#[derive(Debug, Clone)]
pub struct GlpkSolver {
name: String,
command_name: String,
temp_solution_file: Option<PathBuf>,
}
impl Default for GlpkSolver {
fn default() -> Self {
Self::new()
}
}
impl GlpkSolver {
pub fn new() -> GlpkSolver {
GlpkSolver {
name: "Glpk".to_string(),
command_name: "glpsol".to_string(),
temp_solution_file: None,
}
}
pub fn command_name(&self, command_name: String) -> GlpkSolver {
GlpkSolver {
name: self.name.clone(),
command_name,
temp_solution_file: self.temp_solution_file.clone(),
}
}
pub fn with_temp_solution_file(&self, temp_solution_file: String) -> GlpkSolver {
GlpkSolver {
name: self.name.clone(),
command_name: self.command_name.clone(),
temp_solution_file: Some(temp_solution_file.into()),
}
}
}
impl SolverWithSolutionParsing for GlpkSolver {
fn read_specific_solution<'a, P: LpProblem<'a>>(
&self,
f: &File,
_problem: Option<&'a P>,
) -> Result<Solution, String> {
fn read_size(line: Option<Result<String, Error>>) -> Result<usize, String> {
match line {
Some(Ok(l)) => match l.split_whitespace().nth(1) {
Some(value) => match value.parse::<usize>() {
Ok(v) => Ok(v),
_ => Err("Incorrect solution format".to_string()),
},
_ => Err("Incorrect solution format".to_string()),
},
_ => Err("Incorrect solution format".to_string()),
}
}
let mut vars_value: HashMap<_, _> = HashMap::new();
let file = BufReader::new(f);
let mut iter = file.lines();
let row = match read_size(iter.nth(1)) {
Ok(value) => value,
Err(e) => return Err(e),
};
let col = match read_size(iter.next()) {
Ok(value) => value,
Err(e) => return Err(e),
};
let status = match iter.nth(1) {
Some(Ok(status_line)) => match &status_line[12..] {
"INTEGER OPTIMAL" | "OPTIMAL" => Status::Optimal,
"INFEASIBLE (FINAL)" | "INTEGER EMPTY" => Status::Infeasible,
"UNDEFINED" => Status::NotSolved,
"INTEGER UNDEFINED" | "UNBOUNDED" => Status::Unbounded,
_ => return Err("Incorrect solution format: Unknown solution status".to_string()),
},
_ => return Err("Incorrect solution format: No solution status found".to_string()),
};
let mut result_lines = iter.skip(row + 7);
for _ in 0..col {
let line = match result_lines.next() {
Some(Ok(l)) => l,
_ => {
return Err("Incorrect solution format: Not all columns are present".to_string())
}
};
let result_line: Vec<_> = line.split_whitespace().collect();
if result_line.len() >= 4 {
match result_line[3].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: Column specification has to few fields".to_string(),
);
}
}
Ok(Solution::new(status, vars_value))
}
}
impl SolverProgram for GlpkSolver {
fn command_name(&self) -> &str {
&self.command_name
}
fn arguments(&self, lp_file: &Path, solution_file: &Path) -> Vec<OsString> {
vec![
"--lp".into(),
lp_file.into(),
"-o".into(),
solution_file.into(),
]
}
fn preferred_temp_solution_file(&self) -> Option<&Path> {
self.temp_solution_file.as_deref()
}
}