mirror of
https://github.com/ANL-CEEESA/MIPLearn.git
synced 2025-12-06 09:28:51 -06:00
Include additional features in instance.features
This commit is contained in:
@@ -84,16 +84,14 @@ class FeaturesExtractor:
|
|||||||
self.solver = internal_solver
|
self.solver = internal_solver
|
||||||
|
|
||||||
def extract(self, instance: "Instance") -> None:
|
def extract(self, instance: "Instance") -> None:
|
||||||
instance.features.variables = self._extract_variables(instance)
|
instance.features.variables = self.solver.get_variables()
|
||||||
instance.features.constraints = self._extract_constraints(instance)
|
instance.features.constraints = self.solver.get_constraints()
|
||||||
instance.features.instance = self._extract_instance(instance, instance.features)
|
self._extract_user_features_vars(instance)
|
||||||
|
self._extract_user_features_constrs(instance)
|
||||||
|
self._extract_user_features_instance(instance)
|
||||||
|
|
||||||
def _extract_variables(
|
def _extract_user_features_vars(self, instance: "Instance"):
|
||||||
self,
|
for (var_name, var) in instance.features.variables.items():
|
||||||
instance: "Instance",
|
|
||||||
) -> Dict[VariableName, Variable]:
|
|
||||||
result: Dict[VariableName, Variable] = {}
|
|
||||||
for var_name in self.solver.get_variable_names():
|
|
||||||
user_features: Optional[List[float]] = None
|
user_features: Optional[List[float]] = None
|
||||||
category: Category = instance.get_variable_category(var_name)
|
category: Category = instance.get_variable_category(var_name)
|
||||||
if category is not None:
|
if category is not None:
|
||||||
@@ -115,20 +113,12 @@ class FeaturesExtractor:
|
|||||||
f"Found {type(v).__name__} instead "
|
f"Found {type(v).__name__} instead "
|
||||||
f"for var={var_name}."
|
f"for var={var_name}."
|
||||||
)
|
)
|
||||||
result[var_name] = Variable(
|
var.category = category
|
||||||
category=category,
|
var.user_features = user_features
|
||||||
user_features=user_features,
|
|
||||||
)
|
|
||||||
return result
|
|
||||||
|
|
||||||
def _extract_constraints(
|
def _extract_user_features_constrs(self, instance: "Instance"):
|
||||||
self,
|
|
||||||
instance: "Instance",
|
|
||||||
) -> Dict[str, Constraint]:
|
|
||||||
has_static_lazy = instance.has_static_lazy_constraints()
|
has_static_lazy = instance.has_static_lazy_constraints()
|
||||||
constraints = self.solver.get_constraints()
|
for (cid, constr) in instance.features.constraints.items():
|
||||||
|
|
||||||
for (cid, constr) in constraints.items():
|
|
||||||
user_features = None
|
user_features = None
|
||||||
category = instance.get_constraint_category(cid)
|
category = instance.get_constraint_category(cid)
|
||||||
if category is not None:
|
if category is not None:
|
||||||
@@ -147,18 +137,13 @@ class FeaturesExtractor:
|
|||||||
f"Constraint features must be a list of floats. "
|
f"Constraint features must be a list of floats. "
|
||||||
f"Found {type(user_features[0]).__name__} instead for cid={cid}."
|
f"Found {type(user_features[0]).__name__} instead for cid={cid}."
|
||||||
)
|
)
|
||||||
constraints[cid].category = category
|
|
||||||
constraints[cid].user_features = user_features
|
|
||||||
if has_static_lazy:
|
if has_static_lazy:
|
||||||
constraints[cid].lazy = instance.is_constraint_lazy(cid)
|
constr.lazy = instance.is_constraint_lazy(cid)
|
||||||
return constraints
|
constr.category = category
|
||||||
|
constr.user_features = user_features
|
||||||
|
|
||||||
@staticmethod
|
def _extract_user_features_instance(self, instance: "Instance"):
|
||||||
def _extract_instance(
|
assert instance.features.constraints is not None
|
||||||
instance: "Instance",
|
|
||||||
features: Features,
|
|
||||||
) -> InstanceFeatures:
|
|
||||||
assert features.constraints is not None
|
|
||||||
user_features = instance.get_instance_features()
|
user_features = instance.get_instance_features()
|
||||||
if isinstance(user_features, np.ndarray):
|
if isinstance(user_features, np.ndarray):
|
||||||
user_features = user_features.tolist()
|
user_features = user_features.tolist()
|
||||||
@@ -172,10 +157,10 @@ class FeaturesExtractor:
|
|||||||
f"Found {type(v).__name__} instead."
|
f"Found {type(v).__name__} instead."
|
||||||
)
|
)
|
||||||
lazy_count = 0
|
lazy_count = 0
|
||||||
for (cid, cdict) in features.constraints.items():
|
for (cid, cdict) in instance.features.constraints.items():
|
||||||
if cdict.lazy:
|
if cdict.lazy:
|
||||||
lazy_count += 1
|
lazy_count += 1
|
||||||
return InstanceFeatures(
|
instance.features.instance = InstanceFeatures(
|
||||||
user_features=user_features,
|
user_features=user_features,
|
||||||
lazy_constraint_count=lazy_count,
|
lazy_constraint_count=lazy_count,
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -9,34 +9,55 @@ from miplearn.features import (
|
|||||||
Constraint,
|
Constraint,
|
||||||
)
|
)
|
||||||
from miplearn.solvers.gurobi import GurobiSolver
|
from miplearn.solvers.gurobi import GurobiSolver
|
||||||
|
from miplearn.solvers.tests import assert_equals
|
||||||
|
|
||||||
|
|
||||||
def test_knapsack() -> None:
|
def test_knapsack() -> None:
|
||||||
for solver_factory in [GurobiSolver]:
|
solver = GurobiSolver()
|
||||||
solver = solver_factory()
|
instance = solver.build_test_instance_knapsack()
|
||||||
instance = solver.build_test_instance_knapsack()
|
model = instance.to_model()
|
||||||
model = instance.to_model()
|
solver.set_instance(instance, model)
|
||||||
solver.set_instance(instance, model)
|
FeaturesExtractor(solver).extract(instance)
|
||||||
FeaturesExtractor(solver).extract(instance)
|
assert_equals(
|
||||||
assert instance.features.variables == {
|
instance.features.variables,
|
||||||
|
{
|
||||||
"x[0]": Variable(
|
"x[0]": Variable(
|
||||||
category="default",
|
category="default",
|
||||||
|
lower_bound=0.0,
|
||||||
|
obj_coeff=505.0,
|
||||||
|
type="B",
|
||||||
|
upper_bound=1.0,
|
||||||
user_features=[23.0, 505.0],
|
user_features=[23.0, 505.0],
|
||||||
),
|
),
|
||||||
"x[1]": Variable(
|
"x[1]": Variable(
|
||||||
category="default",
|
category="default",
|
||||||
|
lower_bound=0.0,
|
||||||
|
obj_coeff=352.0,
|
||||||
|
type="B",
|
||||||
|
upper_bound=1.0,
|
||||||
user_features=[26.0, 352.0],
|
user_features=[26.0, 352.0],
|
||||||
),
|
),
|
||||||
"x[2]": Variable(
|
"x[2]": Variable(
|
||||||
category="default",
|
category="default",
|
||||||
|
lower_bound=0.0,
|
||||||
|
obj_coeff=458.0,
|
||||||
|
type="B",
|
||||||
|
upper_bound=1.0,
|
||||||
user_features=[20.0, 458.0],
|
user_features=[20.0, 458.0],
|
||||||
),
|
),
|
||||||
"x[3]": Variable(
|
"x[3]": Variable(
|
||||||
category="default",
|
category="default",
|
||||||
|
lower_bound=0.0,
|
||||||
|
obj_coeff=220.0,
|
||||||
|
type="B",
|
||||||
|
upper_bound=1.0,
|
||||||
user_features=[18.0, 220.0],
|
user_features=[18.0, 220.0],
|
||||||
),
|
),
|
||||||
}
|
},
|
||||||
assert instance.features.constraints == {
|
)
|
||||||
|
assert_equals(
|
||||||
|
instance.features.constraints,
|
||||||
|
{
|
||||||
"eq_capacity": Constraint(
|
"eq_capacity": Constraint(
|
||||||
lhs={
|
lhs={
|
||||||
"x[0]": 23.0,
|
"x[0]": 23.0,
|
||||||
@@ -50,8 +71,12 @@ def test_knapsack() -> None:
|
|||||||
category="eq_capacity",
|
category="eq_capacity",
|
||||||
user_features=[0.0],
|
user_features=[0.0],
|
||||||
)
|
)
|
||||||
}
|
},
|
||||||
assert instance.features.instance == InstanceFeatures(
|
)
|
||||||
|
assert_equals(
|
||||||
|
instance.features.instance,
|
||||||
|
InstanceFeatures(
|
||||||
user_features=[67.0, 21.75],
|
user_features=[67.0, 21.75],
|
||||||
lazy_constraint_count=0,
|
lazy_constraint_count=0,
|
||||||
)
|
),
|
||||||
|
)
|
||||||
|
|||||||
Reference in New Issue
Block a user