12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697 |
- # Licensed to the Apache Software Foundation (ASF) under one
- # or more contributor license agreements. See the NOTICE file
- # distributed with this work for additional information
- # regarding copyright ownership. The ASF licenses this file
- # to you under the Apache License, Version 2.0 (the
- # "License"); you may not use this file except in compliance
- # with the License. You may obtain a copy of the License at
- #
- # http://www.apache.org/licenses/LICENSE-2.0
- #
- # Unless required by applicable law or agreed to in writing,
- # software distributed under the License is distributed on an
- # "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
- # KIND, either express or implied. See the License for the
- # specific language governing permissions and limitations
- # under the License.
- from __future__ import annotations
- import functools
- import importlib
- import sys
- import warnings
- from types import ModuleType
- def getattr_with_deprecation(
- imports: dict[str, str],
- module: str,
- override_deprecated_classes: dict[str, str],
- extra_message: str,
- name: str,
- ):
- """
- Retrieve the imported attribute from the redirected module and raises a deprecation warning.
- :param imports: dict of imports and their redirection for the module
- :param module: name of the module in the package to get the attribute from
- :param override_deprecated_classes: override target classes with deprecated ones. If target class is
- found in the dictionary, it will be displayed in the warning message.
- :param extra_message: extra message to display in the warning or import error message
- :param name: attribute name
- :return:
- """
- target_class_full_name = imports.get(name)
- if not target_class_full_name:
- raise AttributeError(f"The module `{module!r}` has no attribute `{name!r}`")
- warning_class_name = target_class_full_name
- if override_deprecated_classes and name in override_deprecated_classes:
- warning_class_name = override_deprecated_classes[name]
- message = f"The `{module}.{name}` class is deprecated. Please use `{warning_class_name!r}`."
- if extra_message:
- message += f" {extra_message}."
- warnings.warn(message, DeprecationWarning, stacklevel=2)
- new_module, new_class_name = target_class_full_name.rsplit(".", 1)
- try:
- return getattr(importlib.import_module(new_module), new_class_name)
- except ImportError as e:
- error_message = (
- f"Could not import `{new_module}.{new_class_name}` while trying to import `{module}.{name}`."
- )
- if extra_message:
- error_message += f" {extra_message}."
- raise ImportError(error_message) from e
- def add_deprecated_classes(
- module_imports: dict[str, dict[str, str]],
- package: str,
- override_deprecated_classes: dict[str, dict[str, str]] | None = None,
- extra_message: str | None = None,
- ):
- """
- Add deprecated class PEP-563 imports and warnings modules to the package.
- :param module_imports: imports to use
- :param package: package name
- :param override_deprecated_classes: override target classes with deprecated ones. If module +
- target class is found in the dictionary, it will be displayed in the warning message.
- :param extra_message: extra message to display in the warning or import error message
- """
- for module_name, imports in module_imports.items():
- full_module_name = f"{package}.{module_name}"
- module_type = ModuleType(full_module_name)
- if override_deprecated_classes and module_name in override_deprecated_classes:
- override_deprecated_classes_for_module = override_deprecated_classes[module_name]
- else:
- override_deprecated_classes_for_module = {}
- # Mypy is not able to derive the right function signature https://github.com/python/mypy/issues/2427
- module_type.__getattr__ = functools.partial( # type: ignore[assignment]
- getattr_with_deprecation,
- imports,
- full_module_name,
- override_deprecated_classes_for_module,
- extra_message or "",
- )
- sys.modules.setdefault(full_module_name, module_type)
|