Metadata-Version: 2.1
Name: telegram-garnet
Version: 0.2.2
Summary: Garnet - cool add-on for Telethon
Home-page: https://github.com/uwinx/garnet
Keywords: telegram,telethon,telegram-bot,bot
Author: Martin Winks
Author-email: cat@snejugal.ru
Requires-Python: >=3.7,<4.0
Classifier: Framework :: AsyncIO
Classifier: Intended Audience :: Developers
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.7
Classifier: Programming Language :: Python :: 3.8
Provides-Extra: aiohttp
Provides-Extra: cryptg
Provides-Extra: hachoir
Provides-Extra: orjson
Provides-Extra: redis
Provides-Extra: ujson
Requires-Dist: Telethon (>=1.10.8,<2.0)
Project-URL: Repository, https://github.com/uwinx/garnet
Description-Content-Type: text/x-rst

🍷 Garnet
===================================

Garnet — bot-friendly telethon
-----------------------------------

.. invisible-content-till-nel
.. _aioredis: https://github.com/aio-libs/aioredis
.. _cryptg: https://pypi.org/project/cryptg/
.. _telethon: https://pypi.org/project/Telethon/
.. _orjson: https://pypi.org/project/orjson/
.. _ujson: https://pypi.org/project/ujson/
.. _hachoir: https://pypi.org/project/hachoir/
.. _aiohttp: https://pypi.org/project/aiohttp/
.. _Alex: https://github.com/JrooTJunior

.. image:: https://raw.githubusercontent.com/uwinx/garnet/master/static/pomegranate.jpg

.. image:: https://img.shields.io/badge/code%20style-black-000000.svg
    :target: https://github.com/python/black
    :alt: aioqiwi-code-style


Install::

    pip install telegram-garnet


**Dependencies:**
    - ``telethon`` - main dependency telethon_
**Extras:**
    - ``aioredis`` - redis driver if you use RedisStorage* aioredis_
    - ``orjson`` || ``ujson`` - RedisStorage/JSONStorage de-&& serialization orjson_ ujson_
    - ``cryptg``, ``hachoir``, ``aiohttp`` - boost telethon itself cryptg_ hachoir_ aiohttp_

---------------------------------
🌚 🌝 FSM-Storage types
---------------------------------

- File - json storage, the main idea behind JSON storage is a custom reload of file and usage of memory session for writing, so the data in json storage not always actual

- Memory - powerful in-memory map<str> based storage, only thing - not persistent

- Redis - (requires aioredis_) - redis is fast key-value storage, if you're using your data is durable and persistent


Pomegranate implements updates dispatching and checks callback's filters wrapping all callbacks into ``garnet::Callback`` object

----------------
😋 Filters
----------------

``Filter`` object is the essential part of Pomegranate, the do state checking and other stuff.

Basically, it's ``func`` from ``MyEventBuilder(func=lambda self: <bool>)`` but a way more complicated and not stored in EventBuilder, it's stored in callback object


Useful filters

1) 📨 **Messages**


`Following examples will include pattern`


.. code:: python

    from garnet import MessageText, TelegramClient
    bot = TelegramClient.from_env().start_as_bot()

    # // our code here //

    bot.run_until_disconnected()

.. code:: python

    # handling /start and /help
    @bot.on(MessageText.commands("help", "start"))
    async def cmd_handler(message: custom.Message):
        await message.reply("Hey there!")

    # handling exact words
    my_secret = "abcxyz"
    @bot.on(MessageText.exact(my_secret))
    async def secret_handler(message: custom.Message):
        await message.reply("Secret entered correctly! Welcome!")

MessageText or text(``from garnet import text``) includes following comparisons all returning <bool>
 - ``.exact(x)`` -> ``event.raw_text == x``
 - ``.commands(*x)`` -> ``event.raw_text.split()[0][1:] in x``
 - ``.match(x)`` -> ``re.compile(x).match(event.raw_text)``
 - ``.between(*x)`` -> ``event.raw_text in x``
 - ``.isdigit()`` -> ``(event.raw_text or "").isdigit()``
 - ``.startswith(x)`` -> ``event.raw_text.startswith(x)``



2) 👀 **CurrentState class**  [``from garnet import CurrentState``]

Once great minds decided that state checking will be in filters without adding ``state`` as handler decorator parameter and further storing state in ``callback.(arg)``
``CurrentState`` class methods return ``Filter``. There are two problems that Filter object really solves, ``Filter``'s function can be any kind of callable(async,sync), filters also have a flag ``requires_context``, FSMProxy is passed if true

See `FSM example <https://github.com/uwinx/garnet/blob/master/examples/fsm.py>`_ to understand how CurrentState works

Includes following methods all returning <bool>
 - ``.exact(x)`` or ``CurrentState == x`` -> ``await context.get_state() == x``
 - ``CurrentState == [x, y, z]`` -> ``await context.get_state() in [x, y, z]``
 - ``CurrentState == all`` or ``CurrentState == any`` -> ``await context.get_state() is not None``


3) 🦔 Custom **Filter**

If you want to write your own filter, do it.


