async-stagger


Nameasync-stagger JSON
Version 0.4.0.post1 PyPI version JSON
download
home_pagehttps://github.com/twisteroidambassador/async_stagger
SummaryHappy eyeballs and underlying scheduling algorithm in asyncio
upload_time2024-09-02 12:15:02
maintainerNone
docs_urlNone
authortwisteroid ambassador
requires_python>=3.11
licenseMIT
keywords happy-eyeballs dual-stack tcp
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            async_stagger: Happy Eyeballs in ``asyncio``
############################################

Quick, what's the situation?
============================

To get all the benefits of Happy Eyeballs connection establishment algorithm,
simply use ``async_stagger.open_connection`` like you would use
``asyncio.open_connection``::

    reader, writer = await async_stagger.open_connection('www.example.com', 80)

Now your connections are more dual-stack friendly and will complete faster!
A replacement for ``loop.create_connection`` is also provided.

The long version
================

What is Happy Eyeballs, and why should I use it?
------------------------------------------------

Happy Eyeballs is an algorithm for establishing TCP connections to destinations
specified by host names. It is described in :rfc:`6555` and :rfc:`8305`. The
primary benefit is that when host name resolution returns multiple addresses,
and some of the address are unreachable, Happy Eyeballs will establish the
connection much faster than conventional algorithms. For more information,
check the `Wikipedia article on Happy Eyeballs`_.

.. _Wikipedia article on Happy Eyeballs: https://en.wikipedia.org/wiki/Happy_Eyeballs

Python's standard library provides several high-level methods of establishing
TCP connections towards a host name: The **socket** module has
``socket.create_connection``,
and **asyncio** has ``loop.create_connection`` and ``asyncio.open_connection``.
By default,
these methods have the same behavior when a host name resolves to several IP
addresses: they try to connect to the first address in the list,
and only after the attempt fails (which may take tens of seconds) will
the second one be tried, and so on. In contrast, the Happy Eyeballs algorithm
will start an attempt with the second IP address in parallel to the first one
hasn't completed after some time, typically around 300 milliseconds.
As a result several attempts may be in flight at the same time, and whenever
one of the attempts succeed, all other connections are cancelled, and the
winning connection is used.
This means a much shorter wait before one of the IP addresses connect
successfully.

Happy Eyeballs is particularly important for dual-stack clients, when some hosts
may have resolvable IPv6 addresses that are somehow unreachable.

Starting from Python 3.8, stock **asyncio** also supports Happy Eyeballs.
See below for a comparison.


What does ``async_stagger`` has to offer?
-----------------------------------------

``async_stagger`` provides ``open_connection`` and
``create_connection`` with Happy Eyeballs support. They are mostly drop-in
replacements for their ``asyncio`` counterparts, and support most existing
arguments.
(There are small differences: ``create_connection`` takes
a *loop* argument instead of being a method on an event loop.
Also, these two methods do not support the *sock* argument.)
Another public coroutine ``create_connected_sock`` returns a connected
``socket.socket`` object.
Check the documentation for details.

These methods implements many features specified in :rfc:`8305` Happy Eyeballs
v2, which extends and obsoletes :rfc:`6555`. In particular, asynchronous
address resolution, destination address interleaving by family and staggered
connection attempts are implemented.


Happy Eyeballs sounds great! I want to use similar logic somewhere else!
------------------------------------------------------------------------

You're in luck! ``async_stagger`` actually exposes the underlying scheduling
logic as a reusable block: ``staggered_race``. It can be use when:

* There are several ways to achieve one goal. Some of the ways may fail, but
  you have to try it to find out.

* Making attempts strictly in sequence is too slow.

* You want to parallelize, but also don't want to start the attempts all
  at the same time. Maybe you want to give preference to some of the attempts,
  so they should be started earlier and given more time to complete. Maybe you
  want to avoid straining the system with simultaneous attempts.

* An attempt done half-way can be rolled back safely.


Where can I get it?
-------------------

``async_stagger`` requires Python 3.11 or later from v0.4.0 onwards.
Please use v0.3.1 for Python 3.6 - 3.10.
It does not have any external dependencies.
Install it from PyPI the usual way::

    pip install async-stagger

The documentation can be found here:
http://async-stagger.readthedocs.io/en/latest/

This project is under active development, and APIs may change in the future.
Check out the Changelog in the documentation.

This project is licensed under the MIT license.


Python 3.8 Has Native Happy Eyeballs Now
========================================

I contributed an implementation of Happy Eyeballs to upstream **asyncio**,
and it landed in Python 3.8: see `the docs`__ for details.

__ https://docs.python.org/3/library/asyncio-eventloop.html#asyncio.loop.create_connection

That implementation is essentially an early version of this package,
so it lacks these features:

* Async address resolution
* Detailed exception report
* The ``local_addrs`` argument (as opposed to ``local_addr``)

Still, it should be sufficient for most scenarios, and it's right there in the standard library.


Miscellaneous Remarks
=====================

