Skip to main content

An easy async decorators and promises.

Project description

EAsync – easy async decorator and promises

https://travis-ci.org/complynx/easync.svg?branch=master

@async

Decorator around the functions for them to be asynchronous.

Used as a simple decorator or along with named arguments. Preserves function’s or method’s docs, names and other stuff using functools.wraps.

When the wrapped function is called, it produces the Promise of itself.

Usage:

>>> from easync import async
>>> @async
>>> def f1():
>>>    # some heavy-weight code
>>> @async(daemon=True, print_exception=False)
>>> def f2():
>>>     # some faulty daemon we don't really care about
>>> @async(print_exception=logging.WARNING)
>>> def f3():
>>>     # setting another logging level
>>> f1().wait()
<result>
>>> f2()  # started daemon
easync.Promise(...)
>>> f3().then(callback, error)  # notice, here the print_exception is suppressed (see Promise doc for more)
easync.Promise(...)
param function:

The function to be decorated.

param Boolean daemon:

Optional. Create the daemon thread, that will die when no other threads left.

param print_exception:

Log level to log the exception, if any, or None to mute it. See logging.

param no_promise:

Will not return a thing, instead of returning Promise, good for decorating, for example, Tornado handlers.

return:

Wrapped function, Promise generator.

Promise

This is a threading wrapper that performs asynchronous call of the provided function. The behaviour is inspired by JavaScript Promises, but differs in several points.

First, the resolution is based upon the return of the function beneath. On successful return, the result is stored in Promise.result On exception, exception is stored in Promise.exception More about how Promise resolves in Resolving Promises.

You can add callbacks by using methods Promise.then or Promise.catch They will create a new Promise resolving in either the resolution of the previous one or the underlying callback.

If at the final stage no exception is processed or were generated new ones, they will be printed.

Basic usage:

>>> from easync import Promise
>>> promise = Promise(func)(...args)
>>> promise.wait()
<result>  # if any
>>> promise.result
<result>  # if any
>>> promise.exception
Exception  # if something went wrong

Callbacks usage:

>>> def callback(result):
>>>     do_stuff(result)
>>> def error(exception):
>>>     my_log(exception)
>>> def some_final(_):
>>>     cleanup()
>>> Promise(func)(...args).then(callback, error).then(some_final)
Promise(...)
Promise.Callable

The to-be-runned function.

Promise.result

This parameter holds the result the underlying function returned.

Promise.exception

This parameter holds the exception if the underlying function fails.

Promise.resolved

True if the function resolves. More about it in Resolving Promises.

Promise.started

threading.Event, sets when thread is started.

Promise.finished

threading.Event, sets when thread is finished, no matter where had it been resolved or not.

Promise.print_exception

Enables exception printout if caught. Uses logging to do so and has to be one of logging levels. False or None disable. Also automatically sets to False when a dependent Promise created through Promise.then or Promise.catch.

Resolving Promises

Functions

If the Promise is constructed with a function, it will resolve in it’s return or reject with the caught exception.

Other Promises

Just resolves in the same way the other one is resolved. Printouts will be suppressed in the first one.

Events and Conditions

If Promise is based on threading.Event or threading.Condition, it is resolved when the underlying Event or Condition occurs. The type testing is duck-type for having the wait method, so anything using the interface of waiting can be resolved, for example other Promises, or threads.

The resolving is based on testing is_failed on the object, and if that one returns, the Promise rejects. Otherwise, the get_result is called to obtain the result. Both are duck-type thingeys.

Anything else

Resolves successfully with the result equals to the passed-in argument.

Promise.__init__

__init__(function[, daemon=False, print_exception=logging.ERROR])

The constructor creates a threading.Thread wrapping the function. To start it, call the resulting object as a function with it’s arguments. (Explained in Promise.__call__)

>>> promise = Promise(func, print_exception=None)
>>> promise()
param function:

Function, Event, Condition, or anything else to resolve.

param daemon:

Sets up daemon flag in the thread. May be set later. Optional.

param print_exception:

Sets up the final exception printing level. Pass False to suppress.

Promise.__call__

__call__(*args, **kwargs)

Starts the thread and passes the arguments of the function into it. Returns self, for simple adding Promise.then, Promise.wait or Promise.catch.

Promise.wait

wait([timeout=None])

Pauses the current thread to wait until the underlying promise resolves.

If timeout is set, raises easync.TimeoutError if it’s reached.

Returns result of the underlying function if there’s any.

Promise.then

then([resolved=None, rejected=None, print_exception=Promise.print_exception])

This method sets callbacks for a Promise.

NOTE this method suppresses the Promise default error handling by setting Promise.print_exception to False. You can then re-enable printouts manually, overriding the Promise.print_exception yourself.

NOTE calling this method twice on the same Promise object will result in duplicated exception printouts unless changed.

The result is a new Promise which resolves in:

callback exception:

If the called callback (either resolved or rejected) failed or raised anything.

reject:

If the underlying Promise rejected and no rejected callback was passed.

callback return:

The result of the called callback.

resolve:

The result of the underlying Promise if it resolves and no resolved callback was passed.

This is done to have this kind of behaviour:

>>> Promise(action)(...args).then(parse_result).then(parse_one_more_result).catch(any_exception).then(cleanup)
function resolved(result):

The positive callback for the Promise. Has to accept one positional argument - the result.

function rejected(exception):

The negative callback for the Promise. Has to accept one positional argument - the caught exception.

print_exception:

Passed into the corresponding argument of the newly created Promise.

return:

New Promise.

Promise.catch

catch([callback=None, print_exception=Promise.print_exception])

The same as Promise.then (resolved=None, callback, print_exception).

Promise static methods

Promise.resolve

Promise.resolve(thing)

Resolves thing, regardless of what it is, to result.

param thing:

any

return:

resolved Promise with the Promise.result equals to thing.

Promise.reject

Promise.reject(thing)

Rejects thing, regardless of what it is.

param thing:

any

return:

rejected Promise with the Promise.exception equals to thing.

Promise.all

Promise.all(things)

Resolves when all the items in the enumerate(things) are resolved. Or rejects when any of the items is rejected.

param things:

list of things or anything to be enumerate’d.

result:

list of results of all the Promises for each of the items.

exception:

first caught exception.

Promise.race

Promise.race(things)

Resolves when any of the items in the enumerate(things) is resolved. Or rejects when any of the items is rejected.

param things:

list of things or anything to be enumerate’d.

result:

the result of the first resolved item.

exception:

first caught exception.

Other functions

get_result

get_result(obj)

Returns the first found attribute of result or success of the object obj, if any. Otherwise returns None.

is_failed

is_failed(obj)

Returns:

found property:

if one of error, exception, failure is found.

True:

if one of failed or is_failed is true.

True:

if success is present and is False.

None:

Otherwise

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

easync-1.0.5b1.tar.gz (11.8 kB view hashes)

Uploaded Source

Built Distribution

easync-1.0.5b1-py2.py3-none-any.whl (12.5 kB view hashes)

Uploaded Python 2 Python 3

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page