#!/usr/bin/env python3
# Copyright (c) Meta Platforms, Inc. and affiliates.
#
# This source code is licensed under the MIT license found in the
# LICENSE file in the root directory of this source tree.
from itertools import groupby
from typing import Dict, List, Optional
from ax.core.metric import Metric
from ax.core.objective import MultiObjective, Objective, ScalarizedObjective
from ax.core.outcome_constraint import (
ComparisonOp,
ObjectiveThreshold,
OutcomeConstraint,
ScalarizedOutcomeConstraint,
)
from ax.exceptions.core import UserInputError
from ax.utils.common.base import Base
from ax.utils.common.logger import get_logger
logger = get_logger(__name__)
TRefPoint = List[ObjectiveThreshold]
OC_TEMPLATE: str = (
"{cls_name}(objective={objective}, outcome_constraints=[{constraints}])"
)
MOOC_TEMPLATE: str = (
"{cls_name}(objective={objective}, outcome_constraints=[{constraints}], "
"objective_thresholds=[{thresholds}])"
)
[docs]class OptimizationConfig(Base):
"""An optimization configuration, which comprises an objective
and outcome constraints.
There is no minimum or maximum number of outcome constraints, but an
individual metric can have at most two constraints--which is how we
represent metrics with both upper and lower bounds.
"""
def __init__(
self,
objective: Objective,
outcome_constraints: Optional[List[OutcomeConstraint]] = None,
) -> None:
"""Inits OptimizationConfig.
Args:
objective: Metric+direction to use for the optimization.
outcome_constraints: Constraints on metrics.
"""
constraints: List[OutcomeConstraint] = (
[] if outcome_constraints is None else outcome_constraints
)
self._validate_optimization_config(
objective=objective, outcome_constraints=constraints
)
self._objective: Objective = objective
self._outcome_constraints: List[OutcomeConstraint] = constraints
[docs] def clone(self) -> "OptimizationConfig":
"""Make a copy of this optimization config."""
return self.clone_with_args()
[docs] def clone_with_args(
self,
objective: Optional[Objective] = None,
outcome_constraints: Optional[List[OutcomeConstraint]] = None,
) -> "OptimizationConfig":
"""Make a copy of this optimization config."""
objective = objective or self.objective.clone()
outcome_constraints = outcome_constraints or [
constraint.clone() for constraint in self.outcome_constraints
]
return OptimizationConfig(
objective=objective, outcome_constraints=outcome_constraints
)
@property
def objective(self) -> Objective:
"""Get objective."""
return self._objective
@objective.setter
def objective(self, objective: Objective) -> None:
"""Set objective if not present in outcome constraints."""
self._validate_optimization_config(objective, self.outcome_constraints)
self._objective = objective
@property
def all_constraints(self) -> List[OutcomeConstraint]:
"""Get outcome constraints."""
return self.outcome_constraints
@property
def outcome_constraints(self) -> List[OutcomeConstraint]:
"""Get outcome constraints."""
return self._outcome_constraints
@property
def metrics(self) -> Dict[str, Metric]:
constraint_metrics = {
oc.metric.name: oc.metric
for oc in self._outcome_constraints
if not isinstance(oc, ScalarizedOutcomeConstraint)
}
scalarized_constraint_metrics = {
metric.name: metric
for oc in self._outcome_constraints
if isinstance(oc, ScalarizedOutcomeConstraint)
for metric in oc.metrics
}
objective_metrics = {metric.name: metric for metric in self.objective.metrics}
return {
**constraint_metrics,
**scalarized_constraint_metrics,
**objective_metrics,
}
@property
def is_moo_problem(self) -> bool:
return self.objective is not None and isinstance(self.objective, MultiObjective)
@outcome_constraints.setter
def outcome_constraints(self, outcome_constraints: List[OutcomeConstraint]) -> None:
"""Set outcome constraints if valid, else raise."""
self._validate_optimization_config(
objective=self.objective, outcome_constraints=outcome_constraints
)
self._outcome_constraints = outcome_constraints
@staticmethod
def _validate_optimization_config(
objective: Objective,
outcome_constraints: Optional[List[OutcomeConstraint]] = None,
) -> None:
"""Ensure outcome constraints are valid.
Either one or two outcome constraints can reference one metric.
If there are two constraints, they must have different 'ops': one
LEQ and one GEQ.
If there are two constraints, the bound of the GEQ op must be less
than the bound of the LEQ op.
Args:
outcome_constraints: Constraints to validate.
"""
if type(objective) == MultiObjective:
# Raise error on exact equality; `ScalarizedObjective` is OK
raise ValueError(
(
"OptimizationConfig does not support MultiObjective. "
"Use MultiObjectiveOptimizationConfig instead."
)
)
outcome_constraints = outcome_constraints or []
# Only vaidate `outcome_constraints`
outcome_constraints = [
constraint
for constraint in outcome_constraints
if isinstance(constraint, ScalarizedOutcomeConstraint) is False
]
unconstrainable_metrics = objective.get_unconstrainable_metrics()
OptimizationConfig._validate_outcome_constraints(
unconstrainable_metrics=unconstrainable_metrics,
outcome_constraints=outcome_constraints,
)
@staticmethod
def _validate_outcome_constraints(
unconstrainable_metrics: List[Metric],
outcome_constraints: List[OutcomeConstraint],
) -> None:
constraint_metrics = [
constraint.metric.name for constraint in outcome_constraints
]
for metric in unconstrainable_metrics:
if metric.name in constraint_metrics:
raise ValueError("Cannot constrain on objective metric.")
def get_metric_name(oc: OutcomeConstraint) -> str:
return oc.metric.name
sorted_constraints = sorted(outcome_constraints, key=get_metric_name)
for metric_name, constraints_itr in groupby(
sorted_constraints, get_metric_name
):
constraints: List[OutcomeConstraint] = list(constraints_itr)
constraints_len = len(constraints)
if constraints_len == 2:
if constraints[0].op == constraints[1].op:
raise ValueError(f"Duplicate outcome constraints {metric_name}")
lower_bound_idx = 0 if constraints[0].op == ComparisonOp.GEQ else 1
upper_bound_idx = 1 - lower_bound_idx
lower_bound = constraints[lower_bound_idx].bound
upper_bound = constraints[upper_bound_idx].bound
if lower_bound >= upper_bound:
raise ValueError(
f"Lower bound {lower_bound} is >= upper bound "
+ f"{upper_bound} for {metric_name}"
)
elif constraints_len > 2:
raise ValueError(f"Duplicate outcome constraints {metric_name}")
def __repr__(self) -> str:
return OC_TEMPLATE.format(
cls_name=self.__class__.__name__,
objective=repr(self.objective),
constraints=", ".join(
constraint.__repr__() for constraint in self.outcome_constraints
),
)
[docs]class MultiObjectiveOptimizationConfig(OptimizationConfig):
"""An optimization configuration for multi-objective optimization,
which comprises multiple objective, outcome constraints, and objective
thresholds.
There is no minimum or maximum number of outcome constraints, but an
individual metric can have at most two constraints--which is how we
represent metrics with both upper and lower bounds.
ObjectiveThresholds should be present for every objective. A good
rule of thumb is to set them 10% below the minimum acceptable value
for each metric.
"""
def __init__(
self,
objective: Objective,
outcome_constraints: Optional[List[OutcomeConstraint]] = None,
objective_thresholds: Optional[List[ObjectiveThreshold]] = None,
) -> None:
"""Inits OptimizationConfig.
Args:
objective: Metric+direction to use for the optimization.
outcome_constraints: Constraints on metrics.
objective_thesholds: Thresholds objectives must exceed. Used for
multi-objective optimization and for calculating frontiers
and hypervolumes.
"""
constraints: List[OutcomeConstraint] = (
[] if outcome_constraints is None else outcome_constraints
)
objective_thresholds = objective_thresholds or []
self._validate_optimization_config(
objective=objective,
outcome_constraints=constraints,
objective_thresholds=objective_thresholds,
)
self._objective: Objective = objective
self._outcome_constraints: List[OutcomeConstraint] = constraints
self._objective_thresholds: List[ObjectiveThreshold] = objective_thresholds
[docs] def clone_with_args(
self,
objective: Optional[Objective] = None,
outcome_constraints: Optional[List[OutcomeConstraint]] = None,
objective_thresholds: Optional[List[ObjectiveThreshold]] = None,
) -> "MultiObjectiveOptimizationConfig":
"""Make a copy of this optimization config."""
objective = objective or self.objective.clone()
outcome_constraints = outcome_constraints or [
constraint.clone() for constraint in self.outcome_constraints
]
objective_thresholds = objective_thresholds or [
ot.clone() for ot in self.objective_thresholds
]
return MultiObjectiveOptimizationConfig(
objective=objective,
outcome_constraints=outcome_constraints,
objective_thresholds=objective_thresholds,
)
@property
def objective(self) -> Objective:
"""Get objective."""
return self._objective
@objective.setter
def objective(self, objective: Objective) -> None:
"""Set objective if not present in outcome constraints."""
self._validate_optimization_config(
objective=objective,
outcome_constraints=self.outcome_constraints,
objective_thresholds=self.objective_thresholds,
)
self._objective = objective
@property
def all_constraints(self) -> List[OutcomeConstraint]:
"""Get all constraints and thresholds."""
# pyre-ignore[58]: `+` not supported for Lists of different types.
return self.outcome_constraints + self.objective_thresholds
@property
def metrics(self) -> Dict[str, Metric]:
constraint_metrics = {
oc.metric.name: oc.metric
for oc in self.all_constraints
if not isinstance(oc, ScalarizedOutcomeConstraint)
}
scalarized_constraint_metrics = {
metric.name: metric
for oc in self.all_constraints
if isinstance(oc, ScalarizedOutcomeConstraint)
for metric in oc.metrics
}
objective_metrics = {metric.name: metric for metric in self.objective.metrics}
return {
**constraint_metrics,
**scalarized_constraint_metrics,
**objective_metrics,
}
@property
def objective_thresholds(self) -> List[ObjectiveThreshold]:
"""Get objective thresholds."""
return self._objective_thresholds
@objective_thresholds.setter
def objective_thresholds(
self, objective_thresholds: List[ObjectiveThreshold]
) -> None:
"""Set outcome constraints if valid, else raise."""
self._validate_optimization_config(
objective=self.objective, objective_thresholds=objective_thresholds
)
self._objective_thresholds = objective_thresholds
@property
def objective_thresholds_dict(self) -> Dict[str, ObjectiveThreshold]:
"""Get a mapping from objective metric name to the corresponding
threshold.
"""
return {ot.metric.name: ot for ot in self._objective_thresholds}
@staticmethod
def _validate_optimization_config(
objective: Objective,
outcome_constraints: Optional[List[OutcomeConstraint]] = None,
objective_thresholds: Optional[List[ObjectiveThreshold]] = None,
) -> None:
"""Ensure outcome constraints are valid.
Either one or two outcome constraints can reference one metric.
If there are two constraints, they must have different 'ops': one
LEQ and one GEQ.
If there are two constraints, the bound of the GEQ op must be less
than the bound of the LEQ op.
Args:
outcome_constraints: Constraints to validate.
"""
if not isinstance(objective, (MultiObjective, ScalarizedObjective)):
raise TypeError(
(
"`MultiObjectiveOptimizationConfig` requires an objective "
"of type `MultiObjective` or `ScalarizedObjective`. "
"Use `OptimizationConfig` instead if using a "
"single-metric objective."
)
)
outcome_constraints = outcome_constraints or []
objective_thresholds = objective_thresholds or []
if isinstance(objective, MultiObjective):
objectives_by_name = {obj.metric.name: obj for obj in objective.objectives}
check_objective_thresholds_match_objectives(
objectives_by_name=objectives_by_name,
objective_thresholds=objective_thresholds,
)
unconstrainable_metrics = objective.get_unconstrainable_metrics()
OptimizationConfig._validate_outcome_constraints(
unconstrainable_metrics=unconstrainable_metrics,
outcome_constraints=outcome_constraints,
)
def __repr__(self) -> str:
return MOOC_TEMPLATE.format(
cls_name=self.__class__.__name__,
objective=repr(self.objective),
constraints=", ".join(
constraint.__repr__() for constraint in self.outcome_constraints
),
thresholds=", ".join(
threshold.__repr__() for threshold in self.objective_thresholds
),
)
[docs]def check_objective_thresholds_match_objectives(
objectives_by_name: Dict[str, Objective],
objective_thresholds: List[ObjectiveThreshold],
) -> None:
"""Error if thresholds on objective_metrics bound from the wrong direction or
if there is a mismatch between objective thresholds and objectives.
"""
obj_thresh_metrics = set()
for threshold in objective_thresholds:
metric_name = threshold.metric.name
if metric_name not in objectives_by_name:
raise UserInputError(
f"Objective threshold {threshold} is on metric '{metric_name}', "
f"but that metric is not among the objectives."
)
if metric_name in obj_thresh_metrics:
raise UserInputError(
"More than one objective threshold specified for metric "
f"{metric_name}."
)
obj_thresh_metrics.add(metric_name)
if metric_name in objectives_by_name:
minimize = objectives_by_name[metric_name].minimize
bounded_above = threshold.op == ComparisonOp.LEQ
is_aligned = minimize == bounded_above
if not is_aligned:
raise UserInputError(
f"Objective threshold on {metric_name} bounds from "
f"{'above' if bounded_above else 'below'} "
f"but {metric_name} is being "
f"{'minimized' if minimize else 'maximized'}."
)
obj_metrics = set(objectives_by_name.keys())
if objective_thresholds and obj_thresh_metrics.symmetric_difference(obj_metrics):
raise UserInputError(
f"Objective thresholds: {obj_thresh_metrics} do not match objectives: "
f"{obj_metrics}."
)