pixelocr/pixelocr/utils.py
2014-09-03 21:06:17 +02:00

71 lines
1.9 KiB
Python

# Copyright (C) 2014 Andrey Golovizin
#
# 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 3 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, see <http://www.gnu.org/licenses/>.
import functools
import itertools
from collections import defaultdict
from threading import Lock
def cached_property(fun):
"""A memoize decorator for class properties."""
lock = Lock()
@functools.wraps(fun)
def get(self):
with lock:
try:
obj_locks = self._locks
except AttributeError:
obj_locks = self._locks = defaultdict(Lock)
obj_lock = obj_locks[fun]
with obj_lock:
try:
cache = self._cache
except AttributeError:
cache = self._cache = {}
try:
ret = cache[fun]
except KeyError:
ret = cache[fun] = fun(self)
return ret
return property(get)
def collect_iterable(func):
@functools.wraps(func)
def collect(*args, **kwargs):
return list(func(*args, **kwargs))
return collect
def pairwise(iterable):
a, b = itertools.tee(iterable)
next(b, None)
return itertools.zip_longest(a, b)
def neighbourhood(lst, index, window=5):
"""Return adjacent list items."""
before = lst[:index]
after = lst[index + 1:]
return before[-window:] + after[:window]
def pipe(value, *funcs):
for func in funcs:
value = func(value)
return value