Asynchronous address resolution is added in v0.2.1. With that, I feel that
the package should be fairly feature-complete.

I have implemented Happy Eyeballs-like algorithms in some of my other projects,
and this module reflects the things I have learned. However I have yet to
eat my own dog food and actually import this module from those other projects.
I would love to hear people's experience using this module in real world
conditions.


Acknowledgments
===============

The Happy Eyeballs scheduling algorithm implementation is inspired by
`the implementation in trio`__.

__ https://github.com/python-trio/trio/pull/145/files


            

Raw data

            {
    "_id": null,
    "home_page": "https://github.com/twisteroidambassador/async_stagger",
    "name": "async-stagger",
    "maintainer": null,
    "docs_url": null,
    "requires_python": ">=3.11",
    "maintainer_email": null,
    "keywords": "happy-eyeballs dual-stack tcp",
    "author": "twisteroid ambassador",
    "author_email": "twisteroid.ambassador@gmail.com",
    "download_url": "https://files.pythonhosted.org/packages/43/4e/69a1be89fe93020650d4b715ab7809da20b55af439f1f3b3b34d7a318d92/async_stagger-0.4.0.post1.tar.gz",
    "platform": null,
    "description": "async_stagger: Happy Eyeballs in ``asyncio``\r\n############################################\r\n\r\nQuick, what's the situation?\r\n============================\r\n\r\nTo get all the benefits of Happy Eyeballs connection establishment algorithm,\r\nsimply use ``async_stagger.open_connection`` like you would use\r\n``asyncio.open_connection``::\r\n\r\n    reader, writer = await async_stagger.open_connection('www.example.com', 80)\r\n\r\nNow your connections are more dual-stack friendly and will complete faster!\r\nA replacement for ``loop.create_connection`` is also provided.\r\n\r\nThe long version\r\n================\r\n\r\nWhat is Happy Eyeballs, and why should I use it?\r\n------------------------------------------------\r\n\r\nHappy Eyeballs is an algorithm for establishing TCP connections to destinations\r\nspecified by host names. It is described in :rfc:`6555` and :rfc:`8305`. The\r\nprimary benefit is that when host name resolution returns multiple addresses,\r\nand some of the address are unreachable, Happy Eyeballs will establish the\r\nconnection much faster than conventional algorithms. For more information,\r\ncheck the `Wikipedia article on Happy Eyeballs`_.\r\n\r\n.. _Wikipedia article on Happy Eyeballs: https://en.wikipedia.org/wiki/Happy_Eyeballs\r\n\r\nPython's standard library provides several high-level methods of establishing\r\nTCP connections towards a host name: The **socket** module has\r\n``socket.create_connection``,\r\nand **asyncio** has ``loop.create_connection`` and ``asyncio.open_connection``.\r\nBy default,\r\nthese methods have the same behavior when a host name resolves to several IP\r\naddresses: they try to connect to the first address in the list,\r\nand only after the attempt fails (which may take tens of seconds) will\r\nthe second one be tried, and so on. In contrast, the Happy Eyeballs algorithm\r\nwill start an attempt with the second IP address in parallel to the first one\r\nhasn't completed after some time, typically around 300 milliseconds.\r\nAs a result several attempts may be in flight at the same time, and whenever\r\none of the attempts succeed, all other connections are cancelled, and the\r\nwinning connection is used.\r\nThis means a much shorter wait before one of the IP addresses connect\r\nsuccessfully.\r\n\r\nHappy Eyeballs is particularly important for dual-stack clients, when some hosts\r\nmay have resolvable IPv6 addresses that are somehow unreachable.\r\n\r\nStarting from Python 3.8, stock **asyncio** also supports Happy Eyeballs.\r\nSee below for a comparison.\r\n\r\n\r\nWhat does ``async_stagger`` has to offer?\r\n-----------------------------------------\r\n\r\n``async_stagger`` provides ``open_connection`` and\r\n``create_connection`` with Happy Eyeballs support. They are mostly drop-in\r\nreplacements for their ``asyncio`` counterparts, and support most existing\r\narguments.\r\n(There are small differences: ``create_connection`` takes\r\na *loop* argument instead of being a method on an event loop.\r\nAlso, these two methods do not support the *sock* argument.)\r\nAnother public coroutine ``create_connected_sock`` returns a connected\r\n``socket.socket`` object.\r\nCheck the documentation for details.\r\n\r\nThese methods implements many features specified in :rfc:`8305` Happy Eyeballs\r\nv2, which extends and obsoletes :rfc:`6555`. In particular, asynchronous\r\naddress resolution, destination address interleaving by family and staggered\r\nconnection attempts are implemented.\r\n\r\n\r\nHappy Eyeballs sounds great! I want to use similar logic somewhere else!\r\n------------------------------------------------------------------------\r\n\r\nYou're in luck! ``async_stagger`` actually exposes the underlying scheduling\r\nlogic as a reusable block: ``staggered_race``. It can be use when:\r\n\r\n* There are several ways to achieve one goal. Some of the ways may fail, but\r\n  you have to try it to find out.\r\n\r\n* Making attempts strictly in sequence is too slow.\r\n\r\n* You want to parallelize, but also don't want to start the attempts all\r\n  at the same time. Maybe you want to give preference to some of the attempts,\r\n  so they should be started earlier and given more time to complete. Maybe you\r\n  want to avoid straining the system with simultaneous attempts.\r\n\r\n* An attempt done half-way can be rolled back safely.\r\n\r\n\r\nWhere can I get it?\r\n-------------------\r\n\r\n``async_stagger`` requires Python 3.11 or later from v0.4.0 onwards.\r\nPlease use v0.3.1 for Python 3.6 - 3.10.\r\nIt does not have any external dependencies.\r\nInstall it from PyPI the usual way::\r\n\r\n    pip install async-stagger\r\n\r\nThe documentation can be found here:\r\nhttp://async-stagger.readthedocs.io/en/latest/\r\n\r\nThis project is under active development, and APIs may change in the future.\r\nCheck out the Changelog in the documentation.\r\n\r\nThis project is licensed under the MIT license.\r\n\r\n\r\nPython 3.8 Has Native Happy Eyeballs Now\r\n========================================\r\n\r\nI contributed an implementation of Happy Eyeballs to upstream **asyncio**,\r\nand it landed in Python 3.8: see `the docs`__ for details.\r\n\r\n__ https://docs.python.org/3/library/asyncio-eventloop.html#asyncio.loop.create_connection\r\n\r\nThat implementation is essentially an early version of this package,\r\nso it lacks these features:\r\n\r\n* Async address resolution\r\n* Detailed exception report\r\n* The ``local_addrs`` argument (as opposed to ``local_addr``)\r\n\r\nStill, it should be sufficient for most scenarios, and it's right there in the standard library.\r\n\r\n\r\nMiscellaneous Remarks\r\n=====================\r\n\r\nAsynchronous address resolution is added in v0.2.1. With that, I feel that\r\nthe package should be fairly feature-complete.\r\n\r\nI have implemented Happy Eyeballs-like algorithms in some of my other projects,\r\nand this module reflects the things I have learned. However I have yet to\r\neat my own dog food and actually import this module from those other projects.\r\nI would love to hear people's experience using this module in real world\r\nconditions.\r\n\r\n\r\nAcknowledgments\r\n===============\r\n\r\nThe Happy Eyeballs scheduling algorithm implementation is inspired by\r\n`the implementation in trio`__.\r\n\r\n__ https://github.com/python-trio/trio/pull/145/files\r\n\r\n",
    "bugtrack_url": null,
    "license": "MIT",
    "summary": "Happy eyeballs and underlying scheduling algorithm in asyncio",
    "version": "0.4.0.post1",
    "project_urls": {
        "Documentation": "https://async-stagger.readthedocs.io",
        "Homepage": "https://github.com/twisteroidambassador/async_stagger"
    },
    "split_keywords": [
        "happy-eyeballs",
        "dual-stack",
        "tcp"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "d435bcc9d235c5f4188e53074911c45ea3e65c3f0492fe8ee26b8c088b94aaeb",
                "md5": "79ecbe04335a156834dda4dab481febc",
                "sha256": "65658cc5743e1c9b991562e1c3b3a33eb1fdd78084a39df3b7e287fd417a9dcb"
            },
            "downloads": -1,
            "filename": "async_stagger-0.4.0.post1-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "79ecbe04335a156834dda4dab481febc",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.11",
            "size": 23640,
            "upload_time": "2024-09-02T12:15:00",
            "upload_time_iso_8601": "2024-09-02T12:15:00.593360Z",
            "url": "https://files.pythonhosted.org/packages/d4/35/bcc9d235c5f4188e53074911c45ea3e65c3f0492fe8ee26b8c088b94aaeb/async_stagger-0.4.0.post1-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "434e69a1be89fe93020650d4b715ab7809da20b55af439f1f3b3b34d7a318d92",
                "md5": "3997b67aef97e0c7b02ea8cef52ab4df",
                "sha256": "b378c8db915b2851fbb1fd0a89a5c8b2178b9cd62f3171c473e47a7f43cbac5c"
            },
            "downloads": -1,
            "filename": "async_stagger-0.4.0.post1.tar.gz",
            "has_sig": false,
            "md5_digest": "3997b67aef97e0c7b02ea8cef52ab4df",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.11",
            "size": 21932,
            "upload_time": "2024-09-02T12:15:02",
            "upload_time_iso_8601": "2024-09-02T12:15:02.278305Z",
            "url": "https://files.pythonhosted.org/packages/43/4e/69a1be89fe93020650d4b715ab7809da20b55af439f1f3b3b34d7a318d92/async_stagger-0.4.0.post1.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-09-02 12:15:02",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "twisteroidambassador",
    "github_project": "async_stagger",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": false,
    "lcname": "async-stagger"
}
        
Elapsed time: 0.33580s