import functools
import inspect
import textwrap
from typing import Any
from typing import Callable
import warnings
from optuna.exceptions import ExperimentalWarning
_EXPERIMENTAL_NOTE_TEMPLATE = """
.. note::
Added in v{ver} as an experimental feature. The interface may change in newer versions
without prior notice. See https://github.com/optuna/optuna/releases/tag/v{ver}.
"""
def _validate_version(version: str) -> None:
if not isinstance(version, str) or len(version.split(".")) != 3:
raise ValueError(
"Invalid version specification. Must follow `x.y.z` format but `{}` is given".format(
version
)
)
def _get_docstring_indent(docstring: str) -> str:
return docstring.split("\n")[-1] if "\n" in docstring else ""
def experimental(version: str, name: str = None) -> Any:
"""Decorate class or function as experimental.
Args:
version: The first version that supports the target feature.
name: The name of the feature. Defaults to the function or class name. Optional.
"""
_validate_version(version)
def _experimental_wrapper(f: Any) -> Any:
# f is either func or class.
def _experimental_func(func: Callable[[Any], Any]) -> Callable[[Any], Any]:
if func.__doc__ is None:
func.__doc__ = ""
note = _EXPERIMENTAL_NOTE_TEMPLATE.format(ver=version)
indent = _get_docstring_indent(func.__doc__)
func.__doc__ = func.__doc__.strip() + textwrap.indent(note, indent) + indent
# TODO(crcrpar): Annotate this correctly.
@functools.wraps(func)
def new_func(*args: Any, **kwargs: Any) -> Any:
warnings.warn(
"{} is experimental (supported from v{}). "
"The interface can change in the future.".format(
name if name is not None else func.__name__, version
),
ExperimentalWarning,
stacklevel=2,
)
return func(*args, **kwargs) # type: ignore
return new_func
def _experimental_class(cls: Any) -> Any:
"""Decorates a class as experimental.
This decorator is supposed to be applied to the experimental class.
"""
_original_init = cls.__init__
@functools.wraps(_original_init)
def wrapped_init(self, *args, **kwargs) -> None: # type: ignore
warnings.warn(
"{} is experimental (supported from v{}). "
"The interface can change in the future.".format(
name if name is not None else cls.__name__, version
),
ExperimentalWarning,
stacklevel=2,
)
_original_init(self, *args, **kwargs)
cls.__init__ = wrapped_init
if cls.__doc__ is None:
cls.__doc__ = ""
note = _EXPERIMENTAL_NOTE_TEMPLATE.format(ver=version)
indent = _get_docstring_indent(cls.__doc__)
cls.__doc__ = cls.__doc__.strip() + textwrap.indent(note, indent) + indent
return cls
return _experimental_class(f) if inspect.isclass(f) else _experimental_func(f)
return _experimental_wrapper