About
=====
.. _imaplib2: https://sourceforge.net/projects/imaplib2/
.. _imaplib: https://docs.python.org/3/library/imaplib.html
.. _asyncio: https://docs.python.org/3/library/asyncio.html
.. image:: https://github.com/bamthomas/aioimaplib/workflows/tests/badge.svg
:alt: Build status
:target: https://github.com/bamthomas/aioimaplib/actions/
.. image:: https://coveralls.io/repos/github/bamthomas/aioimaplib/badge.svg
:target: https://coveralls.io/github/bamthomas/aioimaplib
This library is inspired by imaplib_ and imaplib2_ from Piers Lauder, Nicolas Sebrecht, Sebastian Spaeth. Some utilities functions are taken from imaplib/imaplib2 thanks to them.
Thank you to all contributors for your time and interest :pray: :point_right:
The aim is to port the imaplib with asyncio_, to benefit from the sleep or treat model. It has no dependencies for runtime (the core library).
It is tested against python 3.9, 3.10, 3.11 matrix. But the library itself *could* run with other python versions.
Example
-------
.. code-block:: python
import asyncio
from aioimaplib import aioimaplib
async def check_mailbox(host, user, password):
imap_client = aioimaplib.IMAP4_SSL(host=host)
await imap_client.wait_hello_from_server()
await imap_client.login(user, password)
res, data = await imap_client.select()
print('there is %s messages INBOX' % data[0])
await imap_client.logout()
if __name__ == '__main__':
loop = asyncio.get_event_loop()
loop.run_until_complete(check_mailbox('my.imap.server', 'user', 'pass'))
**Beware** that the IMAP4.close() function is an IMAP function that is closing the selected mailbox, thus passing from SELECTED state to AUTH state. It **does not close** the TCP connection.
The way to close TCP connection properly is to logout.
IDLE command
------------
.. _RFC2177: https://tools.ietf.org/html/rfc2177
The RFC2177_ is implemented, to be able to wait for new mail messages without using CPU. The responses are pushed in an async queue, and it is possible to read them in real time. To leave the IDLE mode, it is necessary to send a "DONE" command to the server.
.. code-block:: python
async def wait_for_new_message(host, user, password):
imap_client = aioimaplib.IMAP4_SSL(host=host)
await imap_client.wait_hello_from_server()
await imap_client.login(user, password)
await imap_client.select()
idle = await imap_client.idle_start(timeout=10)
while imap_client.has_pending_idle():
msg = await imap_client.wait_server_push()
print(msg)
if msg == STOP_WAIT_SERVER_PUSH:
imap_client.idle_done()
await asyncio.wait_for(idle, 1)
await imap_client.logout()
if __name__ == '__main__':
loop = asyncio.get_event_loop()
loop.run_until_complete(wait_for_new_message('my.imap.server', 'user', 'pass'))
Or in a more event based style (the IDLE command is closed at each message from server):
.. code-block:: python
async def idle_loop(host, user, password):
imap_client = aioimaplib.IMAP4_SSL(host=host, timeout=30)
await imap_client.wait_hello_from_server()
await imap_client.login(user, password)
await imap_client.select()
while True:
print((await imap_client.uid('fetch', '1:*', 'FLAGS')))
idle = await imap_client.idle_start(timeout=60)
print((await imap_client.wait_server_push()))
imap_client.idle_done()
await asyncio.wait_for(idle, 30)
Threading
---------
.. _asyncio.Event: https://docs.python.org/3.4/library/asyncio-sync.html#event
.. _asyncio.Condition: https://docs.python.org/3.4/library/asyncio-sync.html#condition
.. _supervisor: http://supervisord.org/
The IMAP4ClientProtocol class is not thread safe, it uses asyncio.Event_ and asyncio.Condition_ that are not thread safe, and state change for pending commands is not locked.
It is possible to use threads but each IMAP4ClientProtocol instance should run in the same thread:
.. image:: images/thread_imap_protocol.png
Each color rectangle is an IMAP4ClientProtocol instance piece of code executed by the thread asyncio loop until it reaches a yield, waiting on I/O.
For example, it is possible to launch 4 mono-threaded mail-fetcher processes on a 4 cores server with supervisor_, and use a distribution function like len(email) % (process_num) or whatever to share equally a mail account list between the 4 processes.
IMAP command concurrency
------------------------
IMAP protocol allows to run some commands in parallel. Four rules are implemented to ensure responses consistency:
1. if a sync command is running, the following requests (sync or async) must wait
2. if an async command is running, same async commands (or with the same untagged response type) must wait
3. async commands can be executed in parallel
4. sync command must wait pending async commands to finish
Logging
-------
.. _howto: https://docs.python.org/3.4/howto/logging.html#configuring-logging-for-a-library
As said in the logging howto_ the logger is defined with
.. code-block:: python
logger = logging.getLogger(__name__)
Where name is 'aioimaplib.aioimaplib'. You can set the logger parameters, either by python API
.. code-block:: python
aioimaplib_logger = logging.getLogger('aioimaplib.aioimaplib')
sh = logging.StreamHandler()
sh.setLevel(logging.DEBUG)
sh.setFormatter(logging.Formatter("%(asctime)s %(levelname)s [%(module)s:%(lineno)d] %(message)s"))
aioimaplib_logger.addHandler(sh)
Or loading config file (for example with logging.config.dictConfig(yaml.load(file))) with this piece of yaml file
.. code-block:: yaml
loggers:
...
aioimaplib.aioimaplib:
level: DEBUG
handlers: [syslog]
propagate: no
...
Authentication with OAuth2
--------------------------
Starting with the 01/01/23 Microsoft Outlook can only be accessed with OAuth2.
You need to register you client to be used with oauth. Find more
:https://learn.microsoft.com/en-us/exchange/client-developer/legacy-protocols/how-to-authenticate-an-imap-pop-smtp-application-by-using-oauth:`here`.
This might be also used with Google Mail, but it is not tested for it.
Tested with
-----------
- dovecot 2.2.13 on debian Jessie
- gmail with imap and SSL
- outlook with SSL
- yahoo with SSL
- free.fr with SSL
- orange.fr with SSL
- mailden.net with SSL
Develop
=======
.. _poetry: https://python-poetry.org/
Developers are welcome! If you want to improve it, fix bugs, test it with other IMAP servers, give feedback, thank you for it.
We use poetry_ for building the library. Just run
.. code-block:: bash
poetry install
poetry run pytest
# or you can create a poetry shell
poetry install
poetry shell
pytest
To add an imaplib or imaplib2 command you can :
- add the function to the testing imapserver with a new imaplib or imaplib2 server test, i.e. test_imapserver_imaplib.py or test_imapserver_imaplib2.py respectively;
- then add the function to the aioimaplib doing almost the same test than above but the async way in test_aioimaplib.py.
Not unit tested
---------------
- PREAUTH
TODO
----
.. _rfc3501: https://tools.ietf.org/html/rfc3501
.. _rfc4978: https://tools.ietf.org/html/rfc4978
.. _rfc4314: https://tools.ietf.org/html/rfc4314
.. _rfc2087: https://tools.ietf.org/html/rfc2087
.. _rfc5256: https://tools.ietf.org/html/rfc5256
.. _rfc2971: https://tools.ietf.org/html/rfc2971
.. _rfc2342: https://tools.ietf.org/html/rfc2342
.. _rfc4469: https://tools.ietf.org/html/rfc4469
- 23/25 IMAP4rev1 commands are implemented from the main rfc3501_. 'STARTTLS' and 'AUTHENTICATE'(except with XOAUTH2) are still missing.
- 'COMPRESS' from rfc4978_
- 'SETACL' 'DELETEACL' 'GETACL' 'MYRIGHTS' 'LISTRIGHTS' from ACL rfc4314_
- 'GETQUOTA': 'GETQUOTAROOT': 'SETQUOTA' from quota rfc2087_
- 'SORT' and 'THREAD' from the rfc5256_
- 'ID' from the rfc2971_
- 'NAMESPACE' from rfc2342_
- 'CATENATE' from rfc4469_
- tests with other servers
If it goes wrong
----------------
Sometimes you break things and you don't understand what's going on (I always do). For this library I have two related tools:
.. role:: bash(code)
:language: bash
- ngrep on the imap test port: :bash:`sudo ngrep -d lo port 12345`
- activate debug logs changing INFO to DEBUG at the top of the mock server and the aioimaplib
Raw data
{
"_id": null,
"home_page": "https://github.com/bamthomas/aioimaplib",
"name": "aioimaplib",
"maintainer": null,
"docs_url": null,
"requires_python": "<4.0,>=3.9",
"maintainer_email": null,
"keywords": "asyncio, mail, imap",
"author": "Bruno THOMAS",
"author_email": "bthomas@iroco.fr",
"download_url": "https://files.pythonhosted.org/packages/ee/da/a454c47fb8522e607425e15bf1f49ccfdb3d75f4071f40b63ebd49573495/aioimaplib-2.0.1.tar.gz",
"platform": null,
"description": "About\n=====\n.. _imaplib2: https://sourceforge.net/projects/imaplib2/\n.. _imaplib: https://docs.python.org/3/library/imaplib.html\n.. _asyncio: https://docs.python.org/3/library/asyncio.html\n\n.. image:: https://github.com/bamthomas/aioimaplib/workflows/tests/badge.svg\n :alt: Build status\n :target: https://github.com/bamthomas/aioimaplib/actions/\n\n.. image:: https://coveralls.io/repos/github/bamthomas/aioimaplib/badge.svg\n :target: https://coveralls.io/github/bamthomas/aioimaplib\n\n\nThis library is inspired by imaplib_ and imaplib2_ from Piers Lauder, Nicolas Sebrecht, Sebastian Spaeth. Some utilities functions are taken from imaplib/imaplib2 thanks to them.\n\nThank you to all contributors for your time and interest :pray: :point_right:\n\nThe aim is to port the imaplib with asyncio_, to benefit from the sleep or treat model. It has no dependencies for runtime (the core library).\n\nIt is tested against python 3.9, 3.10, 3.11 matrix. But the library itself *could* run with other python versions.\n\nExample\n-------\n\n.. code-block:: python\n\n import asyncio\n from aioimaplib import aioimaplib\n\n\n async def check_mailbox(host, user, password):\n imap_client = aioimaplib.IMAP4_SSL(host=host)\n await imap_client.wait_hello_from_server()\n\n await imap_client.login(user, password)\n\n res, data = await imap_client.select()\n print('there is %s messages INBOX' % data[0])\n\n await imap_client.logout()\n\n\n if __name__ == '__main__':\n loop = asyncio.get_event_loop()\n loop.run_until_complete(check_mailbox('my.imap.server', 'user', 'pass'))\n\n**Beware** that the IMAP4.close() function is an IMAP function that is closing the selected mailbox, thus passing from SELECTED state to AUTH state. It **does not close** the TCP connection.\nThe way to close TCP connection properly is to logout.\n\nIDLE command\n------------\n.. _RFC2177: https://tools.ietf.org/html/rfc2177\n\nThe RFC2177_ is implemented, to be able to wait for new mail messages without using CPU. The responses are pushed in an async queue, and it is possible to read them in real time. To leave the IDLE mode, it is necessary to send a \"DONE\" command to the server.\n\n.. code-block:: python\n\n async def wait_for_new_message(host, user, password):\n imap_client = aioimaplib.IMAP4_SSL(host=host)\n await imap_client.wait_hello_from_server()\n\n await imap_client.login(user, password)\n await imap_client.select()\n\n idle = await imap_client.idle_start(timeout=10)\n while imap_client.has_pending_idle():\n msg = await imap_client.wait_server_push()\n print(msg)\n if msg == STOP_WAIT_SERVER_PUSH:\n imap_client.idle_done()\n await asyncio.wait_for(idle, 1)\n\n await imap_client.logout()\n\n if __name__ == '__main__':\n loop = asyncio.get_event_loop()\n loop.run_until_complete(wait_for_new_message('my.imap.server', 'user', 'pass'))\n\nOr in a more event based style (the IDLE command is closed at each message from server):\n\n.. code-block:: python\n\n async def idle_loop(host, user, password):\n imap_client = aioimaplib.IMAP4_SSL(host=host, timeout=30)\n await imap_client.wait_hello_from_server()\n\n await imap_client.login(user, password)\n await imap_client.select()\n\n while True:\n print((await imap_client.uid('fetch', '1:*', 'FLAGS')))\n\n idle = await imap_client.idle_start(timeout=60)\n print((await imap_client.wait_server_push()))\n\n imap_client.idle_done()\n await asyncio.wait_for(idle, 30)\n\nThreading\n---------\n.. _asyncio.Event: https://docs.python.org/3.4/library/asyncio-sync.html#event\n.. _asyncio.Condition: https://docs.python.org/3.4/library/asyncio-sync.html#condition\n.. _supervisor: http://supervisord.org/\n\nThe IMAP4ClientProtocol class is not thread safe, it uses asyncio.Event_ and asyncio.Condition_ that are not thread safe, and state change for pending commands is not locked.\n\nIt is possible to use threads but each IMAP4ClientProtocol instance should run in the same thread:\n\n.. image:: images/thread_imap_protocol.png\n\nEach color rectangle is an IMAP4ClientProtocol instance piece of code executed by the thread asyncio loop until it reaches a yield, waiting on I/O.\n\nFor example, it is possible to launch 4 mono-threaded mail-fetcher processes on a 4 cores server with supervisor_, and use a distribution function like len(email) % (process_num) or whatever to share equally a mail account list between the 4 processes.\n\nIMAP command concurrency\n------------------------\n\nIMAP protocol allows to run some commands in parallel. Four rules are implemented to ensure responses consistency:\n\n1. if a sync command is running, the following requests (sync or async) must wait\n2. if an async command is running, same async commands (or with the same untagged response type) must wait\n3. async commands can be executed in parallel\n4. sync command must wait pending async commands to finish\n\nLogging\n-------\n.. _howto: https://docs.python.org/3.4/howto/logging.html#configuring-logging-for-a-library\n\nAs said in the logging howto_ the logger is defined with\n\n.. code-block:: python\n\n logger = logging.getLogger(__name__)\n\n\nWhere name is 'aioimaplib.aioimaplib'. You can set the logger parameters, either by python API\n\n.. code-block:: python\n\n aioimaplib_logger = logging.getLogger('aioimaplib.aioimaplib')\n sh = logging.StreamHandler()\n sh.setLevel(logging.DEBUG)\n sh.setFormatter(logging.Formatter(\"%(asctime)s %(levelname)s [%(module)s:%(lineno)d] %(message)s\"))\n aioimaplib_logger.addHandler(sh)\n\nOr loading config file (for example with logging.config.dictConfig(yaml.load(file))) with this piece of yaml file\n\n.. code-block:: yaml\n\n loggers:\n ...\n aioimaplib.aioimaplib:\n level: DEBUG\n handlers: [syslog]\n propagate: no\n ...\n\nAuthentication with OAuth2\n--------------------------\n\nStarting with the 01/01/23 Microsoft Outlook can only be accessed with OAuth2. \nYou need to register you client to be used with oauth. Find more \n:https://learn.microsoft.com/en-us/exchange/client-developer/legacy-protocols/how-to-authenticate-an-imap-pop-smtp-application-by-using-oauth:`here`.\n\nThis might be also used with Google Mail, but it is not tested for it.\n\n\nTested with\n-----------\n\n- dovecot 2.2.13 on debian Jessie\n- gmail with imap and SSL\n- outlook with SSL\n- yahoo with SSL\n- free.fr with SSL\n- orange.fr with SSL\n- mailden.net with SSL\n\nDevelop\n=======\n.. _poetry: https://python-poetry.org/\n\nDevelopers are welcome! If you want to improve it, fix bugs, test it with other IMAP servers, give feedback, thank you for it.\n\nWe use poetry_ for building the library. Just run\n\n.. code-block:: bash\n\n poetry install\n poetry run pytest\n\n # or you can create a poetry shell\n poetry install\n poetry shell\n pytest\n\nTo add an imaplib or imaplib2 command you can :\n\n- add the function to the testing imapserver with a new imaplib or imaplib2 server test, i.e. test_imapserver_imaplib.py or test_imapserver_imaplib2.py respectively;\n- then add the function to the aioimaplib doing almost the same test than above but the async way in test_aioimaplib.py.\n\nNot unit tested\n---------------\n- PREAUTH\n\nTODO\n----\n.. _rfc3501: https://tools.ietf.org/html/rfc3501\n.. _rfc4978: https://tools.ietf.org/html/rfc4978\n.. _rfc4314: https://tools.ietf.org/html/rfc4314\n.. _rfc2087: https://tools.ietf.org/html/rfc2087\n.. _rfc5256: https://tools.ietf.org/html/rfc5256\n.. _rfc2971: https://tools.ietf.org/html/rfc2971\n.. _rfc2342: https://tools.ietf.org/html/rfc2342\n.. _rfc4469: https://tools.ietf.org/html/rfc4469\n\n- 23/25 IMAP4rev1 commands are implemented from the main rfc3501_. 'STARTTLS' and 'AUTHENTICATE'(except with XOAUTH2) are still missing.\n- 'COMPRESS' from rfc4978_\n- 'SETACL' 'DELETEACL' 'GETACL' 'MYRIGHTS' 'LISTRIGHTS' from ACL rfc4314_\n- 'GETQUOTA': 'GETQUOTAROOT': 'SETQUOTA' from quota rfc2087_\n- 'SORT' and 'THREAD' from the rfc5256_\n- 'ID' from the rfc2971_\n- 'NAMESPACE' from rfc2342_\n- 'CATENATE' from rfc4469_\n- tests with other servers\n\nIf it goes wrong\n----------------\nSometimes you break things and you don't understand what's going on (I always do). For this library I have two related tools:\n\n.. role:: bash(code)\n :language: bash\n\n- ngrep on the imap test port: :bash:`sudo ngrep -d lo port 12345`\n- activate debug logs changing INFO to DEBUG at the top of the mock server and the aioimaplib\n",
"bugtrack_url": null,
"license": "GPL-3.0",
"summary": "Python asyncio IMAP4rev1 client library",
"version": "2.0.1",
"project_urls": {
"Homepage": "https://github.com/bamthomas/aioimaplib",
"Repository": "https://github.com/bamthomas/aioimaplib"
},
"split_keywords": [
"asyncio",
" mail",
" imap"
],
"urls": [
{
"comment_text": "",
"digests": {
"blake2b_256": "135248aaa287fb3c4c995edcb602370b10d182dc5c48371df7cb3a404356733f",
"md5": "fe2bac1e7fc2e89bc04025ea7dd11020",
"sha256": "727e00c35cf25106bd34611dddd6e2ddf91a5f1a7e72d9269f3ce62486b31e14"
},
"downloads": -1,
"filename": "aioimaplib-2.0.1-py3-none-any.whl",
"has_sig": false,
"md5_digest": "fe2bac1e7fc2e89bc04025ea7dd11020",
"packagetype": "bdist_wheel",
"python_version": "py3",
"requires_python": "<4.0,>=3.9",
"size": 34729,
"upload_time": "2025-01-16T10:38:20",
"upload_time_iso_8601": "2025-01-16T10:38:20.427870Z",
"url": "https://files.pythonhosted.org/packages/13/52/48aaa287fb3c4c995edcb602370b10d182dc5c48371df7cb3a404356733f/aioimaplib-2.0.1-py3-none-any.whl",
"yanked": false,
"yanked_reason": null
},
{
"comment_text": "",
"digests": {
"blake2b_256": "eedaa454c47fb8522e607425e15bf1f49ccfdb3d75f4071f40b63ebd49573495",
"md5": "8a6b82f06bbc2a2f54b2160113b99d85",
"sha256": "5a494c3b75f220977048f5eb2c7ba9c0570a3148aaf38bee844e37e4d7af8648"
},
"downloads": -1,
"filename": "aioimaplib-2.0.1.tar.gz",
"has_sig": false,
"md5_digest": "8a6b82f06bbc2a2f54b2160113b99d85",
"packagetype": "sdist",
"python_version": "source",
"requires_python": "<4.0,>=3.9",
"size": 35555,
"upload_time": "2025-01-16T10:38:23",
"upload_time_iso_8601": "2025-01-16T10:38:23.140386Z",
"url": "https://files.pythonhosted.org/packages/ee/da/a454c47fb8522e607425e15bf1f49ccfdb3d75f4071f40b63ebd49573495/aioimaplib-2.0.1.tar.gz",
"yanked": false,
"yanked_reason": null
}
],
"upload_time": "2025-01-16 10:38:23",
"github": true,
"gitlab": false,
"bitbucket": false,
"codeberg": false,
"github_user": "bamthomas",
"github_project": "aioimaplib",
"travis_ci": false,
"coveralls": false,
"github_actions": true,
"lcname": "aioimaplib"
}