Source code for ax.benchmark.benchmark_problem

# 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 dataclasses import dataclass, field
from typing import List

from ax.core.metric import Metric

from ax.core.objective import MultiObjective, Objective
from ax.core.optimization_config import (
    MultiObjectiveOptimizationConfig,
    ObjectiveThreshold,
    OptimizationConfig,
)
from ax.core.parameter import ParameterType, RangeParameter
from ax.core.runner import Runner
from ax.core.search_space import SearchSpace
from ax.core.types import ComparisonOp
from ax.metrics.botorch_test_problem import BotorchTestProblemMetric
from ax.runners.botorch_test_problem import BotorchTestProblemRunner
from ax.utils.common.base import Base
from botorch.test_functions.base import BaseTestProblem
from botorch.test_functions.multi_objective import MultiObjectiveTestProblem
from botorch.test_functions.synthetic import SyntheticTestFunction

# NOTE: Do not add `from __future__ import annotatations` to this file. Adding
# `annotations` postpones evaluation of types and will break FBLearner's usage of
# `BenchmarkProblem` as return type annotation, used for serialization and rendering
# in the UI.


[docs]@dataclass(frozen=True) class BenchmarkProblem(Base): """Benchmark problem, represented in terms of Ax search space, optimization config, and runner. """ name: str search_space: SearchSpace optimization_config: OptimizationConfig runner: Runner num_trials: int tracking_metrics: List[Metric] = field(default_factory=list)
[docs] @classmethod def from_botorch( cls, test_problem: BaseTestProblem, num_trials: int ) -> "BenchmarkProblem": """Create a BenchmarkProblem from a BoTorch BaseTestProblem using specialized Metrics and Runners. The test problem's result will be computed on the Runner and retrieved by the Metric. """ search_space = SearchSpace( parameters=[ RangeParameter( name=f"x{i}", parameter_type=ParameterType.FLOAT, lower=test_problem._bounds[i][0], upper=test_problem._bounds[i][1], ) for i in range(test_problem.dim) ] ) optimization_config = OptimizationConfig( objective=Objective( metric=BotorchTestProblemMetric( name=f"{test_problem.__class__.__name__}", noise_sd=(test_problem.noise_std or 0), ), minimize=True, ) ) return cls( name=f"{test_problem.__class__.__name__}", search_space=search_space, optimization_config=optimization_config, runner=BotorchTestProblemRunner(test_problem=test_problem), num_trials=num_trials, )
[docs]@dataclass(frozen=True, init=False) class SingleObjectiveBenchmarkProblem(BenchmarkProblem): """The most basic BenchmarkProblem, with a single objective and a known optimal value. """ optimal_value: float = field() # pyre-fixme[2]: Parameter must be annotated. def __init__(self, optimal_value: float, **kwargs) -> None: super().__init__(**kwargs) object.__setattr__(self, "optimal_value", optimal_value)
[docs] @classmethod def from_botorch_synthetic( cls, test_problem: SyntheticTestFunction, num_trials: int ) -> "SingleObjectiveBenchmarkProblem": """Create a BenchmarkProblem from a BoTorch BaseTestProblem using specialized Metrics and Runners. The test problem's result will be computed on the Runner and retrieved by the Metric. """ problem = BenchmarkProblem.from_botorch( test_problem=test_problem, num_trials=num_trials ) return cls( name=f"{test_problem.__class__.__name__}", search_space=problem.search_space, optimization_config=problem.optimization_config, runner=problem.runner, num_trials=num_trials, optimal_value=test_problem.optimal_value, )
[docs]@dataclass(frozen=True, init=False) class MultiObjectiveBenchmarkProblem(BenchmarkProblem): """A BenchmarkProblem support multiple objectives. Rather than knowing each objective's optimal value we track a known maximum hypervolume computed from a given reference point. """ maximum_hypervolume: float = field() reference_point: List[float] = field() def __init__( self, maximum_hypervolume: float, reference_point: List[float], # pyre-fixme[2]: Parameter must be annotated. **kwargs, ) -> None: super().__init__(**kwargs) object.__setattr__(self, "maximum_hypervolume", maximum_hypervolume) object.__setattr__(self, "reference_point", reference_point)
[docs] @classmethod def from_botorch_multi_objective( cls, test_problem: MultiObjectiveTestProblem, num_trials: int, ) -> "MultiObjectiveBenchmarkProblem": """Create a BenchmarkProblem from a BoTorch BaseTestProblem using specialized Metrics and Runners. The test problem's result will be computed on the Runner once per trial and each Metric will retrieve its own result by index. """ problem = BenchmarkProblem.from_botorch( test_problem=test_problem, num_trials=num_trials ) metrics = [ BotorchTestProblemMetric( name=f"{test_problem.__class__.__name__}_{i}", noise_sd=(test_problem.noise_std or 0), index=i, ) for i in range(test_problem.num_objectives) ] optimization_config = MultiObjectiveOptimizationConfig( objective=MultiObjective( objectives=[ Objective( metric=metric, minimize=True, ) for metric in metrics ] ), objective_thresholds=[ ObjectiveThreshold( metric=metrics[i], # pyre-fixme[6]: For 2nd param expected `float` but got `Tensor`. bound=test_problem.ref_point[i], relative=False, op=ComparisonOp.LEQ, ) for i in range(test_problem.num_objectives) ], ) return cls( name=f"{test_problem.__class__.__name__}", search_space=problem.search_space, optimization_config=optimization_config, runner=problem.runner, num_trials=num_trials, maximum_hypervolume=test_problem.max_hv, reference_point=test_problem._ref_point, )