.. code:: python

    from garnet import Filter, FSMProxy

    async def myFunc(event): ...
    async def myFuncContextRequires(event, context: FSMProxy): ...
    def normal_func(event): ...

    @bot.on(Filter(normal_func), Filter(myFunc), Filter(myFuncContextRequires, requires_context=True))
    async def handler(event, context: FSMProxy): ...
    # same as
    @bot.on(normal_func, myFunc, Filter(myFuncContextRequires, requires_context=True))
    async def handler(event): ...

So the handler can take strict ``context`` argument and also ignore it

-----------------------
On start|finish
-----------------------

``garnet::TelegramClient`` contains three lists on_start on_background and on_finish, their instance is ``PseudoFrozenList`` which freezes at calling ``.run_until_disconnected``
``PseudoFrozenList`` has three main methods::

    .append(*items)
    .remove(*items)
    .freeze()
    .__call__(func)   # for shiny decorator

``items`` in case of TelegramClient means unpacked container of async-defined functions taking on position arguments

Usage example:

.. code-block:: python

    # my_module.py
    class MyPostgresDatabase:
        ...
        async def close_pool(self, bot): await self.pool.safe_close()
        async def open_pool(self, bot): await self.pool.open_conn_pool()

    # garnethon.py
    from garnet import TelegramClient
    from my_module import MyPostgresDatabase

    db = MyPostgresDatabase()
    bot = TelegramClient.from_env().start_as_bot()
    bot.on_start.append(db.open_pool)
    bot.on_finish.append(db.close_pool)
    ...

    @bot.on_background
    async def xyz(cl: TelegramClient):
        while True:
           ...

    bot.run_until_connected()


-------------------------------------------------
📦 Router and Migrating to garnet using Router
-------------------------------------------------

Think of router as just a dummy container of handlers(callbacks)

`garnet::router::Router` may be helpful if you have telethon's `event.register` registered handlers. One thing: Router, I believe, is correct and more obvious way of registering event handlers. Example:

**Migrate from telethon to garnet, also use for bot.on cases(soon better example)**

.. code-block:: python

    # my_handlers.py

    # telethon register(bad) will raise Warning in garnet
    from telethon import events

    @events.register(event_type)
    async def handler(event): ...

    # garnet's telethon-like router
    from garnet.router import TelethonRouter

    router = TelethonRouter()

    @router(event_type)
    async def handler(event): ...



The advantage of routers is evidence of registering handlers when you have module-separated handlers. `events.register` was doing well, but blindly importing modules to register handlers and don't use them(modules) doesn't seem like a good idea.


Example of registering router in bot application


.. code-block:: python

    # handlers/messages.py
    from garnet.router import Router

    router = Router()

    @router()
    async def handler(event): ...

    # handlers/cb_query.py
    from garnet.events import CallbackQuery
    from garnet.router import Router

    router = Router()

    @router(event=CallbackQuery())
    async def handler(event): ...

    # entry.py ()
    from garnet import TelegramClient

    from handlers import messages, cb_query

    tg = TelegramClient.from_env().start_as_bot()
    tg.bind_routers(messages, cb_query)
    ...

`TelethonRouter` and `Router` both have following remarkable methods:

::

    .message_handler(*filters)
    .callback_query_handler(*filters)
    .chat_action_handler(*filters)
    .message_edited_handler(*filters)
    .album_handler(*filters)

--------------------
🍬 Context magic
--------------------

One of the sweetest parts of garnet. Using `contextvars` we reach incredibly beautiful code :D
*this is not FSMContext don't confuse with context magic provided by contextvars*

As an example, bot that doesn't requires `TelegramClient` to answer messages directly.

.. code-block:: python

    from garnet.functions.messages import reply, message, respond

    @bot.message_handler()
    async def handler():
        # message() - function to get current Message event
        await message().respond("ok")
        await message().reply("ok")
        # the same result, but shortcuts
        await respond("ok")
        await reply("Ok")



-----------------
What's more ❓
-----------------

Class-based handlers are also can be implemented with garnet conveniently. Use your imagination and ``garnet::callbacks::base::Callback`` as a parent class

Awesome bitwise operation supported filters(I highly recommend to use them)::

    # & (conjunction), | (disjunction), ~ (inversion), ^ (exclusive disjunction)
    # also: ==, != (idk why)
    @bot.on(MessageText.exact(".") | MessageText.exact(".."))


``Len`` attribute in ``MessageText`` which has cmp methods::


    @bot.on((MessageText.Len <= 14) | (MessageText.Len >= 88))


Using `client = TelegramClient.start` assignment and start client on the fly, make annotation or typing.cast to have better hints.

---------------
About
---------------

You can find me in tg by `@martin_winks <https://telegram.me/martin_winks>`_ and yeah I receive donates as well as all contributors do(support `lonamiwebs <http://paypal.me/lonamiwebs>`_ and `JRootJunior <https://opencollective.com/aiogram/organization/0/website>`_).


---------------------
🤗 Credits
---------------------

Finite-state machine was ported from cool BotAPI library 'aiogram', special thanks to Alex_

