PEP8 cleanup of functional.py

This commit is contained in:
Preston Holmes 2013-03-06 16:09:36 -08:00
parent 0ea5bf88dd
commit 876fc39128
2 changed files with 27 additions and 16 deletions

View File

@ -1,18 +1,20 @@
import copy import copy
import operator import operator
from functools import wraps, update_wrapper from functools import wraps
import sys import sys
from django.utils import six from django.utils import six
# You can't trivially replace this with `functools.partial` because this binds # You can't trivially replace this with `functools.partial` because this binds
# to classes and returns bound instances, whereas functools.partial (on # to classes and returns bound instances, whereas functools.partial (on
# CPython) is a type and its instances don't bind. # CPython) is a type and its instances don't bind.
def curry(_curried_func, *args, **kwargs): def curry(_curried_func, *args, **kwargs):
def _curried(*moreargs, **morekwargs): def _curried(*moreargs, **morekwargs):
return _curried_func(*(args+moreargs), **dict(kwargs, **morekwargs)) return _curried_func(*(args + moreargs), **dict(kwargs, **morekwargs))
return _curried return _curried
def memoize(func, cache, num_args): def memoize(func, cache, num_args):
""" """
Wrap a function so that results for any argument tuple are stored in Wrap a function so that results for any argument tuple are stored in
@ -31,6 +33,7 @@ def memoize(func, cache, num_args):
return result return result
return wrapper return wrapper
class cached_property(object): class cached_property(object):
""" """
Decorator that converts a method with a single self argument into a Decorator that converts a method with a single self argument into a
@ -45,6 +48,7 @@ class cached_property(object):
res = instance.__dict__[self.func.__name__] = self.func(instance) res = instance.__dict__[self.func.__name__] = self.func(instance)
return res return res
class Promise(object): class Promise(object):
""" """
This is just a base class for the proxy class created in This is just a base class for the proxy class created in
@ -53,6 +57,7 @@ class Promise(object):
""" """
pass pass
def lazy(func, *resultclasses): def lazy(func, *resultclasses):
""" """
Turns any callable into a lazy evaluated callable. You need to give result Turns any callable into a lazy evaluated callable. You need to give result
@ -88,9 +93,9 @@ def lazy(func, *resultclasses):
cls.__dispatch[resultclass] = {} cls.__dispatch[resultclass] = {}
for type_ in reversed(resultclass.mro()): for type_ in reversed(resultclass.mro()):
for (k, v) in type_.__dict__.items(): for (k, v) in type_.__dict__.items():
# All __promise__ return the same wrapper method, but they # All __promise__ return the same wrapper method, but
# also do setup, inserting the method into the dispatch # they also do setup, inserting the method into the
# dict. # dispatch dict.
meth = cls.__promise__(resultclass, k, v) meth = cls.__promise__(resultclass, k, v)
if hasattr(cls, k): if hasattr(cls, k):
continue continue
@ -111,8 +116,8 @@ def lazy(func, *resultclasses):
__prepare_class__ = classmethod(__prepare_class__) __prepare_class__ = classmethod(__prepare_class__)
def __promise__(cls, klass, funcname, method): def __promise__(cls, klass, funcname, method):
# Builds a wrapper around some magic method and registers that magic # Builds a wrapper around some magic method and registers that
# method for the given type and method name. # magic method for the given type and method name.
def __wrapper__(self, *args, **kw): def __wrapper__(self, *args, **kw):
# Automatically triggers the evaluation of a lazy value and # Automatically triggers the evaluation of a lazy value and
# applies the given magic method of the result type. # applies the given magic method of the result type.
@ -176,9 +181,11 @@ def lazy(func, *resultclasses):
return __wrapper__ return __wrapper__
def _lazy_proxy_unpickle(func, args, kwargs, *resultclasses): def _lazy_proxy_unpickle(func, args, kwargs, *resultclasses):
return lazy(func, *resultclasses)(*args, **kwargs) return lazy(func, *resultclasses)(*args, **kwargs)
def allow_lazy(func, *resultclasses): def allow_lazy(func, *resultclasses):
""" """
A decorator that allows a function to be called with one or more lazy A decorator that allows a function to be called with one or more lazy
@ -197,6 +204,8 @@ def allow_lazy(func, *resultclasses):
return wrapper return wrapper
empty = object() empty = object()
def new_method_proxy(func): def new_method_proxy(func):
def inner(self, *args): def inner(self, *args):
if self._wrapped is empty: if self._wrapped is empty:
@ -204,6 +213,7 @@ def new_method_proxy(func):
return func(self._wrapped, *args) return func(self._wrapped, *args)
return inner return inner
class LazyObject(object): class LazyObject(object):
""" """
A wrapper for another class that can be used to delay instantiation of the A wrapper for another class that can be used to delay instantiation of the
@ -246,6 +256,7 @@ class LazyObject(object):
# Workaround for http://bugs.python.org/issue12370 # Workaround for http://bugs.python.org/issue12370
_super = super _super = super
class SimpleLazyObject(LazyObject): class SimpleLazyObject(LazyObject):
""" """
A lazy object initialised from any function. A lazy object initialised from any function.
@ -288,8 +299,8 @@ class SimpleLazyObject(LazyObject):
# Because we have messed with __class__ below, we confuse pickle as to what # Because we have messed with __class__ below, we confuse pickle as to what
# class we are pickling. It also appears to stop __reduce__ from being # class we are pickling. It also appears to stop __reduce__ from being
# called. So, we define __getstate__ in a way that cooperates with the way # called. So, we define __getstate__ in a way that cooperates with the way
# that pickle interprets this class. This fails when the wrapped class is a # that pickle interprets this class. This fails when the wrapped class is
# builtin, but it is better than nothing. # a builtin, but it is better than nothing.
def __getstate__(self): def __getstate__(self):
if self._wrapped is empty: if self._wrapped is empty:
self._setup() self._setup()
@ -314,8 +325,8 @@ class SimpleLazyObject(LazyObject):
repr_attr = self._wrapped repr_attr = self._wrapped
return '<SimpleLazyObject: %r>' % repr_attr return '<SimpleLazyObject: %r>' % repr_attr
# Need to pretend to be the wrapped class, for the sake of objects that care # Need to pretend to be the wrapped class, for the sake of objects that
# about this (especially in equality tests) # care about this (especially in equality tests)
__class__ = property(new_method_proxy(operator.attrgetter("__class__"))) __class__ = property(new_method_proxy(operator.attrgetter("__class__")))
__eq__ = new_method_proxy(operator.eq) __eq__ = new_method_proxy(operator.eq)
__hash__ = new_method_proxy(hash) __hash__ = new_method_proxy(hash)
@ -343,6 +354,7 @@ class lazy_property(property):
return getattr(instance, name)() return getattr(instance, name)()
return property(fget, fset, fdel, doc) return property(fget, fset, fdel, doc)
def partition(predicate, values): def partition(predicate, values):
""" """
Splits the values into two sets, based on the return value of the function Splits the values into two sets, based on the return value of the function
@ -356,7 +368,7 @@ def partition(predicate, values):
results[predicate(item)].append(item) results[predicate(item)].append(item)
return results return results
if sys.version_info >= (2,7,2): if sys.version_info >= (2, 7, 2):
from functools import total_ordering from functools import total_ordering
else: else:
# For Python < 2.7.2. Python 2.6 does not have total_ordering, and # For Python < 2.7.2. Python 2.6 does not have total_ordering, and

