lbry-sdk/lbrynet/undecorated.py

64 lines
1.9 KiB
Python
Raw Normal View History

2017-03-23 01:27:01 +01:00
# -*- coding: utf-8 -*-
# Copyright 2016-2017 Ionuț Arțăriși <ionut@artarisi.eu>
# 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
2017-04-25 20:31:05 +02:00
if isinstance(o, type):
2017-03-23 01:27:01 +01:00
return o
try:
# python2
closure = o.func_closure
except AttributeError:
pass
2018-07-22 14:54:51 +02:00
try:
# python3
closure = o.__closure__
except AttributeError:
return
2017-03-23 01:27:01 +01:00
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
2017-04-25 20:08:33 +02:00
return o
2017-03-23 01:27:01 +01:00
def looks_like_a_decorator(a):
2017-04-25 20:08:33 +02:00
return isfunction(a) or ismethod(a) or isclass(a)