<div dir="ltr">Do you know about run_in_executor?<div><br></div><div><a href="http://docs.python.org/3.4/library/asyncio-eventloop.html#asyncio.BaseEventLoop.run_in_executor">http://docs.python.org/3.4/library/asyncio-eventloop.html#asyncio.BaseEventLoop.run_in_executor</a><br>
</div></div><div class="gmail_extra"><br><br><div class="gmail_quote">2014-02-12 10:24 GMT+01:00 David Foster <span dir="ltr"><<a href="mailto:davidfstr@gmail.com" target="_blank">davidfstr@gmail.com</a>></span>:<br>
<blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">There has been much talk of the use of async libraries (such as Tulip) within the general programming community to avoid the "christmas tree" pattern of code indentation in code that uses a lot of callbacks. For example, consider the following callback-based code:<br>

<br>
def duplicate_file_async(filepath, done):<br>
    def exists_async(does_exist):<br>
        if not does_exist:<br>
            done(False)<br>
            return<br>
        def entire_file_read(contents):<br>
            if not contents:<br>
                done(False)<br>
                return<br>
            def file_written(success):<br>
                done(success)<br>
            write_entire_file_async(<u></u>filepath + ' copy', contents, file_written)<br>
        read_entire_file_async(<u></u>filepath, entire_file_read)<br>
    exists_async(filepath)<br>
<br>
This code be rewritten in Tulip as:<br>
<br>
@async.coroutine<br>
def duplicate_file_async(filepath)<u></u>:<br>
    if not yield from exists_async(filepath):<br>
        return False<br>
    contents = yield from read_entire_file_async(<u></u>filepath)<br>
    if not contents:<br>
        return False<br>
    return yield from write_entire_file_async(<u></u>filepath + ' copy', contents)<br>
<br>
Much more clear, no?<br>
<br>
Tulip, however, does not appear to address a slightly different problem I run into where I need different parts of the same conceptual function to run on specific *different* threads. Consider the following code which needs different parts executed on a UI thread, database thread, and background thread:<br>

<br>
def tree_node_clicked(tree_node):<br>
    print('Clicked: ' + repr(tree_node))<br>
    def db_task():<br>
        db_node = fetch_node_from_db(tree_node)<br>
<br>
        def process_node(db_node):<br>
            tree_node.set_contents(db_<u></u>node.contents)<br>
            # (done)<br>
<br>
        if db_node is not None:<br>
            def bg_task():<br>
                db_node = fetch_node_from_internet(tree_<u></u>node.url)<br>
                def db_task():<br>
                    insert_into_db(db_node)<br>
                    ui_call_soon(process_node, db_node)<br>
                db_call_soon(db_task)<br>
            bg_call_soon(bg_task)<br>
        else:<br>
            ui_call_soon(process_node, db_node)<br>
    db_call_soon(db_task)<br>
<br>
Imagine if you could write this function like the following:<br>
<br>
ui_loop = ...<br>
db_loop = ...<br>
bg_loop = ...<br>
<br>
@async.coroutine<br>
def tree_node_clicked(tree_node):<br>
    print('Clicked: ' + repr(tree_node))<br>
<br>
    yield from switch_to_loop(db_loop)<br>
    db_node = fetch_node_from_db(tree_node)<br>
    if db_node is not None:<br>
        yield from switch_to_loop(bg_loop)<br>
        db_node = fetch_node_from_internet(tree_<u></u>node.url)<br>
<br>
        yield from switch_to_loop(db_loop)<br>
        insert_into_db(db_node)<br>
<br>
    yield from switch_to_loop(ui_loop)<br>
    tree_node.set_contents(db_<u></u>node.contents)<br>
<br>
Or even better: If you created a decorators like @loop_affinity(*_loop) that automatically called switch_to_loop(...) if the current event loop wasn't correct, you could even write the following even-more simplified version:<br>

<br>
@async.coroutine<br>
def tree_node_clicked(tree_node):<br>
    print('Clicked: ' + repr(tree_node))<br>
    db_node = yield from fetch_node_from_db(tree_node)<br>
    if db_node is not None:<br>
        db_node = yield from fetch_node_from_internet(tree_<u></u>node.url)<br>
        yield from insert_into_db(db_node)<br>
    yield from tree_node.set_contents(db_<u></u>node.contents)<br>
<br>
@async.loop_affinity(db_loop)<br>
def fetch_node_from_db(...): ...<br>
<br>
@async.loop_affinity(bg_loop)<br>
def fetch_node_from_internet(...): ...<br>
<br>
@async.loop_affinity(db_loop)<br>
def insert_into_db(...): ...<br>
<br>
@async.loop_affinity(ui_loop)<br>
def set_contents(...): ...<br>
<br>
Wouldn't that be just grand? I have prototyped switch_to_loop(...) successfully in Tulip, albeit with a race condition I was unable to isolate.<br>
<br>
How about some equivalent to switch_to_loop(...) and loop_affinity(...) in a future version of Tulip?<span class="HOEnZb"><font color="#888888"><br>
<br>
-- <br>
David Foster<br>
<a href="http://dafoster.net/" target="_blank">http://dafoster.net/</a><br>
______________________________<u></u>_________________<br>
Python-ideas mailing list<br>
<a href="mailto:Python-ideas@python.org" target="_blank">Python-ideas@python.org</a><br>
<a href="https://mail.python.org/mailman/listinfo/python-ideas" target="_blank">https://mail.python.org/<u></u>mailman/listinfo/python-ideas</a><br>
Code of Conduct: <a href="http://python.org/psf/codeofconduct/" target="_blank">http://python.org/psf/<u></u>codeofconduct/</a><br>
</font></span></blockquote></div><br></div>