View File

@ -3,7 +3,6 @@ from __future__ import unicode_literals
import copy import copy
import pickle import pickle
from django.test.utils import str_prefix
from django.utils import six from django.utils import six
from django.utils.unittest import TestCase from django.utils.unittest import TestCase
from django.utils.functional import SimpleLazyObject, empty from django.utils.functional import SimpleLazyObject, empty
@ -67,7 +66,7 @@ class TestUtilsSimpleLazyObject(TestCase):
self.assertEqual(empty, x._wrapped) self.assertEqual(empty, x._wrapped)
# Second, for an evaluated SimpleLazyObject # Second, for an evaluated SimpleLazyObject
name = x.name # evaluate name = x.name # evaluate
self.assertTrue(isinstance(x._wrapped, _ComplexObject)) self.assertTrue(isinstance(x._wrapped, _ComplexObject))
# __repr__ contains __repr__ of wrapped object # __repr__ contains __repr__ of wrapped object
self.assertEqual("<SimpleLazyObject: %r>" % x._wrapped, repr(x)) self.assertEqual("<SimpleLazyObject: %r>" % x._wrapped, repr(x))
@ -97,14 +96,14 @@ class TestUtilsSimpleLazyObject(TestCase):
self.assertEqual(s2, complex_object()) self.assertEqual(s2, complex_object())
# Second, for an evaluated SimpleLazyObject # Second, for an evaluated SimpleLazyObject
name = s.name # evaluate name = s.name # evaluate
self.assertIsNot(s._wrapped, empty) self.assertIsNot(s._wrapped, empty)
s3 = copy.deepcopy(s) s3 = copy.deepcopy(s)
self.assertEqual(s3, complex_object()) self.assertEqual(s3, complex_object())
def test_none(self): def test_none(self):
i = [0] i = [0]
def f(): def f():
i[0] += 1 i[0] += 1
return None return None