You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
ORPA-pyOpenRPA/Resources/WPy64-3720/python-3.7.2.amd64/Lib/site-packages/pydash/functions.py

1031 lines
27 KiB

# -*- coding: utf-8 -*-
"""
Functions that wrap other functions.
.. versionadded:: 1.0.0
"""
from __future__ import absolute_import
import itertools
import time
import pydash as pyd
from ._compat import _range, getfullargspec, iteritems
__all__ = (
"after",
"ary",
"before",
"conjoin",
"curry",
"curry_right",
"debounce",
"delay",
"disjoin",
"flip",
"flow",
"flow_right",
"iterated",
"juxtapose",
"negate",
"once",
"over_args",
"partial",
"partial_right",
"rearg",
"spread",
"throttle",
"unary",
"wrap",
)
class After(object):
"""Wrap a function in an after context."""
def __init__(self, func, n):
try:
n = int(n)
assert n >= 0
except (ValueError, TypeError, AssertionError):
n = 0
self.n = n
self.func = func
def __call__(self, *args, **kwargs):
"""Return results of :attr:`func` after :attr:`n` calls."""
self.n -= 1
if self.n <= 0:
return self.func(*args, **kwargs)
class Ary(object):
"""Wrap a function in an ary context."""
def __init__(self, func, n):
try:
n = int(n)
assert n >= 0
except (ValueError, TypeError, AssertionError):
n = None
self.n = n
self.func = func
def __call__(self, *args, **kwargs):
"""
Return results of :attr:`func` with arguments capped to :attr:`n`.
Only positional arguments are capped. Any number of keyword arguments are allowed.
"""
if self.n is not None:
args = args[: self.n]
return self.func(*args, **kwargs)
class Before(After):
"""Wrap a function in a before context."""
def __call__(self, *args, **kwargs):
self.n -= 1
if self.n > 0:
return self.func(*args, **kwargs)
class Flow(object):
"""Wrap a function in a flow context."""
def __init__(self, *funcs, **kwargs):
self.funcs = funcs
self.from_right = kwargs.get("from_right", True)
def __call__(self, *args, **kwargs):
"""Return results of composing :attr:`funcs`."""
funcs = list(self.funcs)
from_index = -1 if self.from_right else 0
result = None
while funcs:
result = funcs.pop(from_index)(*args, **kwargs)
args = (result,)
kwargs = {}
return result
class Conjoin(object):
"""Wrap a set of functions in a conjoin context."""
def __init__(self, *funcs):
self.funcs = funcs
def __call__(self, obj):
"""Return result of conjoin `obj` with :attr:`funcs` predicates."""
def iteratee(item):
return pyd.every(self.funcs, lambda func: func(item))
return pyd.every(obj, iteratee)
class Curry(object):
"""Wrap a function in a curry context."""
def __init__(self, func, arity, args=None, kwargs=None):
self.func = func
self.arity = len(getfullargspec(func).args) if arity is None else arity
self.args = () if args is None else args
self.kwargs = {} if kwargs is None else kwargs
def __call__(self, *args, **kwargs):
"""Store `args` and `kwargs` and call :attr:`func` if we've reached or exceeded the function
arity."""
args = self.compose_args(args)
kwargs.update(self.kwargs)
if (len(args) + len(kwargs)) >= self.arity:
args_arity = self.arity - len(kwargs)
args = args[: (args_arity if args_arity > 0 else 0)]
curried = self.func(*args, **kwargs)
else:
# NOTE: Use self.__class__ so that subclasses will use their own
# class to generate next iteration of call.
curried = self.__class__(self.func, self.arity, args, kwargs)
return curried
def compose_args(self, new_args):
"""Combine `self.args` with `new_args` and return."""
return tuple(list(self.args) + list(new_args))
class CurryRight(Curry):
"""Wrap a function in a curry-right context."""
def compose_args(self, new_args):
return tuple(list(new_args) + list(self.args))
class Debounce(object):
"""Wrap a function in a debounce context."""
def __init__(self, func, wait, max_wait=False):
self.func = func
self.wait = wait
self.max_wait = max_wait
self.last_result = None
# Initialize last_* times to be prior to the wait periods so that func
# is primed to be executed on first call.
self.last_call = pyd.now() - self.wait
self.last_execution = pyd.now() - max_wait if pyd.is_number(max_wait) else None
def __call__(self, *args, **kwargs):
"""
Execute :attr:`func` if function hasn't been called witinin last :attr:`wait` milliseconds
or in last :attr:`max_wait` milliseconds.
Return results of last successful call.
"""
present = pyd.now()
if (present - self.last_call) >= self.wait or (
self.max_wait and (present - self.last_execution) >= self.max_wait
):
self.last_result = self.func(*args, **kwargs)
self.last_execution = present
self.last_call = present
return self.last_result
class Disjoin(object):
"""Wrap a set of functions in a disjoin context."""
def __init__(self, *funcs):
self.funcs = funcs
def __call__(self, obj):
"""Return result of disjoin `obj` with :attr:`funcs` predicates."""
def iteratee(item):
return pyd.some(self.funcs, lambda func: func(item))
return pyd.some(obj, iteratee)
class Flip(object):
"""Wrap a function in a flip context."""
def __init__(self, func):
self.func = func
def __call__(self, *args, **kwargs):
return self.func(*reversed(args), **kwargs)
class Iterated(object):
"""Wrap a function in an iterated context."""
def __init__(self, func):
self.func = func
def _iteration(self, initial):
"""Iterator that composing :attr:`func` with itself."""
value = initial
while True:
value = self.func(value)
yield value
def __call__(self, initial, n):
"""Return value of calling :attr:`func` `n` times using `initial` as seed value."""
value = initial
iteration = self._iteration(value)
for _ in _range(n):
value = next(iteration)
return value
class Juxtapose(object):
"""Wrap a function in a juxtapose context."""
def __init__(self, *funcs):
self.funcs = funcs
def __call__(self, *objs):
return [func(*objs) for func in self.funcs]
class OverArgs(object):
"""Wrap a function in a over_args context."""
def __init__(self, func, *transforms):
self.func = func
self.transforms = pyd.flatten(transforms)
def __call__(self, *args):
args = (self.transforms[idx](args) for idx, args in enumerate(args))
return self.func(*args)
class Negate(object):
"""Wrap a function in a negate context."""
def __init__(self, func):
self.func = func
def __call__(self, *args, **kwargs):
"""Return negated results of calling :attr:`func`."""
return not self.func(*args, **kwargs)
class Once(object):
"""Wrap a function in a once context."""
def __init__(self, func):
self.func = func
self.result = None
self.called = False
def __call__(self, *args, **kwargs):
"""Return results from the first call of :attr:`func`."""
if not self.called:
self.result = self.func(*args, **kwargs)
self.called = True
return self.result
class Partial(object):
"""Wrap a function in a partial context."""
def __init__(self, func, args, kwargs=None, from_right=False):
self.func = func
self.args = args
self.kwargs = kwargs or {}
self.from_right = from_right
def __call__(self, *args, **kwargs):
"""
Return results from :attr:`func` with :attr:`args` + `args`.
Apply arguments from left or right depending on :attr:`from_right`.
"""
if self.from_right:
args = itertools.chain(args, self.args)
else:
args = itertools.chain(self.args, args)
kwargs = dict(itertools.chain(iteritems(self.kwargs), iteritems(kwargs)))
return self.func(*args, **kwargs)
class Rearg(object):
"""Wrap a function in a rearg context."""
def __init__(self, func, *indexes):
self.func = func
# Index `indexes` by the index value so we can do a lookup mapping by walking the function
# arguments.
self.indexes = dict(
(src_index, dest_index) for dest_index, src_index in enumerate(pyd.flatten(indexes))
)
def __call__(self, *args, **kwargs):
"""Return results from :attr:`func` using rearranged arguments."""
reargs = {}
rest = []
# Walk arguments to ensure each one is added to the final argument list.
for src_index, arg in enumerate(args):
# NOTE: dest_index will range from 0 to len(indexes).
dest_index = self.indexes.get(src_index)
if dest_index is not None:
# Remap argument index.
reargs[dest_index] = arg
else:
# Argumnet index is not contained in `indexes` so stick in the back.
rest.append(arg)
args = itertools.chain((reargs[key] for key in sorted(reargs)), rest)
return self.func(*args, **kwargs)
class Spread(object):
"""Wrap a function in a spread context."""
def __init__(self, func):
self.func = func
def __call__(self, args):
"""Return results from :attr:`func` using array of `args` provided."""
return self.func(*args)
class Throttle(object):
"""Wrap a function in a throttle context."""
def __init__(self, func, wait):
self.func = func
self.wait = wait
self.last_result = None
self.last_execution = pyd.now() - self.wait
def __call__(self, *args, **kwargs):
"""
Execute :attr:`func` if function hasn't been called witinin last :attr:`wait` milliseconds.
Return results of last successful call.
"""
present = pyd.now()
if (present - self.last_execution) >= self.wait:
self.last_result = self.func(*args, **kwargs)
self.last_execution = present
return self.last_result
def after(func, n):
"""
Creates a function that executes `func`, with the arguments of the created function, only after
being called `n` times.
Args:
func (callable): Function to execute.
n (int): Number of times `func` must be called before it is executed.
Returns:
After: Function wrapped in an :class:`After` context.
Example:
>>> func = lambda a, b, c: (a, b, c)
>>> after_func = after(func, 3)
>>> after_func(1, 2, 3)
>>> after_func(1, 2, 3)
>>> after_func(1, 2, 3)
(1, 2, 3)
>>> after_func(4, 5, 6)
(4, 5, 6)
.. versionadded:: 1.0.0
.. versionchanged:: 3.0.0
Reordered arguments to make `func` first.
"""
return After(func, n)
def ary(func, n):
"""
Creates a function that accepts up to `n` arguments ignoring any additional arguments. Only
positional arguments are capped. All keyword arguments are allowed through.
Args:
func (callable): Function to cap arguments for.
n (int): Number of arguments to accept.
Returns:
Ary: Function wrapped in an :class:`Ary` context.
Example:
>>> func = lambda a, b, c=0, d=5: (a, b, c, d)
>>> ary_func = ary(func, 2)
>>> ary_func(1, 2, 3, 4, 5, 6)
(1, 2, 0, 5)
>>> ary_func(1, 2, 3, 4, 5, 6, c=10, d=20)
(1, 2, 10, 20)
.. versionadded:: 3.0.0
"""
return Ary(func, n)
def before(func, n):
"""
Creates a function that executes `func`, with the arguments of the created function, until it
has been called `n` times.
Args:
func (callable): Function to execute.
n (int): Number of times `func` may be executed.
Returns:
Before: Function wrapped in an :class:`Before` context.
Example:
>>> func = lambda a, b, c: (a, b, c)
>>> before_func = before(func, 3)
>>> before_func(1, 2, 3)
(1, 2, 3)
>>> before_func(1, 2, 3)
(1, 2, 3)
>>> before_func(1, 2, 3)
>>> before_func(1, 2, 3)
.. versionadded:: 1.1.0
.. versionchanged:: 3.0.0
Reordered arguments to make `func` first.
"""
return Before(func, n)
def conjoin(*funcs):
"""
Creates a function that composes multiple predicate functions into a single predicate that tests
whether **all** elements of an object pass each predicate.
Args:
*funcs (callable): Function(s) to conjoin.
Returns:
Conjoin: Function(s) wrapped in a :class:`Conjoin` context.
Example:
>>> conjoiner = conjoin(lambda x: isinstance(x, int), lambda x: x > 3)
>>> conjoiner([1, 2, 3])
False
>>> conjoiner([1.0, 2, 1])
False
>>> conjoiner([4.0, 5, 6])
False
>>> conjoiner([4, 5, 6])
True
.. versionadded:: 2.0.0
"""
return Conjoin(*funcs)
def curry(func, arity=None):
"""
Creates a function that accepts one or more arguments of `func` that when invoked either
executes `func` returning its result (if all `func` arguments have been provided) or returns a
function that accepts one or more of the remaining `func` arguments, and so on.
Args:
func (callable): Function to curry.
arity (int, optional): Number of function arguments that can be accepted by curried
function. Default is to use the number of arguments that are accepted by `func`.
Returns:
Curry: Function wrapped in a :class:`Curry` context.
Example:
>>> func = lambda a, b, c: (a, b, c)
>>> currier = curry(func)
>>> currier = currier(1)
>>> assert isinstance(currier, Curry)
>>> currier = currier(2)
>>> assert isinstance(currier, Curry)
>>> currier = currier(3)
>>> currier
(1, 2, 3)
.. versionadded:: 1.0.0
"""
return Curry(func, arity)
def curry_right(func, arity=None):
"""
This method is like :func:`curry` except that arguments are applied to `func` in the manner of
:func:`partial_right` instead of :func:`partial`.
Args:
func (callable): Function to curry.
arity (int, optional): Number of function arguments that can be accepted by curried
function. Default is to use the number of arguments that are accepted by `func`.
Returns:
CurryRight: Function wrapped in a :class:`CurryRight` context.
Example:
>>> func = lambda a, b, c: (a, b, c)
>>> currier = curry_right(func)
>>> currier = currier(1)
>>> assert isinstance(currier, CurryRight)
>>> currier = currier(2)
>>> assert isinstance(currier, CurryRight)
>>> currier = currier(3)
>>> currier
(3, 2, 1)
.. versionadded:: 1.1.0
"""
return CurryRight(func, arity)
def debounce(func, wait, max_wait=False):
"""
Creates a function that will delay the execution of `func` until after `wait` milliseconds have
elapsed since the last time it was invoked. Subsequent calls to the debounced function will
return the result of the last `func` call.
Args:
func (callable): Function to execute.
wait (int): Milliseconds to wait before executing `func`.
max_wait (optional): Maximum time to wait before executing `func`.
Returns:
Debounce: Function wrapped in a :class:`Debounce` context.
.. versionadded:: 1.0.0
"""
return Debounce(func, wait, max_wait=max_wait)
def delay(func, wait, *args, **kwargs):
"""
Executes the `func` function after `wait` milliseconds. Additional arguments will be provided to
`func` when it is invoked.
Args:
func (callable): Function to execute.
wait (int): Milliseconds to wait before executing `func`.
*args (optional): Arguments to pass to `func`.
**kwargs (optional): Keyword arguments to pass to `func`.
Returns:
mixed: Return from `func`.
.. versionadded:: 1.0.0
"""
time.sleep(wait / 1000.0)
return func(*args, **kwargs)
def disjoin(*funcs):
"""
Creates a function that composes multiple predicate functions into a single predicate that tests
whether **any** elements of an object pass each predicate.
Args:
*funcs (callable): Function(s) to disjoin.
Returns:
Disjoin: Function(s) wrapped in a :class:`Disjoin` context.
Example:
>>> disjoiner = disjoin(lambda x: isinstance(x, float),\
lambda x: isinstance(x, int))
>>> disjoiner([1, '2', '3'])
True
>>> disjoiner([1.0, '2', '3'])
True
>>> disjoiner(['1', '2', '3'])
False
.. versionadded:: 2.0.0
"""
return Disjoin(*funcs)
def flip(func):
"""
Creates a function that invokes the method with arguments reversed.
Args:
func (callable): Function to flip arguments for.
Returns:
function: Function wrapped in a :class:`Flip` context.
Example:
>>> flipped = flip(lambda *args: args)
>>> flipped(1, 2, 3, 4)
(4, 3, 2, 1)
>>> flipped = flip(lambda *args: [i * 2 for i in args])
>>> flipped(1, 2, 3, 4)
[8, 6, 4, 2]
.. versionadded:: 4.0.0
"""
return Flip(func)
def flow(*funcs):
"""
Creates a function that is the composition of the provided functions, where each successive
invocation is supplied the return value of the previous. For example, composing the functions
``f()``, ``g()``, and ``h()`` produces ``h(g(f()))``.
Args:
*funcs (callable): Function(s) to compose.
Returns:
Flow: Function(s) wrapped in a :class:`Flow` context.
Example:
>>> mult_5 = lambda x: x * 5
>>> div_10 = lambda x: x / 10.0
>>> pow_2 = lambda x: x ** 2
>>> ops = flow(sum, mult_5, div_10, pow_2)
>>> ops([1, 2, 3, 4])
25.0
.. versionadded:: 2.0.0
.. versionchanged:: 2.3.1
Added :func:`pipe` as alias.
.. versionchanged:: 4.0.0
Removed alias ``pipe``.
"""
return Flow(*funcs, from_right=False)
def flow_right(*funcs):
"""
This function is like :func:`flow` except that it creates a function that invokes the provided
functions from right to left. For example, composing the functions ``f()``, ``g()``, and ``h()``
produces ``f(g(h()))``.
Args:
*funcs (callable): Function(s) to compose.
Returns:
Flow: Function(s) wrapped in a :class:`Flow` context.
Example:
>>> mult_5 = lambda x: x * 5
>>> div_10 = lambda x: x / 10.0
>>> pow_2 = lambda x: x ** 2
>>> ops = flow_right(mult_5, div_10, pow_2, sum)
>>> ops([1, 2, 3, 4])
50.0
.. versionadded:: 1.0.0
.. versionchanged:: 2.0.0
Added :func:`flow_right` and made :func:`compose` an alias.
.. versionchanged:: 2.3.1
Added :func:`pipe_right` as alias.
.. versionchanged:: 4.0.0
Removed aliases ``pipe_right`` and ``compose``.
"""
return Flow(*funcs, from_right=True)
def iterated(func):
"""
Creates a function that is composed with itself. Each call to the iterated function uses the
previous function call's result as input. Returned :class:`Iterated` instance can be called with
``(initial, n)`` where `initial` is the initial value to seed `func` with and `n` is the number
of times to call `func`.
Args:
func (callable): Function to iterate.
Returns:
Iterated: Function wrapped in a :class:`Iterated` context.
Example:
>>> doubler = iterated(lambda x: x * 2)
>>> doubler(4, 5)
128
>>> doubler(3, 9)
1536
.. versionadded:: 2.0.0
"""
return Iterated(func)
def juxtapose(*funcs):
"""
Creates a function whose return value is a list of the results of calling each `funcs` with the
supplied arguments.
Args:
*funcs (callable): Function(s) to juxtapose.
Returns:
Juxtapose: Function wrapped in a :class:`Juxtapose` context.
Example:
>>> double = lambda x: x * 2
>>> triple = lambda x: x * 3
>>> quadruple = lambda x: x * 4
>>> juxtapose(double, triple, quadruple)(5)
[10, 15, 20]
.. versionadded:: 2.0.0
"""
return Juxtapose(*funcs)
def negate(func):
"""
Creates a function that negates the result of the predicate `func`. The `func` function is
executed with the arguments of the created function.
Args:
func (callable): Function to negate execute.
Returns:
Negate: Function wrapped in a :class:`Negate` context.
Example:
>>> not_is_number = negate(lambda x: isinstance(x, (int, float)))
>>> not_is_number(1)
False
>>> not_is_number('1')
True
.. versionadded:: 1.1.0
"""
return Negate(func)
def once(func):
"""
Creates a function that is restricted to execute `func` once. Repeat calls to the function will
return the value of the first call.
Args:
func (callable): Function to execute.
Returns:
Once: Function wrapped in a :class:`Once` context.
Example:
>>> oncer = once(lambda *args: args[0])
>>> oncer(5)
5
>>> oncer(6)
5
.. versionadded:: 1.0.0
"""
return Once(func)
def over_args(func, *transforms):
"""
Creates a function that runs each argument through a corresponding transform function.
Args:
func (callable): Function to wrap.
*transforms (callable): Functions to transform arguments, specified as individual functions
or lists of functions.
Returns:
OverArgs: Function wrapped in a :class:`OverArgs` context.
Example:
>>> squared = lambda x: x ** 2
>>> double = lambda x: x * 2
>>> modder = over_args(lambda x, y: [x, y], squared, double)
>>> modder(5, 10)
[25, 20]
.. versionadded:: 3.3.0
.. versionchanged:: 4.0.0
Renamed from ``mod_args`` to ``over_args``.
"""
return OverArgs(func, *transforms)
def partial(func, *args, **kwargs):
"""
Creates a function that, when called, invokes `func` with any additional partial arguments
prepended to those provided to the new function.
Args:
func (callable): Function to execute.
*args (optional): Partial arguments to prepend to function call.
**kwargs (optional): Partial keyword arguments to bind to function call.
Returns:
Partial: Function wrapped in a :class:`Partial` context.
Example:
>>> dropper = partial(lambda array, n: array[n:], [1, 2, 3, 4])
>>> dropper(2)
[3, 4]
>>> dropper(1)
[2, 3, 4]
>>> myrest = partial(lambda array, n: array[n:], n=1)
>>> myrest([1, 2, 3, 4])
[2, 3, 4]
.. versionadded:: 1.0.0
"""
return Partial(func, args, kwargs)
def partial_right(func, *args, **kwargs):
"""
This method is like :func:`partial` except that partial arguments are appended to those provided
to the new function.
Args:
func (callable): Function to execute.
*args (optional): Partial arguments to append to function call.
**kwargs (optional): Partial keyword arguments to bind to function call.
Returns:
Partial: Function wrapped in a :class:`Partial` context.
Example:
>>> myrest = partial_right(lambda array, n: array[n:], 1)
>>> myrest([1, 2, 3, 4])
[2, 3, 4]
.. versionadded:: 1.0.0
"""
return Partial(func, args, kwargs, from_right=True)
def rearg(func, *indexes):
"""
Creates a function that invokes `func` with arguments arranged according to the specified
indexes where the argument value at the first index is provided as the first argument, the
argument value at the second index is provided as the second argument, and so on.
Args:
func (callable): Function to rearrange arguments for.
*indexes (int): The arranged argument indexes.
Returns:
Rearg: Function wrapped in a :class:`Rearg` context.
Example:
>>> jumble = rearg(lambda *args: args, 1, 2, 3)
>>> jumble(1, 2, 3)
(2, 3, 1)
>>> jumble('a', 'b', 'c', 'd', 'e')
('b', 'c', 'd', 'a', 'e')
.. versionadded:: 3.0.0
"""
return Rearg(func, *indexes)
def spread(func):
"""
Creates a function that invokes `func` with the array of arguments provided to the created
function.
Args:
func (callable): Function to spread.
Returns:
Spread: Function wrapped in a :class:`Spread` context.
Example:
>>> greet = spread(lambda *people: 'Hello ' + ', '.join(people) + '!')
>>> greet(['Mike', 'Don', 'Leo'])
'Hello Mike, Don, Leo!'
.. versionadded:: 3.1.0
"""
return Spread(func)
def throttle(func, wait):
"""
Creates a function that, when executed, will only call the `func` function at most once per
every `wait` milliseconds. Subsequent calls to the throttled function will return the result of
the last `func` call.
Args:
func (callable): Function to throttle.
wait (int): Milliseconds to wait before calling `func` again.
Returns:
mixed: Results of last `func` call.
.. versionadded:: 1.0.0
"""
return Throttle(func, wait)
def unary(func):
"""
Creates a function that accepts up to one argument, ignoring any additional arguments.
Args:
func (callable): Function to cap arguments for.
Returns:
Ary: Function wrapped in an :class:`Ary` context.
Example:
>>> func = lambda a, b=1, c=0, d=5: (a, b, c, d)
>>> unary_func = unary(func)
>>> unary_func(1, 2, 3, 4, 5, 6)
(1, 1, 0, 5)
>>> unary_func(1, 2, 3, 4, 5, 6, b=0, c=10, d=20)
(1, 0, 10, 20)
.. versionadded:: 4.0.0
"""
return Ary(func, 1)
def wrap(value, func):
"""
Creates a function that provides value to the wrapper function as its first argument. Additional
arguments provided to the function are appended to those provided to the wrapper function.
Args:
value (mixed): Value provided as first argument to function call.
func (callable): Function to execute.
Returns:
Partial: Function wrapped in a :class:`Partial` context.
Example:
>>> wrapper = wrap('hello', lambda *args: args)
>>> wrapper(1, 2)
('hello', 1, 2)
.. versionadded:: 1.0.0
"""
return Partial(func, (value,))