use crate::error::check_ampl_error;
use crate::ffi;
use crate::ampl::Ampl;
use crate::dataframe::DataFrame;
use crate::variableinstance::Variableinstance;
use crate::tuple::Tuple;
use libc::c_char;
use std::ffi::{CStr, CString};
use std::ptr;
pub struct Variable {
pub(crate) ampl: *mut Ampl,
pub(crate) name: String,
}
impl Variable {
pub fn print(&self) {
println!("Variable: {}", self.name);
}
pub fn value(&self) -> f64 {
let name = CString::new(&*self.name).unwrap();
let mut value: f64 = 0.0;
let err = unsafe { ffi::AMPL_VariableGetValue((*self.ampl).raw, name.as_ptr(), &mut value as *mut f64) };
unsafe { check_ampl_error(err) };
value
}
pub fn fix(&self) {
let name = CString::new(&*self.name).unwrap();
let err = unsafe { ffi::AMPL_VariableFix((*self.ampl).raw, name.as_ptr()) };
unsafe { check_ampl_error(err) };
}
pub fn fix_with_value(&self, value: f64) {
let name = CString::new(&*self.name).unwrap();
let err = unsafe { ffi::AMPL_VariableFixWithValue((*self.ampl).raw, name.as_ptr(), value) };
unsafe { check_ampl_error(err) };
}
pub fn unfix(&self) {
let name = CString::new(&*self.name).unwrap();
let err = unsafe { ffi::AMPL_VariableUnfix((*self.ampl).raw, name.as_ptr()) };
unsafe { check_ampl_error(err) };
}
pub fn set_value(&self, value: f64) {
let name = CString::new(&*self.name).unwrap();
let err = unsafe { ffi::AMPL_VariableSetValue((*self.ampl).raw, name.as_ptr(), value) };
unsafe { check_ampl_error(err) };
}
pub fn indexarity(&self) -> usize {
let name = CString::new(&*self.name).unwrap();
let mut indexarity: usize = 0;
let err = unsafe { ffi::AMPL_EntityGetIndexarity((*self.ampl).raw, name.as_ptr(), &mut indexarity as *mut usize) };
unsafe { check_ampl_error(err) };
indexarity
}
pub fn instances(&self) -> Vec<Variableinstance> {
let name = CString::new(&*self.name).unwrap();
let mut instances_ptr: *mut *mut ffi::AMPL_TUPLE = ptr::null_mut();
let mut num_instances: usize = 0;
unsafe {
let err = ffi::AMPL_EntityGetTuples((*self.ampl).raw, name.as_ptr(), &mut instances_ptr, &mut num_instances as *mut usize);
check_ampl_error(err);
let mut instances = Vec::new();
for i in 0..num_instances {
let raw = *instances_ptr.add(i);
let tuple = Tuple {raw: raw};
let instance = Variableinstance::new(&mut *self.ampl, self.name.clone(), tuple);
instances.push(instance);
}
instances
}
}
pub fn num_instances(&self) -> usize {
let name = CString::new(&*self.name).unwrap();
let mut num_instances: usize = 0;
let err = unsafe { ffi::AMPL_EntityGetNumInstances((*self.ampl).raw, name.as_ptr(), &mut num_instances as *mut usize) };
unsafe { check_ampl_error(err) };
num_instances
}
pub fn declaration(&mut self) -> String {
let name = CString::new(&*self.name).unwrap();
let mut value_ptr: *mut c_char = ptr::null_mut();
unsafe {
let err = ffi::AMPL_EntityGetDeclaration((*self.ampl).raw, name.as_ptr(), &mut value_ptr);
check_ampl_error(err);
if value_ptr.is_null() {
return String::new();
}
let value_str = CStr::from_ptr(value_ptr).to_str().unwrap().to_string();
ffi::AMPL_StringFree(&mut value_ptr);
value_str
}
}
pub fn get_values(&self) -> DataFrame {
let name = CString::new(&*self.name).unwrap();
let mut df: *mut ffi::AMPL_DATAFRAME = ptr::null_mut();
let err = unsafe {
ffi::AMPL_EntityGetValues((*self.ampl).raw, name.as_ptr(), ptr::null(), 0, &mut df)
};
unsafe { check_ampl_error(err) };
DataFrame { raw: df }
}
pub fn drop(&self) {
let name = CString::new(&*self.name).unwrap();
let err = unsafe { ffi::AMPL_EntityDrop((*self.ampl).raw, name.as_ptr()) };
unsafe { check_ampl_error(err) };
}
pub fn restore(&self) {
let name = CString::new(&*self.name).unwrap();
let err = unsafe { ffi::AMPL_EntityRestore((*self.ampl).raw, name.as_ptr()) };
unsafe { check_ampl_error(err) };
}
}