# universalasync
[![CircleCI](https://circleci.com/gh/bitcart/universalasync.svg?style=svg)](https://circleci.com/gh/bitcart/universalasync)
[![Codecov](https://img.shields.io/codecov/c/github/bitcart/universalasync?style=flat-square)](https://codecov.io/gh/bitcart/universalasync)
[![PyPI version](https://img.shields.io/pypi/v/universalasync.svg?style=flat-square)](https://pypi.python.org/pypi/universalasync/)
[![Read the Docs](https://img.shields.io/readthedocs/universalasync?style=flat-square)](https://universalasync.bitcart.ai)
A library to help automate the creation of universal python libraries
## Overview
Have you ever been frustrated that you need to maintain both sync and async versions of your library, even thought their code differs by just async and await?
You might have came up to rewriting your code before release or other unreliable solutions.
This library helps you to focus only on the main async implementation of your library: sync one will be created automatically
Via decorating all your public methods, the wrapped functions automatically detect different conditions and run the functions accordingly.
If user uses your library in async context, minimal overhead is added, it just returns the coroutine right away.
Otherwise the library calls the coroutine via various loop methods, like as if you did it manually.
There should be no issues at all, the only limitation is that signals and async subprocesses are supported only when running in the main thread.
Also note that when run from a different os thread, the library will create a new event loop there and run coroutines.
This means that you might need to adapt your code a bit in case you use some resources bound to a certain event loop (like `aiohttp.ClientSession`).
You can see an example of how this could be solved [here](https://github.com/bitcart/bitcart-sdk/blob/4a425f80f62a0c90f8c5fa19ccb7e578590dcead/bitcart/providers/jsonrpcrequests.py#L51-L58)
For API reference see [the docs](https://universalasync.bitcart.ai)
## Installation
`pip install universalasync`
## Example of usage
```python
# wrap needed methods one by one
class Client:
@async_to_sync_wraps
async def help():
...
@async_to_sync_wraps
@property
async def async_property():
...
# or wrap whole classes
@wrap
class Client:
async def help(self):
...
@property
async def async_property():
...
client = Client()
def sync_call():
client.help()
client.async_property
async def async_call():
await client.help()
await client.async_property
# works in all cases
sync_call()
asyncio.run(async_call())
threading.Thread(target=sync_call).start()
threading.Thread(target=asyncio.run, args=(async_call(),)).start()
```
## Copyright and License
Copyright (C) 2021 MrNaif2018
Universal async-sync wrapper initially based on [this implementation](https://github.com/pyrogram/pyrogram/blob/master/pyrogram/sync.py)
Licensed under the [MIT license](LICENSE)
Raw data
{
"_id": null,
"home_page": null,
"name": "universalasync",
"maintainer": null,
"docs_url": null,
"requires_python": ">=3.9",
"maintainer_email": null,
"keywords": "async, asyncio, asynctosync, await, bitcart, sync, synctoasync, universal",
"author": null,
"author_email": "MrNaif2018 <chuff184@gmail.com>",
"download_url": "https://files.pythonhosted.org/packages/0e/b0/4f9831dbd39b82190cc7eb9d3b89e37dc7c4ecb16f7e4a2ed2dbd38c647c/universalasync-0.4.0.1.tar.gz",
"platform": null,
"description": "# universalasync\n\n[![CircleCI](https://circleci.com/gh/bitcart/universalasync.svg?style=svg)](https://circleci.com/gh/bitcart/universalasync)\n[![Codecov](https://img.shields.io/codecov/c/github/bitcart/universalasync?style=flat-square)](https://codecov.io/gh/bitcart/universalasync)\n[![PyPI version](https://img.shields.io/pypi/v/universalasync.svg?style=flat-square)](https://pypi.python.org/pypi/universalasync/)\n[![Read the Docs](https://img.shields.io/readthedocs/universalasync?style=flat-square)](https://universalasync.bitcart.ai)\n\nA library to help automate the creation of universal python libraries\n\n## Overview\n\nHave you ever been frustrated that you need to maintain both sync and async versions of your library, even thought their code differs by just async and await?\nYou might have came up to rewriting your code before release or other unreliable solutions.\n\nThis library helps you to focus only on the main async implementation of your library: sync one will be created automatically\n\nVia decorating all your public methods, the wrapped functions automatically detect different conditions and run the functions accordingly.\n\nIf user uses your library in async context, minimal overhead is added, it just returns the coroutine right away.\n\nOtherwise the library calls the coroutine via various loop methods, like as if you did it manually.\n\nThere should be no issues at all, the only limitation is that signals and async subprocesses are supported only when running in the main thread.\n\nAlso note that when run from a different os thread, the library will create a new event loop there and run coroutines.\n\nThis means that you might need to adapt your code a bit in case you use some resources bound to a certain event loop (like `aiohttp.ClientSession`).\n\nYou can see an example of how this could be solved [here](https://github.com/bitcart/bitcart-sdk/blob/4a425f80f62a0c90f8c5fa19ccb7e578590dcead/bitcart/providers/jsonrpcrequests.py#L51-L58)\n\nFor API reference see [the docs](https://universalasync.bitcart.ai)\n\n## Installation\n\n`pip install universalasync`\n\n## Example of usage\n\n```python\n# wrap needed methods one by one\nclass Client:\n @async_to_sync_wraps\n async def help():\n ...\n\n @async_to_sync_wraps\n @property\n async def async_property():\n ...\n\n# or wrap whole classes\n@wrap\nclass Client:\n async def help(self):\n ...\n\n @property\n async def async_property():\n ...\n\nclient = Client()\n\ndef sync_call():\n client.help()\n client.async_property\n\nasync def async_call():\n await client.help()\n await client.async_property\n\n# works in all cases\nsync_call()\nasyncio.run(async_call())\nthreading.Thread(target=sync_call).start()\nthreading.Thread(target=asyncio.run, args=(async_call(),)).start()\n```\n\n## Copyright and License\n\nCopyright (C) 2021 MrNaif2018\n\nUniversal async-sync wrapper initially based on [this implementation](https://github.com/pyrogram/pyrogram/blob/master/pyrogram/sync.py)\n\nLicensed under the [MIT license](LICENSE)\n",
"bugtrack_url": null,
"license": null,
"summary": "A library to help automate the creation of universal python libraries",
"version": "0.4.0.1",
"project_urls": {
"Changelog": "https://github.com/bitcart/universalasync/blob/master/CHANGELOG.md",
"Homepage": "https://github.com/bitcart/universalasync",
"Source": "https://github.com/bitcart/universalasync"
},
"split_keywords": [
"async",
" asyncio",
" asynctosync",
" await",
" bitcart",
" sync",
" synctoasync",
" universal"
],
"urls": [
{
"comment_text": null,
"digests": {
"blake2b_256": "7970b927626118b4571c1b3b494a33372d37c3d264f2c62e0c3284b0ff5d3866",
"md5": "a37bc89ed31661982aab6e5deae3395f",
"sha256": "55691b23f8265ff9a658d5eaee5a0375d75ac46432862198178c9a360062ba52"
},
"downloads": -1,
"filename": "universalasync-0.4.0.1-py3-none-any.whl",
"has_sig": false,
"md5_digest": "a37bc89ed31661982aab6e5deae3395f",
"packagetype": "bdist_wheel",
"python_version": "py3",
"requires_python": ">=3.9",
"size": 5873,
"upload_time": "2024-12-16T15:28:10",
"upload_time_iso_8601": "2024-12-16T15:28:10.829683Z",
"url": "https://files.pythonhosted.org/packages/79/70/b927626118b4571c1b3b494a33372d37c3d264f2c62e0c3284b0ff5d3866/universalasync-0.4.0.1-py3-none-any.whl",
"yanked": false,
"yanked_reason": null
},
{
"comment_text": null,
"digests": {
"blake2b_256": "0eb04f9831dbd39b82190cc7eb9d3b89e37dc7c4ecb16f7e4a2ed2dbd38c647c",
"md5": "871a2abf8f8047be50cbad9431b0ece3",
"sha256": "aadf1f20170366b76fb0317a343775c403842c8db25062f7d68184a09b3e882c"
},
"downloads": -1,
"filename": "universalasync-0.4.0.1.tar.gz",
"has_sig": false,
"md5_digest": "871a2abf8f8047be50cbad9431b0ece3",
"packagetype": "sdist",
"python_version": "source",
"requires_python": ">=3.9",
"size": 58741,
"upload_time": "2024-12-16T15:28:12",
"upload_time_iso_8601": "2024-12-16T15:28:12.908360Z",
"url": "https://files.pythonhosted.org/packages/0e/b0/4f9831dbd39b82190cc7eb9d3b89e37dc7c4ecb16f7e4a2ed2dbd38c647c/universalasync-0.4.0.1.tar.gz",
"yanked": false,
"yanked_reason": null
}
],
"upload_time": "2024-12-16 15:28:12",
"github": true,
"gitlab": false,
"bitbucket": false,
"codeberg": false,
"github_user": "bitcart",
"github_project": "universalasync",
"travis_ci": false,
"coveralls": false,
"github_actions": false,
"circle": true,
"lcname": "universalasync"
}