Module sverchok.utils.decorators

Expand source code
# ##### BEGIN GPL LICENSE BLOCK #####
#
#  This program is free software; you can redistribute it and/or
#  modify it under the terms of the GNU General Public License
#  as published by the Free Software Foundation; either version 2
#  of the License, or (at your option) any later version.
#
#  This program is distributed in the hope that it will be useful,
#  but WITHOUT ANY WARRANTY; without even the implied warranty of
#  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
#  GNU General Public License for more details.
#
#  You should have received a copy of the GNU General Public License
#  along with this program; if not, write to the Free Software Foundation,
#  Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
#
# ##### END GPL LICENSE BLOCK #####

import time
import functools
import inspect
import warnings
from sverchok.utils.sv_logging import sv_logger
from sverchok.utils.ascii_print import str_color

string_types = (type(b''), type(u''))


def deprecated(argument):
    """
    This is a decorator which can be used to mark functions
    as deprecated. It will result in a warning being emitted
    when the function is used.
    """

    def format_message(object, reason):
        if inspect.isclass(object):
            if reason is None:
                fmt = "Call to deprecated class {name}."
                return fmt.format(name = object.__name__)
            else:
                fmt = "Call to deprecated class {name} ({reason})."
                return fmt.format(name = object.__name__, reason = reason)
        elif inspect.isfunction(object):
            if reason is None:
                fmt = "Call to deprecated function {name}."
                return fmt.format(name = object.__name__)
            else: 
                fmt = "Call to deprecated function {name} ({reason})."
                return fmt.format(name = object.__name__, reason = reason)
        else:
            raise TypeError("Unexpected deprecated object type: " + repr(type(object)))

    if isinstance(argument, string_types):

        # The @deprecated is used with a 'reason'.
        #
        # .. code-block:: python
        #
        #    @deprecated("please, use another function")
        #    def old_function(x, y):
        #      pass

        reason = argument

        def decorator(func):
            @functools.wraps(func)
            def wrapped(*args, **kwargs):
                warnings.simplefilter('always', DeprecationWarning)
                warnings.warn(
                    format_message(func, reason),
                    category=DeprecationWarning,
                    stacklevel=2
                )
                warnings.simplefilter('default', DeprecationWarning)
                return func(*args, **kwargs)
            return wrapped
        return decorator

    else:
        # The @deprecated is used without any 'reason'.
        #
        # .. code-block:: python
        #
        #    @deprecated
        #    def old_function(x, y):
        #      pass

        func = argument
        reason = None

        @functools.wraps(func)
        def wrapped(*args, **kwargs):
            warnings.simplefilter('always', DeprecationWarning)
            warnings.warn(
                format_message(func, reason),
                category=DeprecationWarning,
                stacklevel=2
            )
            warnings.simplefilter('default', DeprecationWarning)
            return func(*args, **kwargs)

        return wrapped

        
def duration(func):

    @functools.wraps(func)
    def wrapped(*args, **kwargs):

        start_time = time.time()
        result = func(*args, **kwargs)
        duration = (time.time() - start_time) * 1000

        # display_args = (f"\n    {args=}" if args else "")
        # display_kwargs = (f"\n    {kwargs=}" if kwargs else "")
        func_name = str_color(func.__name__, 31)
        duration = str_color(f"{duration:.5g} ms", 32)
        
        msg = f"\n{func_name}: {duration}" # + display_args + display_kwargs
        try:
            sv_logger.info(msg)
        except:
            print(msg)

        return result
    return wrapped

Functions

def deprecated(argument)

This is a decorator which can be used to mark functions as deprecated. It will result in a warning being emitted when the function is used.

Expand source code
def deprecated(argument):
    """
    This is a decorator which can be used to mark functions
    as deprecated. It will result in a warning being emitted
    when the function is used.
    """

    def format_message(object, reason):
        if inspect.isclass(object):
            if reason is None:
                fmt = "Call to deprecated class {name}."
                return fmt.format(name = object.__name__)
            else:
                fmt = "Call to deprecated class {name} ({reason})."
                return fmt.format(name = object.__name__, reason = reason)
        elif inspect.isfunction(object):
            if reason is None:
                fmt = "Call to deprecated function {name}."
                return fmt.format(name = object.__name__)
            else: 
                fmt = "Call to deprecated function {name} ({reason})."
                return fmt.format(name = object.__name__, reason = reason)
        else:
            raise TypeError("Unexpected deprecated object type: " + repr(type(object)))

    if isinstance(argument, string_types):

        # The @deprecated is used with a 'reason'.
        #
        # .. code-block:: python
        #
        #    @deprecated("please, use another function")
        #    def old_function(x, y):
        #      pass

        reason = argument

        def decorator(func):
            @functools.wraps(func)
            def wrapped(*args, **kwargs):
                warnings.simplefilter('always', DeprecationWarning)
                warnings.warn(
                    format_message(func, reason),
                    category=DeprecationWarning,
                    stacklevel=2
                )
                warnings.simplefilter('default', DeprecationWarning)
                return func(*args, **kwargs)
            return wrapped
        return decorator

    else:
        # The @deprecated is used without any 'reason'.
        #
        # .. code-block:: python
        #
        #    @deprecated
        #    def old_function(x, y):
        #      pass

        func = argument
        reason = None

        @functools.wraps(func)
        def wrapped(*args, **kwargs):
            warnings.simplefilter('always', DeprecationWarning)
            warnings.warn(
                format_message(func, reason),
                category=DeprecationWarning,
                stacklevel=2
            )
            warnings.simplefilter('default', DeprecationWarning)
            return func(*args, **kwargs)

        return wrapped
def duration(func)
Expand source code
def duration(func):

    @functools.wraps(func)
    def wrapped(*args, **kwargs):

        start_time = time.time()
        result = func(*args, **kwargs)
        duration = (time.time() - start_time) * 1000

        # display_args = (f"\n    {args=}" if args else "")
        # display_kwargs = (f"\n    {kwargs=}" if kwargs else "")
        func_name = str_color(func.__name__, 31)
        duration = str_color(f"{duration:.5g} ms", 32)
        
        msg = f"\n{func_name}: {duration}" # + display_args + display_kwargs
        try:
            sv_logger.info(msg)
        except:
            print(msg)

        return result
    return wrapped