Source code for spinn_utilities.overrides
# Copyright (c) 2017 The University of Manchester
#
# Licensed 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
#
# https://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 types import FunctionType, MethodType
from typing import Any, Callable, TypeVar
#: :meta private:
Method = TypeVar("Method", bound=Callable[..., Any])
# pylint: disable=invalid-name
[docs]
class overrides(object):
"""
A decorator for indicating that a method overrides another method in
a superclass. This checks that the method does actually exist,
copies the doc-string for the method, and enforces that the method
overridden is specified, making maintenance easier.
"""
__slots__ = [
# The method in the superclass that this method overrides
"_superclass_method",
# True if the doc string is to be extended, False to set if not set
"_extend_doc",
]
def __init__(
self, super_class_method: Callable, *, extend_doc: bool = True):
"""
:param super_class_method: The method to override in the superclass
:param extend_doc:
True the method doc string should be appended to the super-method
doc string, False if the documentation should be set to the
super-method doc string only if there isn't a doc string already
"""
if isinstance(super_class_method, property):
super_class_method = super_class_method.fget
if not isinstance(super_class_method, (FunctionType, MethodType)):
raise TypeError("may only decorate method declarations; "
f"this is a {type(super_class_method)}")
self._superclass_method = super_class_method
self._extend_doc = bool(extend_doc)
def __call__(self, method: Method) -> Method:
"""
Apply the decorator to the given method.
"""
# Check and fail if this is a property
if isinstance(method, property):
raise AttributeError(
f"Please ensure that the {self.__class__.__name__} decorator "
"is the last decorator before the method declaration")
# Check that the name matches
if (method.__name__ != self._superclass_method.__name__):
raise AttributeError(
f"Name mismatch: {self._superclass_method.__name__} "
f" !=- {method.__name__}")
if (self._superclass_method.__doc__ is not None and
method.__doc__ is None):
method.__doc__ = self._superclass_method.__doc__
elif (self._extend_doc and
self._superclass_method.__doc__ is not None):
method.__doc__ = (
self._superclass_method.__doc__ + (method.__doc__ or ""))
return method