2014-05-10 16 views
5

Sto usando un decoratore:Come usare doctest con una funzione decorata in python?

class Memoized(object): 

    __cache = {} 

    def __init__(self, func): 
     self.func = func 
     key = (func.__module__, func.__name__) 
     # print key 
     if key not in self.__cache: 
      self.__cache[key] = {} 
     self.mycache = self.__cache[key] 

    def __call__(self, *args): 
     try: 
      return self.mycache[args] 
     except KeyError: 
      value = self.func(*args) 
      self.mycache[args] = value 
      return value 
     except TypeError: 
      return self.func(*args) 

    def __get__(self, obj, objtype): 
     return functools.partial(self.__call__, obj) 

    def reset(self): 
     for v in self.__cache.itervalues(): 
      v.clear() 

e una funzione:

@Memoized 
def is_tile_inside_border(x, y, z, border): 
    '''Checks if a tile is inside border or not 
    >>> is_tile_inside_border(85,53,7,'iran') 
    3 
    >>> is_tile_inside_border(85,15,7,'iran') 
    0 
    ''' 
    binary_data = get_border_binary(border, z) 
    return isInside((x, y), binary_data) 

Ma quando si utilizza doctest modulo (python mycode.py -v):

if __name__ == '__main__': 
    import doctest 
    doctest.testmod() 

Python può' t trovare il test in doc-string. So che questo è un problema con il decoratore. Ma come posso aggiustarlo?

PS: functools.update_wrapper(self, func) non funziona!

+1

Hai provato la functools.wraps' decoratore '? http://stackoverflow.com/questions/308999/what-does-functools-wraps-do –

+0

[ticket rilevante] (http://bugs.python.org/issue1108) – shx2

+0

@AndrewGorcester Ho studiato il collegamento, grazie. Ma io sto ferendo come posso usare * wraps * in un decoratore CLASS? – Farsheed

risposta

2

Che ne dici di eludere il problema utilizzando un decoratore memo che restituisce una funzione invece di un'istanza di classe:

import functools 

def decorator(d): 
    """Make function d a decorator: d wraps a function fn. 
    Authors: Peter Norvig and Darius Bacon""" 
    def _d(fn): 
     return functools.update_wrapper(d(fn), fn) 
    functools.update_wrapper(_d, d) 
    return _d 

@decorator 
def memo(f): 
    # by Peter Norvig 
    """Decorator that caches the return value for each call to f(args). 
    Then when called again with same args, we can just look it up.""" 
    cache = {} 

    def _f(*args): 
     try: 
      return cache[args] 
     except KeyError: 
      cache[args] = result = f(*args) 
      return result 
     except TypeError: 
      # some element of args can't be a dict key 
      return f(*args) 
    _f.cache = cache 
    return _f 

@memo 
def is_tile_inside_border(x, y, z, border): 
    '''Checks if a tile is inside border or not 
    >>> is_tile_inside_border(85,53,7,'iran') 
    3 
    >>> is_tile_inside_border(85,15,7,'iran') 
    0 
    ''' 
    # binary_data = get_border_binary(border, z) 
    # return isInside((x, y), binary_data) 
    return True 

class Foo(object): 
    @memo 
    def bar(self, x): 
     """ 
     >>> Foo().bar(1) 
     2 
     """ 
     return x 

if __name__ == '__main__': 
    import doctest 
    doctest.testmod()