Source code for baybe.recommenders.pure.bayesian.base

"""Base class for all Bayesian recommenders."""

from abc import ABC

import pandas as pd
from attrs import define, field

from baybe.acquisition.acqfs import qLogExpectedImprovement
from baybe.acquisition.base import AcquisitionFunction
from baybe.acquisition.utils import convert_acqf
from baybe.exceptions import DeprecationError
from baybe.objectives.base import Objective
from baybe.recommenders.pure.base import PureRecommender
from baybe.searchspace import SearchSpace
from baybe.surrogates import CustomONNXSurrogate, GaussianProcessSurrogate
from baybe.surrogates.base import Surrogate
from baybe.utils.dataframe import to_tensor


[docs] @define class BayesianRecommender(PureRecommender, ABC): """An abstract class for Bayesian Recommenders.""" surrogate_model: Surrogate = field(factory=GaussianProcessSurrogate) """The used surrogate model.""" acquisition_function: AcquisitionFunction = field( converter=convert_acqf, factory=qLogExpectedImprovement ) """The used acquisition function class.""" _botorch_acqf = field(default=None, init=False) """The current acquisition function.""" acquisition_function_cls: str | None = field(default=None, kw_only=True) "Deprecated! Raises an error when used." @acquisition_function_cls.validator def _validate_deprecated_argument(self, _, value) -> None: """Raise DeprecationError if old acquisition_function_cls parameter is used.""" if value is not None: raise DeprecationError( "Passing 'acquisition_function_cls' to the constructor is deprecated. " "The parameter has been renamed to 'acquisition_function'." ) def _setup_botorch_acqf( self, searchspace: SearchSpace, objective: Objective, measurements: pd.DataFrame, ) -> None: """Create the acquisition function for the current training data.""" # noqa: E501 # TODO: Transition point from dataframe to tensor needs to be refactored. # Currently, surrogate models operate with tensors, while acquisition # functions with dataframes. train_x = searchspace.transform(measurements, allow_extra=True) train_y = objective.transform(measurements) self.surrogate_model._fit(searchspace, *to_tensor(train_x, train_y)) self._botorch_acqf = self.acquisition_function.to_botorch( self.surrogate_model, searchspace, train_x, train_y )
[docs] def recommend( # noqa: D102 self, batch_size: int, searchspace: SearchSpace, objective: Objective | None = None, measurements: pd.DataFrame | None = None, ) -> pd.DataFrame: # See base class. if objective is None: raise NotImplementedError( f"Recommenders of type '{BayesianRecommender.__name__}' require " f"that an objective is specified." ) if (measurements is None) or (len(measurements) == 0): raise NotImplementedError( f"Recommenders of type '{BayesianRecommender.__name__}' do not support " f"empty training data." ) if isinstance(self.surrogate_model, CustomONNXSurrogate): CustomONNXSurrogate.validate_compatibility(searchspace) self._setup_botorch_acqf(searchspace, objective, measurements) return super().recommend( batch_size=batch_size, searchspace=searchspace, objective=objective, measurements=measurements, )