# Copyright 2016-2017 Ionuț Arțăriși # 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 # 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. # This came from https://github.com/mapleoin/undecorated from inspect import isfunction, ismethod, isclass __version__ = '0.3.0' def undecorated(o): """Remove all decorators from a function, method or class""" # class decorator if isinstance(o, type): return o try: # python2 closure = o.func_closure except AttributeError: pass try: # python3 closure = o.__closure__ except AttributeError: return if closure: for cell in closure: # avoid infinite recursion if cell.cell_contents is o: continue # check if the contents looks like a decorator; in that case # we need to go one level down into the dream, otherwise it # might just be a different closed-over variable, which we # can ignore. # Note: this favors supporting decorators defined without # @wraps to the detriment of function/method/class closures if looks_like_a_decorator(cell.cell_contents): undecd = undecorated(cell.cell_contents) if undecd: return undecd return o def looks_like_a_decorator(a): return isfunction(a) or ismethod(a) or isclass(a)